CoordinateLayout的实现原理

引言

CoordinateLayout的主要用法:
1、作为顶层应用的装饰或者chrome布局
2、作为一个能响应特定的一个或多个子视图交互的容器
也就是说,CoordinateLayout本身不具备布局的能力,它只是作为一个将Behavior和子View绑定的容器,将收到的事件几乎原封不动的分发给子View对应的Behavior。

例如,BottomSheetDialog的布局:

<FrameLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:fitsSystemWindows="true">

  <androidx.coordinatorlayout.widget.CoordinatorLayout
      android:id="@+id/coordinator"
      android:layout_width="match_parent"
      android:layout_height="match_parent"
      android:fitsSystemWindows="true">

    <View
        android:id="@+id/touch_outside"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:importantForAccessibility="no"
        android:soundEffectsEnabled="false"
        tools:ignore="UnusedAttribute"/>

    <FrameLayout
        android:id="@+id/design_bottom_sheet"
        style="?attr/bottomSheetStyle"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_gravity="center_horizontal|top"
        app:layout_behavior="@string/bottom_sheet_behavior"/>

  </androidx.coordinatorlayout.widget.CoordinatorLayout>

</FrameLayout>

我们将子View add到design_bottom_sheet的FrameLayout中之后,就能实现从底部弹窗的效果。
我们找到这个Behavior发现对应了一个java类,发现了里面包含了大量事件处理的函数。

<string name="bottom_sheet_behavior" translatable="false">com.google.android.material.bottomsheet.BottomSheetBehavior</string>

引发我们思考几个问题:
1、这个layout_behavior属性为什么加在子View上面?
2、上面的Behavior又是如何跟对应的View关联的,为何Behavoir能处理子View对应的行为?

实现原理

针对上述原理,我们进一步探究CoordinateLayout的实现原理。
如引言所说,coordinateLayout只是一个代理,只是仅仅作为事件转发给对应的Behavior,那么是如何转发的呢?以及Behavior又是如何跟对应的子View绑定的呢?
针对第2个问题:
我们回到bottomSheet那个例子,发现在对应的子View上指定了layout_behavior这个属性。
事实上coordianteLayout将子View上的属性解析出来,然后获取layout_behavior上的值,也就是对应的class类名。
之后就可以通过反射class对应的类,构造Behavior对象。
CoordinateLayout继承于ViewGroup,拥有响应事件的能力,因此,它将事件进一步转发给子类的所有Behavior。

其中还有一个关键的问题,就是CoordinateBehavior何时能够获取到子View的Attribute。
事实上,子View在被addView到Parent中的时候,会先看下子View有没有提供LayoutParams,如果没有提供,就会调用父ViewGroup的generatorLayoutParams()生成对应的LayoutParams,在此时接收了子View的Attribute参数。

伪代码:

public class CoordinateLayout extends ViewGroup {
  // 拿到AttributeSet,生成自己的LayoutParams。
  @Override
  public LayoutParams generateLayoutParams(AttributeSet attrs) {
      return new LayoutParams(getContext(), attrs);
  }
  
  private static class LayoutParams {
    Behavor behavior;
    public LayoutParams(Context context, AttributeSet attrs) {
      // get layout_behavior attribute from attrs
      TypedArray typedArray = c.obtainStyledAttributes(attrs, R.styleable.BehaviorCoordinateLayout);
      String behaviorClassName = typedArray.getString(R.styleable.BehaviorCoordinateLayout_layout_behavior);
      // Generate the Behavior object use reflection
      behavior = parseBehavior(behaviorClassName, c, attrs);
      typedArray.recycle();
    }
  }
    

  public void onTouchEvent(MotionEvent event) {
        for (int i = 0; i < getChildCount(); i++) {
            View child = getChildAt(i);
            LayoutParams lp = (LayoutParams) child.getLayoutParams();
            if (lp.behavior != null) {
                lp.behavior.onTouchEvent(this, child, event);
            }
        }
        return false;
  }

  // .. 其他代码省略
}

我们通过在generatorLayoutParams中生成自己的LayoutParams返回,并在LayoutParams中绑定自己对应的Behavior,之后在CoordinateLayout收到某个事件的时候,直接传递给所有子View的Behavior。
需要消费该事件的子View只需要实现Behavior对应的方法,即可完成事件的处理。
Behavior对应的伪代码很简单:

public class Behavior {
    public Behavior(Context context, AttributeSet attrs) {}
    public void onTouchEvent(View parent, View child, MotionEvent ev) { }
}

例如,在BottomSheetDialog中,子View就绑定对应的BottomSheetBehavior,进而在该Behavior中进一步通过viewDragHelper来完成手势等操作,置于viewDragHelper如何使用请自行参考其他文章。

总结

本文主要介绍了CoordinateLayout的原理,描述了其如何绑定子View对应的Behavior,以及如何将对应的Event转发给对应的子View进行处理。

posted @ 2021-09-02 01:28  、、、路遥  阅读(557)  评论(0编辑  收藏  举报