Skip to content

Commit 281d0dd

Browse files
authored
Merge pull request #847 from vector-im/feature/bma/blockUserUx
Improve block/unblock user ux
2 parents 087f5a2 + 38b91a7 commit 281d0dd

File tree

9 files changed

+128
-34
lines changed

9 files changed

+128
-34
lines changed

features/roomdetails/impl/src/main/kotlin/io/element/android/features/roomdetails/impl/blockuser/BlockUserSection.kt

Lines changed: 50 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -25,28 +25,65 @@ import androidx.compose.ui.res.stringResource
2525
import io.element.android.features.roomdetails.impl.R
2626
import io.element.android.features.roomdetails.impl.members.details.RoomMemberDetailsEvents
2727
import io.element.android.features.roomdetails.impl.members.details.RoomMemberDetailsState
28+
import io.element.android.libraries.architecture.Async
29+
import io.element.android.libraries.core.bool.orFalse
2830
import io.element.android.libraries.designsystem.components.dialogs.ConfirmationDialog
31+
import io.element.android.libraries.designsystem.components.dialogs.RetryDialog
2932
import io.element.android.libraries.designsystem.components.preferences.PreferenceCategory
3033
import io.element.android.libraries.designsystem.components.preferences.PreferenceText
34+
import io.element.android.libraries.ui.strings.CommonStrings
3135

