Activity-setContentView(int resId)源码分析
入口–Activity–》setContentView()
public void setContentView(@LayoutRes int layoutResID) {
getWindow().setContentView(layoutResID);
initWindowDecorActionBar();
}
大家应该都知道这里的getWindow()返回的是一个PhoneWindow对象,所以直接跳到PhoneWindow中去查看setContentView(layoutResID)代码:
@Override
public void setContentView(int layoutResID) {
if (mContentParent == null) {
installDecor(); 初始化mContentParent
} else if (!hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
mContentParent.removeAllViews();
}
if (hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
final Scene newScene = Scene.getSceneForLayout(mContentParent, layoutResID,
getContext());
transitionTo(newScene);
} else {
//加载xml界面并添加到mContentParent中。
mLayoutInflater.inflate(layoutResID, mContentParent);
}
mContentParent.requestApplyInsets();
。。。。。。
}
代码很简单,首先是判断contentParent是否实例化了?如果没有就执行installDecor()去实例化、以下是installDecor()代码:
private void installDecor() {
mForceDecorInstall = false;
if (mDecor == null) {
mDecor = generateDecor(-1);
mDecor.setDescendantFocusability(ViewGroup.FOCUS_AFTER_DESCENDANTS);
mDecor.setIsRootNamespace(true);
if (!mInvalidatePanelMenuPosted && mInvalidatePanelMenuFeatures != 0) {
mDecor.postOnAnimation(mInvalidatePanelMenuRunnable);
}
} else {
mDecor.setWindow(this);
}
if (mContentParent == null) {
mContentParent = generateLayout(mDecor);
,,,,,,
这里的代码也不难,就是说如果mDecor如果为null就去generateDecor(-1)得到一个DecorView,大家可以点进去看就是return了一个new DecorView回来。那么往下看,如果mContentParent为null,就回去调用
generateLayout(mDecor)方法,将前面刚刚加载的mDecor当做参数传递过去,看一下这个代码:
protected ViewGroup generateLayout(DecorView decor) {
-------
int layoutResource;
int features = getLocalFeatures();
------加载layoutResource,实际是一个安卓自带的布局文件
//通过判断在mDecor中加载layoutResource布局,其中包含着一个contentParent。
mDecor.onResourcesLoaded(mLayoutInflater, layoutResource);
ViewGroup contentParent = (ViewGroup)findViewById(ID_ANDROID_CONTENT);
--------- 对mDecor的一些添加。
return contentParent;
}
public static final int ID_ANDROID_CONTENT = com.android.internal.R.id.content;
这个方法内部的代码很长,前前后后几十行。这边主要讲两处,第一处是通过较多的判断来为layoutResource赋值一个安卓自带的xml布局文件,然后调用onResourcesLoaded来将这个xml文件加载到mDecor中(可以自己查看源码,很简单,调用的addView方法)。
第二处就是这个findViewById了,点进去看就是获取到外围的DecorView去获取ContentParent。
所以其实在第一处加载到mDecor中的layoutResource布局中就包含了一个R.id.content的frameLayout。
@Nullable
public View findViewById(@IdRes int id) {
return getDecorView().findViewById(id);
}
这样的话contentParent和mDecorView就都有了,且contentParent已经加载到了mDecorView中,接下来再看一遍setContentView源码:
@Override
public void setContentView(int layoutResID) {
if (mContentParent == null) {
installDecor(); 初始化mContentParent
} else if (!hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
mContentParent.removeAllViews();
}
if (hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
final Scene newScene = Scene.getSceneForLayout(mContentParent, layoutResID,
getContext());
transitionTo(newScene);
} else {
//加载xml界面并添加到mContentParent中。
mLayoutInflater.inflate(layoutResID, mContentParent);
}
mContentParent.requestApplyInsets();
。。。。。。
}
加载出来并将加载出来的view添加到mContentParent中,(这块如果有问题可以看我对LayoutInflater加载view的源码分析的文章。)
这样就形成了这样的view结构: