zoukankan      html  css  js  c++  java
  • Android GUI之View绘制流程

      在上篇文章中,我们通过跟踪源代码,我们了解了Activity、Window、DecorView以及View之间的关系(查看文章:http://blog.csdn.net/jerehedu/article/details/47021541)。

    那么整个Activity的界面究竟是怎样绘制出来的呢?既然DecorView作为Activity的顶层界面视图。那么整个界面的绘制工作应该从它開始,以下我们继续跟踪源代码,看看是不是这种。

      Activity在启动过程中会调用主线程ActivityThread中的方法performLaunchActivity和handleResumeActivity。在方法handleResumeActivity中会将创建的DecorView和WindowManagerImpl对象关联起来。关键源代码部分例如以下:

    public final class ActivityThread {
       ……
        final void handleResumeActivity(IBinder token,
                boolean clearHide, boolean isForward, boolean reallyResume) {
          ……
                if (r.window == null && !a.mFinished && willBeVisible) {
                    r.window = r.activity.getWindow();
                    View decor = r.window.getDecorView();
                    decor.setVisibility(View.INVISIBLE);
                    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) {
                        a.mWindowAdded = true;
                        wm.addView(decor, l);
                    }
    
                } 
    ……
        }
    }

      WindowManagerImpl关键代码:

        public final class WindowManagerImpl implements WindowManager {
    private final WindowManagerGlobal mGlobal = WindowManagerGlobal.getInstance();
    ……
    @Override
    public void addView(View view, ViewGroup.LayoutParams params) {
          mGlobal.addView(view, params, mDisplay, mParentWindow);
    }
    ……
    }

      WindowManagerGlobal关键代码:

    public final class WindowManagerGlobal {
        ……
        public void addView(View view, ViewGroup.LayoutParams params,
                Display display, Window parentWindow) {
               ……
            ViewRootImpl root;
            View panelParentView = null;
    ……
            root.setView(view, wparams, panelParentView);
               ……
        }
    }

      依据源代码调用关系。可得下图:

      从图中。我们能够看出在ActivityThread中生成的DecorView经过WindowManagerImpl、WindowManagerGlobal,终于调用了ViewRootImpl中的setView方法。将DecorView设置赋值给了ViewRootImpl中的mView属性。

    通过追踪ViewRootImpl我们发现终于调用了performTraversals方法。该方法关键代码例如以下:

    private void performTraversals() {
            // cache mView since it is used so much below...
            final View host = mView;
            ……
            performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
            ……
            performLayout(lp, desiredWindowWidth, desiredWindowHeight);
            ……
            performDraw();
            ……
    }

      从上述源代码中我们能够看出。performTraversals实际上依次调用了三个关键的方法,各自是performMeasure,performLayout、performDraw。

      1、方法performMeasure,内部实际上调用了mView.measure(childWidthMeasureSpec, childHeightMeasureSpec);不要忘了此处的mView正是我们传递进来的DecorView,该方法用于測量View的大小。

    关键源代码例如以下:

            private void performMeasure(int childWidthMeasureSpec, int childHeightMeasureSpec) {
            Trace.traceBegin(Trace.TRACE_TAG_VIEW, "measure");
            try {
                mView.measure(childWidthMeasureSpec, childHeightMeasureSpec);
            } finally {
                Trace.traceEnd(Trace.TRACE_TAG_VIEW);
            }
        }

      2、方法performLayout,内部实际上调用了host.layout(0, 0, host.getMeasuredWidth(), host.getMeasuredHeight());该方法用于确定视图的位置。

    关键源代码例如以下:

        private void performLayout(WindowManager.LayoutParams lp, int desiredWindowWidth,
                int desiredWindowHeight) {
            mLayoutRequested = false;
            mScrollMayChange = true;
            mInLayout = true;
    
            final View host = mView;
            ……
            try {
                host.layout(0, 0, host.getMeasuredWidth(), host.getMeasuredHeight());
    
                mInLayout = false;
                int numViewsRequestingLayout = mLayoutRequesters.size();
                if (numViewsRequestingLayout > 0) {
                  ……
                    if (validLayoutRequesters != null) {
                        // Set this flag to indicate that any further requests are happening during
                       ……
                        host.layout(0, 0, host.getMeasuredWidth(), host.getMeasuredHeight());
    
                       ……                   
     }
                    }
    
                }
            } finally {
                Trace.traceEnd(Trace.TRACE_TAG_VIEW);
            }
            mInLayout = false;
        }

      3、方法performDraw。用于绘制视图。追踪源代码发现,终于调用了mView.draw(canvas)方法。用于绘制。

      经过上述过程基本上能够确定View的绘制流程,流程图详细例如以下:

      疑问咨询或技术交流。请增加官方QQ群:JRedu技术交流 (452379712)

    作者:杰瑞教育
    出处:http://blog.csdn.net/jerehedu/ 
    本文版权归烟台杰瑞教育科技有限公司和CSDN共同拥有,欢迎转载。但未经作者允许必须保留此段声明。且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

     
  • 相关阅读:
    Luogu6300 悔改 [FFT,阈值法]
    CF1016G Appropriate Team [Pollard-rho,FMT]
    AGC021F Trinity【计数,NTT】
    CF578F Mirror Box 【图论,Matrix-Tree】
    [ARC083]Collecting Balls
    HNCPC2019部分题解
    [LuoguP1829]Crash的文明表格(二次扫描与换根+第二类斯特林数)
    [CF960G]Bandit Blues(第一类斯特林数+分治卷积)
    [CF804F]Fake bullions
    [CF643E]Bear and Destroying Subtrees(期望,忽略误差)
  • 原文地址:https://www.cnblogs.com/cynchanpin/p/7354053.html
Copyright © 2011-2022 走看看