Android View 的绘制流程 - 开篇 MeasureSpec
Android View 的绘制流程 01 - 前置流程
Android View 的绘制流程 02 - performMeasure
Android View 的绘制流程 03 - performLayout
Android View 的绘制流程 04 - performDraw
耗时1个多月, 期间断断续续的看和学习. 总算是把Android View 的绘制流程弄清楚了.
不过仅仅只是流程, 还有很多细节没有学习到. 这就需要后面的不断积累和查看源码了. 有了流程, 再去看源码, 就会觉得清晰了很多.
经过这次学习, 包括前面学习的 setContentView 流程, 再结合网上的一些博客, 大概的弄清了 Activity, DecorView, Window, ViewRoot 他们之间的关系
绘制了一张图, 来代表 Android View 绘制的全部过程, 从 Activity 启动流程中调用 ActivityThread.handleLaunchActivity() 方法开始, 一直到 preforDraw() 结束. 图中, 有什么不对的地方, 欢迎大家指正, 一起学习,共同进步.
现在需要对 Android View 绘制流程做一些总结和一些网上收集到的.
只有在 onMeasure 测量的时候, 是一直遍历到最后才开始测量叶子 View, 然后一级一级的再向上传递叶子 View 的测量结果.根据所有叶子View 的测量结果来测量父容器.
measure 过程中, 子View 是要根据父容器的测量规格与自己的属性来确定自己的测量规格.
measure 完成后得到的是对每个 View 经测量过的 measuredWidth 和 measuredHeight.
layout 布局摆放的时候, 真正执行布局摆放的方法是 View.setFrame() 方法. 根据 onMeasure 测量结果进行布局摆放.
layout 完成之后得到的是对每个 View 进行位置分配后的 mLeft, mTop, mRight, mBottom, 这些值都是相对于父View来说的.
onDraw 需要我们自己去实现,从而绘制内容.
dispatchDraw 在 ViewGroup 已经实现好了, 默认会调用子 View 的 draw() 方法.
View 绘制的顺序, 就是我们给一个 ViewGroup 添加子View 的顺序.
layout 是对本身 View 的布局, onLayout, 是对所有子 View进行布局. layout 中 就是通过 setFrame() 方法设定本身 View 的四个顶点的位置, 这四个位置确定后, 本身 View 的位置就固定了. 然后调用 onLayout 来布局子View, View 和 ViewGroup 的 onLayout 方法都没有实现, 留给我们自己给子View布局
ViewGroup 默认是不会执行 onDraw 方法的. 因为 ViewGroup 多数只是负责布局, 不负责 draw. 如果想要 ViewGroup 进行 onDraw 怎么办, 可以调用
setWillNotDraw(false)
方法, 里面参数传 false 即可. ViewGroup 在初始化的时候, 默认这个标志位是 true. 如果我们自定义的 View 不需要 draw 的话, 就可以调用这个方法, 设置为 true. 这样系统就知道这个 View 是不需要绘制的. 可以优化执行速度. View 中 这个标志位一半都是关闭的(false状态).-
invalidate 方法和 View 绘制的关系.
invalidate 方法会会请求重绘 View 树. 调用 draw 方法. 如果 View 的大小没有发生变化就不会调用 layout 的流程, 并且只会绘制那些需要重绘的 View, 也就是哪个 View 请求 invalidate, 就绘制该View. (View 就只绘制该 View, ViewGroup 就会绘制整个 View Group)
直接调用 invalidate 方法, 会绘制调用者本身.
触发 setSelection 方法, 会绘制调用者本身.
触发 setVisibility 方法,
当 View 可视状态在 INVISIBLE 转换 VISIBLE 时会间接调用 invalidate 方法,绘制该View.
当 View 的可视状态在 INVISIBLE\VISIBLE 转换为 GONE 状态时会间接调用 requestLayout 和 invalidate 方法,同时由于 View 树大小发生了变化,所以会请求 measure 过程以及 draw 过程,同样只绘制需要“重新绘制”的视图。触发 setEnabled 方法, 不会重新绘制任何 View 包括当前调用者本身.
触发 requestFocus 方法, 只会绘制需要重绘的 View .