Activity、Window、View之间的关系
- Activity,控制模型,控制Window。
- Window,承载模型,负责承载视图(View)。
- View,显示模型,用于显示。
Activity与Window
每一次创建Activity实例后,接着会调用Activity.attach()来初始化一些内容,而Window对象就是在attach里进行创建初始化赋值的。在attach()里系统会创建Activity所属的Window对象并为其设置回调接口。由于Activity实现了Window的Callback接口,因此当Window接收到外界的状态改变就会回调到Activity的方法。
Window与View
ViewRoot(ViewRootImpl),DecorView
在了解Window与View之间的关系之前我们需要知道ViewRoot(ViewRootImpl)到底是什么。它是连接WindowManager和DecorView的纽带。
概括
- ViewRoot是一个ViewTree的管理者,而不是ViewTree的根节点。
- 严格意义上说,ViewTree的根节点只有DecorView。
- ViewRoot将DecorView和PhoneWindow(Activity创建的Window实例)“组合”起来。
而View的绘制三大流程(measure,layout,draw)均是通过ViewRoot来完成的。
执行的流程
在ActivityThread中,当Activity对象被创建完毕后,会将DecorVidew添加到Window中,同时会创建ViewRootImpl对象,利用ViewRootImpl对象来Window对象和DecorView之间的关系。
Window与View
View是Android中的视图呈现方式,但是View不能单独存在,它必须附着在Window这个抽象的概念上面,因此有视图的地方就有Window。
最有效的例子
最熟悉的方法就是在Activity.onCreate()方法中调用setContentView()方法,传入指定的布局文件。Activity将具体实现交给了Window处理,而Window的具体实现是PhoneWindow在setContentView中创建了DecorView,DecorView是整棵View树的根View,然后将制定的布局文件添加到DecorView的mContentParent中。
DecorView
DecorView作为根View,它内部会包含一个竖直方向的LinearLayout,在这个LinearLayout里面有上下两个部分,上面是标题栏(ActionBar Container),下面是内容(FrameLayout content),其中标题栏一般由Activity的Theme样式所决定。在Activity中我们通过setContentView所设置的布局文件其实就是被加到内容部分之中的。
设置Activity全屏
实现Activity全屏实际上就是:隐藏ActionBar和StatusBar。
通过上述的分析,有关DecorView的ContentView部分我们可以通过布局文件自定义。那么如何修改ActionBar的样式,甚至StatusBar呢?上面已经提到可以通过Theme来修改。而Theme是针对Window的一种属性集合,而通过Window.Attributes()获得的是WindowManager.LayoutParams对象。与此同时我在一些demo中遇到了Window.setFlags(WindowManager.LayoutParams,WindowManager.LayoutParams)方法。所以我们可以通过Window.setFlags()方法直接修改Theme属性值,实现Activity全屏的效果。
实践
在Activity的onCreate()中调用
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);
这也非常好的解释了为什么需要在setContentView()方法之前调用Window.setFlags()方法(设置某些Flag,例如隐藏ActionBar)。因为调用Activity.onCreate()后会通过与Activity绑定的Window对象创建DecorView。所以要实现设置好一系列的规则来创建这个DecorView。同样道理,requestWindowFeature()方法。
以上是我的个人理解,没有阅读过任何源码。水平不够,YY的。有不对的请指出