Android控件架构
View的测量与绘制
ViewGroup的测量与绘制
自定义控件的三种方式
事件的拦截机制
3.1 Android控件架构
每个控件在界面中占据一块矩形的区域,控件大致可以分为两类:ViewGroup控件与View控件;ViewGroup作为父控件可以包含多个View控件,负责下层控件的测量与绘制,并传递交互事件
Android界面的架构图
每个Activity都包含一个Window对象,Window对象由PhoneWindow实现;PhoneWindow将一个DecorView设置为整个应用窗口的根View;DecorView包含了TitleView和ContentView
3.2 View的测量
Android提供了一个MeasureSpec类,通过它来测量View。MeasureSpec是一个32位的int值,高2位是测量的模式,低30位是测量的大小;测量的模式分为三种:
1.EXACTLY:精确值模式(默认模式)将控件的“layout_height”,"layout_width"属性指定为具体值的时候就是EXACTLY模式
2.AT_MOST:最大值模式当宽度高度指定为“wrap_content”时,控件大小一般随控件的子空间或内容的变化而变化,此时控件的尺寸不能超过父控件允许的最大尺寸
3.UNSPECIFIED:不指定其大小测量模式 总结:View的onMeasure()方法只支持EXACTLY模式,所以如果要让自定义View支持"wrap_content"属性就要重写onMeasure()方法来指定wrap_content时的大小。
源码分析:
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
}
进入onMeasure();方法
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec),
getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec));
}
我们可以看到源代码是通过setMeasuredDimension来测量大小的
代码示例
package com.example.view;
import android.content.Context;
import android.graphics.Paint;
import android.util.AttributeSet;
import android.view.View;
/**
* Created by 小新 on 2016/6/4.
*/
public class Myview extends View {
public Myview(Context context) {
super(context);
}
public Myview(Context context, AttributeSet attrs) {
super(context, attrs);
}
public Myview(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
setMeasuredDimension(widthMeasure(widthMeasureSpec),heightMeasure(heightMeasureSpec));
}
private int heightMeasure(int heightMeasureSpec) {
int result = 0;
int specMode = MeasureSpec.getMode(heightMeasureSpec);
int specSize = MeasureSpec.getSize(heightMeasureSpec);
if(specMode==MeasureSpec.EXACTLY){
result=specSize;
}else{
result=200;
if(specMode==MeasureSpec.AT_MOST){
result=Math.min(result,specSize);
}
}
return result;
}
private int widthMeasure(int widthMeasureSpec) {
int result = 0;
int specMode = MeasureSpec.getMode(widthMeasureSpec);
int specSize = MeasureSpec.getSize(widthMeasureSpec);
if(specMode==MeasureSpec.EXACTLY){
result=specSize;
}else{
result=200;
if(specMode==MeasureSpec.AT_MOST){
result=Math.min(result,specSize);
}
}
return result;
}
}
这样当我们自定义宽高为"wrap_content的时候"默认的大小是200px;不设置onMeasure()方法的时候我们使用“wrap_content”默认的大小是充满父控件