Activity 从加载布局文件到显示的过程分析

Activity 生命周期函数执行过程详解中介绍了ActivityThread、Instrumentation、ActivityManagerService启动activity的过程,本文主要介绍Activity从加载布局文件到显示的过程。

SetContentView


在activity的onCreate()生命周期函数中,会调用setContentView()来设置我们的布局文件,先来了解下setContentView()函数

  public void setContentView(@LayoutRes int layoutResID) {
        getWindow().setContentView(layoutResID);//调用了getWindow()
        initWindowDecorActionBar();
    }

间接调用了getWindow()的setContentView()函数,来看看getWindow()

 public Window getWindow() {
        return mWindow;
    }

来看看mWindow是什么时候初始化的

 final void attach(Context context, ActivityThread aThread,
            Instrumentation instr, IBinder token, int ident,
            Application application, Intent intent, ActivityInfo info,
            CharSequence title, Activity parent, String id,
            NonConfigurationInstances lastNonConfigurationInstances,
            Configuration config, String referrer, IVoiceInteractor voiceInteractor) {
        attachBaseContext(context);
        ...
        mWindow = new PhoneWindow(this);
        ...
    }

在activity的attach()函数中初始化,是一个PhoneWindow,那接着而看看PhoneWindow的setContentView()方法

    @Override
    public void setContentView(int layoutResID) {

        // before this happens.
        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();
        }
    }

mContentParent为null时先调用 installDecor(),然后在通过mLayoutInflater.inflate(layoutResID, mContentParent);进行设置,先来看看installDecor()

private void installDecor() {
            //mDecor为null,进行初始化
        if (mDecor == null) {
            mDecor = generateDecor();
            mDecor.setDescendantFocusability(ViewGroup.FOCUS_AFTER_DESCENDANTS);
            mDecor.setIsRootNamespace(true);
            if (!mInvalidatePanelMenuPosted && mInvalidatePanelMenuFeatures != 0) {
                mDecor.postOnAnimation(mInvalidatePanelMenuRunnable);
            }
        }
        //mContentParent为null,进行初始化
        if (mContentParent == null) {
            mContentParent = generateLayout(mDecor);

            // Set up decor part of UI to ignore fitsSystemWindows if appropriate.
            mDecor.makeOptionalFitsSystemWindows();

            final DecorContentParent decorContentParent = (DecorContentParent) mDecor.findViewById(
                    R.id.decor_content_parent);

            if (decorContentParent != null) {
                mDecorContentParent = decorContentParent;
                mDecorContentParent.setWindowCallback(getCallback());
                if (mDecorContentParent.getTitle() == null) {
                    mDecorContentParent.setWindowTitle(mTitle);
                }

                final int localFeatures = getLocalFeatures();
                for (int i = 0; i < FEATURE_MAX; i++) {
                    if ((localFeatures & (1 << i)) != 0) {
                        mDecorContentParent.initFeature(i);
                    }
                }

                mDecorContentParent.setUiOptions(mUiOptions);

                if ((mResourcesSetFlags & FLAG_RESOURCE_SET_ICON) != 0 ||
                        (mIconRes != 0 && !mDecorContentParent.hasIcon())) {
                    mDecorContentParent.setIcon(mIconRes);
                } else if ((mResourcesSetFlags & FLAG_RESOURCE_SET_ICON) == 0 &&
                        mIconRes == 0 && !mDecorContentParent.hasIcon()) {
                    mDecorContentParent.setIcon(
                            getContext().getPackageManager().getDefaultActivityIcon());
                    mResourcesSetFlags |= FLAG_RESOURCE_SET_ICON_FALLBACK;
                }
                if ((mResourcesSetFlags & FLAG_RESOURCE_SET_LOGO) != 0 ||
                        (mLogoRes != 0 && !mDecorContentParent.hasLogo())) {
                    mDecorContentParent.setLogo(mLogoRes);
                }


                PanelFeatureState st = getPanelState(FEATURE_OPTIONS_PANEL, false);
                if (!isDestroyed() && (st == null || st.menu == null) && !mIsStartingWindow) {
                    invalidatePanelMenu(FEATURE_ACTION_BAR);
                }
            } else {
               // mContentParent不为null
                mTitleView = (TextView)findViewById(R.id.title);
                if (mTitleView != null) {
                    mTitleView.setLayoutDirection(mDecor.getLayoutDirection());
                     // 不显示标题的时候隐藏
                    if ((getLocalFeatures() & (1 << FEATURE_NO_TITLE)) != 0) {
                        View titleContainer = findViewById(
                                R.id.title_container);
                        if (titleContainer != null) {
                            titleContainer.setVisibility(View.GONE);
                        } else {
                            mTitleView.setVisibility(View.GONE);
                        }
                        if (mContentParent instanceof FrameLayout) {
                            ((FrameLayout)mContentParent).setForeground(null);
                        }
                    } else {
                     // 设置标题
                        mTitleView.setText(mTitle);
                    }
                }
            }
        }
    }