3236
@Composable
3337
internal fun BlockUserSection(state: RoomMemberDetailsState, modifier: Modifier = Modifier) {
3438
PreferenceCategory(showDivider = false, modifier = modifier) {
35-
if (state.isBlocked) {
36-
PreferenceText(
37-
title = stringResource(R.string.screen_dm_details_unblock_user),
38-
icon = Icons.Outlined.Block,
39-
onClick = { state.eventSink(RoomMemberDetailsEvents.UnblockUser(needsConfirmation = true)) },
40-
)
41-
} else {
42-
PreferenceText(
43-
title = stringResource(R.string.screen_dm_details_block_user),
44-
icon = Icons.Outlined.Block,
45-
tintColor = MaterialTheme.colorScheme.error,
46-
onClick = { state.eventSink(RoomMemberDetailsEvents.BlockUser(needsConfirmation = true)) },
47-
)
39+
when (state.isBlocked) {
40+
is Async.Failure -> PreferenceBlockUser(isBlocked = state.isBlocked.prevData, isLoading = false, eventSink = state.eventSink)
41+
is Async.Loading -> PreferenceBlockUser(isBlocked = state.isBlocked.prevData, isLoading = true, eventSink = state.eventSink)
42+
is Async.Success -> PreferenceBlockUser(isBlocked = state.isBlocked.data, isLoading = false, eventSink = state.eventSink)
43+
Async.Uninitialized -> PreferenceBlockUser(isBlocked = null, isLoading = true, eventSink = state.eventSink)
4844
}
4945
}
46+
if (state.isBlocked is Async.Failure) {
47+
RetryDialog(
48+
content = stringResource(CommonStrings.error_unknown),
49+
onDismiss = { state.eventSink(RoomMemberDetailsEvents.ClearBlockUserError) },
50+
onRetry = {
51+
val event = when (state.isBlocked.prevData) {
52+
true -> RoomMemberDetailsEvents.UnblockUser(needsConfirmation = false)
53+
false -> RoomMemberDetailsEvents.BlockUser(needsConfirmation = false)
54+
null -> /*Should not happen */ RoomMemberDetailsEvents.ClearBlockUserError
55+
}
56+
state.eventSink(event)
57+
},
58+
)
59+
}
60+
}
61+
62+
@Composable
63+
private fun PreferenceBlockUser(
64+
isBlocked: Boolean?,
65+
isLoading: Boolean,
66+
eventSink: (RoomMemberDetailsEvents) -> Unit,
67+
modifier: Modifier = Modifier,
68+
) {
69+
if (isBlocked.orFalse()) {
70+
PreferenceText(
71+
title = stringResource(R.string.screen_dm_details_unblock_user),
72+
icon = Icons.Outlined.Block,
73+
onClick = { if (!isLoading) eventSink(RoomMemberDetailsEvents.UnblockUser(needsConfirmation = true)) },
74+
loadingCurrentValue = isLoading,
75+
modifier = modifier,
76+
)
77+
} else {
78+
PreferenceText(
79+
title = stringResource(R.string.screen_dm_details_block_user),
80+
icon = Icons.Outlined.Block,
81+
tintColor = MaterialTheme.colorScheme.error,
82+
onClick = { if (!isLoading) eventSink(RoomMemberDetailsEvents.BlockUser(needsConfirmation = true)) },
83+
loadingCurrentValue = isLoading,
84+
modifier = modifier,
85+
)
86+
}
5087
}
5188

5289
@Composable

features/roomdetails/impl/src/main/kotlin/io/element/android/features/roomdetails/impl/members/details/RoomMemberDetailsEvents.kt

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -19,5 +19,6 @@ package io.element.android.features.roomdetails.impl.members.details
1919
sealed interface RoomMemberDetailsEvents {
2020
data class BlockUser(val needsConfirmation: Boolean = false) : RoomMemberDetailsEvents
2121
data class UnblockUser(val needsConfirmation: Boolean = false) : RoomMemberDetailsEvents
22+
object ClearBlockUserError : RoomMemberDetailsEvents
2223
object ClearConfirmationDialog : RoomMemberDetailsEvents
2324
}

features/roomdetails/impl/src/main/kotlin/io/element/android/features/roomdetails/impl/members/details/RoomMemberDetailsPresenter.kt

Lines changed: 33 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@ import androidx.compose.runtime.setValue
2828
import dagger.assisted.Assisted
2929
import dagger.assisted.AssistedInject
3030
import io.element.android.features.roomdetails.impl.members.details.RoomMemberDetailsState.ConfirmationDialog
31+
import io.element.android.libraries.architecture.Async
3132
import io.element.android.libraries.architecture.Presenter
3233
import io.element.android.libraries.core.bool.orFalse
3334
import io.element.android.libraries.matrix.api.MatrixClient
@@ -53,8 +54,13 @@ class RoomMemberDetailsPresenter @AssistedInject constructor(
5354
var confirmationDialog by remember { mutableStateOf<ConfirmationDialog?>(null) }
5455
val roomMember by room.getRoomMemberAsState(roomMemberId)
5556
// the room member is not really live...
56-
val isBlocked = remember {
57-
mutableStateOf(roomMember?.isIgnored.orFalse())
57+
val isBlocked: MutableState<Async<Boolean>> = remember(roomMember) {
58+
val isIgnored = roomMember?.isIgnored
59+
if (isIgnored == null) {
60+
mutableStateOf(Async.Uninitialized)
61+
} else {
62+
mutableStateOf(Async.Success(isIgnored))
63+
}
5864
}
5965
LaunchedEffect(Unit) {
6066
room.updateMembers()
@@ -79,6 +85,9 @@ class RoomMemberDetailsPresenter @AssistedInject constructor(
7985
}
8086
}
8187
RoomMemberDetailsEvents.ClearConfirmationDialog -> confirmationDialog = null
88+
RoomMemberDetailsEvents.ClearBlockUserError -> {
89+
isBlocked.value = Async.Success(isBlocked.value.dataOrNull().orFalse())
90+
}
8291
}
8392
}
8493

@@ -105,20 +114,31 @@ class RoomMemberDetailsPresenter @AssistedInject constructor(
105114
)
106115
}
107116

