说明
之前【Android源码:xml文件是如何加载到屏幕上的(一)】中,从setContentView这个方法出发,找到了DecorView的初始化过程,也知道了xml文件是如何通过inflate方法,添加到mContentParent中的。
联系到之前Activity的创建过程中描述的,Activity是在ActivityThread中创建的,现在回到ActivityThread类中,接着看xml文件被添加到mContentParent中以后,是如何绘制到屏幕上的。
首先回到ActivityThread类中的performLaunchActivity,之前看Activity的创建过程的时候,我们已经知道了会调用这个方法,然后在这个方法中创建出一个activity。
activity = mInstrumentation.newActivity(
cl, component.getClassName(), r.intent);
在这行代码往后看,可以找到一个attach方法。在attach方法中,创建了一个Window对象,并且将ActivityThread中的参数保存到了Activity当中。最后在performLaunchActivity方法中,将创建出来的Activity返回了出来。
在创建完Activity之后,系统又调用了一个handleResumeActivity方法:
if (r.window == null && !a.mFinished && willBeVisible) {
//拿到activity中创建的Window对象
r.window = r.activity.getWindow();
//拿到Window中的decor
View decor = r.window.getDecorView();
decor.setVisibility(View.INVISIBLE);
ViewManager wm = a.getWindowManager();
WindowManager.LayoutParams l = r.window.getAttributes();
//将Window中的decor赋值给Window,作为activity的decor
a.mDecor = decor;
l.type = WindowManager.LayoutParams.TYPE_BASE_APPLICATION;
l.softInputMode |= forwardBit;
//将Window的decor添加到WindowManager中
if (a.mVisibleFromClient) {
a.mWindowAdded = true;
wm.addView(decor, l);
}
}
有之前了解的Activity的创建过程,已经知道了在performLaunchActivity中,会调用到Activity的onCreate方法,而在上一篇绘制流程中,已经知道了,在onCreate方法中的setContentView中,会创建出DecorView和Window对象,而在handleResumeActivity中,我们发现,activity的Window对象被保存了起来,而activity的Window中的decor也被保存了起来,并且添加到了ViewManager中。
所以,现在我们已经清楚的知道,我们的xml文件被填充出来后,最终添加到了WindowManager中。看到这里,我们还没有找到View的绘制,那么我们可以猜测View的绘制过程一定发生在WindowManager中,首先看WindowManager的实现类。
我们发现WindowManager有两个实现类,分别是WindowManagerGlobal和WindowManagerImpl,但是WindowManagerImpl其实还是持有的WindowManagerGlobal对象,和Context的实现类一样。
WindowManagerGlobal:
WindowManagerGlobal的addView逻辑也很简单:
ViewRootImpl root;
View panelParentView = null;
//首先构建出一个ViewRootImpl对象
root = new ViewRootImpl(view.getContext(), display);
//将传入属性设置给传入的的view
view.setLayoutParams(wparams);
//最后将传入的view和wparams添加到内部的集合对象mViews,mParams
//将创建出来的ViewRootImpl添加到另一个内部集合
mViews.add(view);
mRoots.add(root);
mParams.add(wparams);
在做完这些以后,调用了ViewRootImpl的setView方法:
root.setView(view, wparams, panelParentView);
在ViewRootImpl中,先将传入的view保存到了mView对象中,然后调用到requestLayout方法。
public void requestLayout() {
if (!mHandlingLayoutInLayoutRequest) {
checkThread();
mLayoutRequested = true;
scheduleTraversals();
}
}
void scheduleTraversals() {
......
mChoreographer.postCallback(
Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
}
在这里,我们发现他执行了一个mTraversalRunnable:
final TraversalRunnable mTraversalRunnable = new TraversalRunnable();
final class TraversalRunnable implements Runnable {
@Override
public void run() {
doTraversal();
}
}
void doTraversal() {
......
performTraversals();
}
最后,在performTraversals这个方法中,终于可以看到它对view的处理了,包括取出view的属性等等一些操作,然后在其中可以看到:
performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
performLayout(lp, desiredWindowWidth, desiredWindowHeight);
performDraw();
看到这里,我们终于找到view最初被绘制到界面上的发起者了。