一、VIew 中对 子视图 进行 measure() 操作:
1.measureChildren() : 内部调用 measureChild() 对每一个子视图进行 measure 操作
2.measureChild() : 为指定的子视图进行measure操作
3.measureChildWithMargins() : measure 时考虑把 margin 及 padding 也作为子视图大小的一部分
二、measureChildWithMargins() 执行流程:
1.调用child.getLayoutParams()获得子视图的LayoutParams属性
1.1 该方法需要重载 generateDefaultLayoutParams() 返回一个继承自 ViewGroup.MarginLayoutParams 的 布局类(例如 LinearLayout.LayoutParams),如果未重载将导致 本 View 的XML属性 layout_margin 失效并且不能使用 measureChildWithMargins()
2.调用两次 getChildMeasureSpec()函数,分别计算出孩子视图的宽度和高度的 Spec(确定子视图的测量规格).
3.调用child.measure()函数,确定子视图的最终布局大小。
三、最后调用resolveSize()获取高度,然后调用 setMeasureDimension()设置该本身所占用的布局高度。
android onmeasure在View.java中的定义:
protected void onMeasure(int widthMeasureSpec,int heightMeasureSpec) {
setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec),
getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec));
}
public static int getDefaultSize(int size, int measureSpec) {
int result = size;
int specMode = MeasureSpec.getMode(measureSpec);
int specSize = MeasureSpec.getSize(measureSpec);
switch (specMode) {
case MeasureSpec.UNSPECIFIED: //未指定
result = size;
break;
case MeasureSpec.AT_MOST: //至多
case MeasureSpec.EXACTLY: //精确
result = specSize;
break;
}
return result;
}
在ViewGroup中:
protected void measureChildren(int widthMeasureSpec, int heightMeasureSpec) {
final int size = mChildrenCount;
final View[] children = mChildren;
for (int i = 0; i < size; ++i) {
final View child = children[i];
if ((child.mViewFlags & VISIBILITY_MASK) != GONE) {
measureChild(child, widthMeasureSpec, heightMeasureSpec);
}
}
}
protected void measureChild(View child, int parentWidthMeasureSpec,
int parentHeightMeasureSpec) {
final LayoutParams lp = child.getLayoutParams();
final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec,
mPaddingLeft + mPaddingRight, lp.width);
final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec,
mPaddingTop + mPaddingBottom, lp.height);
child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
}
protected void measureChildWithMargins(View child,
int parentWidthMeasureSpec, int widthUsed,
int parentHeightMeasureSpec, int heightUsed) {
final MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams();
final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec,
mPaddingLeft + mPaddingRight + lp.leftMargin + lp.rightMargin
+ widthUsed, lp.width);
final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec,
mPaddingTop + mPaddingBottom + lp.topMargin + lp.bottomMargin
+ heightUsed, lp.height);
child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
}
说明:
-
measure是final修饰的方法,不可被重写。
在外部调用时,直接调用view.measure(int wSpec, int hSpec)。
measure中调用了onMeasure。
自定义view时,重写onMeasure即可。
2.MeasureSpec 这是一个含mode和size的结合体,不需要我们来具体的关心。
当在测量时,可以调用MeasureSpec.getSize|getMode 得到相应的size和mode。
然后使用MeasureSpec.makeMeasureSpec(size,mode); 来创建MeasureSpec对象。
那么mode是怎么来的呢?是根据使用该自定义view时的layoutWith|height参数决定的,所以不能自己随便new一个。
而size可以自己指定,也可以直接使用 measureSpec.getSize。
3.如果是一个View,重写onMeasure时要注意:
如果在使用自定义view时,用了wrap_content。那么在onMeasure中就要调用setMeasuredDimension,
来指定view的宽高。如果使用的fill_parent或者一个具体的dp值。那么直接使用super.onMeasure即可。
4.如果是一个ViewGroup,重写onMeasure时要注意:
首先,结合上面两条,来测量自身的宽高。
然后,需要测量子View的宽高。
测量子view的方式有:
-
getChildAt(int index).可以拿到index上的子view。
通过getChildCount得到子view的数目,再循环遍历出子view。 接着,subView.measure(int wSpec, int hSpec); //使用子view自身的测量方法
-
或者调用viewGroup的测量子view的方法:
//某一个子view,多宽,多高, 内部加上了viewGroup的padding值
measureChild(subView, int wSpec, int hSpec);
-
//所有子view 都是 多宽,多高, 内部调用了measureChild方法
measureChildren(int wSpec, int hSpec);
-
//某一个子view,多宽,多高, 内部加上了viewGroup的padding值、margin值和传入的宽高wUsed、hUsed
measureChildWithMargins(subView, intwSpec, int wUsed, int hSpec, int hUsed);