android.app.View 就是手机的UI,View 负责绘制UI,处理事件(evnet),Android 利用 View 打造出所 Widgets,利用 Widget 可打造出互动式的使用者介面,每个View 负责一定区域的绘制。
一张图理解常用控件层级关系
View的宽高是有top、left、right、bottom参数决定的 而X,Y和translationX,和translationY则负责View位置的改变。
从Android3.0开始,加入了translation的概念,即相对于父容器的偏移量以及X,Y坐标的概念,X,Y代表左上顶点的横纵坐标。当View在发生平移时,getX,getY,setX,setY
get/setTranslationX/Y来获得当前左上点的坐标。
X=left+translationX Y同理。
注意:在View发生改变的过程中,top,left等值代表原始位置,是不会改变的。改变的只有X,Y,translationX/Y。
Category | Methods | Description |
---|---|---|
Creation | Constructors | 几个View的构造函数 |
onFinishInflate() | 当系统解析完View之后调用onFinishInflate方法 | |
Layout | onMeasure(int, int) | 确定所有子View的大小 |
onLayout(boolean, int, int, int, int) | 当ViewGroup分配所有的子View的大小和位置时触发 | |
onSizeChanged(int, int, int, int) | 当view的大小发生变化时触发 | |
Drawing | onDraw(android.graphics.Canvas) | view渲染内容的细节 |
Event processing | onKeyDown(int, KeyEvent) | 有按键按下后触发 |
onKeyUp(int, KeyEvent) | 有按键按下后弹起时触发 | |
onTrackballEvent(MotionEvent) | 轨迹球事件 | |
onTouchEvent(MotionEvent) | 触屏事件 | |
Focus | onFocusChanged(boolean, int, android.graphics.Rect) | 当View获取或失去焦点时触发 |
onWindowFocusChanged(boolean) | 当窗口包含的view获取或失去焦点时触发 | |
Attaching | onAttachedToWindow() | 当view被附着到一个窗口时触发 |
onDetachedFromWindow() | 当view离开附着的窗口时触发,该方法和 onAttachedToWindow() 是相反 | |
onWindowVisibilityChanged(int) | 当窗口中包含的可见的view发生变化时触发 |
对实现自定义View,不需要重写所有这些方法。事实上,你可以只onDraw(android.graphics.Canvas)
public MyView(Context context)
public MyView(Context context, AttributeSet attrs)
public MyView(Context context, AttributeSet attrs, int defStyleAttr)
@TargetApi(Build.VERSION_CODES.LOLLIPOP)
public MyView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes)
这里我们先简单理解View 的绘制,后续文章我们会深入理解。
1.测量——onMeasure():决定View的大小
2.布局——onLayout():决定View在ViewGroup中的位置
3.绘制——onDraw():如何绘制这个View。
override
父类的方法,如 onDraw,(onMeasure)
等attrs.xml
文件,在其中定义属性通过context.obtainStyledAttributes将构造函数中的attrs进行解析出来,就可以拿到相对应的属性.
TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.MyView);
mColor = typedArray.getColor(R.styleable.MyView_myColor, 0XFF00FF00);【注意】三个函数获取尺寸的区别:
getDimension()
是基于当前DisplayMetrics进行转换,获取指定资源id对应的尺寸getDimensionPixelSize()
与getDimension()
功能类似,不同的是将结果转换为int,并且小数部分四舍五入getDimensionPixelOffset()
与getDimension()
功能类似,不同的是将结果转换为int,取整去除小数。举个例子
列如getDimension()
返回结果是20.5f,那么getDimensionPixelSize()
返回结果就是 21,getDimensionPixelOffset()
返回结果就是20。
res-atuo
命名空间,就不用在添加自定义View全类名。xmlns:app="http://schemas.android.com/apk/res-auto"
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
|
/**
* Created by fuchenxuan on 16/6/4.
*/
public
class
MyView
extends
View {
private
int
mRadius=
200
;
private
int
mColor;
public
MyView(Context context) {
this
(context,
null
);
}
public
MyView(Context context, AttributeSet attrs) {
this
(context, attrs,
0
);
}
public
MyView(Context context, AttributeSet attrs,
int
defStyleAttr) {
super
(context, attrs, defStyleAttr);
//read custom attrs
TypedArray t = context.obtainStyledAttributes(attrs,
R.styleable.rainbowbar,
0
,
0
);
mRadius = t.getDimensionPixelSize(R.styleable.coutom_radius, (
int
) hSpace);
t.getDimensionPixelOffset(R.styleable.coutom_at1, (
int
) vSpace);
mColor=t.getColor(R.styleable.color, barColor);
t.recycle();
// we should always recycle after used
}
@Override
protected
void
onMeasure(
int
widthMeasureSpec,
int
heightMeasureSpec) {
//super.onMeasure(widthMeasureSpec, heightMeasureSpec);
int
widthMode = MeasureSpec.getMode(widthMeasureSpec);
int
widthSize = MeasureSpec.getSize(widthMeasureSpec);
int
heightMode = MeasureSpec.getMode(heightMeasureSpec);
int
heightSize = MeasureSpec.getSize(heightMeasureSpec);
//set size
setMeasuredDimension(widthMode == MeasureSpec.AT_MOST ? (
int
) mRadius *
3
: widthSize, heightMode == MeasureSpec.AT_MOST ? (
int
) mRadius *
3
: heightSize);
}
//draw be invoke clire.
int
index =
0
;
@Override
protected
void
onDraw(Canvas canvas) {
//super.onDraw(canvas);
mPaint =
new
Paint();
mPaint.setColor(mColor);
mPaint.setAntiAlias(
true
);
canvas.drawCircle(mRadius, mRadius, mRadius, mPaint);
}
}
|
这里是一个普通的自定义View,里面画了圆,根据不同的模式设置了父View的大小。
关于View重写onMeasure()
时机:
如果用了wrap_content
。那么在onMeasure()
中就要调用setMeasuredDimension()
,
来指定view的宽高。如果使用的是match_parent
或者一个具体的dp值。那么直接使用super.onMeasure()
即可。
onLayout()
函数
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
|
/**
* Created by fuchenxuan on 16-6-6.
*/
public
class
CostumViewGroup
extends
ViewGroup {
public
CostumViewGroup(Context context) {
super
(context);
}
public
CostumViewGroup(Context context, AttributeSet attrs) {
super
(context, attrs);
}
@Override
protected
void
onMeasure(
int
widthMeasureSpec,
int
heightMeasureSpec) {
super
.onMeasure(widthMeasureSpec, heightMeasureSpec);
int
childCount = getChildCount();
for
(
int
i =
0
; i < childCount; i++) {
View childView = getChildAt(i);
measureChild(childView, widthMeasureSpec, heightMeasureSpec);
}
}
@Override
protected
void
onLayout(
boolean
changed,
int
l,
int
t,
int
r,
int
b) {
if
(changed) {
int
childCount = getChildCount();
for
(
int
i =
0
; i < childCount; i++) {
View childView = getChildAt(i);
childView.layout(i * childView.getMeasuredWidth(),
0
, (i +
1
) * childView.getMeasuredWidth(), childView.getMeasuredHeight());
}
}
}
}
|
这里是一个简单的自定义ViewGroup,实现类似LinearLayout 横向排放子View位置。这就是一个简单的ViewGroup过程。
View的大小不仅由自身所决定,同时也会受到父控件的影响,为了我们的控件能更好的适应各种情况,一般会自己进行测量。他们是由 mode+size两部分组成的。widthMeasureSpec和heightMeasureSpec转化成二进制数字表示,他们都是30位的。前两位代表mode(测量模 式),后面28位才是他们的实际数值(size);MeasureSpec.getMode()
获取模式,MeasureSpec.getSize()
获取尺寸
测量View大小使用的是onMeasure函数,所以我们需要了解三种测量模式:
EXACTLY
:一般是设置了明确的值(100dp)或者是MATCH_PARENT
AT_MOST
:表示子布局限制在一个最大值内,一般为WARP_CONTENT
UNSPECIFIED
:表示子布局想要多大就多大,很少使用关于ViewGroup重写onMeasure()
时机:
getChildAt(int index)
可以拿到index上的子view。getChildCount()
得到子view的个数,再循环遍历出子view。measureChild(subView, int wSpec, int hSpec);
measureChildren(int wSpec, int hSpec);
measureChildWithMargins(subView, intwSpec, int wUsed, int hSpec, int hUsed);
测量某一个子view,多宽,多高, 内部加上了viewGroup的padding值、margin值和传入的宽高wUsed、hUsed