View的事件传递和工作中的分配任务类似。
比如说大领导有任务需要人完成,将其分配给了经理,然后经理将任务分发给了底下的员工,员工完成任务后向经理交付任务,最后经理再向领导汇报任务完成。整个过程就类似于事件的传递,我画了一张图来说明。
下面我们就用自定义控件来验证一下。
先创建HighViewGroup(类似于领导),继承LinearLayout,并重写dispatchTouchEvent(),onInterceptTouchEvent(),onTouchEvent()方法。
public class HighViewGroup extends LinearLayout {
public HighViewGroup(Context context) {
super(context);
}
public HighViewGroup(Context context, AttributeSet attrs) {
super(context, attrs);
}
public HighViewGroup(Context context, AttributeSet attrs,
int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
private static final String TAG = "view";
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
Log.d(TAG, "HighViewGroup dispatchTouchEvent: ");
return super.dispatchTouchEvent(ev);
}
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
Log.d(TAG, "HighViewGroup onInterceptTouchEvent: ");
return super.onInterceptTouchEvent(ev);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
Log.d(TAG, "HighViewGroup onTouchEvent: ");
return super.onTouchEvent(event);
}
}
然后创建MiddleViewGroup(类似于经理),继承LinearLayout,并重写dispatchTouchEvent(),onInterceptTouchEvent(),onTouchEvent()方法。
public class MiddleViewGroup extends LinearLayout {
public MiddleViewGroup(Context context) {
super(context);
}
public MiddleViewGroup(Context context, AttributeSet attrs) {
super(context, attrs);
}
public MiddleViewGroup(Context context, AttributeSet attrs,
int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
private static final String TAG = "view";
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
Log.d(TAG, "MiddleViewGroup dispatchTouchEvent: ");
return super.dispatchTouchEvent(ev);
}
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
Log.d(TAG, "MiddleViewGroup onInterceptTouchEvent: ");
return super.onInterceptTouchEvent(ev);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
Log.d(TAG, "MiddleViewGroup onTouchEvent: ");
return super.onTouchEvent(event);
}
}
最后创建BaseView(类似于员工),继承View,并重写dispatchTouchEvent(),onTouchEvent()方法。
public class BaseView extends View {
public BaseView(Context context) {
super(context);
}
public BaseView(Context context, AttributeSet attrs) {
super(context, attrs);
}
public BaseView(Context context, AttributeSet attrs,
int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
private static final String TAG = "view";
@Override
public boolean onTouchEvent(MotionEvent event) {
Log.d(TAG, "BaseView onTouchEvent: ");
return super.onTouchEvent(event);
}
@Override
public boolean dispatchTouchEvent(MotionEvent event) {
Log.d(TAG, "BaseView dispatchTouchEvent: ");
return super.dispatchTouchEvent(event);
}
}
在MainActivity布局文件中使用自定义控件并设置背景色。
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
>
<com.example.other.HighViewGroup
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#ff0000"
>
<com.example.other.MiddleViewGroup
android:layout_width="match_parent"
android:layout_height="200dp"
android:background="#ffff00"
>
<com.example.other.BaseView
android:layout_width="200dp"
android:layout_height="150dp"
android:background="#ff00ff"/>
</com.example.other.MiddleViewGroup>
</com.example.other.HighViewGroup>
</RelativeLayout>
运行程序,界面如下。
然后点击紫色区域,发现log如下:
看的出来,HighViewGroup最先捕捉到了触摸事件,然后将其分发给了MiddleViewGroup,MiddleViewGroup再将其分发给了BaseView。BaseView响应了触摸事件,并将事件返回给了MiddleViewGroup,然后MiddleViewGroup再将事件返回给HighViewGroup流程完成。
onTouchEvent()中的返回值表示是否消费此事件,返回false表示不消费,事件继续往上传递;返回true表示消费此事件,事件则不再往上传递。
我们将BaseView中 onTouchEvent()方法返回true运行程序点击紫色区域发现log如下:
即触摸事件被BaseView消费了,事件没有再往上传递。
然后再将MiddleViewGroup中onInterceptTouchEvent()方法返回true,再次运行程序,点击紫色区域。发现并没有触发BaseView的方法。
onInterceptTouchEvent()方法返回true表示拦截此事件,不会再往下传递,所以也自然不会触发BaseView的方法。就好比大领导把任务给了经理,经理觉得任务很简单,没必要再分配给员工,自己把它完成了。所以就没有员工什么事了。
以上就是View触摸事件的传递,如有错误请指正,谢谢!