使用StaggeredGridLayoutManager的RecylerView在DiffUtil更改后跳转到2个元素

时间:2017-12-15 17:39:35

标签: android android-recyclerview staggeredgridlayoutmanager

我有基于RecyclerView的2列StaggeredGridLayoutManager

更改1项的内容并触发DiffUtil时,RecyclerView会跳转到2项。但是当向上滚动时,可以看到1个元素和空白空间,在滚动项目转换为自然顺序期间。此外,差异列表中的项目数量相同。

如何在diffUtil期间避免这种恼人的行为并保持滚动位置?

val layoutManager = StaggeredGridLayoutManager(2, StaggeredGridLayoutManager.VERTICAL)
    recyclerView.layoutManager = layoutManager
    recyclerView.itemAnimator = null
    recyclerView.setHasFixedSize(true)
    recyclerView.setItemViewCacheSize(if (App.isTablet()) 3 else 2)

public void diffUpdate(List<T> newList) {
    if (getCollection().size() == 0) {
      addAll(newList);
      notifyDataSetChanged();
    } else {
      DiffCallback diffCallback = new DiffCallback<T>(getCollection(), newList);
      DiffUtil.DiffResult diffResult = DiffUtil.calculateDiff(diffCallback);
      clear();
      addAll(newList);
      diffResult.dispatchUpdatesTo(this);
    }
  }

1 个答案:

答案 0 :(得分:0)

问题在于DiffCallback的实现。早些时候我用过哈希:

    @Override public boolean areItemsTheSame(int oldItemPosition, int newItemPosition) {
      T oldItem = oldList.get(oldItemPosition);
      T newItem = newList.get(newItemPosition);
      boolean areTheSameInstance = oldItem == newItem;
      boolean hasTheSameType = oldItem.getClass().equals(newItem.getClass());
      boolean hasTheSameHash = oldItem.hashCode() == newItem.hashCode();
      return areTheSameInstance || hasTheSameType && hasTheSameHash;
    }

这触发了RecyclerView中整个视图的重新创建,并可能在StaggeredGridLayoutManager中进行测量。

在我将其更改为id检查后,diffCalbback已开始仅触发RecyclerView中视图的渲染。

return ((Model) oldItem).id() == ((Model) newItem).id();