Android源码:xml文件是如何加载到屏幕上的(二)

说明

之前【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最初被绘制到界面上的发起者了。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值