Android事件分发详解(四)——事件传递基础示例

MainActivity如下:

package com.cn;
import android.app.Activity;
import android.os.Bundle;
import android.view.MotionEvent;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.View.OnTouchListener;
/**
 * Demo描述:
 * Android事件传递基础
 *
 * 背景知识:
 * 事件的传递方向为:从最外层(Activity)传递至最内层(某个View)
*  事件的消费方向为:从最内层(某个View)传递至最外层(Activity)
*
*
 *
 * Demo构成:
 * 1 一个普通的Activity中包含一个自定义Button.
 * 2 覆写Activity的dispatchTouchEvent和onTouchEvent
 * 3 自定义Button中覆写dispatchTouchEvent和onTouchEvent
 *   两者的返回值均为默认值(true).
 * 4 在Activity中为自定义Button注册TouchListener,
 *   覆写里面的onTouch(),该方法默认返回false表示事件未消耗.
 * 5 在Activity中为自定义Button注册ClickListener,
 *
 *
 * 点击屏幕上的Button进行测试
 *
 *
 * Touch事件的处理顺序为:
 * 1 Activity的dispatchTouchEvent()方法
 *   如果它返回false表示不分发事件,则2,3,4不会执行
 * 2 自定义Button的dispatchTouchEvent()方法
 *   如果它返回false表示不分发事件,则3,4不会执行
 * 3 Activity中自定义Button的TouchListener执行
 *   如果它返回true表示事件已经消耗,则4不会执行
 * 4 自定义Button的onTouchEvent()方法
 *   如果它返回false(事件未被消费)即返回给自定义Button的dispatchTouchEvent()方法一个false.
 *   所以自定义Button会中断事件的分发,即每次只分发ACTION_DOWN事件,不分发ACTION_MOVE和ACTION_UP.
 *   所以,每次自定义Button只会响应ACTION_DOWN,不会收到ACTION_MOVE和ACTION_UP.
 *   也正因为这些事件没有被消耗,所以事件会向上传递,于是Activity的onTouchEvent()方法
 *   可以响应到一系列的ACTION_DOWN和ACTION_MOVE以及ACTION_UP.
 *   如果它返回true(事件被消费),那么事件就不会向上传递给Activity了.
 *   Activity自然也收到不到一系列的ACTION_DOWN和ACTION_MOVE以及ACTION_UP.
 *
 *   小结:
 *   (1) 每一次的ACTION_DOWN和ACTION_MOVE以及ACTION_UP都会引起每一层的dispatchTouchEvent()
 *   (2) 但是每一层的onTouchEvent()就不一定会执行了.比如下层已经消耗掉了事件,那么上层就不会响应onTouchEvent()了.
 *
 * 5 自定义Button的ClickListener中的onClick()方法
 *
 *
 * 小结:
 * 1 因为自定义Button的onTouchEvent默认返回为true.
 *   表示事件已经消耗,不会继续向上层传递,所以Activity中的onTouchEvent方法不会调用
 * 2 再次证明:
 *   View中onTouch()先于onTouchEvent()执行
 *   View中onTouchEvent()先于onClick()执行,或者说在onTouchEvent()中调用了onClick()
 * 3 1和2代表的是:事件的传递方向-->从最外层(Activity)传递至最内层(某个View)
 *
 *
 *
 *
 *
 * 常量对应:
 * ACTION_DOWN----->0
 * ACTION_UP------->1
 * ACTION_MOVE----->2
 *
 * 参考资料:
 * 1 http://blog.csdn.net/bigconvience/article/details/26391743
 * 2 http://blog.csdn.net/siobhan/article/details/8257334
 * 3 http://blog.csdn.net/drunkcello/article/details/39892761
 * 4 http://blog.csdn.net/yuanzeyao/article/details/37961997
 * 5 http://blog.csdn.net/lmj623565791/article/details/38960443
 * 6 http://blog.csdn.net/lmj623565791/article/details/39102591
 *   Thank you very much
 *
 */
public class MainActivity extends Activity {
	private EventButton mEventButton;
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.main);
		init();
	}
   private void init(){
	   mEventButton=(EventButton) findViewById(R.id.button);
	   mEventButton.setOnTouchListener(new OnTouchListener() {
		@Override
		public boolean onTouch(View v, MotionEvent event) {
			System.out.println("---> 按钮的OnTouchListener  "+event.getAction());
			return false;
		}
	   });
	   mEventButton.setOnClickListener(new OnClickListener() {
		@Override
		public void onClick(View view) {
			System.out.println("---> 按钮的OnClickListener  点击了Button");
		}
	   });

   }

    @Override
    public boolean dispatchTouchEvent(MotionEvent ev) {
    	System.out.println("---> Activity dispatchTouchEvent "+ev.getAction()+"  , it defaulst is true");
    	return super.dispatchTouchEvent(ev);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
    	System.out.println("---> Activity onTouchEvent "+event.getAction()+"  , it defaulst is false");
    	return super.onTouchEvent(event);
    }

}

EventButton如下:

package com.cn;
import android.content.Context;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.widget.Button;

