从setContentView出发,分析UI的绘制流程,大致可分为如下几个步骤:
1、生成DecorView(通过LayoutInflater解析xml文件,并添加到DecorView中)
2、把生成的DecorView添加到Window中(这里涉及到Activity的启动流程)
3、通过ViewRootImpl完成View的measure,layout和draw操作。
一:生成DecorView:
从activity 的 onCreate 中的 setContentView开始
通过阅读源码发现,window是一个抽象类,并没有setContentView方法的实现,但通过类说明可以知道,Window的唯一实现类是PhoneWindow,PhoneWindow的setContentView方法
installDecor()方法生成DecorView:
generateLayout(DecorView decor)方法会初始化一些Window相关的属性信息,如:无标题,全屏等,所以如果想实现这些特性的话,需要在setContentView方法之前执行,否则在setContentView方法之后执行如requestFeature(FEATURE_NO_TITLE);方法会报错,因为你Window相关的属性已经初始化完成了,此方法还做了一件事情就是会根据设置的相关的属性,去加载一个基础的xml布局文件,如:screen_simple.xml。
当前这里竟然在加载布局文件,并且生成了一个view, 但是好像貌似不是我门自己的view
这是DecorView默认的一个渲染,然后我门自己的布局都是渲染到她的FrameLayout上的
那么在这里我门现在能够明白,installDector其实实际上是在初始化两个视图容器,然后加载系统的R资源及特征,产生了一个基本布局
那么接着回到之前我门关注的另外一个方法mLayoutInflater.inflate(layoutResID, mContentParent);
至此,我们所有的View对象都已经实例化完成,所有设置的View属性也都解析完成,存放在各自对象的属性当中。接下来分析生成的DecorView是如何添加到Window中的,这就涉及到Activity的启动流程了,Activity的启动流程与ActivityThread有关,可以看到里面有一个handleLaunchActivity(ActivityClientRecord r, Intent customIntent, String reason)方法,从此处入手,分析DecorView是如何添加到Window中的。
二、把生成的DecorView添加到Window中(这里涉及到Activity的启动流程)
ActivityThread的handleLaunchActivity方法:
performLaunchActivity方法部分代码,主要实现的功能有初始化Activity对象,初始化Window和WindowManager对象,执行Activity的onCreate生命周期方法,onRestore方法等
handleResumeActivity方法也会调用performResumeActivity方法,去执行Activity的onResume生命周期的方法,然后在handleResumeActivity方法中,添加DecorView到Window中
WindowManager是一个接口,实现类为WindowManagerImpl,WindowManager的addView方法:
WindowManagerGlobal的addView方法:
ViewRootImpl的setView方法 最终会执行requestLayout();方法,开始进行UI的绘制流程,这里有一个细节需要注意一下 ,在setView方法里有一行代码view.assignParent(this),即给DecorView设置了一个父容器就是ViewRootImpl,所以当我们调用View的requestLayout方法的时候,最终都会调用到ViewRootImpl的requestLayout方法。
三、通过ViewRootImpl完成View的measure,layout和draw操作。
在performTraversals方法中,我发现了整体的绘制流程的调用,当前里面依次是用了:performTraversals()开始执行UI绘制,在performTraversals方法里依次执行了performMeasure,performLayout和performDraw方法进行View的measure、layout和draw操作
UI绘制先回去测量布局,然后在进行布局的摆放,当所有的布局测量摆放完毕之后,进行绘制。
至此整体UI绘制过程我们就已经非常清楚了。