你可能误会了!原来自定义LayoutManager可以这么简单

参考资料

参考资料1;
参考资料2
参考资料3
参考资料4

背景介绍

RecyclerView由于其强大的扩展性,现在已经逐步的取代了ListViewGridView了。为了实现不同的布局效果,我们会用到官方提供的LinearLayoutManagerGridLayoutManagerStaggeredGridLayoutManager。但这些布局只能满足日常需求,在一些比较复杂的布局中,它们就力不从心了,强行拼凑实现,带来的后果就是较差的体验和性能。所以能够自定义LayoutManager还是十分必要的,它能够解放创造力,构造复杂的、流畅的滑动列表。上面几篇参考资料中就实现了一些不寻常的效果,我们可以看到,这些效果如果用常规的方案去实现将会十分蹩脚。

揭开LayoutManager中不为人知的秘密

自定义LayoutManager主要要求我们完成三件事情:

  • 计算每个ItemView的位置;
  • 处理滑动事件;
  • 缓存并重用ItemView;

而我们比较重要的工作是在onLayoutChildern() 这个回调方法中完成的。

下面我们就来一一解析。

预先准备

当我们extends RecyclerView.LayoutManager是,我们会被强制要求重写generateDefaultLayoutParams()方法,如方法名字一样,我们需要提供一个默认的LayoutParams,这里为我们的每个ItemView提供默认的LayoutParams,所以它能够直接影响到我们的布局效果,这里我们设置成WRAP_CONTENT,让ItemView获得决定权。

@Override
  public RecyclerView.LayoutParams generateDefaultLayoutParams() {
    return new RecyclerView.LayoutParams(RecyclerView.LayoutParams.WRAP_CONTENT,
        RecyclerView.LayoutParams.WRAP_CONTENT);
  }

计算ItemView的位置

1.实现简单的LayoutManager

先看效果图:

简单LayoutManager

再看代码:

@Override
  public void onLayoutChildren(RecyclerView.Recycler recycler, RecyclerView.State state) {
    super.onLayoutChildren(recycler, state);
    // 先把所有的View先从RecyclerView中detach掉,然后标记为"Scrap"状态,表示这些View处于可被重用状态(非显示中)。
    // 实际就是把View放到了Recycler中的一个集合中。
    detachAndScrapAttachedViews(recycler);
    calculateChildrenSite(recycler);
    // 回收和填充Item
    recycleAndFillView(recycler, state);
  }

private void calculateChildrenSite(RecyclerView.Recycler recycler) {
        totalHeight = 0;
        boolean needNew = true;
        int width = 0;
        int height = 0;
        for (int i = 0; i < getItemCount(); i++) {
            // 没有会创建
            if (needNew) {
                View view = recycler.getViewForPosition(i);
                measureChildWithMargins(view, 0, 0);
                calculateItemDecorationsForChild(view, new Rect());
                width = getDecoratedMeasuredWidth(view);
                height = getDecoratedMeasuredHeight(view);
                addView(view);
            }
            if (totalHeight > getHeight() + height) {
                needNew = false;
            }
            Rect mTmpRect = allItemRects.get(i);
            if (mTmpRect == null) {
                mTmpRect = new Rect();
            }
            mTmpRect.set(0, totalHeight, DisplayUtils.getScreenWidth(), totalHeight + height);
            totalHeight = totalHeight + height;
            // 保存ItemView的位置信息
            allItemRects.put(i, mTmpRect);
            // 由于之前调用过detachAndScrapAttachedViews(recycler),所以此时item都是不可见的
            itemStates.put(i, false);
        }
    }

这段代码逻辑简单,它实现的其实就是一个简单的垂直线性布局,当然现在还不能滑动,也没有缓存机制。在这段代码中,我们先调用detachAndScrapAttachedViews(recycler);将所有的ItemView标记为Scrap状态,然后在挨个取出来,计算他们应该布局到什么位置,并用成员变量totalHeight记录总高度,最后调用recycleAndFillView()将ItemView布局上去。

