Android 中与 Touch 事件相关的方法包括:dispatchTouchEvent(MotionEvent ev)、onInterceptTouchEvent(MotionEvent ev)、onTouchEvent(MotionEvent ev);能够响应这些方法的控件包括:ViewGroup、View、Activity。方法与控件的对应关系如下表所示:
Touch 事件相关方法 | 方法功能 |
ViewGroup |
View |
Activity |
public boolean dispatchTouchEvent(MotionEvent ev) | 事件分发 |
Yes | Yes | Yes |
public boolean onInterceptTouchEvent(MotionEvent ev) |
事件拦截 |
Yes | No | No |
public boolean onTouchEvent(MotionEvent ev) | 事件响应 |
Yes | Yes | Yes |
1)public boolean dispatchTouchEvent(MotionEvent ev) 这个方法用来分发TouchEvent
2)public boolean onInterceptTouchEvent(MotionEvent ev) 这个方法用来拦截TouchEvent
3)public boolean onTouchEvent(MotionEvent ev) 这个方法用来处理TouchEvent
当TouchEvent发生时,首先Activity将TouchEvent传递给最顶层的View, TouchEvent最先到达最顶层 view 的 dispatchTouchEvent ,然后由 dispatchTouchEvent 方法进行分发,如果dispatchTouchEvent返回true ,则交给这个view的onTouchEvent处理,如果dispatchTouchEvent返回 false ,则交给这个 view 的 interceptTouchEvent 方法来决定是否要拦截这个事件,如果 interceptTouchEvent 返回 true ,也就是拦截掉了,则交给它的 onTouchEvent 来处理,如果 interceptTouchEvent 返回 false ,那么就传递给子 view ,由子 view 的 dispatchTouchEvent 再来开始这个事件的分发。如果事件传递到某一层的子 view 的 onTouchEvent 上了,这个方法返回了 false ,那么这个事件会从这个 view 往上传递,都是 onTouchEvent 来接收。而如果传递到最上面的 onTouchEvent 也返回 false 的话,这个事件就会“消失”,而且接收不到下一次事件。
ViewGroup的相关事件有三个:onInterceptTouchEvent、dispatchTouchEvent、onTouchEvent。Activity和View的相关事件只有两个:dispatchTouchEvent、onTouchEvent。
一、Touch 事件分析
▐ 事件分发:public boolean dispatchTouchEvent(MotionEvent ev)
Touch 事件发生时 Activity 的 dispatchTouchEvent(MotionEvent ev) 方法会以隧道方式(从根元素依次往下传递直到最内层子元素或在中间某一元素中由于某一条件停止传递)将事件传递给最外层 View 的 dispatchTouchEvent(MotionEvent ev) 方法,并由该 View 的 dispatchTouchEvent(MotionEvent ev) 方法对事件进行分发。dispatchTouchEvent 的事件分发逻辑如下:
▐ 事件拦截:public boolean onInterceptTouchEvent(MotionEvent ev)
在外层 View 的 dispatchTouchEvent(MotionEvent ev) 方法返回系统默认的 super.dispatchTouchEvent(ev) 情况下,事件会自动的分发给当前 View 的 onInterceptTouchEvent 方法。onInterceptTouchEvent 的事件拦截逻辑如下:
▐ 事件响应:public boolean onTouchEvent(MotionEvent ev)
在 dispatchTouchEvent 返回 super.dispatchTouchEvent(ev) 并且 onInterceptTouchEvent 返回 true 或返回 super.onInterceptTouchEvent(ev) 的情况下 onTouchEvent 会被调用。onTouchEvent 的事件响应逻辑如下:
到这里,与 Touch 事件相关的三个方法就分析完毕了。下面的内容会通过各种不同的的测试案例来验证上文中三个方法对事件的处理逻辑。
package com.example.ontouch;
import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
public class MainActivity extends Activity {
private MyListView mListView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mListView=(MyListView) findViewById(R.id.list);
mListView.setAdapter(new MyAdapter(this));
mListView.setOnItemClickListener(new OnItemClickListener() {
@Override
public void onItemClick(AdapterView> parent, View view, int position, long id) {
Log.d("ee", "onItemClick:"+view+" "+position+" "+id);
}
});
}
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
int action = ev.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
Log.d("ee", "activit----dispatchTouchEvent-----action:ACTION_DOWN");
break;
case MotionEvent.ACTION_MOVE:
Log.d("ee", "activit----dispatchTouchEvent-----action:ACTION_MOVE");
break;
case MotionEvent.ACTION_UP:
Log.d("ee", "activit----dispatchTouchEvent-----action:ACTION_UP");
break;
case MotionEvent.ACTION_CANCEL:
Log.d("ee", "activit----dispatchTouchEvent-----action:ACTION_CANCEL");
break;
}
return super.dispatchTouchEvent(ev);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
int action = event.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
Log.d("ee", "activit----onTouchEvent-----action:ACTION_DOWN");
break;
case MotionEvent.ACTION_MOVE:
Log.d("ee", "activit----onTouchEvent-----action:ACTION_MOVE");
break;
case MotionEvent.ACTION_UP:
Log.d("ee", "activit----onTouchEvent-----action:ACTION_UP");
break;
case MotionEvent.ACTION_CANCEL:
Log.d("ee", "activit----onTouchEvent-----action:ACTION_CANCEL");
break;
}
return super.onTouchEvent(event);
}
}
package com.example.ontouch;
import android.content.Context;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.widget.ListView;
public class MyListView extends ListView{
public MyListView(Context context) {
super(context);
}
public MyListView(Context context, AttributeSet attrs) {
super(context, attrs);
}
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
int action = ev.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
Log.d("ee", "groupView----dispatchTouchEvent-----action:ACTION_DOWN");
break;
case MotionEvent.ACTION_MOVE:
Log.d("ee", "groupView----dispatchTouchEvent-----action:ACTION_MOVE");
break;
case MotionEvent.ACTION_UP:
Log.d("ee", "groupView----dispatchTouchEvent-----action:ACTION_UP");
break;
case MotionEvent.ACTION_CANCEL:
Log.d("ee", "groupView----dispatchTouchEvent-----action:ACTION_CANCEL");
break;
}
return super.dispatchTouchEvent(ev);
}
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
int action = ev.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
Log.d("ee", "groupView----onInterceptTouchEvent-----action:ACTION_DOWN");
break;
case MotionEvent.ACTION_MOVE:
Log.d("ee", "groupView----onInterceptTouchEvent-----action:ACTION_MOVE");
break;
case MotionEvent.ACTION_UP:
Log.d("ee", "groupView----onInterceptTouchEvent-----action:ACTION_UP");
break;
case MotionEvent.ACTION_CANCEL:
Log.d("ee", "groupView----onInterceptTouchEvent-----action:ACTION_CANCEL");
break;
}
return false;
}
@Override
public boolean onTouchEvent(MotionEvent ev) {
int action = ev.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
Log.d("ee", "groupView----onTouchEvent-----action:ACTION_DOWN");
break;
case MotionEvent.ACTION_MOVE:
Log.d("ee", "groupView----onTouchEvent-----action:ACTION_MOVE");
break;
case MotionEvent.ACTION_UP:
Log.d("ee", "groupView----onTouchEvent-----action:ACTION_UP");
break;
case MotionEvent.ACTION_CANCEL:
Log.d("ee", "groupView----onTouchEvent-----action:ACTION_CANCEL");
break;
}
return true;
}
}
package com.example.ontouch;
import android.content.Context;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.widget.Button;
public class MyButton extends Button{
public MyButton(Context context) {
super(context);
}
public MyButton(Context context, AttributeSet attrs) {
super(context, attrs);
}
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
int action = ev.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
Log.d("ee", "groupView----dispatchTouchEvent-----action:ACTION_DOWN");
break;
case MotionEvent.ACTION_MOVE:
Log.d("ee", "groupView----dispatchTouchEvent-----action:ACTION_MOVE");
break;
case MotionEvent.ACTION_UP:
Log.d("ee", "groupView----dispatchTouchEvent-----action:ACTION_UP");
break;
case MotionEvent.ACTION_CANCEL:
Log.d("ee", "groupView----dispatchTouchEvent-----action:ACTION_CANCEL");
break;
}
return super.dispatchTouchEvent(ev);
}
@Override
public boolean onTouchEvent(MotionEvent ev) {
int action = ev.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
Log.d("ee", "View----onTouchEvent-----action:ACTION_DOWN");
break;
case MotionEvent.ACTION_MOVE:
Log.d("ee", "View----onTouchEvent-----action:ACTION_MOVE");
break;
case MotionEvent.ACTION_UP:
Log.d("ee", "View----onTouchEvent-----action:ACTION_UP");
break;
case MotionEvent.ACTION_CANCEL:
Log.d("ee", "View----onTouchEvent-----action:ACTION_CANCEL");
break;
}
return true;
}
}
点击button响应事件:
0,所有的都返回super.onTouchEvent(event)
view响应所有TouchEven事件
1,groupview.onInterceptTouchEvent返回false&&groupview.onTouchEvent返回true&&view.onTouchEvent返回true
2,groupview.onInterceptTouchEvent返回false&&groupview.onTouchEvent返回true&&view.onTouchEvent返回false
view只处理了action_down,groupview处理了所有TouchEvent事件
3,groupview.onInterceptTouchEvent返回true&&groupview.onTouchEvent返回true&&view.onTouchEvent返回false
groupview处理了所有TouchEvent事件
4,groupview.onInterceptTouchEvent返回true&&groupview.onTouchEvent返回false&&view.onTouchEvent返回false
groupview只处理了down事件,activity处理了所有事件