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 属性值

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,670评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,928评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,926评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,238评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,112评论 4 356
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,138评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,545评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,232评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,496评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,596评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,369评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,226评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,600评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,906评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,185评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,516评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,721评论 2 335

推荐阅读更多精彩内容