MainActivity如下:
EventButton如下:
main.xml如下:
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>