Activity -> PhoneWindow-> DecorView-> TitleView(ActionBarContainer->ActionBar)、ContentView(FrameLayout、RelativeLayout)
PhoneWindow这个类是Framework为我们提供的Android窗口概念的具体实现。调用setContentView()方法设置Activity的用户界面时,实际上就完成了对所关联的PhoneWindow的ViewTree的设置。
每个应用程序窗口的decorView都有一个与之关联的ViewRoot对象,这种关联关系是由WindowManager来维护的。decorView与ViewRoot的关联关系是Activity启动时,在ActivityThread.handleResumeActivity()方法中建立的。
绘制起点:requestLayout
绘制流程的三个阶段:
测量 measure: 判断是否需要重新计算View的大小,需要的话则计算
布局 layout: 判断是否需要重新计算View的位置,需要的话则计算
绘制 draw: 判断是否需要重新绘制View,需要的话则重绘制。
一、测量 measure
1.MeasureSpec 的理解
MeasureSpec中的值是一个整型(32位),其中高两位是mode,后面30位存的是size。
一个MeasureSpec封装了从父容器传递给子容器的布局要求,是由父View的MeasureSpec和子View的LayoutParams通过简单的计算得出一个针对子View的测量要求。
MeasureSpec一共有三种模式:
UPSPECIFIED : 对于子容器没有任何限制,通常是不确定的值,需要对子容器进一步测量才能确定。
EXACTLY: 为View和子容器设置了精确尺寸。
AT_MOST:View和子容器可以是该限定大小内的任意值。
2.测量流程的理解
如果我们在xml 的layout_width或者layout_height 把值都写死,那么测量完全就不需要了。
之所以要测量,是因为 match_parent 就是充满父容器,wrap_content 就是自己多大就多大。
这个测量过程,就是计算出来的父View的MeasureSpec不断往子View传递,结合子View的LayoutParams 一起再算出子View的MeasureSpec,子View有了MeasureSpec才能测量自己和自己的子View。
2.1 父View的MeasureSpec 是EXACTLY时
说明父View的大小是确切的,那么它的size 是多大,最后展示到屏幕就是多大。
1、如果子View的大小是MATCH_PARENT,那么子View的大小肯定也是确切的,子View的size=父View的size,mode=EXACTLY。
2、如果子View 的layout_xxxx是WRAP_CONTENT,大小是根据自己的content 来决定的,但是子View毕竟不能超过父View的大小。因此子View MeasureSpec mode的应该是AT_MOST,而size 暂定父View的 size。
3、如果子View 的layout_xxxx是确定的值例如200dp,那么控件最后展示就是200dp,不管父View有多大,也不管自己的content 有多大。MeasureSpec 的mode = EXACTLY 大小size=layout_xxxx 设置的值。
2.2 父View的MeasureSpec 是AT_MOST时
父View的大小是不确定,最大的值是MeasureSpec 的size。
1、如果子View 的layout_xxxx是MATCH_PARENT,由于父View的大小不确定(只限制了最大值),那么子View即使充满父容器,大小肯定也不能确定的。所以子View的mode=AT_MOST,size=父View的size。
子View最终大小由父View测量结果决定。
2、如果子View 的layout_xxxx是WRAP_CONTENT,父View的大小是不确定(只知道最大限定值),那么在子View的Content没算出大小之前,子View MeasureSpec mode的就是AT_MOST,而size 暂定父View的 size。
子View最终大小由计算出的content决定。
2.3 父View的MeasureSpec 是UNSPECIFIED(未指定)时
表示没有任何束缚和约束,不像AT_MOST表示最大只能多大,不也像EXACTLY表示父View确定的大小,子View可以得到任意想要的大小,不受约束。
对于子View来说无论是WRAP_CONTENT还是MATCH_PARENT,子View也是没有任何束缚的,size的值没有任何意义了,所以一般都直接设置成0。
测量工作都是在measure方法中调用onMeasure()做的,measure方法是final的所以这个方法也不可重写,如果想自定义View的测量,应该重写onMeasure()方法。