在Activity 的启动过程中,调用ActivityThread 的handleResumeActivity 方法时,先得到一个与Activity 关联的PhoneWindow 对象,然后通过PhoneWindow 来获取DecorView。
PhoneWindow.java
class="brush:java;gutter:true;">public final View getDecorView() { if (mDecor == null) { installDecor(); } return mDecor; } private void installDecor() { if (mDecor == null) { // 生成DecorView mDecor = generateDecor(); mDecor.setDescendantFocusability(ViewGroup.FOCUS_AFTER_DESCENDANTS); mDecor.setIsRootNamespace(true); } if (mContentParent == null) { // 根据Window 样式确定DecorView 中的布局内容,mContentParent 就是DecorView 的第一个子View,也是我们写的Activity onCreate()中setContentView() 的父View mContentParent = generateLayout(mDecor); final DecorContentParent decorContentParent = (DecorContentParent) mDecor.findViewById(R.id.decor_content_parent); } ... }
DecorView 是PhoneWindow 的一个内部类,继承FrameLayout,generateDecor() 也是像平时我们自定义View 是new 一个DecorView,只不过多了一个featureId 参数,该参数如果是-1,表示这个View 是一个DecorView
protected DecorView generateDecor() { return new DecorView(getContext(), -1); } private final class DecorView extends FrameLayout { public DecorView(Context context, int featureId) { super(context); mFeatureId = featureId; } }
因此,DecorView 中的布局是怎样的,主要是由generateLayout(mDecor) 决定的
protected ViewGroup generateLayout(DecorView decor) { // 获取Window 的样式 TypedArray a = getWindowStyle(); ... final WindowManager windowService = (WindowManager) getContext().getSysteService(Context.WINDOW_SERVICE); if (windowService != null) { final Display display = windowService.getDefaultDisplay(); ... } final Context context = getContext(); final int targetSdk = context.getApplicationInfo().targetSdkVersion(); ... WindowManager.LayoutParams params = getAttributes(); // Inflate the window decor int layoutResource; int features = getLocalFeatures(); if (){ // 判读features 中是否包含FEATURE_SWIPE_TO_DISMISS } else if() { // 如果features 包含自定义的Title layoutResource = R.layout.screen_custom_title; } else { // 如果window 属性没有任何装饰(没有TitleBar,没有ActionBar 等) layoutResource = R.layout.screen_simple; } mDecor.startChanging(); // 可以看到DecorView 也是通过inflate 加载布局的,系统framework 默认的布局资源位于frameworks\base\core\res\res\layout 目录下 View in = mLayoutInflater.inflate(layoutResource, null); decor.addView(in, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT)); mContentRoot = (ViewGroup) in; ViewGroup contentParent = (ViewGroup)findViewById(ID_ANDROID_CONTENT); if (contentParent == null) { // 抛出异常 } ... mDecor.finishChanging(); return contentParent; }
由此得出,DecorView 是根据不同的window 属性通过inflate() 方法加载位于frameworks\base\core\res\res\layout 目录下对应的布局资源生成的。至于LayoutInflate.inflate() 方法是如何加载布局文件并解析生成View 的,将在下篇文章中分析。