通過本文你可能會明白以下幾點
- 1 .setContentView是否可以多次調用
- 2 .為什么requestWindowFeature要在setContentView之前調用
- 3 .關于Activity完整視圖結構的概念
可能很多人都看過和下面類似的圖:
這才是我們一個Activity的完整視圖,我們調用setContentView時,只是將布局放到content中了,何以見得呢,可以參考源碼。關于setContentView的具體流程可以參考我的這篇文章
setContentView最后調用了PhoneWindow的setContentView方法
frameworks\base\core\java\com\android\internal\policy\PhoneWindow.java
@Override
public void setContentView(int layoutResID) {
if (mContentParent == null) {
installDecor();
} else if (!hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
mContentParent.removeAllViews();
}
if (hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
final Scene newScene = Scene.getSceneForLayout(mContentParent, layoutResID,
getContext());
transitionTo(newScene);
} else {
mLayoutInflater.inflate(layoutResID, mContentParent);
}
mContentParent.requestApplyInsets();
final Callback cb = getCallback();
if (cb != null && !isDestroyed()) {
cb.onContentChanged();
}
mContentParentExplicitlySet = true;
}
這里調用了mLayoutInflater.inflate來加載布局,加載到的布局最終通過addView添加到了mContentParent中(詳細過程見這里)。mContentParent是什么呢?它是一個ViewGroup類型的成員變量。在這個方法開頭對他進行的為空判斷,為空時會調用installDecor方法,不為空時先清空其子view(看到這里也就說明activity中setContentView可以多次調用來加載不同布局)。我們來看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);
....
}
這個方法比較長僅截取相關片段。這里并沒有著急初始化mContentParent ,而是初始化了mDecor 。mDecor 就是一個DecorView對象。這里第一次出現了DecorView,DecorView繼承于FrameLayout,是一個Activity的頂級視圖,也就是最外層的布局,如本文開頭圖中所示。先不直接去看這個類,先看看初始化mDecor 的那個方法:
protected DecorView generateDecor(int featureId) {
Context context;
if (mUseDecorContext) {
Context applicationContext = getContext().getApplicationContext();
if (applicationContext == null) {
context = getContext();
} else {
context = new DecorContext(applicationContext, getContext().getResources());
if (mTheme != -1) {
context.setTheme(mTheme);
}
}
} else {
context = getContext();
}
return new DecorView(context, featureId, this, getAttributes());
}
可見僅僅只是簡單new了一個DecorView而已。我們接著看installDecor,在初始化完mDecor 之后,就開始初始化mContentParent 了,調用了generateLayout方法:
protected ViewGroup generateLayout(DecorView decor) {
....
int layoutResource;
int features = getLocalFeatures();
if ((features & (1 << FEATURE_SWIPE_TO_DISMISS)) != 0) {
layoutResource = R.layout.screen_swipe_dismiss;
setCloseOnSwipeEnabled(true);
} else if ((features & ((1 << FEATURE_LEFT_ICON) | (1 << FEATURE_RIGHT_ICON))) != 0) {
if (mIsFloating) {
TypedValue res = new TypedValue();
getContext().getTheme().resolveAttribute(
R.attr.dialogTitleIconsDecorLayout, res, true);
layoutResource = res.resourceId;
} else {
layoutResource = R.layout.screen_title_icons;
}
removeFeature(FEATURE_ACTION_BAR);
} else if ((features & ((1 << FEATURE_PROGRESS) | (1 << FEATURE_INDETERMINATE_PROGRESS))) != 0
&& (features & (1 << FEATURE_ACTION_BAR)) == 0) {
layoutResource = R.layout.screen_progress;
} else if ((features & (1 << FEATURE_CUSTOM_TITLE)) != 0) {
if (mIsFloating) {
TypedValue res = new TypedValue();
getContext().getTheme().resolveAttribute(
R.attr.dialogCustomTitleDecorLayout, res, true);
layoutResource = res.resourceId;
} else {
layoutResource = R.layout.screen_custom_title;
}
removeFeature(FEATURE_ACTION_BAR);
} else if ((features & (1 << FEATURE_NO_TITLE)) == 0) {
if (mIsFloating) {
TypedValue res = new TypedValue();
getContext().getTheme().resolveAttribute(
R.attr.dialogTitleDecorLayout, res, true);
layoutResource = res.resourceId;
} else if ((features & (1 << FEATURE_ACTION_BAR)) != 0) {
layoutResource = a.getResourceId(
R.styleable.Window_windowActionBarFullscreenDecorLayout,
R.layout.screen_action_bar);
} else {
layoutResource = R.layout.screen_title;
}
} else if ((features & (1 << FEATURE_ACTION_MODE_OVERLAY)) != 0) {
layoutResource = R.layout.screen_simple_overlay_action_mode;
} else {
layoutResource = R.layout.screen_simple;
}
mDecor.startChanging();
mDecor.onResourcesLoaded(mLayoutInflater, layoutResource);
ViewGroup contentParent = (ViewGroup)findViewById(ID_ANDROID_CONTENT);
....
return contentParent;
}
這個方法也非常長,這里截取一些重要地方。截取的代碼中,先是在選布局,依據的就是不同的features ,看一下getLocalFeatures方法,這里PhoneWindow沒有實現這個方法,調用的時父類,也就是Window中的:
android\frameworks\base\core\java\android\view\Window.java
protected final int getLocalFeatures()
{
return mLocalFeatures;
}
這里僅是取了mLocalFeatures的值,有一個設置的地方就在requestFeature
public boolean requestFeature(int featureId) {
final int flag = 1<<featureId;
mFeatures |= flag;
mLocalFeatures |= mContainer != null ? (flag&~mContainer.mFeatures) : flag;
return (mFeatures&flag) != 0;
}
看到這里,應該就明白了,為什么我們在取消標題欄時,requestWindowFeature(Window.FEATURE_NO_TITLE);的調用必須放在setContentView之前才會生效,因為setContentView執行過程中,會根據features 選擇DecorView的布局。
我們可以看一下加載的布局,最最常用的R.layout.screen_title:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:fitsSystemWindows="true">
<ViewStub android:id="@+id/action_mode_bar_stub"
android:inflatedId="@+id/action_mode_bar"
android:layout="@layout/action_mode_bar"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:theme="?attr/actionBarTheme" />
<FrameLayout
android:layout_width="match_parent"
android:layout_height="?android:attr/windowTitleSize"
style="?android:attr/windowTitleBackgroundStyle">
<TextView android:id="@android:id/title"
style="?android:attr/windowTitleStyle"
android:background="@null"
android:fadingEdge="horizontal"
android:gravity="center_vertical"
android:layout_width="match_parent"
android:layout_height="match_parent" />
</FrameLayout>
<FrameLayout android:id="@android:id/content"
android:layout_width="match_parent"
android:layout_height="0dip"
android:layout_weight="1"
android:foregroundGravity="fill_horizontal|top"
android:foreground="?android:attr/windowContentOverlay" />
</LinearLayout>
全屏時布局R.layout.screen_simple
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:fitsSystemWindows="true"
android:orientation="vertical">
<ViewStub android:id="@+id/action_mode_bar_stub"
android:inflatedId="@+id/action_mode_bar"
android:layout="@layout/action_mode_bar"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:theme="?attr/actionBarTheme" />
<FrameLayout
android:id="@android:id/content"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:foregroundInsidePadding="false"
android:foregroundGravity="fill_horizontal|top"
android:foreground="?android:attr/windowContentOverlay" />
</LinearLayout>
可見全屏時,就是去掉了@android:id/title,正常時是一個LinearLayout。自上到下是@android:id/title和@android:id/content。和我們文章開頭的圖中一樣。
確定完布局后就調用了DecorView的onResourcesLoaded方法把布局添加到DecorView中,不要忘了DecorView是一個FrameLayout,開始里面是沒有任何東西的。
添加完布局后,緊接著就調用了findViewById方法,那個常量ID_ANDROID_CONTENT在父類Window中定義著。之后在generateLayout方法最后返回了contentParent 。最后在setContentView中,將布局加載到這個FrameLayout中。至于title,在installDecor中根據實際情況實例化了mTitleView,他的title由setTitle設置,我們在activity中調用的setTitle會最后間接調用PhoneWindow中的setTitle。
setContentView執行完畢后,我們一個activity的完整視圖就完成了。總結一下,最外層為一個DecorView,他是一個FrameLayout,內部根據具體情況加載不同布局,但必有一個id為@android:id/content的FrameLayout,這里用來加載我們自己要顯示的布局。