2.两列式的LayoutManager

先看效果图:

效果图

有了上例的基础,我们只需要稍作调整,直接看下面代码,注意注释部分。

private void calculateChildrenSite(RecyclerView.Recycler recycler) {
        totalHeight = 0;
        boolean needNew = true;
        int width = 0;
        int height = 0;
        for (int i = 0; i < getItemCount(); i++) {
            // 没有会创建
            if (needNew) {
                View view = recycler.getViewForPosition(i);
                measureChildWithMargins(view, DisplayUtils.getScreenWidth() / 2, 0);
                calculateItemDecorationsForChild(view, new Rect());
                width = getDecoratedMeasuredWidth(view);
                height = getDecoratedMeasuredHeight(view);
                addView(view);
            }
            if (totalHeight > getHeight() + height) {
                needNew = false;
            }
            Rect mTmpRect = allItemRects.get(i);
            if (mTmpRect == null) {
                mTmpRect = new Rect();
            }

            if (i % 2 == 0) { // 当i能被2整除时,是左,否则是右。
                // 左
                mTmpRect.set(0, totalHeight, DisplayUtils.getScreenWidth() / 2, totalHeight + height);
            } else {
                // 右,需要换行
                mTmpRect.set(DisplayUtils.getScreenWidth() / 2, totalHeight, DisplayUtils.getScreenWidth(),
                        totalHeight + height);
                totalHeight = totalHeight + height;
            }
            // 保存ItemView的位置信息
            allItemRects.put(i, mTmpRect);
            // 由于之前调用过detachAndScrapAttachedViews(recycler),所以此时item都是不可见的
            itemStates.put(i, false);
        }

    }

处理滑动

先来看一下效果:

效果图

滑动事件主要涉及到4个方法需要重写,我们直接来看代码:

@Override
  public boolean canScrollVertically() {
    //返回true表示可以纵向滑动
    return true;
  }

  @Override
  public int scrollVerticallyBy(int dy, RecyclerView.Recycler recycler, RecyclerView.State state) {
    //列表向下滚动dy为正,列表向上滚动dy为负,这点与Android坐标系保持一致。
    //实际要滑动的距离
    int travel = dy;

    LogUtils.e("dy = " + dy);
    //如果滑动到最顶部
    if (verticalScrollOffset + dy < 0) {
      travel = -verticalScrollOffset;
    } else if (verticalScrollOffset + dy > totalHeight - getVerticalSpace()) {//如果滑动到最底部
      travel = totalHeight - getVerticalSpace() - verticalScrollOffset;
    }

    //将竖直方向的偏移量+travel
    verticalScrollOffset += travel;

    // 调用该方法通知view在y方向上移动指定距离
    offsetChildrenVertical(-travel);

    return travel;
  }

  private int getVerticalSpace() {
    //计算RecyclerView的可用高度,除去上下Padding值
    return getHeight() - getPaddingBottom() - getPaddingTop();
  }

  @Override
  public boolean canScrollHorizontally() {
    //返回true表示可以横向滑动
    return super.canScrollHorizontally();
  }

  @Override
  public int scrollHorizontallyBy(int dx, RecyclerView.Recycler recycler, RecyclerView.State state) {
    //在这个方法中处理水平滑动
    return super.scrollHorizontallyBy(dx, recycler, state);
  }

缓存并重用ItemView

在上面代码的基础上我们稍作改动,加入缓存,先看下面的log信息,它显示虽然有100个Item,但childCount稳定在26:

log

下面来看看代码的变化,我展示了完整的代码,留心注释。

public class CustomLayoutManager extends RecyclerView.LayoutManager {
  /** 用于保存item的位置信息 */
  private SparseArray<Rect> allItemRects = new SparseArray<>();
  /** 用于保存item是否处于可见状态的信息 */
  private SparseBooleanArray itemStates = new SparseBooleanArray();

  public int totalHeight = 0;
  private int verticalScrollOffset;

