item高度不同時Recyclerview獲取滑動距離的方法
前言
最近遇到需求,要計算RecyclerView滑動距離,用提供的computeVerticalScrollOffset()方法得到的值不是很準(zhǔn)確。是基于item的平均高度算得,如果列表中item高度一致可以用此方法。問題來了,我的應(yīng)用場景是各item高度不一,這時就只能另找方法了。
方法一:
網(wǎng)上找的方法,用一個變量去統(tǒng)計,每次滑動的時候累加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()方法,既然原生方法是按平均高度計算的,那重寫該計算邏輯,就能達(dá)到我們想要的效果。
1.統(tǒng)計列表已展示過的item的高度,在每次布局完成的時候,用一個map記錄positon位置item對應(yīng)的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(),找到當(dāng)前屏幕第一個可見item的position,通過heightMap循環(huán)累加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; }
3.最終代碼
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));
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
android實現(xiàn)獲取正在運行的應(yīng)用程序
android如何獲取正在運行的應(yīng)用程序,因為在framework中想添加這個功能,所以寫了個appliction來實現(xiàn)一下獲取正在運行的應(yīng)用程序2013-01-01Android WebView如何判定網(wǎng)頁加載的錯誤
這篇文章主要介紹了Android WebView如何判定網(wǎng)頁加載的錯誤,幫助大家更好的理解和學(xué)習(xí)使用Android,感興趣的朋友可以了解下2021-04-04Android Studio 超級簡單的打包生成apk的方法
本篇文章主要介紹了Android Studio 超級簡單的打包生成apk的方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-10-10自定義toast外形,多次點擊不會總是彈出toast的實現(xiàn)方法
下面小編就為大家?guī)硪黄远xtoast外形,多次點擊不會總是彈出toast的實現(xiàn)方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-04-04Android.mk文件中添加第三方j(luò)ar文件的方法
這篇文章主要介紹了Android.mk文件中添加第三方j(luò)ar文件及引用第三方j(luò)ar包的方法,需要的朋友可以參考下2018-01-01