在Activity 的启动过程中,调用ActivityThread 的handleResumeActivity 方法时,先得到一个与Activity 关联的PhoneWindow 对象,然后通过PhoneWindow 来获取DecorView。
PhoneWindow.java
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 的,将在下篇文章中分析。