事件分发相对于ViewGroup来讲的,节点View不能分发事件。
1.dispatchTouchEvent(MotionEvent event)
2.onInterceptTouchEvent(MotionEvent event)
3.onTouchEvent(MotionEvent event)
事件——MotionEvent
public boolean dispatchTouchEvent(MotionEvent ev)
用于对事件进行分发。如果事件能被传递给当前View,那么该方法一定会被调用并且是首先被调用,返回值代表当前事件是否被消耗,
public boolean onInterceptTouchEvent(MotionEvent ev)
用于判断是否拦截某个事件,在上述方法的内部被调用。如果View拦截了某个事件,那么在同一事件序列中,此方法不会被再次调用。
public boolean onTouchEvent(MotionEvent ev)
用于对事件的具体处理,返回值表示是否消耗当前事件。如果onTouchEvent()方法未消耗当前事件,则在同一事件序列中,当前View无法再次接收到事件。
onInterceptTouchEvent(MotionEvent event)是ViewGroup的一个方法,系统向ViewGroup及其所有子视图传递事件之前进行一次拦截,不能包含子view的控件是没有这个方法的,如TextView就没有。
Event Down事件首先会传递到onInterceptTouchEvent(MotionEvent event)
Event Down
return false/true;
Move
return false/true;
Up
return false/true;
如果ViewGroup的onInterceptTouchEvent(Event event)接收到Down
事件处理完成之后
return true
那么MotionEvent后续的Move, Up等事件将不再分发,不再由onInterceptTouchEvent()分发下去了,而是和Down事件一起传递给ViewGroup
onTouchEvent(MotionEvent event)处理,注意,目标view将接收不到任何事件。
onTouchEvent(MotionEvent event)
Event Down
return false/true;
Move
return false/true;
Up
return false/true;
如果ViewGroup的onInterceptTouchEvent()在接收到
down事件
处理完成之后
return false
那么后续的move, up等事件将继续由onInterceptTouchEvent()分发,之后跟down事件一样传递给最终的目标view的onTouchEvent()处理。
这里需要说明onInterceptTouchEvent()在接收到Move事件完成后return true
如果最终需要处理事件的view的onTouchEvent()
Down
处理完事件
return false
那么该事件将被传递至其上一层次的view的onTouchEvent()处理。
如果最终需要处理事件的view的onTouchEvent()
Down
处理完事件
return true
那么后续事件将继续传递给该view的onTouchEvent()处理。
如果ViewGroup中onInterceptTouchEvent()拦截Down事件
return true,说明拦截了干事件由我来处理,不会向子View分发了。所以不会再由onInterceptTouchEvent()函数分发事件,也就不会再调用该函数,子View无法接受到任何的Touch事件,导致目标View不会收到Touch事件。全部由该ViewGroup中的onTouchEvent消耗掉。
如果onTouchEvent拦截
Down事件
return true,则该ViewGroup中结束全部事件,即Move 、UP结束,不会执行Move及Up事件。
如果该ViewGroup有父View,把Down Move Up三种事件返回给父类View处理,父View没有子View则无需调用onInterceptTouchEvent()分发函数,没有父类,则结束事件。
//拦截按下事件
@Override
public boolean onInterceptTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN: {
Log.d(TAG, "MotionEvent.ACTION_DOWN");
return true;
}
}
//所有Touch事件被该onTouchEvent消耗
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN: {
Log.d(TAG, "MotionEvent.ACTION_DOWN");
return true;
}
case MotionEvent.ACTION_MOVE: {
Log.d(TAG, "MotionEvent.ACTION_MOVE");
return true;
}
case MotionEvent.ACTION_UP: {
Log.d(TAG, "MotionEvent.ACTION_UP");
return true;
}
}
return true;
}
参考文章
View的绘制
看了大神们写的书,没有理解的很透,故记录在案:
View的工作原理/View的底层工作原理
自定义View/自定义控件
书中有句"ViewRoot对应于ViewRootImpl类"我当时理解为ViewRootImpl继承于ViewRoot,但是看过源码后,其实不是这样的,并不存在ViewRoot这个类。
ViewRootImpl.java
//执行遍历方法,贴出了主要代码片段,该方法中很长很长
private void performTraversals() {
//缓存mView,很多地方需要用到该对象
final View host = mView;
performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
performLayout(lp, desiredWindowWidth, desiredWindowHeight);
performDraw();
}
//查看performMeasure方法,这里贴出了全部的代码,该方法中调用了rmView的measure()方法。
private void performMeasure(int childWidthMeasureSpec, int childHeightMeasureSpec) {
Trace.traceBegin(Trace.TRACE_TAG_VIEW, "measure");
try {
mView.measure(childWidthMeasureSpec, childHeightMeasureSpec);
} finally {
Trace.traceEnd(Trace.TRACE_TAG_VIEW);
}
}
测量流程--measure
测量完成后,计算出View的宽高。
测量的结果可通过getMeasureWidth(),getMeasureHeight()获得测量的宽高值。
MeasureSpec
DecorView的MeasureSpec产生过程
ViewRootImpl.java中的方法measureHierarchy()
private boolean measureHierarchy(final View host, final WindowManager.LayoutParams lp,
final Resources res, final int desiredWindowWidth, final int desiredWindowHeight) {
childWidthMeasureSpec = getRootMeasureSpec(desiredWindowWidth, lp.width);
childHeightMeasureSpec = getRootMeasureSpec(desiredWindowHeight, lp.height);
}
private static int getRootMeasureSpec(int windowSize, int rootDimension) {
int measureSpec;
switch (rootDimension) {
case ViewGroup.LayoutParams.MATCH_PARENT:
// Window can't resize. Force root view to be windowSize.
measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.EXACTLY);
break;
case ViewGroup.LayoutParams.WRAP_CONTENT:
// Window can resize. Set max size for root view.
measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.AT_MOST);
break;
default:
// Window wants to be an exact size. Force root view to be that size.
measureSpec = MeasureSpec.makeMeasureSpec(rootDimension, MeasureSpec.EXACTLY);
break;
}
return measureSpec;
}
布局流程-layout
布局完成后会固定View的坐标及实际的宽高。
布局的结果可通过getLeft(),getTop(),getRight(), getBottom()获得布局的的左上右下的坐标。
getWidth(),getHeight()获取View的实际宽高值。
绘制流程-draw
Demo:滑块随着手指移动