Android View 触摸事件传递
2017-11-10 02:22
183 查看
View的事件传递和工作中的分配任务类似。
比如说大领导有任务需要人完成,将其分配给了经理,然后经理将任务分发给了底下的员工,员工完成任务后向经理交付任务,最后经理再向领导汇报任务完成。整个过程就类似于事件的传递,我画了一张图来说明。
下面我们就用自定义控件来验证一下。
先创建HighViewGroup(类似于领导),继承LinearLayout,并重写dispatchTouchEvent(),onInterceptTouchEvent(),onTouchEvent()方法。
然后创建MiddleViewGroup(类似于经理),继承LinearLayout,并重写dispatchTouchEvent(),onInterceptTouchEvent(),onTouchEvent()方法。
最后创建BaseView(类似于员工),继承View,并重写dispatchTouchEvent(),onTouchEvent()方法。
在MainActivity布局文件中使用自定义控件并设置背景色。
看的出来,HighViewGroup最先捕捉到了触摸事件,然后将其分发给了MiddleViewGroup,MiddleViewGroup再将其分发给了BaseView。BaseView响应了触摸事件,并将事件返回给了MiddleViewGroup,然后MiddleViewGroup再将事件返回给HighViewGroup流程完成。
onTouchEvent()中的返回值表示是否消费此事件,返回false表示不消费,事件继续往上传递;返回true表示消费此事件,事件则不再往上传递。
比如说大领导有任务需要人完成,将其分配给了经理,然后经理将任务分发给了底下的员工,员工完成任务后向经理交付任务,最后经理再向领导汇报任务完成。整个过程就类似于事件的传递,我画了一张图来说明。
下面我们就用自定义控件来验证一下。
先创建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触摸事件的传递,如有错误请指正,谢谢!
相关文章推荐
- Android ViewGroup 触摸事件传递机制
- Android View触摸事件传递机制 一
- Android ViewGroup 触摸事件传递机制
- Android触摸事件传递机制及viewpager嵌套fragment冲突处理
- Android触摸事件传递机制实践——可拖动、大小切换的SizeSwitchView
- Android View触摸事件传递机制
- android view触摸事件传递机制测试
- Android View之间的触摸事件传递图
- 【Android View】Android中View对触摸事件的处理和传递dispatchTouchEvent、onInterceptTouchEvent
- Android View触摸事件传递机制
- Android View 触摸事件传递机制
- android触摸事件传递
- Android触摸事件传递机制
- [Android 分享] View 和 ViewGroup 事件传递机制
- Android 触摸事件机制(四) ViewGroup中触摸事件详解
- Android触摸事件传递机制简要分析
- Android ViewGroup中事件触发和传递机制
- Android触摸事件源码分析:Activity->ViewGroup->View
- android中的Touch触摸事件传递机制
- Android对touch事件的拦截,在View Tree上的传递顺序