CoordinatorLayout.Behavior源码分析(一)

上一篇自定义CoordinatorLayout.Behavior,总结了CoordinatorLayout.Behavior的使用和自定义。

本文主要分析

  • Behavior的构造函数
  • Property Behavior的关键方法
    • layoutDependsOn
    • onDependentViewChanged

1 Behavior的构造函数

以上一篇自定义CoordinatorLayout.Behavior的Property Behavior TextViewPropertyBehavior为例,两个参数的构造函数的调用栈如下:

LayoutInflater.inflate  (LayoutInflater.java:396)
   LayoutInflater.inflate  (LayoutInflater.java:489)
      LayoutInflater.rInflate  (LayoutInflater.java:748)
         CoordinatorLayout.generateLayoutParams  (CoordinatorLayout.java:92)
            CoordinatorLayout.generateLayoutParams  (CoordinatorLayout.java:1435)
               CoordinatorLayout$LayoutParams.<init>  (CoordinatorLayout.java:2288)
                  CoordinatorLayout.parseBehavior  (CoordinatorLayout.java:585)
                     Constructor.newInstance  (Constructor.java:417)
                        Constructor.constructNative  (Constructor.java:-2)
                           TextViewPropertyBehavior.<init>  (TextViewPropertyBehavior.java:25)

可以看出加载布局的时候,CoordinatorLayout.parseBehavior解析XML中的app:layout_behavior的值(即TextViewPropertyBehavior)的全名,通过反射调用其构造方法。

1.1 获取app:layout_behavior等参数

  1. CoordinatorLayout继承自ViewGroup,is a super-powered FrameLayout
  2. CoordinatorLayout调用generateLayoutParams,获取Layout所需要的参数
  3. generateLayoutParams调用CoordinatorLayout.LayoutParams(design-23.2.1 CoordinatorLayout.java:2267)的构造函数,获取子布局中的参数。这些参数决定子布局在CoordinatorLayout中的布局。
LayoutParams(Context context, AttributeSet attrs) {
    super(context, attrs);
    //获取XML布局属性
    final TypedArray a = context.obtainStyledAttributes(attrs,
            R.styleable.CoordinatorLayout_LayoutParams);

    this.gravity = a.getInteger(
            R.styleable.CoordinatorLayout_LayoutParams_android_layout_gravity,
            Gravity.NO_GRAVITY);
    mAnchorId = a.getResourceId(R.styleable.CoordinatorLayout_LayoutParams_layout_anchor,
            View.NO_ID);
    this.anchorGravity = a.getInteger(
            R.styleable.CoordinatorLayout_LayoutParams_layout_anchorGravity,
            Gravity.NO_GRAVITY);

    this.keyline = a.getInteger(R.styleable.CoordinatorLayout_LayoutParams_layout_keyline,
            -1);

    mBehaviorResolved = a.hasValue(
            R.styleable.CoordinatorLayout_LayoutParams_layout_behavior);
    
    //获取Behavior
    if (mBehaviorResolved) {
        mBehavior = parseBehavior(context, attrs, a.getString(
                R.styleable.CoordinatorLayout_LayoutParams_layout_behavior));
    }

    a.recycle();
}

1.2 实例化Behavior

实例化Behavior的逻辑在parseBehavior函数。

static Behavior parseBehavior(Context context, AttributeSet attrs, String name) {
    //排除参数为空的情况
    if (TextUtils.isEmpty(name)) {
        return null;
    }

    final String fullName;

    //如果以"."开头获取包名,得到类的全名
    if (name.startsWith(".")) {
        // Relative to the app package. Prepend the app package name.
        fullName = context.getPackageName() + name;
    } else if (name.indexOf('.') >= 0) {
        // Fully qualified package name.
        fullName = name;
    } else {
         //只有类名的,在CoordinatorLayout的包中获取预备的Behavior类
        // Assume stock behavior in this package.
        fullName = WIDGET_PACKAGE_NAME + '.' + name;
    }

    //通过反射获取Behavior实现类的构造方法
    try {
        Map<String, Constructor<Behavior>> constructors = sConstructors.get();
        if (constructors == null) {
            constructors = new HashMap<>();
            sConstructors.set(constructors);
        }
        Constructor<Behavior> c = constructors.get(fullName);
        if (c == null) {
            final Class<Behavior> clazz = (Class<Behavior>) Class.forName(fullName, true,
                    context.getClassLoader());
            c = clazz.getConstructor(CONSTRUCTOR_PARAMS);
            c.setAccessible(true);
            constructors.put(fullName, c);
        }
        //调用两个参数的构造方法
        return c.newInstance(context, attrs);
    } catch (Exception e) {
        throw new RuntimeException("Could not inflate Behavior subclass " + fullName, e);
    }
}

2. Property Behavior的关键方法

2.1 layoutDependsOn 依赖条件的判断

开启一个CoordinatorLayout之后layoutDependsOn被多次调用。根据Logger的日志,可知,CoordinatorLayout.onMeasure调用的。也就是说,布局改变会触发layoutDependsOn方法。

View.measure  (View.java:17547)
   CoordinatorLayout.onMeasure  (CoordinatorLayout.java:671)
      CoordinatorLayout.ensurePreDrawListener  (CoordinatorLayout.java:1288)
         CoordinatorLayout.hasDependencies  (CoordinatorLayout.java:1318)
            CoordinatorLayout$LayoutParams.dependsOn  (CoordinatorLayout.java:2461)
               TextViewPropertyBehavior.layoutDependsOn  (TextViewPropertyBehavior.java:16)
                  TextViewPropertyBehavior.layoutDependsOn  (TextViewPropertyBehavior.java:37)
  1. 上文讲到构造函数获取XML中的app:layout_behavior属性值保存在CoordinatorLayout.LayoutParams
  2. CoordinatorLayout.hasDependencies有根据LayoutParams存储的属性值,判断是否存在带有合法依赖关系的子布局。
    1. 条件一app:layout_anchor(设置子View的锚点,即以哪个控件为参照点设置位置。)已经设置。
    2. 条件二:至少一个子View符合dependsOn(CoordinatorLayout:2460)函数的条件
/**
 * Check if an associated child view depends on another child view of the CoordinatorLayout.
 *
 * @param parent the parent CoordinatorLayout
 * @param child the child to check
 * @param dependency the proposed dependency to check
 * @return true if child depends on dependency
 */
boolean dependsOn(CoordinatorLayout parent, View child, View dependency) {
    return dependency == mAnchorDirectChild
            || (mBehavior != null && mBehavior.layoutDependsOn(parent, child, dependency));
}

可以看到这里调用了Behavior.layoutDependsOn方法。

2.2 onDependentViewChanged 依赖的逻辑关系

  1. ensurePreDrawListener函数判断,如果符合依赖条件,给布局添加一个OnPreDrawListner
void addPreDrawListener() {
    if (mIsAttachedToWindow) {
        // Add the listener
        if (mOnPreDrawListener == null) {
            mOnPreDrawListener = new OnPreDrawListener();
        }
        final ViewTreeObserver vto = getViewTreeObserver();
        vto.addOnPreDrawListener(mOnPreDrawListener);
    }

    // Record that we need the listener regardless of whether or not we're attached.
    // We'll add the real listener when we become attached.
    mNeedsPreDrawListener = true;
}

ViewTreeObserver.OnPreDrawListenerCoordinatorLayout重绘前调用。

  1. CoordinatorLayout实现了这个接口,如下:
class OnPreDrawListener implements ViewTreeObserver.OnPreDrawListener {
    @Override
    public boolean onPreDraw() {
        dispatchOnDependentViewChanged(false);
        return true;
    }
}
  1. dispatchOnDependentViewChanged这个函数的命名来看,肯定是调用了onDependentViewChanged了。看(CoordinatorLayout.java 1173)
if (b != null && b.layoutDependsOn(this, checkChild, child)) {
    if (!fromNestedScroll && checkLp.getChangedAfterNestedScroll()) {
        // If this is not from a nested scroll and we have already been changed
        // from a nested scroll, skip the dispatch and reset the flag
        checkLp.resetChangedAfterNestedScroll();
        continue;
    }

    final boolean handled = b.onDependentViewChanged(this, checkChild, child);
    
    //...后面还有处理NestedScroll的响应逻辑
}

因此,每次重绘dispatchOnDependentViewChanged再找出存在依赖关系的子View,上次是为了添加监听器,这次是执行依赖逻辑Behavior.onDependentViewChanged

2.3 mDependencySortedChildren

CoordinatorLayout.dispatchOnDependentViewChanged在获取dependent view(被依赖)和auto view时,都从mDependencySortedChildren获取。CoordinatorLayout到底用什么结构存储这些View和她们的关系呢?

改变mDependencySortedChildren在布局初始化的时候初始化,(L:613)prepareChilden具体实现。

private void prepareChildren() {
    mDependencySortedChildren.clear();
    //首先无差别添加所有的child view(所以,dispatchOnDependentViewChanged之前还要调用layoutDependsOn判断是否符合依赖关系)
    for (int i = 0, count = getChildCount(); i < count; i++) {
        final View child = getChildAt(i);

        final LayoutParams lp = getResolvedLayoutParams(child);
        lp.findAnchorView(this, child);

        mDependencySortedChildren.add(child);
    }
    // We need to use a selection sort here to make sure that every item is compared
    // against each other
    //排序
    selectionSort(mDependencySortedChildren, mLayoutDependencyComparator);
}
  1. 首先无差别添加所有的child view(所以,dispatchOnDependentViewChanged之前还要调用layoutDependsOn判断是否符合依赖关系
  2. 排序,排序规则看mLayoutDependencyComparatorselectionSort的实现

mLayoutDependencyComparator的实现

    final Comparator<View> mLayoutDependencyComparator = new Comparator<View>() {
        @Override
        public int compare(View lhs, View rhs) {
            if (lhs == rhs) {
                return 0;
            } else if (((LayoutParams) lhs.getLayoutParams()).dependsOn(
                    CoordinatorLayout.this, lhs, rhs)) {
                return 1;
            } else if (((LayoutParams) rhs.getLayoutParams()).dependsOn(
                    CoordinatorLayout.this, rhs, lhs)) {
                return -1;
            } else {
                return 0;
            }
        }
    };

dependsOn上文已经提到。

咋看,如果用的是Collection.sort,那么dependent view排在后,auto view排在前。
可是,实际上,selectionSort的逻辑是,通过一维数组存储一对多的数据结构,存储了CoordinatorLayout所有子View的依赖关系。

|0...|i | i+1 |...| ... |||n-1|
|--|-----------:|:-------------:| -----:|
|...|dependency|auto|auto|...|dependency|auto|...|

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

推荐阅读更多精彩内容