了解ViewRoot和DecorView

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。


了解ViewRoot和DecorView_第1张图片
dv.png

参考:《Android开发艺术探索》一书
从ViewRootImpl类分析View绘制的流程

你可能感兴趣的:(了解ViewRoot和DecorView)