[英]Fragment Back Navigation Blocked by ViewModel
所以我试图用导航组件实现 mvvm 架构,这个代码片段工作得很好。问题是当我从 fragmentA 导航到 FragmentB 时,没有办法将 go 返回到 FragmentA,就像按钮被阻止(如果我删除了 ViewModel,导航工作正常,因此它与 ViewModel 有关)。
有人知道发生了什么吗?
private fun initVewModel() {
myViewModel = MyViewModel()
myViewModel = ViewModelProviders
.of(this, createFor(myViewModel))
.get(MyViewModel::class.java)
}
private fun setupObservers() {
myViewModel.myData.observe(viewLifecycleOwner, Observer {
it?.let { resource ->
when (resource.status) {
Status.SUCCESS -> {
resource.data?.let {
val action =
FragmentADirections.actionFragmentAToFragmentB()
Navigation.findNavController(requireView()).navigate(action)
}
}
Status.ERROR -> {
//TODO show an error
loading.visibility = View.GONE
}
Status.LOADING -> {
loading.visibility = View.VISIBLE
}
}
}
})
}
这是我的视图模型
class MyViewModel : ViewModel() {
private val searchText = MutableLiveData<String>()
//This method is called on a search button click
fun setSearchText(text: String) {
searchText.value = text
}
val myData = searchText.switchMap { text ->
liveData(Dispatchers.IO) {
emit(Resource.loading(data = null))
try {
emit(Resource.success(data = dataOrigin.getData(text)))
} catch (exception: Exception) {
exception.printStackTrace()
emit(Resource.error(data = null, message = exception.message ?: "Error Occurred!"))
}
}
}
}
那是因为您的myData
Observer
将立即使用返回导航的最新数据触发并再次触发SUCCESS
分支。
我建议使用SharedFlow
或检查LiveData
的SingleLiveEvent
模式,您只能使用一次事件。
更新:
消耗品模式的可能解决方案:
class Consumable<T>(private val event: T) {
private var isConsumed = false
fun consume(): T? {
if (isConsumed) return null
isConsumed = true
return event
}
fun peek() = event
}
然后在ViewModel中:
val event = MutableLiveData<Consumable<YourEventClass>>()
...
event.value = Consumable(XYEvent)
你会像这样观察它:
viewModel.event.observe{ event ->
event.consume()?.let{
// Do your navigation here
}
}
此外,您可以为可消耗的 livedatas 编写扩展 function .consume()
,它会解开值
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.