Activity-setContentView(int resId)源码分析

来源:互联网 发布:yy mac进入房间失败 编辑:程序博客网 时间:2024/05/22 12:02

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结构:
这里写图片描述

原创粉丝点击