Android:嵌套滑动总结

貌似这个比较复杂,记录一下

解释的比较好的。
https://blog.csdn.net/recall2012/article/details/79474172

一.何为滑动嵌套?

就是就有滑动功能的view,再嵌套另一个具有滑动功能的view,例如recyclerview嵌套recyclerview。

二.CoordinatorLayout是什么?

是协调布局,是一个超级framelayout,可以控制多个view协调运动。通过behavior来实现。主要原理是,子view获取到滑动事件后,在滑动前,先询问一下父view,问下父view要不要先滑,父view如果要滑,那就滑,然后把滑完之后把剩下的距离告诉子view,然后子view再滑。
所以说,如果处理不好,可以滑起来卡卡的,例如父view滑了一点,然后子view又滑了一点,你就会感觉卡卡的,难受。

三.behavior是怎样的?

    public static abstract class Behavior<V extends View> {

        public void onNestedScroll(@NonNull CoordinatorLayout coordinatorLayout, @NonNull V child,
                @NonNull View target, int dxConsumed, int dyConsumed,
                int dxUnconsumed, int dyUnconsumed, @NestedScrollType int type) {
            if (type == ViewCompat.TYPE_TOUCH) {
                onNestedScroll(coordinatorLayout, child, target, dxConsumed, dyConsumed,
                        dxUnconsumed, dyUnconsumed);
            }
        }

        @Deprecated
        public void onNestedPreScroll(@NonNull CoordinatorLayout coordinatorLayout,
                @NonNull V child, @NonNull View target, int dx, int dy, @NonNull int[] consumed) {
            // Do nothing
        }

        public void onNestedPreScroll(@NonNull CoordinatorLayout coordinatorLayout,
                @NonNull V child, @NonNull View target, int dx, int dy, @NonNull int[] consumed,
                @NestedScrollType int type) {
            if (type == ViewCompat.TYPE_TOUCH) {
                onNestedPreScroll(coordinatorLayout, child, target, dx, dy, consumed);
            }
        }


        public boolean onNestedFling(@NonNull CoordinatorLayout coordinatorLayout,
                @NonNull V child, @NonNull View target, float velocityX, float velocityY,
                boolean consumed) {
            return false;
        }


        public boolean onNestedPreFling(@NonNull CoordinatorLayout coordinatorLayout,
                @NonNull V child, @NonNull View target, float velocityX, float velocityY) {
            return false;
        }
.......

是个抽象类,看上去有点像NestedScrollingParent的方法。

例如,我们看看onDependentViewChanged是怎么调用的把


image.png

可以看到,CoordinatorLayout注册了PreDraw的监听器,就是view在绘制前,会有回调函数回调到CoordinatorLayout的onChildViewsChanged,

if (b != null && b.layoutDependsOn(this, checkChild, child)) { //首先检查是不是依赖的view发生了改变
      .....  //依赖的view发生改变了,回调给需要协调运动的view
      handled = b.onDependentViewChanged(this, checkChild, child);
       .....
}

我们再看一种场景,就是使用onNestedPreScroll来实现的协调运动。
例如https://github.com/SheHuan/BehaviorDemo
的第二个test。

看看他的backtrace


image.png

可以看到事件是从recyclerview传过来的,所以说move事件到达recyclerview后,recyclerview作为NestedScrollingChild2,会先循环一下作为NestedScrollingParent2的CoordinatorLayout,你CoordinatorLayout有没有需要滑动的,有没有需要处理move事件的,等CoordinatorLayou处理完之后,recyclerview再继续跑。这样看,像是子view和父view都消费了move事件,挺有意思。

recyclerview调用dispatchNestedPreScroll后,实际用使用NestedScrollingChildHelper来处理,

    public boolean dispatchNestedPreScroll(int dx, int dy, @Nullable int[] consumed,
            @Nullable int[] offsetInWindow, @NestedScrollType int type) {
        if (isNestedScrollingEnabled()) { //判断是否支持嵌套滑动,我们常常可以在应用中关闭嵌套滑动
                ViewParentCompat.onNestedPreScroll(parent, mView, dx, 
    }

然后,最后就要回调到CoordinatorLayout,然后CoordinatorLayout会遍历每个子view,子view可以根据需要来确定是否处理。

四.分析一下BottomSheetBehavior的绘制流程

layout流程
    @Override
    @SuppressWarnings("unchecked")
    protected void onLayout(boolean changed, int l, int t, int r, int b) {

            final Behavior behavior = lp.getBehavior();
           //获取childview的behavior,然后调用behavior的onLayoutChild去布局。如果没有behavior就走CoordinatorLayout自身的onLayoutChild
            if (behavior == null || !behavior.onLayoutChild(this, child, layoutDirection)) {
                onLayoutChild(child, layoutDirection);
            }
        }
    }

可以看到,有配置behavior的view,就走behavior的onLayoutChild方法。所以layout方法被behavior接管了。

  viewDragHelper = ViewDragHelper.create(parent, dragCallback);

最后其实就是用viewdragger来实现拖拉位置的

  @Override
  public boolean onLayoutChild(
      @NonNull CoordinatorLayout parent, @NonNull V child, int layoutDirection) {
    if (state == STATE_EXPANDED) {
      ViewCompat.offsetTopAndBottom(child, getExpandedOffset());
    } else if (state == STATE_HALF_EXPANDED) {
      ViewCompat.offsetTopAndBottom(child, halfExpandedOffset);
    } else if (hideable && state == STATE_HIDDEN) {
      ViewCompat.offsetTopAndBottom(child, parentHeight);
    } else if (state == STATE_COLLAPSED) {
      ViewCompat.offsetTopAndBottom(child, collapsedOffset);
    } else if (state == STATE_DRAGGING || state == STATE_SETTLING) {
      ViewCompat.offsetTopAndBottom(child, savedTop - child.getTop());
    }
  }

五.bottomsheet嵌套recyclerview后,recyclerview关闭嵌套滑动之后,滑recyclerview,bottomsheet会动吗?

会动,别以为关闭嵌套滑动,父view就不拦截事件了,实际上,父view会会直接把在slopmove的时候,直接开始拦截move事件了。


image.png

那拦截之后,是怎么拖到整个bottomsheet的呢?


image.png

其实就是拦截之后,事件去了CoordinatorLayout,然后再调用behavior,behavior使用ViewDragHelper去拖动配置了behavior的view网上挪。

关键是下面的函数,来确认是否拦截,返回true就拦截,因为touchingScrollingChild为false,所以最后返回了true。如果recyclerview设置了NestedScrollingEnabled,则touchingScrollingChild为true,最后这个函数返回false,父view就不拦截,move事件直接到达recyclerview。

        @Override
        public boolean tryCaptureView(@NonNull View child, int pointerId) {
          Log.i("fengfeng", "tryCaptureView start:" + "child=" + child + " pointId=" + pointerId + " state=" + state);
          if (state == STATE_DRAGGING) {
            Log.i("fengfeng", "tryCaptureView false 1");
            return false;
          }
          if (touchingScrollingChild) {
            Log.i("fengfeng", "tryCaptureView false 2");
            return false;
          }
          if (state == STATE_EXPANDED && activePointerId == pointerId) {
            View scroll = nestedScrollingChildRef != null ? nestedScrollingChildRef.get() : null;
            if (scroll != null && scroll.canScrollVertically(-1)) {
              // Let the content scroll up
              Log.i("fengfeng", "tryCaptureView false 3");
              return false;
            }
          }
          Log.i("fengfeng", "tryCaptureView true 4");
          return viewRef != null && viewRef.get() == child;
        }        @Override
        public boolean tryCaptureView(@NonNull View child, int pointerId) {
          Log.i("fengfeng", "tryCaptureView start:" + "child=" + child + " pointId=" + pointerId + " state=" + state);
          if (state == STATE_DRAGGING) {
            Log.i("fengfeng", "tryCaptureView false 1");
            return false;
          }
          if (touchingScrollingChild) {
            Log.i("fengfeng", "tryCaptureView false 2");
            return false;
          }
          if (state == STATE_EXPANDED && activePointerId == pointerId) {
            View scroll = nestedScrollingChildRef != null ? nestedScrollingChildRef.get() : null;
            if (scroll != null && scroll.canScrollVertically(-1)) {
              // Let the content scroll up
              Log.i("fengfeng", "tryCaptureView false 3");
              return false;
            }
          }
          Log.i("fengfeng", "tryCaptureView true 4");
          return viewRef != null && viewRef.get() == child;
        }

touchingScrollingChild在down事件赋值,代表点中了一个可以scroll的view,例如recyclerview就是可以scroll的。

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