public class EventButton extends Button {
	public EventButton(Context context, AttributeSet attrs) {
		super(context, attrs);
	}

	@Override
	public boolean dispatchTouchEvent(MotionEvent event) {
		System.out.println("---> 按钮 dispatchTouchEvent , it defaulst is true");
		return super.dispatchTouchEvent(event);
	}

	@Override
	public boolean onTouchEvent(MotionEvent event) {
		System.out.println("---> 按钮 onTouchEvent  , it defaulst is true");
		return super.onTouchEvent(event);
		//return false;
	}
}

main.xml如下:

<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"
    tools:context=".MainActivity" >

    <com.cn.EventButton
        android:id="@+id/button"
        android:layout_width="250dip"
        android:layout_height="250dip"
        android:layout_centerInParent="true"
        android:text="Touch me"
       />

</RelativeLayout>

时间: 2024-10-01 04:07:12

Android事件分发详解(四)——事件传递基础示例的相关文章

Android Touch事件分发详解

Android Touch事件分发详解 先说一些基本的知识,方便后面分析源码时能更好理解. - 所有Touch事件都被封装成MotionEvent对象,包括Touch的位置.历史记录.第几个手指等. 事件类型分为ACTION_DOWN,ACTION_UP,ACTION_MOVE,ACTION_POINTER_DOWN,ACTION_POINTER_UP,ACTION_CANCEL, 每个 一个完整的事件以ACTION_DOWN开始ACTION_UP结束,并且ACTION_CANCEL只能由代码引

Android事件分发详解(一)——View的事件分发

MainActivity如下: package cc.cv; import android.os.Bundle; import android.view.MotionEvent; import android.view.View; import android.view.View.OnClickListener; import android.view.View.OnTouchListener; import android.widget.Button; import android.widge

Android事件分发详解(五)——Touch事件传递验证

MainActivity如下: package cn.c; import android.os.Bundle; import android.app.Activity; import android.view.MotionEvent; /** * Demo描述: * 分析Android事件分发和处理机制 * * * 总结: * 1 ViewGroup继承自View * 事件的传递方向为:从最外层(Activity)传递至最内层(某个View) * 事件的消费方向为:从最内层(某个View)传递至

Android系统输入事件分发详解

什么是输入事件? 我们知道,运行android系统的设备本质上是一台计算机,使用者在和计算机进行交互的时候可以抽象成简单的对计算机的输入和输出(IO).那么对于运行在计算机上的操作系统来说,操作系统在与使用者进行交互的时候起始也是可以抽象成对外界的输入进行处理,然后在输出返还给使用者.本文只讨论的是android系统中的"输入事件"(因为本文讨论的都是输入事件,所以以下简称"输入事件"为"事件")相关的内容. 根据以上描述,我们就可以回答上面的问

Android事件分发详解(三)——ViewGroup的dispatchTouchEvent()源码学习

package cc.aa; import android.os.Environment; import android.view.MotionEvent; import android.view.View; public class UnderstandDispatchTouchEvent { /** * dispatchTouchEvent()源码学习及其注释 * 常说事件传递中的流程是:dispatchTouchEvent->onInterceptTouchEvent->onTouchE

Android事件分发详解(六)——ACTION_DOWN的消费验证

MainActivity如下: package cn.c; import android.os.Bundle; import android.app.Activity; import android.view.MotionEvent; /** * Demo描述: * 分析Android事件分发和处理机制 * * 在该示例中涉及到三个自定义的View.分别是: * 最外层的布局MyFrameLayout * 内层的布局MyLinearLayout * 最里层的自定义按钮MyButton * * 在

Android清单文件详解(四) ---- backupAgent的用法

在<application>节点中有一个非常重要的属性,那就是backupAgent.这里我们将它单独列出来,从基本含义,用法及其相关属性等方面来详细介绍一下. 1.backupAgent简介 android:backupAgent用来设置备份代理.对于大部分应用程序来说,都或多或少保存着一些持久性的数据,比如数据库和共享文件,或者有自己的配置信息.为了保证这些数据和配置信息的安全性以及完整性,Android提供了这样一个机制. 我们可以通过这个备份机制来保存配置信息和数据以便为应用程序提供恢

Android ViewGroup触摸事件拦截详解

前言 在自定义ViewGroup中,有时候需要实现触摸事件拦截,比如ListView下拉刷新就是典型的触摸事件拦截的例子.触摸事件拦截就是在触摸事件被parent view拦截,而不会分发给其child,即使触摸发生在该child身上.被拦截的事件会转到parent view的onTouchEvent方法中进行处理.但是这个交互过程还是挺复杂的,有多种情况,今天我们就来分析一下吧.这篇分析文章已经放了一段时间了,如果有任何问题请高人指出. 触摸事件的分发 简单来说触摸事件的分发会经过这么几个顺序

cocos2dx+lua注册事件函数详解 事件

coocs2dx 版本 3.1.1 registerScriptTouchHandler             注册触屏事件 registerScriptTapHandler                  注册点击事件 registerScriptHandler                         注册基本事件 包括 触屏 层的进入 退出 事件 registerScriptKeypadHandler           注册键盘事件 registerScriptAccelera