一个 界面的布局关系
界面 --> StatusBar && Activity --> PhoneWindow --> DecorView --> Title && contentView --> layout
Activity 的 setContentView 方法和 AppCompatActivity 的 setContentView 是不同的,接下来逐个分析
Activity 的 setContentView
// Activity
public void setContentView(@LayoutRes int layoutResID) {
getWindow().setContentView(layoutResID); // 调用 PhoneWindow 的 setContentView 方法
initWindowDecorActionBar(); // 初始化 ActionBar
}
// 初始化 ActionBar
private void initWindowDecorActionBar() {
Window window = getWindow();
window.getDecorView();
// 判读 Activity 是否带有 ActionBar,如果没有,直接返回
if (isChild() || !window.hasFeature(Window.FEATURE_ACTION_BAR) || mActionBar != null) {
return;
}
// 如果有 ActionBar 则创建一个默认的 ActionBar ,并为 ActionBar 设置 Activity 相关的 Icon 和 Logo
mActionBar = new WindowDecorActionBar(this);
mActionBar.setDefaultDisplayHomeAsUpEnabled(mEnableDefaultActionBarUp);
mWindow.setDefaultIcon(mActivityInfo.getIconResource());
mWindow.setDefaultLogo(mActivityInfo.getLogoResource());
}
// PhoneWindow
@Override
public void setContentView(int layoutResID) {
if (mContentParent == null) {
installDecor(); // 初始化 DecorView 也就是整个 Window 的根 View,主要过程为 new 一个 DecorView
} else if (!hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
mContentParent.removeAllViews(); // 移除 mContentParent 中的所有子 View mContentParent 为 DecorView 中的用于显示用户设置的布局的 ViewGroup
}
if (hasFeature(FEATURE_CONTENT_TRANSITIONS)) { // 有过度动画时的处理
final Scene newScene = Scene.getSceneForLayout(mContentParent, layoutResID,
getContext());
transitionTo(newScene);
} else {
mLayoutInflater.inflate(layoutResID, mContentParent); // 由 layoutId 初始化 View 并添加到 mContentParent
}
...
}
由此可以看出 Activity 中使用过 PhoneWindow 来向界面中添加用户设置的布局的。并且 DecorView 的创建也是在 PhoneWindow 中完成的。
AppCompatActivity 的 setContentView 方法
// AppCompatActivity
@Override
public void setContentView(@LayoutRes int layoutResID) {
getDelegate().setContentView(layoutResID); // getDelegate() 方法的返回值是一个 AppCompatDelegate 对象
}
// AppCompatActivity
public AppCompatDelegate getDelegate() {
if (mDelegate == null) {
mDelegate = AppCompatDelegate.create(this, this);
}
return mDelegate;
}
// AppCompatDelegate
private static AppCompatDelegate create(Context context, Window window,
AppCompatCallback callback) {
final int sdk = Build.VERSION.SDK_INT;
if (BuildCompat.isAtLeastN()) {
return new AppCompatDelegateImplN(context, window, callback);
} else if (sdk >= 23) {
return new AppCompatDelegateImplV23(context, window, callback);
} else if (sdk >= 14) {
return new AppCompatDelegateImplV14(context, window, callback);
} else if (sdk >= 11) {
return new AppCompatDelegateImplV11(context, window, callback);
} else {
return new AppCompatDelegateImplV9(context, window, callback);
}
}
// AppCompatDelegateImplV9 所有 AppCompatDelegate 子类的 setContentView 继承自了 AppCompatDelegateImplV9 的方法
@Override
public void setContentView(int resId) {
ensureSubDecor();
ViewGroup contentParent = (ViewGroup) mSubDecor.findViewById(android.R.id.content);
contentParent.removeAllViews();
LayoutInflater.from(mContext).inflate(resId, contentParent);
mOriginalWindowCallback.onContentChanged();
}
AppCompatDelegateImplV9 中的 mSubDecor 对象是一个 ViewGroup,其构造过程为根据 Activity 的主题创建一个根 View,其创建好之后,就会调用 PhoneWindow 的 setContentView 方法,将其作为参数添加到 Decorview 中
AppCompatActivity 中是通过 AppCompatDelegateImplV9 这个类来完成向界面中添加布局的
AppCompatDelegateImplV9 的 setContentView 方法中,我们将需要添加的布局添加到 mSubDecor ,也就添加到了 DecorView 中,Activity 的 onResume 方法回调之后也就显示到了界面上。
总结
由源码分析得出,setContentView 方法完成之后,Activity 需要显示的布局已经添加到了 DecorView 中
在 Activity 的 onResume 方法回调之后,布局就会显示到界面中。
有关 LayoutInflater 的工作过程,请期待下篇博客。
setContentView 的重载方法
setContentView(int layoutId);
setContentView(View view);
setContentView(View view,LayoutParams params);
setContentView(int layoutId);
这个方法有三个重载,第一个我们刚才已经分析过了,不管是 Activity 或者是 AppCompatActivity ,在添加是都会使用布局文件中自己设置宽高来构造 LayoutParams ,根布局在界面上的显示的宽高也完全由 xml 中设置的决定。
setContentView(View view);
Activity Activity 的 setContentView 加载是通过 PhoneWindow 的 setContentView 方法实现的,如果 setContentView 的参数为一个 View 对象,不管是否 View 有自己的 LayoutParams 属性,都会将 View 的 LayoutParams 设为新的,并且新的的宽高都为 MATH_PARENT
AppCompatActivity
AppCompatActivity setContentView 在添加 View 时,如果 View 有 LayoutParams ,则使用 View 原有的 LayoutParams ,如果 View 没有 LayoutParams ,则使用宽高都为 MATH_PARENT 的 LayoutParems (在 ViewGroup 添加 View 时如果 View 没有 LayoutParams ,此时应该是 默认 Warp_content,不过这里却是 MATH_PARENT ,在代码中没有找到原因)
setContentView(View view,LayoutParams params);
带 LayoutParams 参数的,不管是 Activity 还是 AppCompatActivity ,都会为 View 设置传入的 LayoutParams 属性值