繁体   English   中英

ViewModel 阻止的片段返回导航

[英]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或检查LiveDataSingleLiveEvent模式,您只能使用一次事件。

更新:

消耗品模式的可能解决方案:

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.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM