3

我有一个消息列表。每条消息都有一个唯一的 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);

    }

会话视图模型


    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);
    }

消息道

    @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它将加载更多如果用户上升或下降。

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

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

如何做到这一点?

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

4

1 回答 1

3

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

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

如果你感到有点困惑,这并没有到此结束,因为即使是关于它是什么initialKey以及如何使用它的解释也没有记录在案。不,认真:Pager 构造函数中的 initialKey 参数有什么作用

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

我希望 Paging 3 文档很快得到改进,以涵盖这些非常基本的问题。

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

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

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

于 2020-11-27T14:33:35.753 回答