具有特定页面索引的分页库数据源

5

关于 Arch. 组件中的分页库,它有三种不同的数据源:

  1. PageKeyedDataSource:当请求需要下一个/上一个索引键时使用。
  2. ItemKeyedDataSource:当请求需要作为键的项目时使用。
  3. PositionalDataSource:当请求需要索引以获取下一批数据时使用。

如果后端响应不支持下一个/上一个索引键,而只支持当前页面,则应该使用哪个数据源?当响应如下所示:

{
    "status": "success",
    "response": "1",
    "message": "Data fetched successfully",
    "data": {
        "total": 2,
        "per_page": "5",
        "page": 1,
        "items": [],
        "last_page": 1,
        "has_more": false
    }
}

你成功让它工作了吗,@mhdtouban? - Levi Moreira
@levi-albuquerque,你提出的解决方案是可行的,但是当我向下滚动等待新数据时,仍然可能会卡住。我知道分页可以帮助在到达边界之前获取数据。 - mhdtouban
1个回答

5
这看起来很像我在这个示例中使用的StackOverflow响应。
{
  "items": [
  ],
  "has_more": true,
  "quota_max": 10000,
  "quota_remaining": 9965
}

我使用的Retrofit接口长这样:

@Headers("Content-Type: application/json", "Accept: application/json")
    @GET("/2.2/users")
    fun getTopUsers(
            @Query("page") page: Int,
            @Query("pagesize") pagesize: Int,
            @Query("order") order: String = "desc",
            @Query("sort") sort: String = "reputation",
            @Query("site") site: String = "stackoverflow"

    ): Single<SOResponse<User>>

因此,我们可以指定页面大小和页面数量,就像在您的情况下一样。

为此,我使用了一个PageKeyedDataSource<Int,User>。您可以通过传递整数来指定要加载的页面。例如,我的初始加载方法如下:

override fun loadInitial(params: LoadInitialParams<Int>, callback: LoadInitialCallback<Int, User>) {
        loadState.onNext(NetworkState.LOADING)
        val disposable = soApi.apiService.getTopUsers(1, params.requestedLoadSize)
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe({
                    callback.onResult(it.items, 1, 2)
                    loadState.onNext(NetworkState.LOADED)
                }, {
                    loadState.onNext(NetworkState.error(it.message))
                })

        disposables.add(disposable)
    }

如果您查看callback.onResult(it.items, 1, 2),您会发现我正在请求下一页,即第2页。
对于其他页面,我使用以下代码:
override fun loadAfter(params: LoadParams<Int>, callback: LoadCallback<Int, User>) {
        loadState.onNext(NetworkState.LOADING)
        val disposable = soApi.apiService.getTopUsers(params.key, params.requestedLoadSize)
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe({
                    callback.onResult(it.items, params.key + 1)
                    loadState.onNext(NetworkState.LOADED)
                }, {
                    loadState.onNext(NetworkState.error(it.message))
                })
        disposables.add(disposable)
    }

这一次,我只是增加了页面号 callback.onResult(it.items, params.key + 1)


如果我们的URL中没有 @Query("page") page: Int 该怎么办?@LeviMoreira - Maulik Dodia

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接