  @Override
  public RecyclerView.LayoutParams generateDefaultLayoutParams() {
    return new RecyclerView.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT,
        ViewGroup.LayoutParams.WRAP_CONTENT);
  }

  @Override
  public void onLayoutChildren(RecyclerView.Recycler recycler, RecyclerView.State state) {
    if (getItemCount() <= 0 || state.isPreLayout()) {
      return;
    }
    super.onLayoutChildren(recycler, state);
    detachAndScrapAttachedViews(recycler);
    /* 这个方法主要用于计算并保存每个ItemView的位置 */
    calculateChildrenSite(recycler);
    recycleAndFillView(recycler, state);
  }

  private void calculateChildrenSite(RecyclerView.Recycler recycler) {
        totalHeight = 0;
        boolean needNew = true;
        int width = 0;
        int height = 0;
        for (int i = 0; i < getItemCount(); i++) {
            // 没有会创建
            if (needNew) {
                View view = recycler.getViewForPosition(i);
                measureChildWithMargins(view, DisplayUtils.getScreenWidth() / 2, 0);
                calculateItemDecorationsForChild(view, new Rect());
                width = getDecoratedMeasuredWidth(view);
                height = getDecoratedMeasuredHeight(view);
                addView(view);
            }
            if (totalHeight > getHeight() + height) {
                needNew = false;
            }
            Rect mTmpRect = allItemRects.get(i);
            if (mTmpRect == null) {
                mTmpRect = new Rect();
            }

            if (i % 2 == 0) { // 当i能被2整除时,是左,否则是右。
                // 左
                mTmpRect.set(0, totalHeight, DisplayUtils.getScreenWidth() / 2, totalHeight + height);
            } else {
                // 右,需要换行
                mTmpRect.set(DisplayUtils.getScreenWidth() / 2, totalHeight, DisplayUtils.getScreenWidth(),
                        totalHeight + height);
                totalHeight = totalHeight + height;
            }
            // 保存ItemView的位置信息
            allItemRects.put(i, mTmpRect);
            // 由于之前调用过detachAndScrapAttachedViews(recycler),所以此时item都是不可见的
            itemStates.put(i, false);
        }

    }


  private void recycleAndFillView(RecyclerView.Recycler recycler, RecyclerView.State state) {
    if (getItemCount() <= 0 || state.isPreLayout()) {
      return;
    }

    // 当前scroll offset状态下的显示区域
    Rect displayRect= new Rect(0, verticalScrollOffset, getHorizontalSpace(),
        verticalScrollOffset + getVerticalSpace());

    /**
     * 将滑出屏幕的Items回收到Recycle缓存中
     */
    Rect childRect = new Rect();
    for (int i = 0; i < getItemCount(); i++) {
      //这个方法获取的是RecyclerView中的View,注意区别Recycler中的View
      //这获取的是实际的View
      View child = recycler.getViewForPosition(i);
      //下面几个方法能够获取每个View占用的空间的位置信息,包括ItemDecorator
      childRect.left = getDecoratedLeft(child);
      childRect.top = getDecoratedTop(child);
      childRect.right = getDecoratedRight(child);
      childRect.bottom = getDecoratedBottom(child);
      //如果Item没有在显示区域,就说明需要回收
      if (!Rect.intersects(displayRect, childRect)) {
        //移除并回收掉滑出屏幕的View
        removeAndRecycleView(child, recycler);
        itemStates.put(i, false); //更新该View的状态为未依附
      }
    }

    //重新显示需要出现在屏幕的子View
    for (int i = 0; i < getItemCount(); i++) {
      //判断ItemView的位置和当前显示区域是否重合
      if (Rect.intersects(displayRect, allItemRects.get(i))) {
        //获得Recycler中缓存的View
        View itemView = recycler.getViewForPosition(i);
        measureChildWithMargins(itemView, DisplayUtils.getScreenWidth() / 2, 0);
        //添加View到RecyclerView上
        addView(itemView);
        //取出先前存好的ItemView的位置矩形
        Rect rect = allItemRects.get(i);
        //将这个item布局出来
        layoutDecoratedWithMargins(itemView,
          rect.left,
          rect.top - verticalScrollOffset,  //因为现在是复用View,所以想要显示在
          rect.right,
          rect.bottom - verticalScrollOffset);
        itemStates.put(i, true); //更新该View的状态为依附
      }
    }
    LogUtils.e("itemCount = " + getChildCount());
  }


  @Override
  public boolean canScrollVertically() {
    // 返回true表示可以纵向滑动
    return true;
  }

  @Override
  public int scrollVerticallyBy(int dy, RecyclerView.Recycler recycler, RecyclerView.State state) {
    //每次滑动时先释放掉所有的View,因为后面调用recycleAndFillView()时会重新addView()。
    detachAndScrapAttachedViews(recycler);
    // 列表向下滚动dy为正,列表向上滚动dy为负,这点与Android坐标系保持一致。
    // 实际要滑动的距离
    int travel = dy;

    LogUtils.e("dy = " + dy);
    // 如果滑动到最顶部
    if (verticalScrollOffset + dy < 0) {
      travel = -verticalScrollOffset;
    } else if (verticalScrollOffset + dy > totalHeight - getVerticalSpace()) {// 如果滑动到最底部
      travel = totalHeight - getVerticalSpace() - verticalScrollOffset;
    }
    // 调用该方法通知view在y方向上移动指定距离
    offsetChildrenVertical(-travel);
    recycleAndFillView(recycler, state); //回收并显示View
    // 将竖直方向的偏移量+travel
    verticalScrollOffset += travel;
    return travel;
  }

  private int getVerticalSpace() {
    // 计算RecyclerView的可用高度,除去上下Padding值
    return getHeight() - getPaddingBottom() - getPaddingTop();
  }

  @Override
  public boolean canScrollHorizontally() {
    // 返回true表示可以横向滑动
    return super.canScrollHorizontally();
  }

  @Override
  public int scrollHorizontallyBy(int dx, RecyclerView.Recycler recycler,
      RecyclerView.State state) {
    // 在这个方法中处理水平滑动
    return super.scrollHorizontallyBy(dx, recycler, state);
  }

  public int getHorizontalSpace() {
    return getWidth() - getPaddingLeft() - getPaddingRight();
  }
}

