Activity.setContentView 方法源码解析

一个 界面的布局关系

界面 --> StatusBar && Activity --> PhoneWindow --> DecorView --> Title && contentView --> layout

Activity 的 setContentView 方法和 AppCompatActivity 的 setContentView 是不同的,接下来逐个分析

Activity 的 setContentView

// Activity
public void setContentView(@LayoutRes int layoutResID) {
    getWindow().setContentView(layoutResID); // 调用 PhoneWindow 的 setContentView 方法
    initWindowDecorActionBar(); // 初始化 ActionBar
}

// 初始化 ActionBar
private void initWindowDecorActionBar() {
    Window window = getWindow();
    window.getDecorView();

    // 判读 Activity 是否带有 ActionBar,如果没有,直接返回
    if (isChild() || !window.hasFeature(Window.FEATURE_ACTION_BAR) || mActionBar != null) {
        return;
    }

    // 如果有 ActionBar 则创建一个默认的 ActionBar ,并为 ActionBar 设置 Activity 相关的 Icon 和 Logo
    mActionBar = new WindowDecorActionBar(this);
    mActionBar.setDefaultDisplayHomeAsUpEnabled(mEnableDefaultActionBarUp);

    mWindow.setDefaultIcon(mActivityInfo.getIconResource());
    mWindow.setDefaultLogo(mActivityInfo.getLogoResource());
}    

// PhoneWindow
@Override
public void setContentView(int layoutResID) {
    
    if (mContentParent == null) {
        installDecor(); // 初始化 DecorView 也就是整个 Window 的根 View,主要过程为 new 一个 DecorView
    } else if (!hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
        mContentParent.removeAllViews(); // 移除 mContentParent 中的所有子 View mContentParent 为 DecorView 中的用于显示用户设置的布局的 ViewGroup
    }

    if (hasFeature(FEATURE_CONTENT_TRANSITIONS)) { // 有过度动画时的处理
        final Scene newScene = Scene.getSceneForLayout(mContentParent, layoutResID,
                getContext());
        transitionTo(newScene);
    } else {
        mLayoutInflater.inflate(layoutResID, mContentParent); // 由 layoutId 初始化 View 并添加到 mContentParent
    }
    ...
}

由此可以看出 Activity 中使用过 PhoneWindow 来向界面中添加用户设置的布局的。并且 DecorView 的创建也是在 PhoneWindow 中完成的。

AppCompatActivity 的 setContentView 方法

// AppCompatActivity
@Override
public void setContentView(@LayoutRes int layoutResID) {
    getDelegate().setContentView(layoutResID); // getDelegate() 方法的返回值是一个 AppCompatDelegate 对象
}

// AppCompatActivity
public AppCompatDelegate getDelegate() {
    if (mDelegate == null) {
        mDelegate = AppCompatDelegate.create(this, this);
    }
    return mDelegate;
}

// AppCompatDelegate
private static AppCompatDelegate create(Context context, Window window,
        AppCompatCallback callback) {
    final int sdk = Build.VERSION.SDK_INT;
    if (BuildCompat.isAtLeastN()) {
        return new AppCompatDelegateImplN(context, window, callback);
    } else if (sdk >= 23) {
        return new AppCompatDelegateImplV23(context, window, callback);
    } else if (sdk >= 14) {
        return new AppCompatDelegateImplV14(context, window, callback);
    } else if (sdk >= 11) {
        return new AppCompatDelegateImplV11(context, window, callback);
    } else {
        return new AppCompatDelegateImplV9(context, window, callback);
    }
}

// AppCompatDelegateImplV9 所有 AppCompatDelegate 子类的 setContentView 继承自了 AppCompatDelegateImplV9 的方法 
@Override
public void setContentView(int resId) {
    ensureSubDecor();
    ViewGroup contentParent = (ViewGroup) mSubDecor.findViewById(android.R.id.content);
    contentParent.removeAllViews();
    LayoutInflater.from(mContext).inflate(resId, contentParent);
    mOriginalWindowCallback.onContentChanged();
}

AppCompatDelegateImplV9 中的 mSubDecor 对象是一个 ViewGroup,其构造过程为根据 Activity 的主题创建一个根 View,其创建好之后,就会调用 PhoneWindow 的 setContentView 方法,将其作为参数添加到 Decorview 中

AppCompatActivity 中是通过 AppCompatDelegateImplV9 这个类来完成向界面中添加布局的

AppCompatDelegateImplV9 的 setContentView 方法中,我们将需要添加的布局添加到 mSubDecor ,也就添加到了 DecorView 中,Activity 的 onResume 方法回调之后也就显示到了界面上。

总结

由源码分析得出,setContentView 方法完成之后,Activity 需要显示的布局已经添加到了 DecorView 中

在 Activity 的 onResume 方法回调之后,布局就会显示到界面中。

有关 LayoutInflater 的工作过程,请期待下篇博客。

setContentView 的重载方法

setContentView(int layoutId);

setContentView(View view);

setContentView(View view,LayoutParams params);

setContentView(int layoutId);

这个方法有三个重载,第一个我们刚才已经分析过了,不管是 Activity 或者是 AppCompatActivity ,在添加是都会使用布局文件中自己设置宽高来构造 LayoutParams ,根布局在界面上的显示的宽高也完全由 xml 中设置的决定。

setContentView(View view);

Activity Activity 的 setContentView 加载是通过 PhoneWindow 的 setContentView 方法实现的,如果 setContentView 的参数为一个 View 对象,不管是否 View 有自己的 LayoutParams 属性,都会将 View 的 LayoutParams 设为新的,并且新的的宽高都为 MATH_PARENT

AppCompatActivity
AppCompatActivity setContentView 在添加 View 时,如果 View 有 LayoutParams ,则使用 View 原有的 LayoutParams ,如果 View 没有 LayoutParams ,则使用宽高都为 MATH_PARENT 的 LayoutParems (在 ViewGroup 添加 View 时如果 View 没有 LayoutParams ,此时应该是 默认 Warp_content,不过这里却是 MATH_PARENT ,在代码中没有找到原因)

setContentView(View view,LayoutParams params);

带 LayoutParams 参数的,不管是 Activity 还是 AppCompatActivity ,都会为 View 设置传入的 LayoutParams 属性值

你可能感兴趣的:(Activity.setContentView 方法源码解析)