item高度不同時(shí)Recyclerview獲取滑動(dòng)距離的方法

前言

創(chuàng)新互聯(lián)建站主要從事成都網(wǎng)站制作、網(wǎng)站建設(shè)、網(wǎng)頁設(shè)計(jì)、企業(yè)做網(wǎng)站、公司建網(wǎng)站等業(yè)務(wù)。立足成都服務(wù)夷陵,10多年網(wǎng)站建設(shè)經(jīng)驗(yàn),價(jià)格優(yōu)惠、服務(wù)專業(yè),歡迎來電咨詢建站服務(wù):18982081108

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

方法一

網(wǎng)上找的方法,用一個(gè)變量去統(tǒng)計(jì),每次滑動(dòng)的時(shí)候累加y軸偏移量。item插入\移動(dòng)\刪除的時(shí)候,需要手動(dòng)去更新totalDy,不然就會一直錯(cuò)下去。

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

方法二:

方法一比較麻煩,而且坑較多。所以考慮重寫LinearLayoutManager的computeVerticalScrollOffset()方法,既然原生方法是按平均高度計(jì)算的,那重寫該計(jì)算邏輯,就能達(dá)到我們想要的效果。

1.統(tǒng)計(jì)列表已展示過的item的高度,在每次布局完成的時(shí)候,用一個(gè)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)前屏幕第一個(gè)可見item的position,通過heightMap循環(huán)累加0到positon的item高度,再加上第一個(gè)可見item不可見部分高度。最終得到整個(gè)列表的滑動(dòng)偏移。

@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í)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。

文章標(biāo)題:item高度不同時(shí)Recyclerview獲取滑動(dòng)距離的方法
當(dāng)前URL:http://bm7419.com/article20/jdigco.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供服務(wù)器托管、建站公司、網(wǎng)站營銷、企業(yè)網(wǎng)站制作、網(wǎng)站排名、商城網(wǎng)站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)

成都定制網(wǎng)站建設(shè)