我正在使用 PagedListAdapter,并且必须在屏幕恢复时重新加载数据。
对于这种情况,我不想在加载数据时清除适配器并将其显示为空或占位符。
我想显示以前的数据,并在收到新数据后用差异更新它。
但现在当我这样做时
dataSource.invalidate()
它会立即提交一个空列表,这会导致清除适配器并显示占位符。
现在我的代码看起来像:
class PortfolioListDataSource(
private val viewModel: PortfolioViewModel,
private val tabType: PortfolioTabType
) : PageKeyedDataSource<Int, Investment>() {
override fun loadInitial(params: LoadInitialParams<Int>, callback: LoadInitialCallback<Int, Investment>) {
val offset = 0
viewModel.loadInvestments(tabType, offset, Consumer {
val nextKey = if (it.data.size < params.requestedLoadSize) null else it.data.size
callback.onResult(it.data, offset, it.total, null, nextKey)
})
}
override fun loadAfter(params: LoadParams<Int>, callback: LoadCallback<Int, Investment>) {
val offset = params.key
viewModel.loadInvestments(tabType, offset, Consumer {
val nextKey = offset + it.data.size
callback.onResult(it.data, nextKey)
})
}
override fun loadBefore(params: LoadParams<Int>, callback: LoadCallback<Int, Investment>) {
}
class Factory(
val viewModel: PortfolioViewModel,
private val tabType: PortfolioTabType
) : DataSource.Factory<Int, Investment>() {
val sourceLiveData = MutableLiveData<PortfolioListDataSource>()
override fun create() = PortfolioListDataSource(viewModel, tabType)
.also { sourceLiveData.postValue(it) }
}
}
这里我初始化了
val pagedListConfig = PagedList.Config.Builder()
.setInitialLoadSizeHint(20)
.setPageSize(20)
.build()
ordersLiveData = LivePagedListBuilder(factory, pagedListConfig).build()
我在这里填充适配器
viewModel.ordersLiveData.observe(this, Observer {
adapter.submitList(it)
})
并用
刷新它// TODO Here I want to fetch new data, but not invalidate current dataset while it is loading
factory.sourceLiveData.value?.invalidate()
如何保留以前的数据并在新数据准备好时更新它?
更新:
我想要类似的东西:
if (!refreshing) {
submitList(it)
}
我的意思是在调用
dataSource.invalidate()
时不要将空数据集发布到适配器。
我知道上面提到的示例将不起作用,因为一旦提交,数据源就会将更新传递到列表的同一实例,并且它被完全封装。
另外,我对数据集本身没有影响。
我尝试使用
PagedList.Callback
它可以让我知道列表何时更新。
但同样,没有机会以某种方式影响更新行为。
使用常规适配器,在收到新数据之前,我不会设置或更新其项目。 我想在这里实现类似的目标。
是的,invalidate() 会清除当前数据集。 因此,您将进入加载/占位符状态。 也不可能更新特定页面。 假设您位于第 5 页,并且想要更新该特定页面中的数据。如果不调用 invalidate() 也是不可能的,它将清除数据集并将 Recyclerview 带到第一页(列表顶部)。
摆脱所有这些问题的唯一方法是引入 Room 的持久层。 https://github.com/googlesamples/android-architecture-components/tree/master/PagingWithNetworkSample#paging-with-database-and-network
根据此处的文档: https://developer.android.com/reference/androidx/paging/DataSource#updating-paged-data
如果您有更细粒度的更新信号,例如网络 API 发出对列表中单个项目的更新信号,建议 将数据从网络加载到内存中。然后将该数据呈现给 PagedList 通过包装内存快照的 DataSource。每次 内存中的副本发生更改,使之前的 DataSource 失效,并且 可以创建包装快照新状态的新状态。
https://developer.android.com/topic/libraries/architecture/paging/data#consider-content-updates
当您构造可观察的 PagedList 对象时,请考虑内容如何 更新工作。如果您直接从 Room 数据库加载数据 更新会自动推送到您应用程序的 UI。
在您的情况下,您始终可以在屏幕恢复时从数据库加载数据,直到从 API 获取数据为止。