ViewRoot和DecorView的概念
在了解View的工作流程之前,有必要搞懂一些基本的概念。
ViewRoot对应于ViewRootImpl类,它是链接WindowManager和DecorView的纽带,View的三大流程均是通过ViewRoot来完成的
ViewRoot是如何链接WindowManager和DecorView的,以及如何完成三大流程的绘制的?
这就要从activity的启动时说起了,当Activity初始化Window和将布局添加到PhoneWindow的内部类DecorView类之后,ActivityThread类会调用handleResumeActivity方法将顶层的DecorView添加到PhoneWindow窗口,下面是handleResumeActivity部分源码
final void handleResumeActivity(IBinder token,
boolean clearHide, boolean isForward, boolean reallyResume) {
..................
if (r.window == null && !a.mFinished && willBeVisible) {
//获得当前Activity的PhoneWindow对象
r.window = r.activity.getWindow();
//获得当前phoneWindow内部类DecorView对象
View decor = r.window.getDecorView();
//设置窗口顶层视图DecorView可见度
decor.setVisibility(View.INVISIBLE);
//得当当前Activity的WindowManagerImpl对象
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) {
//标记根布局DecorView已经添加到窗口
a.mWindowAdded = true;
//将根布局DecorView添加到当前Activity的窗口上面
wm.addView(decor, l);
handlerResumeActivity()方法主要是通过上面代码的最后一行wm.addView()
方法将DecorView添加到窗口视图上的。
addView()源码
@Override
public void addView(View view, ViewGroup.LayoutParams params) {
mGlobal.addView(view, params, mDisplay, mParentWindow);
}
WindowManagerGlobal类看addView()方法源码
public void addView(View view, ViewGroup.LayoutParams params,
Display display, Window parentWindow) {
............
ViewRootImpl root;
View panelParentView = null;
............
//获得ViewRootImpl对象root
root = new ViewRootImpl(view.getContext(), display);
...........
// do this last because it fires off messages to start doing things
try {
//将传进来的参数DecorView设置到root中
root.setView(view, wparams, panelParentView);
} catch (RuntimeException e) {
...........
}
}
该方法创建了一个ViewRootImpl对象root,然后调用ViewRootImpl类中的setView成员方法()。(终于等到你,还好我没放弃/(ㄒoㄒ)/~~),进入ViewRootImpl源码
public void setView(View view, WindowManager.LayoutParams attrs, View panelParentView) {
synchronized (this) {
if (mView == null) {
//将顶层视图DecorView赋值给全局的mView
mView = view;
.............
//标记已添加DecorView
mAdded = true;
.............
//请求布局
requestLayout();
.............
}
}
跟踪代码进入到 requestLayout()方法
@Override
public void requestLayout() {
if (!mHandlingLayoutInLayoutRequest) {
checkThread();
mLayoutRequested = true;
scheduleTraversals();
}
}
................
void scheduleTraversals() {
if (!mTraversalScheduled) {
mTraversalScheduled = true;
mTraversalBarrier = mHandler.getLooper().postSyncBarrier();
mChoreographer.postCallback(
Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
if (!mUnbufferedInputDispatch) {
scheduleConsumeBatchedInput();
}
notifyRendererOfFramePending();
}
}
..............
final class TraversalRunnable implements Runnable {
@Override
public void run() {
doTraversal();
}
}
final TraversalRunnable mTraversalRunnable = new TraversalRunnable();
...............
void doTraversal() {
if (mTraversalScheduled) {
mTraversalScheduled = false;
mHandler.getLooper().removeSyncBarrier(mTraversalBarrier);
try {
performTraversals();
} finally {
Trace.traceEnd(Trace.TRACE_TAG_VIEW);
}
}
}
............
经过一系列的跟踪,终于来到高潮部分了,看下performTraversals()方法的代码:
private void performTraversals() {
...
// Ask host how big it wants to be
// 执行测量操作
performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
...
//执行布局操作
performLayout(lp, desiredWindowWidth, desiredWindowHeight);
...
//执行绘制操作
performDraw();
}
performTraversals()方法有700多行,这里只把几行我们需要的部分拿了出来,能够看出来该方法的主要流程体现了View绘制的三个主要步骤:测量,布局,绘制的三个阶段。
DecorView是啥?
DecorView是整个Window界面的最顶层View
一般情况下DecorView内部会包含一个竖直方向的LinearLayout,这个LinearLayout里面有上下两个部分,上面是标题栏,下面是内容栏。
在Activity中我们通过setContentView所设置的布局文件其实就是加载到内容栏之中的,而内容栏的id是content,因此可以理解为Activity指定布局的方法不叫setview而叫setContentView,因为我们的布局的确加到了id为content的Framelayout中。如何得到content呢?可以通过:
ViewGroup content = findViewById(R.android.id.content)。
如何得到我们设置的View呢?可以通过:
content.getChildAt(0)
在看下findViewById()
public View findViewById(int id) {
return getDecorView().findViewById(id);
}
也是从DecorView最顶层开始搜索的
通过源码我们可以知道,DecorView其实是一个FrameLayout,View层的事件都先经过DectorView,然后才传递给我们的View。
参考:《Android开发艺术探索》一书
从ViewRootImpl类分析View绘制的流程