Android自定义View
Android自定义View时,一般会涉及到onMeasure、onLayout、onSizeChanged、onDraw以及和事件分发传递相关的方法等
Android自定义View之onMeasure
一般Android在绘制View时会调用onMeasure方法来计算其宽高,当控件的layout_width和layout_height设为match_parent或者wrap_content显示的大小由其父容器控件来决定。若设置为固定大小的值,那么就显示其设定的值
MeasureSpec.getSize()会解析MeasureSpec得到测量值的大小
MeasureSpec.getMode()会得到3个int类型的值:
1)MeasureSpec.EXACTLY
父容器决定控件的大小,忽略其本身的大小,当width或height设为match_parent或者固定大小时,模式为EXACTLY,控件去占据父容器的剩余空间
2)MeasureSpec.AT_MOST
控件最大可以达到的指定大小,当width或height设置为wrap_content,模式为AT_MOST
3)MeasureSpec.UNSPECIFIED
父容器不对控件施加任何约束,控件可以是任意大小,一般父容器是AdapterView
setMeasuredDimension方法用于设置实际控件的大小
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
int width = measureSize(px2dp(DEFAULT_WIDTH),widthMeasureSpec);
int height = measureSize(px2dp(DEFAULT_HEIGHT),heightMeasureSpec);
setMeasuredDimension(width,height);
}
private int measureSize(int defaultSize,int spec){
int specMode = MeasureSpec.getMode(spec);
int specSize = MeasureSpec.getSize(spec);
if (specMode == MeasureSpec.EXACTLY){
return specSize;
}else if (specMode == MeasureSpec.AT_MOST){
return Math.min(defaultSize,specSize);
}
return defaultSize;
}
Android自定义View之事件拦截分发机制
//分派事件
public boolean dispatchTouchEvent(MotionEvent ev)
//拦截事件
public boolean onInterceptTouchEvent(MotionEvent ev)
//处理事件
public boolean onTouchEvent(MotionEvent event)
其中View只包含dispatchTouchEvent和onTouchEvent两个方法(另外Activity也只包含这两个方法)
而ViewGroup具有全部3个方法
在ViewGroup中的dispatchTouchEvent方法里会调用其onInterceptTouchEvent方法
如果onInterceptTouchEvent方法返回true(即拦截事件),则会传递到其自己的onTouchEvent方法处理,到此就结束了
那么当onInterceptTouchEvent方法返回false,则会传递到其子控件的dispatchTouchEvent方法处理
如果这个子控件是继承ViewGroup,则继续按照这样的方法执行下去,直到它是最后一个ViewGroup或者View
如果传递到的子控件是个View呢,它也会先执行dispatchTouchEvent方法,然后在其方法中调用到onTouchEvent方法
如果onTouchEvent方法返回true,则代表处理成功,然后一层一层的返回去
如果onTouchEvent方法返回false,则代表处理失败,其会回到其父控件去执行其父控件的onTouchEvent方法,直到onTouchEvent方法返回true
如果所有的onTouchEvent都返回false,那么这个touch事件则会丢失未处理
简单来说执行onInterceptTouchEvent方法返回true,执行自己的onTouchEvent方法
否则就执行其子控件的onInterceptTouchEvent方法(ViewGroup)或者onTouchEvent方法(View)
项目地址
https://github.com/peace710/Widget