实现缓存最主要的就是先把每个ItemView的位置信息保存起来,然后在滑动过程中通过判断每个ItemView的位置是否和当前RecyclerView应该显示的区域有重合,若有就显示它,若没有就移除并回收

总结

实现自己的自定义LayoutManager主要的三个步骤:

  • 计算每个ItemView的位置;
  • 添加滑动事件;
  • 实现缓存。

我们需根据代码多理解,多思考,然后动手写属于自己的LayoutManager

探讨

最近路上留意到很多三轮摩托老司机开车十分的奔放,和拉力赛有得一拼。之前坐过几次,坐的时候因为赶时间,所以当时感觉老司机好负责。但最近作为路人看,老司机开车开的太危险,强行抢道,疯狂按喇叭...整个是横冲直撞的态势。总之觉得很危险。
这件事,你怎么看?

如果你觉得这篇文章对你有帮助的话,点赞走一走,再加个关注,互相交流下。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,717评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,501评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,311评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,417评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,500评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,538评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,557评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,310评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,759评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,065评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,233评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,909评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,548评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,172评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,420评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,103评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,098评论 2 352

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,980评论 25 707
  • 基本使用RecyclerView的基本使用并不复杂,只需要提供一个RecyclerView.Apdater的实现用...
    庞哈哈哈12138阅读 5,997评论 2 46
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,081评论 4 62
  • 我是诗人 我不是精神病 我是这个世界的神 诗人是一个独特的存在 什么都可以做不是无病呻吟 一滴露珠可以发现春天的秘...
    香自苦寒阅读 263评论 2 3
  • 这是一部很感人 很感性的电影,里面描写的不仅仅地震所带来的改变,而是在不同时候,不同事情, 不同地方时的人性,人心...
    blair_c阅读 97评论 0 0