Fix IllegalStateException when clicking login after logout
The previous login result was stored as live data and prevented AuthenticationFragment from being shown properly. However, an attempt to destroy RecipesFragment was made. This attempt caused IllegalStateException when accessing view in onDestroyView.
This commit is contained in:
@@ -9,7 +9,7 @@ import timber.log.Timber
|
||||
class RecipeViewHolder(
|
||||
private val binding: ViewHolderRecipeBinding,
|
||||
private val recipeViewModel: RecipeViewModel,
|
||||
private val clickListener: (RecipeSummaryEntity) -> Unit
|
||||
private val clickListener: (RecipeSummaryEntity) -> Unit,
|
||||
) : RecyclerView.ViewHolder(binding.root) {
|
||||
private val loadingPlaceholder by lazy {
|
||||
binding.root.resources.getString(R.string.view_holder_recipe_text_placeholder)
|
||||
|
||||
@@ -1,8 +1,6 @@
|
||||
package gq.kirmanak.mealient.ui.recipes
|
||||
|
||||
import android.widget.ImageView
|
||||
import androidx.lifecycle.LiveData
|
||||
import androidx.lifecycle.MutableLiveData
|
||||
import androidx.lifecycle.ViewModel
|
||||
import androidx.lifecycle.viewModelScope
|
||||
import androidx.paging.cachedIn
|
||||
@@ -10,34 +8,17 @@ import dagger.hilt.android.lifecycle.HiltViewModel
|
||||
import gq.kirmanak.mealient.data.recipes.RecipeImageLoader
|
||||
import gq.kirmanak.mealient.data.recipes.RecipeRepo
|
||||
import gq.kirmanak.mealient.data.recipes.db.entity.RecipeSummaryEntity
|
||||
import kotlinx.coroutines.channels.Channel
|
||||
import kotlinx.coroutines.flow.Flow
|
||||
import kotlinx.coroutines.flow.collect
|
||||
import kotlinx.coroutines.flow.receiveAsFlow
|
||||
import kotlinx.coroutines.launch
|
||||
import timber.log.Timber
|
||||
import javax.inject.Inject
|
||||
|
||||
@HiltViewModel
|
||||
class RecipeViewModel @Inject constructor(
|
||||
private val recipeRepo: RecipeRepo,
|
||||
recipeRepo: RecipeRepo,
|
||||
private val recipeImageLoader: RecipeImageLoader
|
||||
) : ViewModel() {
|
||||
private var _isRefreshing = MutableLiveData<Boolean>()
|
||||
val isRefreshing: LiveData<Boolean> get() = _isRefreshing
|
||||
|
||||
private val _nextRecipeInfoChannel = Channel<RecipeSummaryEntity>()
|
||||
val nextRecipeInfo: Flow<RecipeSummaryEntity> =
|
||||
_nextRecipeInfoChannel.receiveAsFlow()
|
||||
|
||||
val adapter = RecipesPagingAdapter(this) {
|
||||
Timber.d("onClick: recipe clicked $it")
|
||||
viewModelScope.launch { _nextRecipeInfoChannel.send(it) }
|
||||
}
|
||||
|
||||
init {
|
||||
setupAdapter()
|
||||
}
|
||||
val pagingData = recipeRepo.createPager().flow.cachedIn(viewModelScope)
|
||||
|
||||
fun loadRecipeImage(view: ImageView, recipeSummary: RecipeSummaryEntity?) {
|
||||
Timber.v("loadRecipeImage() called with: view = $view, recipeSummary = $recipeSummary")
|
||||
@@ -45,21 +26,4 @@ class RecipeViewModel @Inject constructor(
|
||||
recipeImageLoader.loadRecipeImage(view, recipeSummary?.slug)
|
||||
}
|
||||
}
|
||||
|
||||
private fun setupAdapter() {
|
||||
with(viewModelScope) {
|
||||
launch {
|
||||
recipeRepo.createPager().flow.cachedIn(this).collect {
|
||||
Timber.d("setupAdapter: received data update")
|
||||
adapter.submitData(it)
|
||||
}
|
||||
}
|
||||
launch {
|
||||
adapter.onPagesUpdatedFlow.collect {
|
||||
Timber.d("setupAdapter: pages have been updated")
|
||||
_isRefreshing.value = false
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -6,17 +6,16 @@ import androidx.appcompat.app.AppCompatActivity
|
||||
import androidx.fragment.app.Fragment
|
||||
import androidx.fragment.app.activityViewModels
|
||||
import androidx.fragment.app.viewModels
|
||||
import androidx.lifecycle.lifecycleScope
|
||||
import androidx.navigation.fragment.findNavController
|
||||
import by.kirich1409.viewbindingdelegate.viewBinding
|
||||
import dagger.hilt.android.AndroidEntryPoint
|
||||
import gq.kirmanak.mealient.R
|
||||
import gq.kirmanak.mealient.data.recipes.db.entity.RecipeSummaryEntity
|
||||
import gq.kirmanak.mealient.databinding.FragmentRecipesBinding
|
||||
import gq.kirmanak.mealient.extensions.refreshesLiveData
|
||||
import gq.kirmanak.mealient.extensions.collectWithViewLifecycle
|
||||
import gq.kirmanak.mealient.extensions.refreshRequestFlow
|
||||
import gq.kirmanak.mealient.ui.auth.AuthenticationState
|
||||
import gq.kirmanak.mealient.ui.auth.AuthenticationViewModel
|
||||
import kotlinx.coroutines.flow.collect
|
||||
import timber.log.Timber
|
||||
|
||||
@AndroidEntryPoint
|
||||
@@ -58,20 +57,19 @@ class RecipesFragment : Fragment(R.layout.fragment_recipes) {
|
||||
|
||||
private fun setupRecipeAdapter() {
|
||||
Timber.v("setupRecipeAdapter() called")
|
||||
binding.recipes.adapter = viewModel.adapter
|
||||
viewModel.isRefreshing.observe(viewLifecycleOwner) {
|
||||
Timber.d("setupRecipeAdapter: isRefreshing = $it")
|
||||
binding.refresher.isRefreshing = it
|
||||
val adapter = RecipesPagingAdapter(viewModel, ::navigateToRecipeInfo)
|
||||
binding.recipes.adapter = adapter
|
||||
collectWithViewLifecycle(viewModel.pagingData) {
|
||||
Timber.v("setupRecipeAdapter: received data update")
|
||||
adapter.submitData(lifecycle, it)
|
||||
}
|
||||
binding.refresher.refreshesLiveData().observe(viewLifecycleOwner) {
|
||||
Timber.d("setupRecipeAdapter: received refresh request")
|
||||
viewModel.adapter.refresh()
|
||||
collectWithViewLifecycle(adapter.onPagesUpdatedFlow) {
|
||||
Timber.v("setupRecipeAdapter: pages updated")
|
||||
binding.refresher.isRefreshing = false
|
||||
}
|
||||
viewLifecycleOwner.lifecycleScope.launchWhenResumed {
|
||||
viewModel.nextRecipeInfo.collect {
|
||||
Timber.d("setupRecipeAdapter: navigating to recipe $it")
|
||||
navigateToRecipeInfo(it)
|
||||
}
|
||||
collectWithViewLifecycle(binding.refresher.refreshRequestFlow()) {
|
||||
Timber.v("setupRecipeAdapter: received refresh request")
|
||||
adapter.refresh()
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user