Search code examples
androidkotlinpaginationandroid-jetpackswiperefreshlayout

Unable to refresh recyclerView with pagination


I have a RecyclerView that gets filled with paginated data.

I need to be able to refresh it via SwipeRefreshLayout, but so far, I am unable to do so.

I have tried plenty references that does similar to this, but without any success. Also, for letting you guys know, When I do refresh, the data gets retrieved correctly, but it is not being displayed on the recyclerView...

Also, when I move from that destination to another via Navigation and then, I hit back, it gets updated. I am not sure what I am missing or what I am doing wrong.

ViewModel:

class ActivityViewModel : ViewModel() {
    var itemDataSourceFactory = OperationsDataSourceFactory()
        private set
    val livePagedList: LiveData<PagedList<Result<Operation?>>>

    init {
        val config = PagedList.Config.Builder()
            .setEnablePlaceholders(false)
            .setInitialLoadSizeHint(10)
            .setPageSize(10)
            .build()

        livePagedList = LivePagedListBuilder(itemDataSourceFactory, config).build()
    }

    fun refresh() {
        itemDataSourceFactory.dataSourceLiveData.value?.endCursor = null
        itemDataSourceFactory.refresh()
    }
}

DataSourceFactory:

class OperationsDataSourceFactory :
    DataSource.Factory<String, Result<Operation?>>() {
    val dataSourceLiveData = MutableLiveData<OperationsDataSource>()
    lateinit var dataSource: OperationsDataSource
    override fun create(): DataSource<String, Result<Operation?>> {
        dataSource = OperationsDataSource()
        dataSourceLiveData.postValue(dataSource)
        return dataSource
    }

    fun refresh(){
        dataSourceLiveData.value?.invalidate()
    }
}

And, on the Fragment, I have something like this:

private fun initView() {
    binding.swipeToRefresh.setOnRefreshListener {
        itemViewModel.refresh()
    }
    getData() //observables with adapter.submitList(list)
}

Solution

  • Ok, after a really long research, might not be the best solution, but developers use this too. We need to "invalidate" the layoutManager and adapter.

    swipeToRefresh.setOnRefreshListener {
        operationsRecycler.apply {    
        layoutManager = null
        adapter = null
        layoutManager = LinearLayoutManager(
                            requireContext(),
                                LinearLayoutManager.VERTICAL,
                                false
                            )
        adapter = myAdapter
    }