108-
private fun CoroutineScope.blockUser(userId: UserId, isBlockedState: MutableState<Boolean>) = launch {
117+
private fun CoroutineScope.blockUser(userId: UserId, isBlockedState: MutableState<Async<Boolean>>) = launch {
118+
isBlockedState.value = Async.Loading(false)
109119
client.ignoreUser(userId)
110-
.map {
111-
isBlockedState.value = true
112-
room.updateMembers()
113-
}
114-
120+
.fold(
121+
onSuccess = {
122+
isBlockedState.value = Async.Success(true)
123+
room.updateMembers()
124+
},
125+
onFailure = {
126+
isBlockedState.value = Async.Failure(it, false)
127+
}
128+
)
115129
}
116130

117-
private fun CoroutineScope.unblockUser(userId: UserId, isBlockedState: MutableState<Boolean>) = launch {
131+
private fun CoroutineScope.unblockUser(userId: UserId, isBlockedState: MutableState<Async<Boolean>>) = launch {
132+
isBlockedState.value = Async.Loading(true)
118133
client.unignoreUser(userId)
119-
.map {
120-
isBlockedState.value = false
121-
room.updateMembers()
122-
}
134+
.fold(
135+
onSuccess = {
136+
isBlockedState.value = Async.Success(false)
137+
room.updateMembers()
138+
},
139+
onFailure = {
140+
isBlockedState.value = Async.Failure(it, true)
141+
}
142+
)
123143
}
124144
}

features/roomdetails/impl/src/main/kotlin/io/element/android/features/roomdetails/impl/members/details/RoomMemberDetailsState.kt

Lines changed: 3 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -16,11 +16,13 @@
1616

1717
package io.element.android.features.roomdetails.impl.members.details
1818

19+
import io.element.android.libraries.architecture.Async
20+
1921
data class RoomMemberDetailsState(
2022
val userId: String,
2123
val userName: String?,
2224
val avatarUrl: String?,
23-
val isBlocked: Boolean,
25+
val isBlocked: Async<Boolean>,
2426
val displayConfirmationDialog: ConfirmationDialog? = null,
2527
val isCurrentUser: Boolean,
2628
val eventSink: (RoomMemberDetailsEvents) -> Unit

features/roomdetails/impl/src/main/kotlin/io/element/android/features/roomdetails/impl/members/details/RoomMemberDetailsStateProvider.kt

Lines changed: 4 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -17,15 +17,17 @@
1717
package io.element.android.features.roomdetails.impl.members.details
1818

1919
import androidx.compose.ui.tooling.preview.PreviewParameterProvider
20+
import io.element.android.libraries.architecture.Async
2021

2122
open class RoomMemberDetailsStateProvider : PreviewParameterProvider<RoomMemberDetailsState> {
2223
override val values: Sequence<RoomMemberDetailsState>
2324
get() = sequenceOf(
2425
aRoomMemberDetailsState(),
2526
aRoomMemberDetailsState().copy(userName = null),
26-
aRoomMemberDetailsState().copy(isBlocked = true),
27+
aRoomMemberDetailsState().copy(isBlocked = Async.Success(true)),
2728
aRoomMemberDetailsState().copy(displayConfirmationDialog = RoomMemberDetailsState.ConfirmationDialog.Block),
2829
aRoomMemberDetailsState().copy(displayConfirmationDialog = RoomMemberDetailsState.ConfirmationDialog.Unblock),
30+
aRoomMemberDetailsState().copy(isBlocked = Async.Loading(true)),
2931
// Add other states here
3032
)
3133
}
@@ -34,7 +36,7 @@ fun aRoomMemberDetailsState() = RoomMemberDetailsState(
3436
userId = "@daniel:domain.com",
3537
userName = "Daniel",
3638
avatarUrl = null,
37-
isBlocked = false,
39+
isBlocked = Async.Success(false),
3840
isCurrentUser = false,
3941
eventSink = {},
4042
)

features/roomdetails/impl/src/test/kotlin/io/element/android/features/roomdetails/members/details/RoomMemberDetailsPresenterTests.kt

Lines changed: 28 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,9 @@ import io.element.android.features.roomdetails.impl.members.aRoomMember
2525
import io.element.android.features.roomdetails.impl.members.details.RoomMemberDetailsEvents
2626
import io.element.android.features.roomdetails.impl.members.details.RoomMemberDetailsPresenter
2727
import io.element.android.features.roomdetails.impl.members.details.RoomMemberDetailsState
28+
import io.element.android.libraries.architecture.Async
2829
import io.element.android.libraries.matrix.api.room.MatrixRoomMembersState
30+
import io.element.android.libraries.matrix.test.A_THROWABLE
2931
import io.element.android.libraries.matrix.test.FakeMatrixClient
3032
import kotlinx.coroutines.ExperimentalCoroutinesApi
3133
import kotlinx.coroutines.test.runTest
@@ -50,7 +52,7 @@ class RoomMemberDetailsPresenterTests {
5052
Truth.assertThat(initialState.userId).isEqualTo(roomMember.userId.value)
5153
Truth.assertThat(initialState.userName).isEqualTo(roomMember.displayName)
5254
Truth.assertThat(initialState.avatarUrl).isEqualTo(roomMember.avatarUrl)
53-
Truth.assertThat(initialState.isBlocked).isEqualTo(roomMember.isIgnored)
55+
Truth.assertThat(initialState.isBlocked).isEqualTo(Async.Success(roomMember.isIgnored))
5456
skipItems(1)
5557
val loadedState = awaitItem()
5658
Truth.assertThat(loadedState.userName).isEqualTo("A custom name")
@@ -129,10 +131,33 @@ class RoomMemberDetailsPresenterTests {
129131
}.test {
130132
val initialState = awaitItem()
131133
initialState.eventSink(RoomMemberDetailsEvents.BlockUser(needsConfirmation = false))
132-
Truth.assertThat(awaitItem().isBlocked).isTrue()
134+
Truth.assertThat(awaitItem().isBlocked.isLoading()).isTrue()
135+
Truth.assertThat(awaitItem().isBlocked.dataOrNull()).isTrue()
133136

134137
initialState.eventSink(RoomMemberDetailsEvents.UnblockUser(needsConfirmation = false))
135-
Truth.assertThat(awaitItem().isBlocked).isFalse()
138+
Truth.assertThat(awaitItem().isBlocked.isLoading()).isTrue()
139+
Truth.assertThat(awaitItem().isBlocked.dataOrNull()).isFalse()
140+
}
141+
}
142+
143+
@Test
144+
fun `present - BlockUser with error`() = runTest {
145+
val room = aMatrixRoom()
146+
val roomMember = aRoomMember()
147+
val matrixClient = FakeMatrixClient()
148+
matrixClient.givenIgnoreUserResult(Result.failure(A_THROWABLE))
149+
val presenter = RoomMemberDetailsPresenter(matrixClient, room, roomMember.userId)
150+
moleculeFlow(RecompositionClock.Immediate) {
151+
presenter.present()
152+
}.test {
153+
val initialState = awaitItem()
154+
initialState.eventSink(RoomMemberDetailsEvents.BlockUser(needsConfirmation = false))
155+
Truth.assertThat(awaitItem().isBlocked.isLoading()).isTrue()
156+
val errorState = awaitItem()
157+
Truth.assertThat(errorState.isBlocked.errorOrNull()).isEqualTo(A_THROWABLE)
158+
// Clear error
159+
initialState.eventSink(RoomMemberDetailsEvents.ClearBlockUserError)
160+
Truth.assertThat(awaitItem().isBlocked).isEqualTo(Async.Success(false))
136161
}
137162
}
138163

libraries/matrix/test/src/main/kotlin/io/element/android/libraries/matrix/test/FakeMatrixClient.kt

Lines changed: 3 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,7 @@ import io.element.android.libraries.matrix.test.room.FakeMatrixRoom
3838
import io.element.android.libraries.matrix.test.room.FakeRoomSummaryDataSource
3939
import io.element.android.libraries.matrix.test.sync.FakeSyncService
4040
import io.element.android.libraries.matrix.test.verification.FakeSessionVerificationService
41+
import io.element.android.tests.testutils.simulateLongTask
4142
import kotlinx.coroutines.delay
4243

4344
class FakeMatrixClient(
@@ -72,11 +73,11 @@ class FakeMatrixClient(
7273
return findDmResult
7374
}
7475

75-
override suspend fun ignoreUser(userId: UserId): Result<Unit> {
76+
override suspend fun ignoreUser(userId: UserId): Result<Unit> = simulateLongTask {
7677
return ignoreUserResult
7778
}
7879

79-
override suspend fun unignoreUser(userId: UserId): Result<Unit> {
80+
override suspend fun unignoreUser(userId: UserId): Result<Unit> = simulateLongTask {
8081
return unignoreUserResult
8182
}
8283

0 commit comments

Comments
 (0)