在recycleview中使用Diffutil,使其在添加新项目时自动滚动

时间:2017-04-17 19:24:20

标签: android android-recyclerview

如果我们使用DiffUtil.Callback,请执行

adapter.setItems(itemList);
diff.dispatchUpdatesTo(adapter);

我们如何确保添加新元素将滚动到新位置。

我有一个案例,我看到项目消失了,新的一个作为第一个元素创建在顶部,但不可见。它会隐藏在顶部,直到您向下滚动以使其可见。 在使用DiffUtil之前,我是手动实现的,在我知道我插入某个位置(在顶部)后,我可以滚动到。

3 个答案:

答案 0 :(得分:34)

如果项目被插入可视区域之外,还有一种简单的方法可以保留用户的滚动位置:

import android.os.Parcelable;

Parcelable recyclerViewState = recyclerView.getLayoutManager().onSaveInstanceState();
// apply diff result here (dispatch updates to the adapter)
recyclerView.getLayoutManager().onRestoreInstanceState(recyclerViewState);              

使用此方法,如果将新项目插入到用户可以看到的位置,则可以看到新项目 - 但如果项目插入到视图之外,则会保留用户的视点。

答案 1 :(得分:14)

您也可以使用dispatchUpdatesTo(ListUpdateCallback)方法。

所以你可以实现一个ListUpdateCallback,它会为你提供插入的第一个元素

class MyCallback implements ListUpdateCallback {
    int firstInsert = -1;
    Adapter adapter = null;
    void bind(Adapter adapter) {
        this.adapter = adapter;
    }
    public void onChanged(int position, int count, Object payload) {
        adapter.notifyItemRangeChanged(position, count, payload);
    }
    public void onInserted(int position, int count) {
        if (firstInsert == -1 || firstInsert > position) {
            firstInsert = position;
        }
        adapter.notifyItemRangeInserted(position, count);
    }
    public void onMoved(int fromPosition, int toPosition) {
        adapter.notifyItemMoved(fromPosition, toPosition);
    }
    public void onRemoved(int position, int count) {
        adapter.notifyItemRangeRemoved(position, count);
    }
}

然后只需手动滚动RecyclerView

myCallback.bind(adapter)
adapter.setItems(itemList);
diff.dispatchUpdatesTo(myCallback);
recycler.smoothScrollToPosition(myCallback.firstInsert);

答案 2 :(得分:2)

我使用RecyclerView.AdapterDataObserver检测何时添加了新项目,然后滚动到列表顶部;

adapter.registerAdapterDataObserver(new RecyclerView.AdapterDataObserver() {
    @Override
    public void onItemRangeInserted(int positionStart, int itemCount) {
        super.onItemRangeInserted(positionStart, itemCount);
        recycleView.smoothScrollToPosition(0);
    }
});