深入理解 DecorView

如果你对 Android 有一定了解的话,你一定知道 View 的树形结构,View 的测量、绘制和事件分发都是从树的根部逐级遍历分发下去的,而这个树形结构的根部就是我们今天要讲的 DecorView。下面是我画的一张围绕 DecorView 的层级关系图,其中最顶层是我们熟知的 Activity,每个 Activity 会有一个 Window 对象,该 Window 对象包含的就是 DecorView:

DecorView

接下来我们就自上而下,从源码的角度看看 DecorView 到底是什么。首先从我们最熟悉的一句代码说起,它就是 Activity 的 setContentView(),我们在 Activity 的 onCreate() 回调里都会调用该方法将布局文件设置给 Activity,那么该方法里面做了什么事情呢?来看下面一段源码:

public class Activity {

    public void setContentView(@LayoutRes int layoutResID) {
        getWindow().setContentView(layoutResID);
    }

}

很简单的一句话,就是获取 Activity 的 Window 对象将布局资源设置给它,Window 是一个抽象类,它的唯一实现类是 PhoneWindow,所以接下来我们就去看看 PhoneWindow 的 setContentView() 方法做了什么:

public class PhoneWindow extends Window {

    @Override
    public void setContentView(int layoutResID) {
        if (mContentParent == null) {
            // 初始化 DecorView
            installDecor();
        } else {
            mContentParent.removeAllViews();
        }
        // 解析我们设置的布局资源并且设置 mContentParent 为父布局
        mLayoutInflater.inflate(layoutResID, mContentParent);
    }

}

PhoneWindow 的 setContentView() 方法主要做了两件事情,初始化 DecorView 然后解析我们设置的布局资源到指定的父布局 mContentParent 中,那么我们先从 installDecor() 方法入手,看下 DecorView 是怎么初始化的,mContentParent 我们留到最后来讲:

public class PhoneWindow extends Window {

    private void installDecor() {
        if (mDecor == null) {
            mDecor = generateDecor();
        }
        if (mContentParent == null) {
            mContentParent = generateLayout(mDecor);
        }
    }

}

DecorView 的初始化分为两步,分别是创建 DecorView 和 初始化 DecorView 的布局,其中创建 DecorView 的方法 generateDecor() 很简单,里面就一句话创建一个新的 DecorView 对象,代码如下所示:

public class PhoneWindow extends Window {

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

}

你一定想知道 DecorView 是什么东西吧?那么我们就来看看下面的 DecorView 源码,其实 DecorView 继承自 FrameLayout,所以它实际上就是一个 ViewGroup。

private final class DecorView extends FrameLayout {}

知道 DecorView 是一个 ViewGroup 之后,我们继续看看它内部都装了什么东西,我们来看 generateLayout() 方法:

public abstract class Window {

    public static final int ID_ANDROID_CONTENT = com.android.internal.R.id.content;

    @Nullable
    public View findViewById(@IdRes int id) {
        return getDecorView().findViewById(id);
    }

}

public class PhoneWindow extends Window implements MenuBuilder.Callback {

    protected ViewGroup generateLayout(DecorView decor) {
        // 此处省去一堆代码,设置窗口属性

        int layoutResource;
        // 此处省去一堆代码,根据不同的主题使用不同的布局资源

        // 这里才是重点,向 DecorView 添加布局,并且从 DecorView 中查找出 contentParent
        View in = mLayoutInflater.inflate(layoutResource, null);
        decor.addView(in, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT));
        ViewGroup contentParent = (ViewGroup)findViewById(ID_ANDROID_CONTENT);
        if (contentParent == null) {
            throw new RuntimeException("Window couldn't find content container view");
        }

        // 此处省去一堆代码设置窗口背景和标题
        
        return contentParent;
    }
    
}

从上面的源码可以看出实际上 DecorView 里面包含了一个系统内置的布局资源,这个布局资源 layoutResource 会根据不同主题变化,其中一个资源是 com.android.internal.R.layout.screen_simple,该资源文件里有一个 ID 为 content 的 FrameLayout,它就是我们前面看到的 mContentParent,我们设置的布局文件就是被解析并添加到 mContentParent 中的:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:fitsSystemWindows="true"
    android:orientation="vertical">
    
    <ViewStub 
        android:id="@+id/action_mode_bar_stub"
        android:inflatedId="@+id/action_mode_bar"
        android:layout="@layout/action_mode_bar"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />
              
    <FrameLayout
        android:id="@android:id/content"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:foregroundInsidePadding="false"
        android:foregroundGravity="fill_horizontal|top"
        android:foreground="?android:attr/windowContentOverlay" />
        
</LinearLayout>

到此为止,我们对 DecorView 的分析就结束了,总结以下几点:

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,506评论 25 707
  • Android控件架构与自定义控件(一) (本文并非原创文章,整理摘抄方便自己查看,原文地址为Android控件架...
    b5e7a6386c84阅读 966评论 0 6
  • View的绘制和事件处理是两个重要的主题,上一篇《图解 Android事件分发机制》已经把事件的分发机制讲得比较详...
    Kelin阅读 119,168评论 100 845
  • 要努力
    不绊阅读 163评论 0 0
  • 站在天底下 拿出来晾晒 破碎残存的些许 灰白的东山欲念疯长 单薄的西河杂念肆意 英国人建的水塔 标志了沧桑的回眸 ...
    营州布衣阅读 205评论 4 10