View的工作原理

1. ViewRoot和DecorView

ViewRoot对应于ViewRootImpl类,它连接WindowManager和DecorView。View的绘制流程从ViewRoot的performTraversals开始,经过measure,layout和draw过程。

Measure决定了View 的宽和高,通过getMeasuredWidth和getMeasuredHeight来获得;

Layout决定了四个顶点的坐标和实际的View宽高,之后可调用getTop, getBottom, getLeft, getRight, getWidth和getHeight;

Draw决定了View的显示。

 

2. MeasureSpec

在测量过程中,系统会将View的LayoutParams根据父容器所施加的规则转换成对应的MeasureSpec,得出View的测量宽高。

MeasureSpec代表一个32位的int值,高2位代表SpecMode,低30位代表SpecSize。

SpecMode取值为EXACTLY,对应LayoutParams的match_parent和具体数值两种方式,表示View的最终大小就是父容器SpecSize所指定的值;

AT_MOST,对应LayoutParams的wrap_content,表示父容器制定了一个SpecSize,View不能大于它。

子元素的MeasureSpec的创建与父容器的MeasureSpec和子元素本身的LayoutParams有关,此外还和View的margin以及padding有关。

如果View为固定宽高,则View的SpecMode都是EXACTLY并且大小为LayoutParams中的大小;

如果View为match_parent,则View的SpecMode与父容器相同,SpecSize也为parentSize;

如果View为wrap_content,则View的SpecMode都是AT_MOST,并且SpecSize为parentSize。

 

3. View的onMeasure

setMeasureDimension设置宽高的测量值,getDefaultSize返回measureSpec中的specSize,getSuggestedMinimumWidth如果View没有设置背景则返回android:minWidth,否则返回minWidth和背景最小宽度中最小的。

直接继承View的自定义控件需要重写onMeasure方法并设置wrap_content时的自身大小,否则默认的wrap_content就相当于match_parent。

重写onMeasure:在AT_MOST的情况下,将尺寸设置为自己内部算好的尺寸。

 

4. ViewGroup的measure

除了完成自己的measure,还会便利调用所有子元素的measure方法。measureChildren方法,对非GONE的子View执行measureChild。

measureChild:根据父节点的MeasureSpec,结合子节点的LayoutParams,创建子节点的MeasureSpec,传入child.measure方法。

某些极端情况下,需要多次measure才能确定最终测量宽高,所以应在onLayout中获取View 的测量宽高或者最终宽高。

以LinearLayout的measureVertical(widthMeasureSpec, heightMeasureSpec)为例,计算每个子元素的初步高度,再累加;

如果它的布局中高度采用的是match_parent或者具体数值,则高度即为specSize,如果采用了wrap_content则高度是累加值但是不能超过父容器的空间剩余;

 

5. 四种方式获取View的尺寸

View的measure过程和Activity的生命周期方法不是同步执行的,所以无法保证在onCreate, onStart, onResume时某个View已经测量完毕。

1)onWindowFocusChanged,View初始化完毕,宽高已经准备好,在Activity的窗口得到焦点和失去时均会被调用;

2)view.post(runnable),通过post将一个runnable投递到消息队列的队尾,然后等待Looper调用;

3)ViewTreeObserver的OnGlobalLayoutListener接口,view树的状态发生改变或者内部View可见性发生改变;

4)view.measure

int widthMeasureSpec = MeasureSpec.makeMeasureSpec((1<<30) - 1, MeasureSpec.AT_MOST);

view.measure(widthMeasureSpec, heightMeasureSpec);

 

6. layout过程

layout的大致流程:首先通过setFrame来设定View的四个顶点位置,接着调用onLayout确定子元素的位置;

以LinerLayout的onLayout中的layoutVertical为例,遍历所有子元素并调用setChildFrame设置子元素的位置,top逐渐增大,setChildFrame内部又调用了child.layout;

测量宽高形成于View的measure过程,最终宽高形成于layout过程。

 

7. draw过程

1)绘制背景background.draw(canvas)

2)绘制自己onDraw(canvas)

3)绘制子元素dispatchDraw(canvas)

4)绘制装饰onDrawScrollBars(canvas)

 

8. 自定义View方式

1)继承View重写onDraw:实现一些不规则的图形,要自己支持wrap_content,并且padding也需要自己处理;

2)继承ViewGroup派生特殊的Layout:处理ViewGroup和子元素的测量和布局两个过程;

3)继承特定的View:例如TextView;

4)继承特定的ViewGroup:例如LinerLayout

 

9. 自定义View须知

1)支持wrap_content

2)直接继承View的控件,需要在draw方法中处理padding,直接继承ViewGroup的控件需要在onMeasure和onLayout中考虑padding和子元素的margin;

3)尽量不要在View中使用Handler,因为View本身支持post方法;

4)View中有线程或者动画:可以在onAttachedToWindow中开启,在onDetachedFromWindow时停止和关闭;

5)滑动嵌套时要解决滑动冲突;

 

10. 自定义属性

1)在values目录下创建attrs_xxxx.xml,声明一个自定义属性集合,在其中定义属性名和格式;

 <resources>

    <declare-styleable name="CircleView">

        <attr name="circle_color" format="color" />

    </declare-styleable>

</resources>

2)在View的构造方法中解析自定义属性的值

public CircleView(Context context, AttributeSet attrs, int defStyleAttr) {

    super(context, attrs, defStyleAttr);

    TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.CircleView);

    mColor = a.getColor(R.styleable.CircleView_circle_color, Color.RED);

    a.recycle();

    init();

}

3)在布局文件中使用自定义属性

xmlns:app="http://schemas.android.com/apk/res-auto"

app:circle_color="@color/light_green"

 

此文章转载http://blog.csdn.net/doom20082004/article/details/53487597

posted @ 2017-12-10 16:05  鹏达君  阅读(220)  评论(0编辑  收藏  举报