如上图所示,几乎所有的Activity的视图树都是从PhoneWindow$DecorView——>LinearLayout——>FrameLayout(id/content)开始的,接下来的内容才是我们在layout中定义的视图的开始,现在从源码中看看这样的结构是怎么形成的。
Activity视图加载总是从setContentView()开始的:
public void setContentView(int layoutResID) { getWindow().setContentView(layoutResID); initActionBar(); }mWindow是Window类型的对象,Window是一个接口,它只有一个实现PhoneWindow,那么我们现在去PhoneWindow.java中看看setContentView()函数。
public Window getWindow() { return mWindow; }
public void setContentView(View view, ViewGroup.LayoutParams params) { if (mContentParent == null) { installDecor(); } else { mContentParent.removeAllViews(); } mContentParent.addView(view, params); final Callback cb = getCallback(); if (cb != null && !isDestroyed()) { cb.onContentChanged(); } } ... private void installDecor() { if (mDecor == null) { mDecor = generateDecor(); mDecor.setDescendantFocusability(ViewGroup.FOCUS_AFTER_DESCENDANTS); mDecor.setIsRootNamespace(true); if (!mInvalidatePanelMenuPosted && mInvalidatePanelMenuFeatures != 0) { mDecor.postOnAnimation(mInvalidatePanelMenuRunnable); } } if (mContentParent == null) { mContentParent = generateLayout(mDecor); ... protected DecorView generateDecor() { return new DecorView(getContext(), -1); }上面这部分代码里,单例模式保证唯一建了两个对象——mDecor(DecorView)和mContentParent(ViewGroup)。generateDecor()函数中创建了DecorView对象,-1表明这是视图树的根视图。
generateLayout(mDecor)函数则揭示了文章开始提到的问题
protected ViewGroup generateLayout(DecorView decor) { ... } else if ((features & (1 << FEATURE_ACTION_MODE_OVERLAY)) != 0) { layoutResource = com.android.internal.R.layout.screen_simple_overlay_action_mode; } else { // Embedded, so no decoration is needed. layoutResource = com.android.internal.R.layout.screen_simple; // System.out.println("Simple!"); } mDecor.startChanging(); Trace.traceBegin(Trace.TRACE_TAG_VIEW, "DecorView-inflate"); View in = mLayoutInflater.inflate(layoutResource, null); Trace.traceEnd(Trace.TRACE_TAG_VIEW); decor.addView(in, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT)); ViewGroup contentParent = (ViewGroup)findViewById(ID_ANDROID_CONTENT); ... return contentParent;layoutResource根据不同的特征加载不同的layout(比如是否有标题栏等)。我们选取screen_simple.xml看看它的内容:
<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" /> <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>这是一个LinearLayout——>FrameLayout结构,且FrameLayout的id为content,这里已经可以与开头的图片上的结构对应起来了。return contentParent表明generateLayout()产生的ViewGroup从FrameLayout开始的。
View in = mLayoutInflater.inflate(layoutResource, null); Trace.traceEnd(Trace.TRACE_TAG_VIEW); decor.addView(in, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT));
总结上述内容,我们要关注的是DecorView对象是如何成为视图树的根(唯一)的,以及接下来的视图是如何展开的,并且还要主义id/content是什么。