最新下载
热门教程
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
Android中在View中如何传递touch事件精讲
时间:2015-09-20 编辑:简简单单 来源:一聚教程网
先来看一张图
版权申明:这是csdn上别人的图,我觉得有用,就拿过来了,
然后简单说明下:
总的来说,触摸事件是从最外层的ViewGroup,一级一级传递进来的
和这相关的每个View的回调方法有三个,dispatchTouchEvent,onInterceptTouchEvent,以及大家都熟悉的ouTouchEvent
事件传递到一个View(ViewGroup),先回调到dispatchTouchEvent()方法,这里如果不super的话,就直接只执行当前代码了,super的话,到onInterceptTouchEvent()方法,这里比较关键,返回false的话,说明不拦截,继续往下传,true的话,说明拦截,直接截断了传递链,然后转到本View的ouTouchEvent方法,整个就结束了。
继续传递的话,如果都没有拦截,就从子View的onTouchEvent方法一直传上来,传到顶层的View,假如传递链中间一个View的onTouchEvent返回true的话,
说明这个View已经处理事件了,就到此为止了,不往上传递了。
Android Touch事件传递机制解析
android系统中的每个View的子类都具有下面三个和TouchEvent处理密切相关的方法:
1)public boolean dispatchTouchEvent(MotionEvent ev) 这个方法用来分发TouchEvent
2)public boolean onInterceptTouchEvent(MotionEvent ev) 这个方法用来拦截TouchEvent
3)public boolean onTouchEvent(MotionEvent ev) 这个方法用来处理TouchEvent
测试程序界面
下述3个Layout包含关系见如下界面图。
状态1:由center处理Touch事件
Xml如下:
注意:只有center这个部分是会处理/消费 Touch事件。
事件传递记录结果如上图。
由于Down、Move、Up事件处理流程略微不同,故分开分析。
ACTION_DOWN事件处理流程:
首先触摸事件发生时(ACTION_DOWN),由系统调用Activity的dispatchTouchEvent方法,分发该事件。根据触摸事件的坐标,将此事件传递给out的dispatchTouchEvent处理,out则调用onInterceptTouchEvent 判断事件是由自己处理,还是继续分发给子View。此处由于out不处理Touch事件,故根据事件发生坐标,将事件传递给out的直接子View(即middle)。
Middle及Center中事件处理过程同上。但是由于Center组件是clickable 表示其能处理Touch事件,故center中的onInterceptTouchEvent方法将事件传递给center自己的onTouchEvent方法处理。至此,此Touch事件已被处理,不继续进行传递。
Move和 up 事件处理流程类似,但是再center内的dispatchTouchEvent方法内被直接分配给onTouchEvent处理,不需经过onInterceptTouchEvent判断。这是由于,android系统中将1个down事件、n个move事件、1个up事件整体作为一次逻辑上的触控操作,Down事件已经确定了处理事件的对象,则后续的move、up事件也确定了处理事件的对象。
状态2:都不处理事件
Xml如下:
轻触center部分logcat输出结果
Action_down事件处理流程:
事件处理流程大致同上,区别是此状态下,所有组件都不会处理事件,事件并不会被center的onTouchEvent方法“消费”,则事件会层层逆向传递回到Activity,若Activity也不对此事件进行处理,此事件相当于消失了(无效果)。
对于后续的move、up事件,由于第一个down事件已经确定由Activity处理事件,故up事有由Activity的dispatchTouchEvent直接分发给自己的onTouchEvent方法处理。
源代码:
package dk.touch; import android.app.Activity; import android.os.Bundle; import android.view.MotionEvent; public class MainActivity extends Activity{ @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); MyLayout out=(MyLayout) findViewById(R.id.out); out.setName("out"); MyLayout middle=(MyLayout) findViewById(R.id.middle); middle.setName("middle"); MyLayout center=(MyLayout) findViewById(R.id.center); center.setName("center"); } @Override public boolean dispatchTouchEvent(MotionEvent ev) { int action=ev.getAction(); String actionName=""; switch(action) { case MotionEvent.ACTION_DOWN: actionName="ACTION_DOWN"; break; case MotionEvent.ACTION_MOVE: actionName="ACTION_MOVE"; break; case MotionEvent.ACTION_UP: actionName="ACTION_UP"; break; } System.out.println("Activity"+"|"+actionName+":dispatchTouchEvent"); return super.dispatchTouchEvent(ev); } @Override public boolean onTouchEvent(MotionEvent event) { int action=event.getAction(); String actionName=""; switch(action) { case MotionEvent.ACTION_DOWN: actionName="ACTION_DOWN"; break; case MotionEvent.ACTION_MOVE: actionName="ACTION_MOVE"; break; case MotionEvent.ACTION_UP: actionName="ACTION_UP"; break; } System.out.println("Activity"+"|"+actionName+":onTouchEvent"); return super.onTouchEvent(event); } } package dk.touch; import android.content.Context; import android.util.AttributeSet; import android.view.MotionEvent; import android.widget.LinearLayout; public class MyLayout extends LinearLayout { private String name=""; public MyLayout(Context context, AttributeSet attrs) { super(context, attrs); } @Override public boolean onTouchEvent(MotionEvent event) { int action=event.getAction(); String actionName=""; switch(action) { case MotionEvent.ACTION_DOWN: actionName="ACTION_DOWN"; break; case MotionEvent.ACTION_MOVE: actionName="ACTION_MOVE"; break; case MotionEvent.ACTION_UP: actionName="ACTION_UP"; break; } System.out.println(name+"|"+actionName+":onTouchEvent"); return super.onTouchEvent(event); } @Override public boolean dispatchTouchEvent(MotionEvent ev) { int action=ev.getAction(); String actionName=""; switch(action) { case MotionEvent.ACTION_DOWN: actionName="ACTION_DOWN"; break; case MotionEvent.ACTION_MOVE: actionName="ACTION_MOVE"; break; case MotionEvent.ACTION_UP: actionName="ACTION_UP"; break; } System.out.println(name+"|"+actionName+":dispatchTouchEvent"); return super.dispatchTouchEvent(ev); } @Override public boolean onInterceptTouchEvent(MotionEvent ev) { int action=ev.getAction(); String actionName=""; switch(action) { case MotionEvent.ACTION_DOWN: actionName="ACTION_DOWN"; break; case MotionEvent.ACTION_MOVE: actionName="ACTION_MOVE"; break; case MotionEvent.ACTION_UP: actionName="ACTION_UP"; break; } System.out.println(name+"|"+actionName+":onInterceptTouchEvent"); return super.onInterceptTouchEvent(ev); } public String getName() { return name; } public void setName(String name) { this.name = name; } }
相关文章
- Android基于hover组件实现监控鼠标移动事件的方法 05-10
- android 向日历中添加事件开发例子 11-18
- Android的View事件传递及传递问题 事件传递机制 08-11
- 安卓开发之Android帧动画的结束事件 07-03
- sora软件价格介绍 02-22
- sora官网入口地址一览 02-22