一個 界面的布局關(guān)系
界面 --> StatusBar && Activity --> PhoneWindow --> DecorView --> Title && contentView --> layout
Activity 的 setContentView 方法和 AppCompatActivity 的 setContentView 是不同的肪康,接下來逐個分析
Activity 的 setContentView
// Activity
public void setContentView(@LayoutRes int layoutResID) {
getWindow().setContentView(layoutResID); // 調(diào)用 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 則創(chuàng)建一個默認的 ActionBar 锭魔,并為 ActionBar 設置 Activity 相關(guān)的 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 的創(chuàng)建也是在 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,其構(gòu)造過程為根據(jù) Activity 的主題創(chuàng)建一個根 View洪己,其創(chuàng)建好之后,就會調(diào)用 PhoneWindow 的 setContentView 方法秩仆,將其作為參數(shù)添加到 Decorview 中
AppCompatActivity 中是通過 AppCompatDelegateImplV9 這個類來完成向界面中添加布局的
AppCompatDelegateImplV9 的 setContentView 方法中码泛,我們將需要添加的布局添加到 mSubDecor ,也就添加到了 DecorView 中澄耍,Activity 的 onResume 方法回調(diào)之后也就顯示到了界面上噪珊。
總結(jié)
由源碼分析得出,setContentView 方法完成之后齐莲,Activity 需要顯示的布局已經(jīng)添加到了 DecorView 中
在 Activity 的 onResume 方法回調(diào)之后痢站,布局就會顯示到界面中。
有關(guān) LayoutInflater 的工作過程选酗,請期待下篇博客阵难。
setContentView 的重載方法
setContentView(int layoutId);
setContentView(View view);
setContentView(View view,LayoutParams params);
setContentView(int layoutId);
這個方法有三個重載,第一個我們剛才已經(jīng)分析過了芒填,不管是 Activity 或者是 AppCompatActivity 呜叫,在添加是都會使用布局文件中自己設置寬高來構(gòu)造 LayoutParams ,根布局在界面上的顯示的寬高也完全由 xml 中設置的決定空繁。
setContentView(View view);
Activity Activity 的 setContentView 加載是通過 PhoneWindow 的 setContentView 方法實現(xiàn)的,如果 setContentView 的參數(shù)為一個 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 參數(shù)的栓拜,不管是 Activity 還是 AppCompatActivity ,都會為 View 設置傳入的 LayoutParams 屬性值