Merge pull request #913 from hoangchungk53qx1/main

Refactor Search Feature
pull/870/merge
Adetunji Dahunsi 1 year ago committed by GitHub
commit 4716c7f841
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

@ -19,7 +19,7 @@ package com.google.samples.apps.nowinandroid.feature.search
import com.google.samples.apps.nowinandroid.core.data.model.RecentSearchQuery import com.google.samples.apps.nowinandroid.core.data.model.RecentSearchQuery
sealed interface RecentSearchQueriesUiState { sealed interface RecentSearchQueriesUiState {
object Loading : RecentSearchQueriesUiState data object Loading : RecentSearchQueriesUiState
data class Success( data class Success(
val recentQueries: List<RecentSearchQuery> = emptyList(), val recentQueries: List<RecentSearchQuery> = emptyList(),

@ -20,16 +20,16 @@ import com.google.samples.apps.nowinandroid.core.model.data.FollowableTopic
import com.google.samples.apps.nowinandroid.core.model.data.UserNewsResource import com.google.samples.apps.nowinandroid.core.model.data.UserNewsResource
sealed interface SearchResultUiState { sealed interface SearchResultUiState {
object Loading : SearchResultUiState data object Loading : SearchResultUiState
/** /**
* The state query is empty or too short. To distinguish the state between the * The state query is empty or too short. To distinguish the state between the
* (initial state or when the search query is cleared) vs the state where no search * (initial state or when the search query is cleared) vs the state where no search
* result is returned, explicitly define the empty query state. * result is returned, explicitly define the empty query state.
*/ */
object EmptyQuery : SearchResultUiState data object EmptyQuery : SearchResultUiState
object LoadFailed : SearchResultUiState data object LoadFailed : SearchResultUiState
data class Success( data class Success(
val topics: List<FollowableTopic> = emptyList(), val topics: List<FollowableTopic> = emptyList(),
@ -42,5 +42,5 @@ sealed interface SearchResultUiState {
* A state where the search contents are not ready. This happens when the *Fts tables are not * A state where the search contents are not ready. This happens when the *Fts tables are not
* populated yet. * populated yet.
*/ */
object SearchNotReady : SearchResultUiState data object SearchNotReady : SearchResultUiState
} }

@ -441,9 +441,7 @@ private fun RecentSearchesBody(
style = MaterialTheme.typography.headlineSmall, style = MaterialTheme.typography.headlineSmall,
modifier = Modifier modifier = Modifier
.padding(vertical = 16.dp) .padding(vertical = 16.dp)
.clickable { .clickable { onRecentSearchClicked(recentSearch) }
onRecentSearchClicked(recentSearch)
}
.fillMaxWidth(), .fillMaxWidth(),
) )
} }

@ -48,10 +48,11 @@ class SearchViewModel @Inject constructor(
private val analyticsHelper: AnalyticsHelper, private val analyticsHelper: AnalyticsHelper,
) : ViewModel() { ) : ViewModel() {
val searchQuery = savedStateHandle.getStateFlow(SEARCH_QUERY, "") val searchQuery = savedStateHandle.getStateFlow(key = SEARCH_QUERY, initialValue = "")
val searchResultUiState: StateFlow<SearchResultUiState> = val searchResultUiState: StateFlow<SearchResultUiState> =
getSearchContentsCountUseCase().flatMapLatest { totalCount -> getSearchContentsCountUseCase()
.flatMapLatest { totalCount ->
if (totalCount < SEARCH_MIN_FTS_ENTITY_COUNT) { if (totalCount < SEARCH_MIN_FTS_ENTITY_COUNT) {
flowOf(SearchResultUiState.SearchNotReady) flowOf(SearchResultUiState.SearchNotReady)
} else { } else {
@ -59,22 +60,17 @@ class SearchViewModel @Inject constructor(
if (query.length < SEARCH_QUERY_MIN_LENGTH) { if (query.length < SEARCH_QUERY_MIN_LENGTH) {
flowOf(SearchResultUiState.EmptyQuery) flowOf(SearchResultUiState.EmptyQuery)
} else { } else {
getSearchContentsUseCase(query).asResult().map { getSearchContentsUseCase(query)
when (it) { .asResult()
is Result.Success -> { .map { result ->
SearchResultUiState.Success( when (result) {
topics = it.data.topics, is Result.Success -> SearchResultUiState.Success(
newsResources = it.data.newsResources, topics = result.data.topics,
newsResources = result.data.newsResources,
) )
}
is Result.Loading -> {
SearchResultUiState.Loading
}
is Result.Error -> { is Result.Loading -> SearchResultUiState.Loading
SearchResultUiState.LoadFailed is Result.Error -> SearchResultUiState.LoadFailed
}
} }
} }
} }
@ -87,7 +83,8 @@ class SearchViewModel @Inject constructor(
) )
val recentSearchQueriesUiState: StateFlow<RecentSearchQueriesUiState> = val recentSearchQueriesUiState: StateFlow<RecentSearchQueriesUiState> =
recentSearchQueriesUseCase().map(RecentSearchQueriesUiState::Success) recentSearchQueriesUseCase()
.map(RecentSearchQueriesUiState::Success)
.stateIn( .stateIn(
scope = viewModelScope, scope = viewModelScope,
started = SharingStarted.WhileSubscribed(5_000), started = SharingStarted.WhileSubscribed(5_000),
@ -107,16 +104,9 @@ class SearchViewModel @Inject constructor(
*/ */
fun onSearchTriggered(query: String) { fun onSearchTriggered(query: String) {
viewModelScope.launch { viewModelScope.launch {
recentSearchRepository.insertOrReplaceRecentSearch(query) recentSearchRepository.insertOrReplaceRecentSearch(searchQuery = query)
} }
analyticsHelper.logEvent( analyticsHelper.logEventSearchTriggered(query = query)
AnalyticsEvent(
type = SEARCH_QUERY,
extras = listOf(
Param(SEARCH_QUERY, query),
),
),
)
} }
fun clearRecentSearches() { fun clearRecentSearches() {
@ -126,6 +116,14 @@ class SearchViewModel @Inject constructor(
} }
} }
private fun AnalyticsHelper.logEventSearchTriggered(query: String) =
logEvent(
event = AnalyticsEvent(
type = SEARCH_QUERY,
extras = listOf(element = Param(key = SEARCH_QUERY, value = query)),
),
)
/** Minimum length where search query is considered as [SearchResultUiState.EmptyQuery] */ /** Minimum length where search query is considered as [SearchResultUiState.EmptyQuery] */
private const val SEARCH_QUERY_MIN_LENGTH = 2 private const val SEARCH_QUERY_MIN_LENGTH = 2

Loading…
Cancel
Save