在Repository和ViewModel之间共享相同的MutableLiveData

时间:2018-06-18 13:23:35

标签: android mvvm kotlin android-architecture-components android-livedata

我正在围绕Architecture Components / MVVM。

假设我有一个存储库,一个ViewModel和一个Fragment。我正在使用Resource类作为包装来公开网络状态,如Guide to architecture components中所述。

我的存储库目前看起来像这样(简化为简洁):

class MyRepository {

  fun getLists(organizationId: String) {
    var data = MutableLiveData<Resource<List<Something>>>()
    data.value = Resource.loading()

    ApolloClient().query(query)
        .enqueue(object : ApolloCall.Callback<Data>() {
            override fun onResponse(response: Response<Data>) {
                response.data()?.let {
                    data.postValue(Resource.success(it))
                }
            }

            override fun onFailure(exception: ApolloException) {
                data.postValue(Resource.exception(exception))
            }
        })
}

然后在ViewModel中,我还声明了一个MutableLiveData:

var myLiveData = MutableLiveData<Resource<List<Something>>>()

fun getLists(organizationId: String, forceRefresh: Boolean = false) {
   myLiveData = myRepository.getLists(organizationId)
}

最后,片段:

viewModel.getLists.observe(this, Observer {
        it?.let {
            if (it.status.isLoading()) showLoading() else hideLoading()

            if (it.status == Status.SUCCESS) {
                it.data?.let {
                    adapter.replaceData(it)
                    setupViews()
                }
            }

            if (it.status == Status.ERROR) {
                // Show error
            }
        }
    })

如您所见,观察者没有被触发会出现问题,因为LiveData变量将在进程中重置(Repository创建一个新实例)。

我正在尝试找出确保在Repository和ViewModel之间使用相同LiveData变量的最佳方法。

我考虑过将LiveData从ViewModel传递给getLists方法,以便Repository将使用ViewModel中的对象,但即使它起作用,这样做似乎也是错误的。

我的意思是这样的:

视图模型

var myLiveData = MutableLiveData<Resource<List<Something>>>()

fun getLists(organizationId: String, forceRefresh: Boolean = false) {
   myRepository.getLists(myLiveData, organizationId)
}

存储库

fun getLists(data: MutableLiveData<Resource<List<Something>>>, organizationId: String) {
    ...
}

1 个答案:

答案 0 :(得分:3)

我想我知道该怎么做,多亏了@NSimon的提示。

我的存储库保持不变,我的ViewModel看起来像这样:

class MyViewModel : ViewModel() {
  private val myRepository = MyRepository()

  private val organizationIdLiveData = MutableLiveData<String>()
  private val lists = Transformations.switchMap(organizationIdLiveData) { organizationId -> myRepository.getLists(organizationId) }

  fun getLists() : LiveData<Resource<MutableList<Something>>> {
    return lists
  }

  fun fetchLists(organizationId: String, forceRefresh: Boolean = false) {
    if (organizationIdLiveData.value == null || forceRefresh) {
      organizationIdLiveData.value = organizationId
    }
  }
}

我在片段中观察getLists(),并在需要数据时调用viewModel.fetchLists(id)。似乎合法吗?