From f21c93e11c6dcace90f7bb0a65921356811f094b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Micha=C5=82=20Saleniuk?= <30429749+saleniuk@users.noreply.github.com> Date: Thu, 9 Jan 2025 15:32:23 +0100 Subject: [PATCH] =?UTF-8?q?fix:=20reusing=20PagingData=20crash=20[WPB-1505?= =?UTF-8?q?5][WPB-15079][WPB-15064]=20=F0=9F=8D=92=20(#3778)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Yamil Medina --- .../common/topappbar/search/SearchBarState.kt | 22 ++- .../android/ui/home/archive/ArchiveScreen.kt | 9 +- .../ConversationListViewModel.kt | 150 ++++++++++-------- .../ConversationsScreenContent.kt | 6 +- .../all/AllConversationsScreen.kt | 9 +- .../common/ConversationList.kt | 13 +- .../ConversationListViewModelTest.kt | 82 +++++++++- 7 files changed, 200 insertions(+), 91 deletions(-) diff --git a/app/src/main/kotlin/com/wire/android/ui/common/topappbar/search/SearchBarState.kt b/app/src/main/kotlin/com/wire/android/ui/common/topappbar/search/SearchBarState.kt index d87a1193371..8646ca95f5c 100644 --- a/app/src/main/kotlin/com/wire/android/ui/common/topappbar/search/SearchBarState.kt +++ b/app/src/main/kotlin/com/wire/android/ui/common/topappbar/search/SearchBarState.kt @@ -29,11 +29,12 @@ import androidx.compose.runtime.setValue @Composable fun rememberSearchbarState( + initialIsSearchActive: Boolean = false, searchQueryTextState: TextFieldState = rememberTextFieldState() ): SearchBarState = rememberSaveable( - saver = SearchBarState.saver(searchQueryTextState) + saver = SearchBarState.saver() ) { - SearchBarState(searchQueryTextState = searchQueryTextState) + SearchBarState(isSearchActive = initialIsSearchActive, searchQueryTextState = searchQueryTextState) } class SearchBarState( @@ -57,14 +58,23 @@ class SearchBarState( } companion object { - fun saver(searchQueryTextState: TextFieldState): Saver = Saver( + fun saver(): Saver = Saver( save = { - listOf(it.isSearchActive) + listOf( + it.isSearchActive, + with(TextFieldState.Saver) { + save(it.searchQueryTextState) + } + ) }, restore = { SearchBarState( - isSearchActive = it[0], - searchQueryTextState = searchQueryTextState + isSearchActive = (it.getOrNull(0) as? Boolean) ?: false, + searchQueryTextState = it.getOrNull(1)?.let { + with(TextFieldState.Saver) { + restore(it) + } + } ?: TextFieldState() ) } ) diff --git a/app/src/main/kotlin/com/wire/android/ui/home/archive/ArchiveScreen.kt b/app/src/main/kotlin/com/wire/android/ui/home/archive/ArchiveScreen.kt index 597f2d26390..07a1c578927 100644 --- a/app/src/main/kotlin/com/wire/android/ui/home/archive/ArchiveScreen.kt +++ b/app/src/main/kotlin/com/wire/android/ui/home/archive/ArchiveScreen.kt @@ -32,7 +32,6 @@ import com.wire.android.ui.home.conversationslist.common.previewConversationFold import com.wire.android.ui.home.conversationslist.model.ConversationsSource import com.wire.android.ui.theme.WireTheme import com.wire.android.util.ui.PreviewMultipleThemes -import kotlinx.coroutines.flow.flowOf @HomeNavGraph @WireDestination @@ -57,7 +56,7 @@ fun PreviewArchiveEmptyScreen() = WireTheme { searchBarState = rememberSearchbarState(), conversationsSource = ConversationsSource.ARCHIVE, emptyListContent = { ArchiveEmptyContent() }, - conversationListViewModel = ConversationListViewModelPreview(flowOf()), + conversationListViewModel = ConversationListViewModelPreview(previewConversationFoldersFlow(list = listOf())), ) } @@ -66,10 +65,10 @@ fun PreviewArchiveEmptyScreen() = WireTheme { fun PreviewArchiveEmptySearchScreen() = WireTheme { ConversationsScreenContent( navigator = rememberNavigator {}, - searchBarState = rememberSearchbarState(searchQueryTextState = TextFieldState(initialText = "er")), + searchBarState = rememberSearchbarState(initialIsSearchActive = true, searchQueryTextState = TextFieldState(initialText = "er")), conversationsSource = ConversationsSource.ARCHIVE, emptyListContent = { ArchiveEmptyContent() }, - conversationListViewModel = ConversationListViewModelPreview(flowOf()), + conversationListViewModel = ConversationListViewModelPreview(previewConversationFoldersFlow(searchQuery = "er", list = listOf())), ) } @@ -78,7 +77,7 @@ fun PreviewArchiveEmptySearchScreen() = WireTheme { fun PreviewArchiveScreen() = WireTheme { ConversationsScreenContent( navigator = rememberNavigator {}, - searchBarState = rememberSearchbarState(searchQueryTextState = TextFieldState(initialText = "er")), + searchBarState = rememberSearchbarState(initialIsSearchActive = true, searchQueryTextState = TextFieldState(initialText = "er")), conversationsSource = ConversationsSource.ARCHIVE, emptyListContent = { ArchiveEmptyContent() }, conversationListViewModel = ConversationListViewModelPreview(previewConversationFoldersFlow(searchQuery = "er")), diff --git a/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/ConversationListViewModel.kt b/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/ConversationListViewModel.kt index 334063536d1..feb163076b6 100644 --- a/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/ConversationListViewModel.kt +++ b/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/ConversationListViewModel.kt @@ -135,7 +135,7 @@ class ConversationListViewModelPreview( class ConversationListViewModelImpl @AssistedInject constructor( @Assisted val conversationsSource: ConversationsSource, @Assisted private val usePagination: Boolean = BuildConfig.PAGINATED_CONVERSATION_LIST_ENABLED, - dispatcher: DispatcherProvider, + private val dispatcher: DispatcherProvider, private val updateConversationMutedStatus: UpdateConversationMutedStatusUseCase, private val getConversationsPaginated: GetConversationsFromSearchUseCase, private val observeConversationListDetailsWithEvents: ObserveConversationListDetailsWithEventsUseCase, @@ -171,6 +171,7 @@ class ConversationListViewModelImpl @AssistedInject constructor( override val closeBottomSheet = MutableSharedFlow() private val searchQueryFlow: MutableStateFlow = MutableStateFlow("") + private val isSelfUserUnderLegalHoldFlow = MutableSharedFlow(replay = 1) private val containsNewActivitiesSection = when (conversationsSource) { ConversationsSource.MAIN, @@ -185,39 +186,38 @@ class ConversationListViewModelImpl @AssistedInject constructor( private val conversationsPaginatedFlow: Flow> = searchQueryFlow .debounce { if (it.isEmpty()) 0L else DEFAULT_SEARCH_QUERY_DEBOUNCE } .onStart { emit("") } + .combine(isSelfUserUnderLegalHoldFlow, ::Pair) .distinctUntilChanged() - .flatMapLatest { searchQuery -> + .flatMapLatest { (searchQuery, isSelfUserUnderLegalHold) -> getConversationsPaginated( searchQuery = searchQuery, fromArchive = conversationsSource == ConversationsSource.ARCHIVE, conversationFilter = conversationsSource.toFilter(), onlyInteractionEnabled = false, newActivitiesOnTop = containsNewActivitiesSection, - ).combine(observeLegalHoldStateForSelfUser()) { conversations, selfUserLegalHoldStatus -> - conversations.map { - it.hideIndicatorForSelfUserUnderLegalHold(selfUserLegalHoldStatus) - } - }.map { - it.insertSeparators { before, after -> - when { - // do not add separators if the list shouldn't show conversations grouped into different folders - !containsNewActivitiesSection -> null - - before == null && after != null && after.hasNewActivitiesToShow -> - // list starts with items with "new activities" - ConversationFolder.Predefined.NewActivities - - before == null && after != null && !after.hasNewActivitiesToShow -> - // list doesn't contain any items with "new activities" - ConversationFolder.Predefined.Conversations - - before != null && before.hasNewActivitiesToShow && after != null && !after.hasNewActivitiesToShow -> - // end of "new activities" section and beginning of "conversations" section - ConversationFolder.Predefined.Conversations - - else -> null + ).map { pagingData -> + pagingData + .map { it.hideIndicatorForSelfUserUnderLegalHold(isSelfUserUnderLegalHold) } + .insertSeparators { before, after -> + when { + // do not add separators if the list shouldn't show conversations grouped into different folders + !containsNewActivitiesSection -> null + + before == null && after != null && after.hasNewActivitiesToShow -> + // list starts with items with "new activities" + ConversationFolder.Predefined.NewActivities + + before == null && after != null && !after.hasNewActivitiesToShow -> + // list doesn't contain any items with "new activities" + ConversationFolder.Predefined.Conversations + + before != null && before.hasNewActivitiesToShow && after != null && !after.hasNewActivitiesToShow -> + // end of "new activities" section and beginning of "conversations" section + ConversationFolder.Predefined.Conversations + + else -> null + } } - } } } .flowOn(dispatcher.io()) @@ -232,45 +232,59 @@ class ConversationListViewModelImpl @AssistedInject constructor( private set init { + observeSelfUserLegalHoldState() if (!usePagination) { - viewModelScope.launch { - searchQueryFlow - .debounce { if (it.isEmpty()) 0L else DEFAULT_SEARCH_QUERY_DEBOUNCE } - .onStart { emit("") } - .distinctUntilChanged() - .flatMapLatest { searchQuery: String -> - observeConversationListDetailsWithEvents( - fromArchive = conversationsSource == ConversationsSource.ARCHIVE, - conversationFilter = conversationsSource.toFilter() - ).combine(observeLegalHoldStateForSelfUser()) { conversations, selfUserLegalHoldStatus -> - conversations.map { conversationDetails -> - conversationDetails.toConversationItem( - userTypeMapper = userTypeMapper, - searchQuery = searchQuery, - selfUserTeamId = observeSelfUser().firstOrNull()?.teamId - ).hideIndicatorForSelfUserUnderLegalHold(selfUserLegalHoldStatus) - } to searchQuery - } - } - .map { (conversationItems, searchQuery) -> - if (searchQuery.isEmpty()) { - conversationItems.withFolders(source = conversationsSource).toImmutableMap() - } else { - searchConversation( - conversationDetails = conversationItems, - searchQuery = searchQuery - ).withFolders(source = conversationsSource).toImmutableMap() - } + observeNonPaginatedSearchConversationList() + } + } + + private fun observeSelfUserLegalHoldState() { + viewModelScope.launch { + observeLegalHoldStateForSelfUser() + .map { it is LegalHoldStateForSelfUser.Enabled } + .flowOn(dispatcher.io()) + .collect { isSelfUserUnderLegalHoldFlow.emit(it) } + } + } + + private fun observeNonPaginatedSearchConversationList() { + viewModelScope.launch { + searchQueryFlow + .debounce { if (it.isEmpty()) 0L else DEFAULT_SEARCH_QUERY_DEBOUNCE } + .onStart { emit("") } + .distinctUntilChanged() + .flatMapLatest { searchQuery: String -> + observeConversationListDetailsWithEvents( + fromArchive = conversationsSource == ConversationsSource.ARCHIVE, + conversationFilter = conversationsSource.toFilter() + ).combine(isSelfUserUnderLegalHoldFlow) { conversations, isSelfUserUnderLegalHold -> + conversations.map { conversationDetails -> + conversationDetails.toConversationItem( + userTypeMapper = userTypeMapper, + searchQuery = searchQuery, + selfUserTeamId = observeSelfUser().firstOrNull()?.teamId + ).hideIndicatorForSelfUserUnderLegalHold(isSelfUserUnderLegalHold) + } to searchQuery } - .flowOn(dispatcher.io()) - .collect { - conversationListState = ConversationListState.NotPaginated( - isLoading = false, - conversations = it, - domain = currentAccount.domain - ) + } + .map { (conversationItems, searchQuery) -> + if (searchQuery.isEmpty()) { + conversationItems.withFolders(source = conversationsSource).toImmutableMap() + } else { + searchConversation( + conversationDetails = conversationItems, + searchQuery = searchQuery + ).withFolders(source = conversationsSource).toImmutableMap() } - } + } + .flowOn(dispatcher.io()) + .collect { + conversationListState = ConversationListState.NotPaginated( + isLoading = false, + conversations = it, + domain = currentAccount.domain + ) + } } } @@ -485,11 +499,13 @@ private fun ConversationsSource.toFilter(): ConversationFilter = when (this) { is ConversationsSource.FOLDER -> ConversationFilter.Folder(folderId = folderId, folderName = folderName) } -private fun ConversationItem.hideIndicatorForSelfUserUnderLegalHold(selfUserLegalHoldStatus: LegalHoldStateForSelfUser) = - // if self user is under legal hold then we shouldn't show legal hold indicator next to every conversation - // the indication is shown in the header of the conversation list for self user in that case and it's enough - when (selfUserLegalHoldStatus) { - is LegalHoldStateForSelfUser.Enabled -> when (this) { +/** + * If self user is under legal hold then we shouldn't show legal hold indicator next to every conversation as in that case + * the legal hold indication is shown in the header of the conversation list for self user in that case and it's enough. + */ +private fun ConversationItem.hideIndicatorForSelfUserUnderLegalHold(isSelfUserUnderLegalHold: Boolean) = + when (isSelfUserUnderLegalHold) { + true -> when (this) { is ConversationItem.ConnectionConversation -> this.copy(showLegalHoldIndicator = false) is ConversationItem.GroupConversation -> this.copy(showLegalHoldIndicator = false) is ConversationItem.PrivateConversation -> this.copy(showLegalHoldIndicator = false) diff --git a/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/ConversationsScreenContent.kt b/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/ConversationsScreenContent.kt index 438d949008d..6f099dfd3bc 100644 --- a/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/ConversationsScreenContent.kt +++ b/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/ConversationsScreenContent.kt @@ -90,7 +90,6 @@ fun ConversationsScreenContent( lazyListState: LazyListState = rememberLazyListState(), loadingListContent: @Composable (LazyListState) -> Unit = { ConversationListLoadingContent(it) }, conversationsSource: ConversationsSource = ConversationsSource.MAIN, - initiallyLoaded: Boolean = LocalInspectionMode.current, conversationListViewModel: ConversationListViewModel = when { LocalInspectionMode.current -> ConversationListViewModelPreview() else -> hiltViewModel( @@ -191,10 +190,7 @@ fun ConversationsScreenContent( when (val state = conversationListViewModel.conversationListState) { is ConversationListState.Paginated -> { val lazyPagingItems = state.conversations.collectAsLazyPagingItems() - var showLoading by remember(conversationsSource) { mutableStateOf(!initiallyLoaded) } - if (lazyPagingItems.loadState.refresh != LoadState.Loading && showLoading) { - showLoading = false - } + val showLoading = lazyPagingItems.loadState.refresh == LoadState.Loading && lazyPagingItems.itemCount == 0 when { // when conversation list is not yet fetched, show loading indicator diff --git a/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/all/AllConversationsScreen.kt b/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/all/AllConversationsScreen.kt index 3f9ff8fe23a..30f77cecae3 100644 --- a/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/all/AllConversationsScreen.kt +++ b/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/all/AllConversationsScreen.kt @@ -34,7 +34,6 @@ import com.wire.android.ui.home.conversationslist.model.ConversationsSource import com.wire.android.ui.theme.WireTheme import com.wire.android.util.ui.PreviewMultipleThemes import com.wire.kalium.logic.data.conversation.ConversationFilter -import kotlinx.coroutines.flow.flowOf @HomeNavGraph(start = true) @WireDestination @@ -70,7 +69,7 @@ fun PreviewAllConversationsEmptyScreen() = WireTheme { searchBarState = rememberSearchbarState(), conversationsSource = ConversationsSource.MAIN, emptyListContent = { ConversationsEmptyContent() }, - conversationListViewModel = ConversationListViewModelPreview(flowOf()), + conversationListViewModel = ConversationListViewModelPreview(previewConversationFoldersFlow(list = listOf())), ) } @@ -79,10 +78,10 @@ fun PreviewAllConversationsEmptyScreen() = WireTheme { fun PreviewAllConversationsEmptySearchScreen() = WireTheme { ConversationsScreenContent( navigator = rememberNavigator {}, - searchBarState = rememberSearchbarState(searchQueryTextState = TextFieldState(initialText = "er")), + searchBarState = rememberSearchbarState(initialIsSearchActive = true, searchQueryTextState = TextFieldState(initialText = "er")), conversationsSource = ConversationsSource.MAIN, emptyListContent = { ConversationsEmptyContent() }, - conversationListViewModel = ConversationListViewModelPreview(flowOf()), + conversationListViewModel = ConversationListViewModelPreview(previewConversationFoldersFlow(searchQuery = "er", list = listOf())), ) } @@ -91,7 +90,7 @@ fun PreviewAllConversationsEmptySearchScreen() = WireTheme { fun PreviewAllConversationsSearchScreen() = WireTheme { ConversationsScreenContent( navigator = rememberNavigator {}, - searchBarState = rememberSearchbarState(searchQueryTextState = TextFieldState(initialText = "er")), + searchBarState = rememberSearchbarState(initialIsSearchActive = true, searchQueryTextState = TextFieldState(initialText = "er")), conversationsSource = ConversationsSource.MAIN, emptyListContent = { ConversationsEmptyContent() }, conversationListViewModel = ConversationListViewModelPreview(previewConversationFoldersFlow("er")), diff --git a/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/common/ConversationList.kt b/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/common/ConversationList.kt index ed04d4d72bc..27e2b7566f1 100644 --- a/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/common/ConversationList.kt +++ b/app/src/main/kotlin/com/wire/android/ui/home/conversationslist/common/ConversationList.kt @@ -30,6 +30,8 @@ import androidx.compose.runtime.snapshots.Snapshot import androidx.compose.ui.Modifier import androidx.compose.ui.platform.LocalContext import androidx.compose.ui.platform.LocalInspectionMode +import androidx.paging.LoadState +import androidx.paging.LoadStates import androidx.paging.PagingData import androidx.paging.compose.LazyPagingItems import androidx.paging.compose.collectAsLazyPagingItems @@ -235,7 +237,16 @@ fun previewConversationList(count: Int, startIndex: Int = 0, unread: Boolean = f fun previewConversationFoldersFlow( searchQuery: String = "", list: List = previewConversationFolders(searchQuery = searchQuery) -) = flowOf(PagingData.from(list)) +) = flowOf( + PagingData.from( + data = list, + sourceLoadStates = LoadStates( + prepend = LoadState.NotLoading(true), + append = LoadState.NotLoading(true), + refresh = LoadState.NotLoading(true), + ) + ) +) fun previewConversationFolders(withFolders: Boolean = true, searchQuery: String = "", unreadCount: Int = 3, readCount: Int = 6) = buildList { diff --git a/app/src/test/kotlin/com/wire/android/ui/home/conversationslist/ConversationListViewModelTest.kt b/app/src/test/kotlin/com/wire/android/ui/home/conversationslist/ConversationListViewModelTest.kt index 70dcead1d5b..2416c9377ad 100644 --- a/app/src/test/kotlin/com/wire/android/ui/home/conversationslist/ConversationListViewModelTest.kt +++ b/app/src/test/kotlin/com/wire/android/ui/home/conversationslist/ConversationListViewModelTest.kt @@ -62,6 +62,9 @@ import io.mockk.coEvery import io.mockk.coVerify import io.mockk.impl.annotations.MockK import kotlinx.coroutines.ExperimentalCoroutinesApi +import kotlinx.coroutines.flow.Flow +import kotlinx.coroutines.flow.MutableSharedFlow +import kotlinx.coroutines.flow.first import kotlinx.coroutines.flow.flowOf import kotlinx.coroutines.test.advanceUntilIdle import kotlinx.coroutines.test.runTest @@ -234,6 +237,77 @@ class ConversationListViewModelTest { coVerify(exactly = 1) { arrangement.unblockUser(userId) } } + @Test + fun `given cached PagingData, when self user legal hold changes, then should call paginated use case again`() = + runTest(dispatcherProvider.main()) { + // given + val conversations = listOf( + TestConversationItem.CONNECTION.copy(conversationId = ConversationId("conn_1", "")), + TestConversationItem.PRIVATE.copy(conversationId = ConversationId("private_1", "")), + TestConversationItem.GROUP.copy(conversationId = ConversationId("group_1", "")), + ).associateBy { it.conversationId } + val selfUserLegalHoldStateFlow = MutableSharedFlow() + val (arrangement, conversationListViewModel) = Arrangement(conversationsSource = ConversationsSource.MAIN) + .withConversationsPaginated(conversations.values.toList()) + .withSelfUserLegalHoldStateFlow(selfUserLegalHoldStateFlow) + .arrange() + advanceUntilIdle() + + (conversationListViewModel.conversationListState as ConversationListState.Paginated).conversations.test { + // initial legal hold state + selfUserLegalHoldStateFlow.emit(LegalHoldStateForSelfUser.Disabled) + advanceUntilIdle() + + // use case is called initially + coVerify(exactly = 1) { + arrangement.getConversationsPaginated(any(), any(), any(), any(), any()) + } + + // when legal hold state is changed + selfUserLegalHoldStateFlow.emit(LegalHoldStateForSelfUser.Enabled) + advanceUntilIdle() + + // then use case should be called again (in total 2 executions) to create new PagingData + coVerify(exactly = 2) { + arrangement.getConversationsPaginated(any(), any(), any(), any(), any()) + } + + cancelAndIgnoreRemainingEvents() + } + } + + @Test + fun `given cached PagingData, when observing twice, then paginated use case should not be called again`() = + runTest(dispatcherProvider.main()) { + // given + val conversations = listOf( + TestConversationItem.CONNECTION.copy(conversationId = ConversationId("conn_1", "")), + TestConversationItem.PRIVATE.copy(conversationId = ConversationId("private_1", "")), + TestConversationItem.GROUP.copy(conversationId = ConversationId("group_1", "")), + ).associateBy { it.conversationId } + val (arrangement, conversationListViewModel) = Arrangement(conversationsSource = ConversationsSource.MAIN) + .withConversationsPaginated(conversations.values.toList()) + .withSelfUserLegalHoldState(LegalHoldStateForSelfUser.Disabled) + .arrange() + advanceUntilIdle() + + // flow is collected first time + (conversationListViewModel.conversationListState as ConversationListState.Paginated).conversations.first() + + // use case is called initially + coVerify(exactly = 1) { + arrangement.getConversationsPaginated(any(), any(), any(), any(), any()) + } + + // flow is collected second time + (conversationListViewModel.conversationListState as ConversationListState.Paginated).conversations.first() + + // use case should NOT be called again, there should be still only one call + coVerify(exactly = 1) { + arrangement.getConversationsPaginated(any(), any(), any(), any(), any()) + } + } + inner class Arrangement(val conversationsSource: ConversationsSource = ConversationsSource.MAIN) { @MockK lateinit var updateConversationMutedStatus: UpdateConversationMutedStatusUseCase @@ -325,8 +399,12 @@ class ConversationListViewModelTest { ) } - fun withSelfUserLegalHoldState(LegalHoldStateForSelfUser: LegalHoldStateForSelfUser) = apply { - coEvery { observeLegalHoldStateForSelfUserUseCase() } returns flowOf(LegalHoldStateForSelfUser) + fun withSelfUserLegalHoldState(legalHoldStateForSelfUser: LegalHoldStateForSelfUser) = apply { + coEvery { observeLegalHoldStateForSelfUserUseCase() } returns flowOf(legalHoldStateForSelfUser) + } + + fun withSelfUserLegalHoldStateFlow(legalHoldStateForSelfUserFlow: Flow) = apply { + coEvery { observeLegalHoldStateForSelfUserUseCase() } returns legalHoldStateForSelfUserFlow } fun arrange() = this to ConversationListViewModelImpl(