对自定义view还不是很了解的码友可以先看自定义View入门这篇文章,本文主要对自定义ViewGroup的过程的梳理,废话不多说。
1.View 绘制流程
ViewGroup也是继承于View,下面看看绘制过程中依次会调用哪些函数。
说明:
measure()
和onMeasure()
在
View.Java
源码中:
public final void measure(int widthMeasureSpec,int heightMeasureSpec){
···
onMeasure
···
}
protected void onMeasure(int widthMeasureSpec,int heightMeasureSpec) {
setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec),
getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec));
}
可以看出measure()是被final修饰的,这是不可被重写。onMeasure在measure方法中调用的,当我们继承View的时候通过重写onMeasure方法来测量控件大小。
layout()和onLayout(),draw()和onDraw()类似。
dispatchDraw()
View 中这个函数是一个空函数,ViewGroup 复写了dispatchDraw()来对其子视图进行绘制。自定义的 ViewGroup 一般不对dispatchDraw()进行复写。
requestLayout()
当布局变化的时候,比如方向变化,尺寸的变化,会调用该方法,在自定义的视图中,如果某些情况下希望重新测量尺寸大小,应该手动去调用该方法,它会触发measure()和layout()过程,但不会进行 draw。
自定义ViewGroup的时候一般复写
onMeasure()方法:
计算childView的测量值以及模式,以及设置自己的宽和高
onLayout()方法,对其所有childView的位置进行定位
View树:
树的遍历是有序的,由父视图到子视图,每一个 ViewGroup 负责测绘它所有的子视图,而最底层的 View 会负责测绘自身。
measure:
自上而下进行遍历,根据父视图对子视图的MeasureSpec以及ChildView自身的参数,通过
getChildMeasureSpec(parentHeightMeasure,mPaddingTop+mPaddingBottom,lp.height)
获取ChildView的MeasureSpec,回调ChildView.measure最终调用setMeasuredDimension得到ChildView的尺寸:
mMeasuredWidth 和 mMeasuredHeight
Layout :
也是自上而下进行遍历的,该方法计算每个ChildView的ChildLeft,ChildTop;与measure中得到的每个ChildView的mMeasuredWidth 和 mMeasuredHeight,来对ChildView进行布局。
child.layout(left,top,left+width,top+height)
2.onMeasure过程
measure过程会为一个View及所有子节点的mMeasuredWidth
和mMeasuredHeight变量赋值,该值可以通过getMeasuredWidth()和getMeasuredHeight()方法获得。
onMeasure过程传递尺寸的两个类:
ViewGroup.LayoutParams (ViewGroup 自身的布局参数)
用来指定视图的高度和宽度等参数,使用 view.getLayoutParams() 方法获取一个视图LayoutParams,该方法得到的就是其所在父视图类型的LayoutParams,比如View的父控件为RelativeLayout,那么得到的 LayoutParams 类型就为RelativeLayoutParams。
①具体值
②MATCH_PARENT 表示子视图希望和父视图一样大(不包含 padding 值)
③WRAP_CONTENT 表示视图为正好能包裹其