声明:本笔记是本人阅读徐宜生的《Android群英传》和《Android开发艺术探索》
Android控件框架
-
Android控件大致分为两类,即ViewGroup和View。ViewGroup作为父结构可以包含多个View,并负责子空间的测量与绘制,传递交互事件。从而形成了控制树,而每棵控制树的顶层都有一个ViewParent。
-
每个Activity都包含一个Window对象(Android中其实就是PhoneWindow),其将一个DecorView设置为整个应用窗口的根。其结构如下图:
-
下图是一个标准的视图树,其在第二层加载了一个LinearLayout作为ViewGroup(默认,现在默认的ViewGroup目测是RelativeLayout)。
View测量
要画一个图案,我们需要的首先就是要测量图案的大小,绘制图案的位置。View的绘制流程是从ViewRoot的performTraversals开始的,途径measure、layout、draw。
其中performTravsersals是ViewRoot的方法。ViewRoot对应于ViewRootImpl类,是WindowManager和DecorView的纽带。三大流程都是通过ViewRoot来完成的。
绘制调用详情见文章
理解MeasureSpec
MeasureSpec很大程度上决定了一个View的尺寸规格(这个过程还受到父容器的影响,父容器影响View的MeasureSpec的创建过程)。它其实是一个32位的int,高两位代表SpecMode,低30位代表SpecSize。
public static int makeMeasureSpec(int size, int mode) {
return size + mode;9025
}
SpecMode的三种模式
1.UNSPECIFIED:比较奇怪的一个模式——不指定其大小测量模式,View想多大就多大,通常在绘制自定义控件时实现。
2.EXACTLY(完全):当我们的空间的layout_width或者layout_height属性指定为具体数值时使用。
3.AT_MOST(至多),layout_width或者layout_height属性指定为wrap_content时,控件大小一般随着控件的子空间或者内容的变化而变化,此时控件只需要不超过父控件的最大尺寸。
View默认的onMeasure()方法只支持EXACTLY模式,所以如果自定控件的时候不重写onMeasure方法的话,就只能使用EXACTLY模式。而View的onMeasure方法最终会调用setMeasuredDimension(int widthMeasureSpec , intheightMeasureSpec)。这是系统最终戴傲勇用以将测量后的宽高值设置进去的方法,用以完成测量工作。
具体的看到一篇简书,讲的比较清楚了点击哦!亲.
View绘制
当测量好了一个View之后,我们就可以简单得重写onDraw()方法,这里一共用到Paint、bitmap、canvas三种东西。
Canvas用来实现绘制的操作,需要四个组件
bitmap:保存着像素
canvas:执行画图的命令
drawing primitive:绘图的原始内容
paint:画笔,设置绘图风格
这样说吧,bitmap相当于存储介质,canvas是在这个介质上的画布,而paint是画笔,将会决定画上去的图形效果。
一般情况下创建Canvas,需要传入一个bitmap对象,将画布与bitmap绑定,在画布上的操作,即对bitmap的操作。
Canvas canvas = new Canvas(bitmap);
ViewGroup测量和绘制
ViewGroup会去管理其子View,其中一个管理项目就是负责子View的显示大小。当ViewGroup的大小为Wrap_content时,ViewGroup就需要对子进行遍历,一遍获得所有字View的大小,从而决定自身的大小。
通过调用子View的Measure来获得每一个子View的测量结果。测量完就需要将子View放到合适的位置,即View的layout过程。ViewGroup调用layout同样特使调用子layout来指定确切的位置。
关于ViewGroup的绘制,通常情况下并不需要绘制,应为它本省就没有需要绘制的东西,除非指定背景颜色。
总结自定义View
在View中通常有以下一些比较重要的回调方法。
- onFinishInflate():从XML加载组件后回调。
- onSizeChanged():组件大小改变回调。
- onMeasure():回调该方法来测量。
- onLayout():回调该方法来确定显示位置。
- onTouchEvent():监听到触摸事件的回调。
当然,穿件View重写特定条件的毁掉方法即可。一般自定义控件有以下三种方法: - 对现有控件进行拓展。