1、创建顶层布局容器DecorView
DecorView对象是ViewGroup容器,继承自FrameLayout,是ForWindow对象持有的一个实例,是所有应用程序的顶层View,在系统内部进行初始化。
DecorView初始化完成后,系统会根据应用程序的主题特性加载一个基础容器,主题可能为noActionBar或者DarkActionBar等,虽然主题可能不同,但基础容器中一定会有一个onDraw.R.id.content的容器FrameLayout。
开发者setContentView设置的xml文件,就上被解析后添加到了FrameLayout中。
2、绘制流程:
1、绘制入口
ActivityThread.handleResumeActivity
---------> WindowManagerImpl.addView(dercorView, layoutParams)
---------> WindowManagerGlobal.addView()
2、绘制的类及方法
ViewRootImpl.setView(decorView, layoutParams, parentView)
---------> ViewRootImpl.requestLayout()
---------> scheduleTraversals()
---------> doTraversal()
---------> performTraversals()
3、绘制三大步骤
测量:ViewRootImpl.performMeasure 测量控件的宽高
布局:ViewRootImpl.performLayout
绘制:ViewRootImpl.performDraw
2.1 测量控件宽高:
MeasureSpec类
View = 模式 + 尺寸 -----》MeasureSpec 32位int值
00000000000000000000000000000000
前两位——模式
后30位——尺寸
父容器不对View做任何限制,系统内部使用
public static final int UNSPECIFIED = 0 << MODE_SHIFT; 00000000000000000000000000000000
父容器检测出View的大小,View的大小就是SpecSize LayoutPamras match_parent 固定大小
public static final int UNSPECIFIED = 1 << MODE_SHIFT; 01000000000000000000000000000000
父容器指定一个可用大小,View的大小不能超过这个值,LayoutPamras wrap_content
public static final int UNSPECIFIED = 2 << MODE_SHIFT; 10000000000000000000000000000000
SpecMode + SpecSize = MeasureSpec
MeasureSpec = mode + size
ViewGroup测量流程:
measure方法
onMeasure方法(测量子控件的宽高:先按照规则获取子控件的测量规格,再调用子控件的measure方法完成子控件测量)
setMeasuredDimension方法(设置自己的宽高 )
setMeasuredDimensionRaw(保存自己的宽高)
View测量流程:
measure方法
onMeasure方法(需要重新确定自己的宽高)
setMeasuredDimension方法
setMeasuredDimensionRaw(保存自己的宽高)
自定义View的时候,如果不重写onMeasure方法,那么match_parent和wrap_content得到的效果是一样的。
2.2 View的布局
ViewGroup布局流程:
layout() 确定自己的位置,4个点的位置
onLayout() 进行子View的布局
View布局流程:只需要确定自己就行
layout() 确定自己的位置,4个点的位置
2.3 绘制过程
ViewGroup绘制流程:
绘制背景 drawBackground(canvas)
绘制自己 onDraw(canvas)
绘制子view dispatchDraw(canvas)
绘制前景,滚动条等装饰 onDrawForeground(canvas)
ViewGroup绘制流程:
绘制背景 drawBackground(canvas)
绘制自己 onDraw(canvas)
绘制前景,滚动条等装饰 onDrawForeground(canvas)
3、自定义UI的步骤
onMeasure()
onLayout() 自定义容器才需要这个函数,自定义View不需要,因为View不含子控件
onDraw() 可选,当绘制容器时,如果容器里都是系统控件,这步可省