精準計算RecyclerView滑動距離

小專欄

前去小專欄查看更多分享

前言

最近遇到需求,要計算RecyclerView滑動距離,用提供的computeVerticalScrollOffset()方法得到的值不是很準確。是基於item的平均高度算得,如果列表中item高度一致可以用此方法。問題來了,我的應用場景是各item高度不一,這時就只能另找方法了。

方法一:

網上找的方法,用一個變量去統計,每次滑動的時候累加y軸偏移量。item插入\移動\刪除的時候,需要手動去更新totalDy,不然就會一直錯下去。

private int totalDy = 0;
mRecycler.addOnScrollListener(new RecyclerView.OnScrollListener() {
    @Override
    public void onScrolled(RecyclerView recyclerview, int dx, int dy) {
        totalDy -= dy;
    }
}

方法二:

方法一比較麻煩,而且坑較多。所以考慮重寫LinearLayoutManager的computeVerticalScrollOffset()方法,既然原生方法是按平均高度計算的,那重寫該計算邏輯,就能達到我們想要的效果。
1.統計列表已展示過的item的高度,在每次佈局完成的時候,用一個map記錄positon位置item對應的view的高度。

private Map<Integer, Integer> heightMap = new HashMap<>();
int count = getChildCount();
for (int i = 0; i < count; i++) {
    View view = getChildAt(i);
    heightMap.put(i, view.getHeight());
}

2.重寫computeVerticalScrollOffset(),找到當前屏幕第一個可見item的position,通過heightMap循環累加0到positon的item高度,再加上第一個可見item不可見部分高度。最終得到整個列表的滑動偏移。

@Override
public int computeVerticalScrollOffset(RecyclerView.State state) {
    if (getChildCount() == 0) {
        return 0;
    }
    int firstVisiablePosition = findFirstVisibleItemPosition();
    View firstVisiableView = findViewByPosition(firstVisiablePosition);
    int offsetY = -(int) (firstVisiableView.getY());
    for (int i = 0; i < firstVisiablePosition; i++) {
        offsetY += heightMap.get(i) == null ? 0 : heightMap.get(i);
    }
    return offsetY;
}

最終完整代碼

public class OffsetLinearLayoutManager extends LinearLayoutManager {

    public OffsetLinearLayoutManager(Context context) {
        super(context);
    }

    private Map<Integer, Integer> heightMap = new HashMap<>();

    @Override
    public void onLayoutCompleted(RecyclerView.State state) {
        super.onLayoutCompleted(state);
        int count = getChildCount();
        for (int i = 0; i < count ; i++) {
            View view = getChildAt(i);
            heightMap.put(i, view.getHeight());
        }
    }

    @Override
    public int computeVerticalScrollOffset(RecyclerView.State state) {
        if (getChildCount() == 0) {
            return 0;
        }
        try {
            int firstVisiablePosition = findFirstVisibleItemPosition();
            View firstVisiableView = findViewByPosition(firstVisiablePosition);
            int offsetY = -(int) (firstVisiableView.getY());
            for (int i = 0; i < firstVisiablePosition; i++) {
                offsetY += heightMap.get(i) == null ? 0 : heightMap.get(i);
            }
            return offsetY;
        } catch (Exception e) {
            return 0;
        }
    }
}

使用姿勢

mRecycler.setLayoutManager(new OffsetLinearLayoutManager(mContext));
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章