Android事件分发,拦截,处理机制

Android 中与 Touch 事件相关的方法包括:dispatchTouchEvent(MotionEvent ev)onInterceptTouchEvent(MotionEvent ev)onTouchEvent(MotionEvent ev);能够响应这些方法的控件包括:ViewGroupViewActivity。方法与控件的对应关系如下表所示:

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 的事件分发逻辑如下:

  • 如果 return true,事件会分发给当前 View 并由 dispatchTouchEvent 方法进行消费,同时事件会停止向下传递;
  • 如果 return false,事件分发分为两种情况:
  1. 如果当前 View 获取的事件直接来自 Activity,则会将事件返回给 Activity 的 onTouchEvent 进行消费;
  2. 如果当前 View 获取的事件来自外层父控件,则会将事件返回给父 View 的  onTouchEvent 进行消费。
  • 如果返回系统默认的 super.dispatchTouchEvent(ev),事件会自动的分发给当前 View 的 onInterceptTouchEvent 方法。

▐ 事件拦截:public boolean onInterceptTouchEvent(MotionEvent ev) 

外层 View 的 dispatchTouchEvent(MotionEvent ev) 方法返回系统默认的 super.dispatchTouchEvent(ev) 情况下,事件会自动的分发给当前 View 的 onInterceptTouchEvent 方法。onInterceptTouchEvent 的事件拦截逻辑如下:

  • 如果 onInterceptTouchEvent 返回 true,则表示将事件进行拦截,并将拦截到的事件交由当前 View 的 onTouchEvent 进行处理;
  • 如果 onInterceptTouchEvent 返回 false,则表示将事件放行,当前 View 上的事件会被传递到子 View 上,再由子 View 的 dispatchTouchEvent 来开始这个事件的分发;
  • 如果 onInterceptTouchEvent 返回 super.onInterceptTouchEvent(ev),事件默认会被拦截,并将拦截到的事件交由当前 View 的 onTouchEvent 进行处理。

▐ 事件响应:public boolean onTouchEvent(MotionEvent ev)

在 dispatchTouchEvent 返回 super.dispatchTouchEvent(ev) 并且 onInterceptTouchEvent 返回 true 或返回 super.onInterceptTouchEvent(ev) 的情况下 onTouchEvent 会被调用。onTouchEvent 的事件响应逻辑如下:

  • 如果事件传递到当前 View 的 onTouchEvent 方法,而该方法返回了 false,那么这个事件会从当前 View 向上传递,并且都是由上层 View 的 onTouchEvent 来接收,如果传递到上面的 onTouchEvent 也返回 false,这个事件就会“消失”,而且接收不到下一次事件。
  • 如果返回了 true 则会接收并消费该事件。
  • 如果返回 super.onTouchEvent(ev) 默认处理事件的逻辑和返回 false 时相同。

到这里,与 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)

Android事件分发,拦截,处理机制_第1张图片

view响应所有TouchEven事件

1,groupview.onInterceptTouchEvent返回false&&groupview.onTouchEvent返回true&&view.onTouchEvent返回true

Android事件分发,拦截,处理机制_第2张图片

view处理了所有TouchEvent事件

2,groupview.onInterceptTouchEvent返回false&&groupview.onTouchEvent返回true&&view.onTouchEvent返回false

Android事件分发,拦截,处理机制_第3张图片

view只处理了action_down,groupview处理了所有TouchEvent事件

3,groupview.onInterceptTouchEvent返回true&&groupview.onTouchEvent返回true&&view.onTouchEvent返回false

Android事件分发,拦截,处理机制_第4张图片

groupview处理了所有TouchEvent事件

4,groupview.onInterceptTouchEvent返回true&&groupview.onTouchEvent返回false&&view.onTouchEvent返回false

Android事件分发,拦截,处理机制_第5张图片

groupview只处理了down事件,activity处理了所有事件

你可能感兴趣的:(Android)