手记

View绘制基本知识点

!通过阅读Android开发艺术探索整理


底层工作原理:测量流程、布局流程、绘制流程

常见回调方法:构造方法 onAttach onVisiblityChanged onDetach

ViewRoot对应ViewRootImpl,连接WindowManager与DecorView的纽带。

performTraversals方法:完成measure、layout、draw流程

传递流程:

ViewGroup                                                                    View

1.performMearsure> mearsure>onMearsure  传递        measure     

2.performLayout>layout>onLayout                 传递       layout

3.performDraw>draw>onDraw                      传递        draw

解读:performTraversals依次调用performMearsure,performLayout,performDraw分别完成顶层View的measure,layout,draw三大流程。其中performMearsure调用measure,measure调用onMeasure,onMeasure中对所有子元素进行measure过程,此时measure流程由父元素传递到子元素,完成一次measure过程。子元素会重复父元素的measure过程,如此反复完成整个View树遍历。performLayout与performDraw流程类似,但是performDraw的传递过程是 在draw方法中通过dispatchDraw完成

方法作用:

measure:决定View的宽高,完成以后通过getMeasuredWidth/getMeasuredHeight获取测量后的宽高,几乎所有情况等同于View的最终宽高,特殊情况除外;

layout:View四个顶点的坐标及实际的View宽、高。四点位置:getTop/getLeft/getRight/getBottom,最终宽高getWidth/getHeight;

draw:决定View 的显示,完成后View才会在屏幕上显示。

DectorView:竖直方向包含一个LinearLayout,包含上下两部分,标题栏,内容栏,setContentView布局添加到内容栏,id为content。ViewGroup conent=findViewById(R.android.id.content),得到content;得到我们设置的View:conent.getChildAt(0);

MeasureSpec: 测量规格,32位int值 ,高两位SpecMode 低30位SpecSize。父容器影响View的MeasureSpec创建过程。测量过程中,系统会将View的LayoutParams根据父容器所施加的规则转换成对应的MeasureSpec,然后再根据MeasureSpec测量View的宽高。

SpecMode分类: 

UNSPECIFIED: 不限定大小,一般用于系统内部表示一种测量状态;

AT_MOST:父容器指定大小SpecSize,View不能超过该值,对应wrap_content;

EXACTLY:精确大小,就是SpecSize值,对应match_parent及具体数值

MeasureSpec与LayoutParams对应关系:

注意:LayoutParams需和父容器一起决定View的MeasureSpec,进一步决定View的宽高

DecorView :由窗口尺寸与LayoutParams共同决定

规则:match_parent>EXACTLY,窗口大小

        wrap_content>AT_MOST  大小不定,但不超过窗口

        固定大小,为LayoutParams指定

方法:ViewRootImpl中measureHierarchy方法,屏幕尺寸desireWindowWidth,desireWindowHeight getRootMeaSureSpec方法实现

普通View:由父容器MeasureSpec与自身LayoutParams决定,MeasureSpec一旦确定,onMeasure就可以确定View的测量宽高

方法:measure之前,通过getChildMeasureSpec得到子元素的MeasureSpec,子元素的MeasureSpec与父元素的MeasureSpec、本身的LayoutParams、padding及margin有关

规则:

1.当View固定宽高时,无论父容器何种MeasureSpec,View 的MeasureSpec都精确模式,大小遵循LayoutParams规定大小;

2.View宽高是match_parent时,父容器精确View也精确且大小是父容器剩余空间;父容器最大时View也最大且大小不超过父容器剩余空间

3.View宽高是wrap_content时,不管父容器是精确还是最大化,View都是最大化且不超过父容器剩余空间。

View工作流程:主要为measure/layout/draw

View 的measure:

measure: final方法,调用onMeasure完成。

1.setMeasuredDimension方法设置View宽高测量值。

2.getDefaultSize,AT_MOST与EXACTLY模式时返回的大小就是MeasureSpec中的specSize,最后测量的大小。最终确认在layout阶段,但几乎所有情况下View的测量大小就是最后的大小。

