您的位置:首页 > 移动开发 > Android开发

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

2016-02-21 12:00 531 查看
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>
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: