RelativeLayout onMeasure()方法分析

一. RelativeLayout.LayoutParams 中的mRules数组

xml中的属性android:layout_toRightOf,android:layout_toLeftOf等在RelativeLayout中都有对应的int类型常量字段:RIGHT_OF,LEFT_OF等。他们都叫做child的rule(也叫constraints,约束)。

某个child的rules都保存在它对应的layoutParams中的mRules中。rule对应的常量就是rule在mRules中保存位置的下标。rule分为两种:

(1)child与parent之间的rule

例如android:layout_alignParentRight(对应常量ALIGN_PARENT_BOTTOM 值为 12);如果child具有这个rule则mRules数组下标为12的位置值为-1,如果没有这个rule则对应位置上的值为0.

(2)child与child之间的rule

例如android:layout_toRightOf="@+id/text1";(对应常量RIGHT_OF值为1)如果child具有这个rule则mRules数组下标为1的位置上的值为View(@+id/text1)的Id值。parent在测量的过程中会用到这个值。如果没有这个rule则对应位置上的值为0.


RelativeLayout onMeasure()方法分析_第1张图片
具有layout_alignParentRight="true"和layout_toRightOf="@+id/text1"属性的View

二.private View getRelatedView(int[] rules, int relation)

rules为某个view对应的mRules数组,relation为LEFT_OF,RIGHT_OF,ABOVE等rules值.
getRelatedView()方法尝试获取某个relation上的"related view",如果找到了则返回这个view的引用,如果没找到则返回null。那么什么叫做"related view"呢?

RelativeLayout onMeasure()方法分析_第2张图片
图2.1

图2.1中view1依赖于view2,view2依赖于view3.view1,view2,view3就构成了一条relation为LEFT_OF的约束链.约束链中距离view1最近且可见性不为GONE的view为view1的"related view",在这里"related view"就是view3.

三.onMeasure的测量流程

RelativeLayout onMeasure()方法分析_第3张图片
图3.1 onMeasure()方法的流程图

在RelativeLayout中所有的child一定会被测量2次!分别在2个循环中进行。这个特性有可能导致性能问题(优化方案)。水平方向的测量和竖直方向的类似,这里只分析竖直方向。

3.1onMeasure中children竖直方向的测量过程

3.1.1applyHorizontalSizeRules()

private void applyHorizontalSizeRules(LayoutParams childParams, int myWidth, int[] rules)方法会尽可能的根据水平方向上的rules给child的childParams的mLeft和mRight设置值。设置规则如下:

(1)如果top方向上有至少一个rule则mTop被设置一个初始值,否则mTop的值为VALUE_NOT_SET
(2)如果bottom方向上有至少一个rule则mBottom被设置一个初始值,否则mBottom的值为VALUE_NOT_SET
 (3) child与child之间的rule与child与parent之间的rule采用不同的应用方式

如果RelativeLayout.LayoutParams的mLeft,mRight,mTop,mBottom中任何一个值为VALUE_NOT_SET,则表示这个方向上一个rule都没有,并且值需要根据其他约束推断。(在positionChildVertical()方法中被推断)

例如一个view具有android:layout_above="@+id/text1";属性。很明显这个view的底部需要参照text1的顶部。这时可以说view的bottom方向上有个ABOVE rule,并且view对应的 layoutParam的mBottom会被设置一次值。

RelativeLayout onMeasure()方法分析_第4张图片
图3.2 将child与child之间的rule应用到layoutParams中的流程

如果child具有一个child之间的rule,applyHorizontalSizeRules()方法会先尝试查找child 的related view。如果找到 了就根据related view 的layoutParams设置当前child的layoutParams;而不会根据这个rule 在mRuless数组中保存的view id 设置当前child view的layoutParams。于是就会出现下面这种情况:

view1(VISIBLE)-layout_below->view2(GONE)-layout_below->view3(VISIBLE)-layout_alignParentTop

view1依赖于view2,view2依赖于view3,view3依赖于parent。显然,view1的related view是view3。所以最终的结果为:
-----------
view3
         view1
-----------

applyHorizontalSizeRules()中计算的这些mLeft,mRight,mTop,mBottom值在后续的步骤中还有可能被调整!

还有一个需要注意的是child的属性优先级的问题。在applyHorizontalSizeRules()(注意!这里限定在applyHorizontalSizeRules()方法中!)如果同一方向上有多个rules,那么在applyHorizontalSizeRules()方法中最后判断的那个rule生效。

3.1.2measureChild()

measureChild(View child, LayoutParams params, int myWidth, int myHeight)方法分别调用getChildMeasureSpec()方法获取child的高宽,并传递给child的onMeasure()方法

3.1.3positionChildVertical()

positionChildVertical(View child, LayoutParams params, int myHeight, boolean wrapContent)方法主要用于推断并设置mTop,mBottom中被设置为VALUE_NOT_SET的变量。

3.1.4其他

步骤(4)(5)属于计算parent 的高宽的中间步骤。步骤6在处理android:ignoreGravity这个属性。

3.2onMeasure中自身高宽的计算

保存在局部变量width和height中,在整个onMeasure()方法中会多次调整。

你可能感兴趣的:(RelativeLayout onMeasure()方法分析)