Activity-setContentView(int resId)源码分析

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

猜你喜欢

转载自blog.csdn.net/m0_37194191/article/details/78455875
今日推荐