DecorView为null时调用generateDecor()进行初始化DecorView,mContentParent为null时调用generateLayout(mDecor)初始化mContentParent,然后设置标题,若不显示标题则隐藏。

先来看看generateDecor()

    protected DecorView generateDecor() {
        return new DecorView(getContext(), -1);
    }

只是进行初始化DecorView,接着看generateLayout(mDecor)

   protected ViewGroup generateLayout(DecorView decor) {
        //...省略代码
        //通过主题加载不同的布局文件
        View in = mLayoutInflater.inflate(layoutResource, null);
        //设置DecorView
        decor.addView(in, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT));
        mContentRoot = (ViewGroup) in;
        //获取contentParent
        ViewGroup contentParent = (ViewGroup)findViewById(ID_ANDROID_CONTENT);
        //...省略代码
        return contentParent;

   }

我们再回到PhoneWindow的setContentView()函数中

  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 {
            //布局文件被设置到ContentParent
            mLayoutInflater.inflate(layoutResID, mContentParent);
        }
        mContentParent.requestApplyInsets();
        final Callback cb = getCallback();
        if (cb != null && !isDestroyed()) {
            cb.onContentChanged();
        }
    }

通过转载网上的一张图片来说明

这里写图片描述

1.activity在onCreate()中间接调用PhoneWindow的setContentView()
2.PhoneWindow根据主题进行初始化DecorView以及title和ContentParent
3.将对应的布局文件设置到ContentParent中

Activity 生命周期函数执行过程详解中我们知道,View的显示是在handleResumeActivity()中处理的,那么我们接着看handleResumeActivity()

handleResumeActivity()


直接看handleResumeActivity()代码是如何展示view的


    final void handleResumeActivity(IBinder token,
            boolean clearHide, boolean isForward, boolean reallyResume) {

             //省略部分代码

            if (r.window == null && !a.mFinished && willBeVisible) {
                //获取PhoneWindow
                r.window = r.activity.getWindow();
                //获取PhoneWindow的DecorView
                View decor = r.window.getDecorView();
                //DecorView设置为不显示INVISIBLE
                decor.setVisibility(View.INVISIBLE);
                ViewManager wm = a.getWindowManager();
                WindowManager.LayoutParams l = r.window.getAttributes();
                a.mDecor = decor;
                l.type = WindowManager.LayoutParams.TYPE_BASE_APPLICATION;
                l.softInputMode |= forwardBit;
                if (a.mVisibleFromClient) {
                    a.mWindowAdded = true;
                    wm.addView(decor, l);
                }

                 //省略部分代码

                    if (r.activity.mVisibleFromClient) {
                        ViewManager wm = a.getWindowManager();
                        //将decorview加入到加入到ViewManager中
                        wm.updateViewLayout(decor, l);
                    }
                }
                r.activity.mVisibleFromServer = true;
                mNumVisibleActivities++;
                if (r.activity.mVisibleFromClient) {
                    //显示view
                    r.activity.makeVisible();
                }
            }
    }

1.将DecorView设置为Invisible
2.将DecorView加入到ViewManger中
3.通过activity的makeVisible()进行显示

来看看makeVisible()函数

   void makeVisible() {
        if (!mWindowAdded) {
            ViewManager wm = getWindowManager();
            wm.addView(mDecor, getWindow().getAttributes());
            mWindowAdded = true;
        }
        mDecor.setVisibility(View.VISIBLE);
    }

设置DecorView为visible,而调用setVisibility会间接触发view的绘制流程,然后更新界面。
至于view的绘制流程以及ViewManager用单独的文章介绍。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值