UNSPECFIDED模式,一般用于系统内部测量,宽/高的测量值就是getSuggestedMininumWidth/getSuggestedMininumHeight的返回值。

getSuggestedMininumWidth:如果View没有设置背景,返回android:minWidth的值,可以为0;设置背景,返回android:minWidth和背景宽度中的最大值。

结论:直接继承View重写onMeasure并设置wrap_content时的大小,否则在布局中使用的wrap_content相当于match_parent

方式:根据需要,给View指定一个默认的内部宽高,在wrap_content时设置此值,非wrap_content情形,沿用系统的测量值。

ViewGroup的measure:

除完成自己的measure,还需遍历调用子元素的measure方法,各子元素递归执行此流程

方法:提供measureChildren方法,对每一个子元素measure,通过measureChild方法取出子元素LayoutParams,通过getChildMeasureSpec创建子元素的MeasureSpec,将MeasureSpec传递给子元素的measure来完成测量

ViewGroup没有定义具体的measure过程,因为是抽象类,测量过程的onMeasure需要子类去完成,不同的子类有不同的布局特性,测量细节不同。

总结:measure完成后,通过getMeasuredWidth/Heigt就可以正确的得到View的测量 宽高。某些极端情况下:系统需要多次测量才能确定宽高,此时onMeasure获取宽高不准确,良好习惯是在onLayout中获取View的测量宽高或最终宽高

layout:

作用:ViewGroup用来确定子元素位置

概述:onLayout遍历调用子元素layout,layout中的onLayout方法继续调用

流程:1.setFrame方法设置View的四个顶点位置,顶点位置一旦确定,View在父容器中的位置就会确定;

2.调用onLayout,父容器确定子元素位置,具体实现与具体布局有关,View与ViewGroup没有具体实现

View的测量宽高与实际宽高区别:默认实现中是相等的,测量宽高成形于View的measure过程,最终宽高形成于layout过程,二者赋值时机不同。日常开发中,我们可以认为测量宽高就等于最终宽高;

注:一些特殊情况下,View会多次测量才会确定自己的测量宽高,前几次测量过程中得出的宽高确实有可能与实际不一致,但最终,二者会相同

draw:

流程:1.绘制背景 background.draw(Canvas)

2.绘制自己 onDraw

3.绘制Childrend dispatchDraw

4.绘制装饰onDrawScrollbars

View绘制过程的传递通过dispathcDraw完成,dispatchDraw会遍历调用所有子元素的draw方法,一层一层传递

特殊方法:setWillNotDraw,View默认没有启用,ViewGroup默认启用,不绘制任何内容时,该标记设为true,系统会进行相应优化;对我们的意义是,当我们自定义控件继承自ViewGroup,本身不具备绘制功能时,可以开启此标记以便系统优化。明确知道ViewGroup需要通过onDraw绘制时,需要显示关闭此标记

自定义View:

分类:

1.继承View重写onDraw :不方便通过布局组合方式,静态或动态显示一些不规则图形,需支持wrap_content,处理padding

2.继承ViewGroup派生特殊Layout:实现自定义布局,当几种View组合在一起时,需要合适处理measure与layout过程,同时处理子元素测量布局过程。更接近View底层

3.继承特殊的View如TextView:扩展已有View功能,不需要自己支持wrap_content与padding

4.继承特殊的ViewGroup如LinearLayout:不需要处理measure与layout过程,当几种View组合时。

须知:1.让View支持wrap_content

2.如有必要,让View支持padding

3.尽量不在View中使用Handler,没必要。除非明确需要发送消息

4.View中如果有动画或线程,及时停止,detachedFromWindow

5.处理好滑动嵌套冲突

思想:首先掌握基本功,如弹性滑动,滑动冲突,绘制原理;面对新的自定义View时能够分类并选择合适的实现思路;平时多积累自定义View的经验,做到融汇贯通

原文链接:http://www.apkbus.com/blog-35555-72610.html

0人推荐
随时随地看视频
慕课网APP