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.


具有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"呢?

图2.1

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

三.onMeasure的测量流程

图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会被设置一次值。

图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()方法中会多次调整。

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

推荐阅读更多精彩内容