分页 3:如何在项目位置或具有特定 id 的项目加载列表

Paging 3: How to load list at item position or at item with specific id

我有一个消息列表。 每条消息都有一个唯一的 GUID。

我的设置适用于正常使用:用户点击对话,列表打开,其中包含属于该对话的所有消息,按最新排序。

对话片段


    @Override
    public void onViewCreated(
        @NonNull View view,
        @Nullable Bundle savedInstanceState
    ) {
        LifecycleOwner lifecycleOwner = getViewLifecycleOwner();
        viewModel = new ViewModelProvider(this).get(ConversationViewModel.class);
        viewModel
            .getMessageList(lifecycleOwner, conversationId) // conversationId is a global variable
            .observe(lifecycleOwner, messagePagingData -> adapter.submitData(
                lifecycleOwner.getLifecycle(),
                messagePagingData 
            ));

        super.onViewCreated(view, savedInstanceState);

    }

ConversationViewModel


    final PagingConfig pagingConfig = new PagingConfig(10, 10, false, 20);
    private final ConversationRepository conversationRepository;

    public ConversationViewModel(@NonNull Application application) {
        super(application);
        conversationRepository = new ConversationRepository(application);
    }

    public LiveData<PagingData<ItemMessage>> getMessageList(
        @NonNull LifecycleOwner lifecycleOwner,
        @NonNull String conversationId
    ) {
        return PagingLiveData.cachedIn(
            PagingLiveData.getLiveData(new Pager<>(pagingConfig, () -> conversationRepository.getMessageList(conversationId))),
            lifecycleOwner.getLifecycle()
        );
    }

对话存储库

    private final MessageDao messageDao;

    public ConversationRepository(@NonNull Context context) {
        AppDatabase database = AppDatabase.getDatabase(context);
        messageDao = database.messageDao();
    }

    public PagingSource<Integer, ItemMessage> getMessageList(@NonNull String conversationId) {
        return messageDao.getMessageList(conversationId);
    }

MessageDao

    @Query(
        "SELECT * FROM Message " +
        "WHERE Message.conversationId = :conversationId " +
        "ORDER BY Message.time DESC"
    )
    public abstract PagingSource<Integer, ItemMessage> getMessageList(String conversationId);

现在我的目标是能够打开已滚动到特定消息的对话。

我也不想加载整个对话然后滚动到消息,有些对话可能很长而且我不想让用户自动滚动可能需要很长时间才能到达特定消息.

理想情况下,我设想正确完成此操作的方式是传递要查看的消息 ID,加载围绕该消息 ID 前后的一大块 X 消息,然后在它已经呈现给用户之后RecyclerView 如果用户上下移动,它会加载更多。

这并不意味着使用网络请求,整个对话已经在数据库中可用,因此它只会使用数据库中已有的信息。

我已经尝试理解使用 ItemKeyedDataSourcePageKeyedDataSource 的示例,但我无处可去,因为每次这些示例仅在 Kotlin 中并且需要 Retrofit 才能工作,而我就是这样做的不使用。因为这些例子对于像我这样使用 Java 并且不使用 Retrofit 的人来说完全没用。

如何实现?

请在 Java 中提供答案,而不仅仅是 Kotlin(只要在 java 中,kotlin 就可以)并且请不要建议新的库。

据我所知,官方文档没有提供任何关于如何解决 Paging + Room 集成问题的线索。事实上,它没有提供任何解决方案来滚动到 PagingDataAdapter 中的项目。

到目前为止,唯一对我有用的是 运行 每次我希望完成此操作时进行两次查询:一个是在结果查询列表中查找项目位置,另一个是实际加载 said在 Pager 构造函数中设置 initialKey 的列表具有我们之前查询的项目位置的值。

如果您感到有点困惑,这并没有到此结束,因为甚至连对什么是 initialKey 以及如何使用它的解释都没有记录在案。不,说真的:

所以这里有两个猜谜游戏:一个是找到从结果列表中查找项目索引的正确方法,另一个是在最终查询中正确设置它。

我希望 Paging 3 文档能尽快改进以涵盖这些非常基本的问题。

最后,这是我如何设法让这个问题为我工作的一个例子,尽管我不知道这是否是正确的方法,因为同样,他们的文档绝对缺乏在这个部门。

  1. 为您想要的列表结果创建两个相同的查询
  2. 其中一个查询仅 returns 基于您将用于唯一标识项目的键的结果的完整列表。在我的例子中是 messageId.
  3. 在 2 中加载查询并使用 for... 循环单独迭代结果列表,直到找到您想要知道其在列表中位置的项目。该位置由您在循环块中使用的迭代器给出。
  4. 将 3 中的项目位置作为 initialKey 参数传递到最终查询的 Pager 构建器中
  5. 您现在收到的第一个数据块将包含您想要的项目
  6. 如果您愿意,现在可以在 RecyclerView 中滚动到该项目,但您必须从适配器中加载的当前项目列表中查询它。请参阅在 PagingAdapter
  7. 中使用 .snapshot()

就是这样,现在我终于可以使用 Paging 3 + Room 在某个位置加载一个项目,完全不知道这是否是正确的方法,这要归功于完全没有相关文档。