View的布局基础:
测量与布局阶段都是一个递归过程。
具体:
(1)重写onMeasure来修改已有的view的尺寸
(2)重写onMeasure来全新计算自定义view的尺寸
(3)重写onMeasure和onlayout来全新计算自定义ViewGroup的内部布局
父view对于子view的限制:
父view结合开发者对于view的限制,计算出精确地view该显示的尺寸。子view通过resolveSize()方法, 会得到满足父view要求的尺寸值。
resolve方法执行的步骤:
MeasureSpec.getMode 获取模式,MeasureSpec.getSize() 获取尺寸,判断mode值,
1)如果mode是UNSPECIFIED,则表示不限制,那么返回的size值是view想要设置的size值。
2)如果mode是AT_MOST,表示限制上限,通过MeasureSpec.getSize() 获取的是父布局对于自己的限制,表示实际可以设置的最大值specSize ,当子view想要设置的尺寸size大于specSize ,那么返回值取specSize ,否则表示view想要设置的尺寸在最大范围值内,所以取值是size。
3)如果mode是EXACTLY,表示精确值,那么返回值取MeasureSpec.getSize() 获取的size,父view限制了固定的尺寸,子view不必再算,或者说算了也没用,父view已经给确定的值。
全新定制尺寸和修改尺寸的重要区别:
需要在计算的同时,保证计算结果要满足父view给出的尺寸限制。
父view的尺寸限制
(1)由来,开发者的要求(布局文件中layout_开头的属性,)经过父view处理计算后更精确的要求。
(2)限制的分类:
UNSPECIFIED:不限制
AT_MOST:限制上限
EXACTLY:限制固定值
全新定义自定义view尺寸的方式
(1)重写onMeasure,并计算出view的尺寸。
(2)使用resolveSize,来让子view的计算结果符合父view的限制(也可以用自己的方式来满足父view的限制)
layout内部布局的自定义
1、重写onMeasure来计算
2、重写onLayout
ViewGroup的onMeasure的重写
(1)调用每个子view的measure(),让子view自我测量
(2)根据子view给出的尺寸,得出子view的位置,并保存他们的位置和尺寸
(3)根据子view的位置和尺寸计算出自己的尺寸,并用setMeasuredDimension保存
可用空间的判断方法:
根据自己的MeasureSpec中mode的不同:
1.EXACTLY:
可用空间:MeasureSpec中的size
2.AT_MOST:
可用空间 :MeasureSpec中的size
3.UNSPECIFIED:
可用空间:无限大