一. 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.
二.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"呢?
图2.1中view1依赖于view2,view2依赖于view3.view1,view2,view3就构成了一条relation为LEFT_OF的约束链.约束链中距离view1最近且可见性不为GONE的view为view1的"related view",在这里"related view"就是view3.
三.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会被设置一次值。
如果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()方法中会多次调整。