`
xixinfei
  • 浏览: 410891 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Android中onInterceptTouchEvent与onTouchEvent(图文)!

 
阅读更多

原文

http://blog.csdn.net/android_tutor/article/details/7193090

 

Hi,大家好,今天给大家分享一下Android中onInterceptTouchEvent与onTouchEvent,,记得楼主以前刚开始 找工作的时候,被人问了关于Android中事件传递,当时还是菜鸟一枚,当然答不上来,被人无情的BS了。好了言归正传讲重点.

onInterceptTouchEvent:

onInterceptTouchEvent 是在ViewGroup里面定义的。Android中的layout布局类一般都是继承此类的。onInterceptTouchEvent是用于拦截手 势事件的,每个手势事件都会先调用onInterceptTouchEvent。

onTouchEvent:

onTouchEvent同样也是在view中定义的一个方法。处理传递到view 的手势事件。手势事件类型包括ACTION_DOWN,ACTION_MOVE,ACTION_UP,ACTION_CANCEL等事件。

其 中Layout里的onInterceptTouchEvent默认返回值是false,这样touch事件会传递到View控件,Layout里的 onTouch默认返回值是false, View里的onTouch默认返回值是true,当我们手指点击屏幕时候,先调用ACTION_DOWN事件,当onTouch里返回值是true的时 候,onTouch回继续调用ACTION_UP事件,如果onTouch里返回值是false,那么onTouch只会调用ACTION_DOWN而不 调用ACTION_UP.

为了让当家更容易理解我写了一个简单的Demo.自定义了Layout与View,Android工程目录如下:

新建一个MyLayout.java代码如下:

 

  1. package com.tutor.touch;  
  2.   
  3. import android.content.Context;  
  4. import android.util.AttributeSet;  
  5. import android.util.Log;  
  6. import android.view.MotionEvent;  
  7. import android.widget.FrameLayout;  
  8.   
  9. public class MyLayout extends FrameLayout {  
  10.   
  11.       
  12.     public MyLayout(Context context){  
  13.         super(context);  
  14.     }  
  15.       
  16.     public MyLayout(Context context, AttributeSet attrs) {  
  17.         super(context, attrs);  
  18.         // TODO Auto-generated constructor stub  
  19.     }  
  20.   
  21.     @Override  
  22.     public boolean onInterceptTouchEvent(MotionEvent ev) {  
  23.         Log.e(TouchDemoActivity.TAG, "MyLayout onInterceptTouchEvent.");  
  24.         Log.e(TouchDemoActivity.TAG,"MyLayout onInterceptTouchEvent default return "   
  25.         + super.onInterceptTouchEvent(ev));  
  26.         return super.onInterceptTouchEvent(ev);  
  27.     }  
  28.       
  29.       
  30.       
  31.     @Override  
  32.     public boolean onTouchEvent(MotionEvent event) {  
  33.         Log.e(TouchDemoActivity.TAG, "MyLayout onTouchEvent.");  
  34.         Log.e(TouchDemoActivity.TAG,"MyLayout onTouchEvent default return "   
  35.         + super.onTouchEvent(event));  
  36.         return super.onTouchEvent(event);  
  37.     }  
  38. }  


然后新建一个MyView.java代码如下:

  1. package com.tutor.touch;  
  2.   
  3. import android.content.Context;  
  4. import android.util.AttributeSet;  
  5. import android.util.Log;  
  6. import android.view.MotionEvent;  
  7. import android.widget.Button;  
  8.   
  9. public class MyView extends Button {  
  10.   
  11.     public MyView(Context context){  
  12.         super(context);  
  13.     }  
  14.       
  15.     public MyView(Context context, AttributeSet attrs) {  
  16.         super(context, attrs);  
  17.     }  
  18.       
  19.     @Override  
  20.     public boolean onTouchEvent(MotionEvent event) {  
  21.         Log.e(TouchDemoActivity.TAG, "MyView onTouchEvent.");  
  22.         Log.e(TouchDemoActivity.TAG,"MyView onTouchEvent default return "   
  23.         + super.onTouchEvent(event));  
  24.         return super.onTouchEvent(event);  
  25.     }  
  26.   
  27. }  

 

其中TouchDemoActivity代码如下:

  1. package com.tutor.touch;  
  2.   
  3. import android.app.Activity;  
  4. import android.os.Bundle;  
  5.   
  6. public class TouchDemoActivity extends Activity {  
  7.     public static final String TAG = "TouchDemoActivity";  
  8.     @Override  
  9.     public void onCreate(Bundle savedInstanceState) {  
  10.         super.onCreate(savedInstanceState);  
  11.         setContentView(R.layout.main);  
  12.     }  
  13. }  


上面所有的布局文件main.xml代码如下:

  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <com.tutor.touch.MyLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     android:layout_width="fill_parent"  
  4.     android:layout_height="fill_parent"  
  5.      >  
  6.   
  7.     <com.tutor.touch.MyView  
  8.         android:layout_width="fill_parent"  
  9.         android:layout_height="wrap_content"  
  10.         android:text="@string/hello" />  
  11.   
  12. </com.tutor.touch.MyLayout>  


运行上述Android工程效果如下:

点击红色区域,触发了MyView里的onTouch事件查看logcat,如下图:

点击绿色区域,则触发了MyLayout里的onTouch事件,查看logcat,如下图:

上 面俩个截图都是用系统默认值,可以得出结论:onInterceptTouchEvent默认返回值是false,MyLayout里 onTouchEvent默认返回值是false,所以只消费了ACTION_DOWN事件,MyView里onTouch默认返回值是true,调用了 俩次:ACTION_DOW,ACTION_UP。

下面我们把MyLayout.java里的onInterceptTouchEvent的return值修改为true,代码如下:

  1. @Override  
  2.     public boolean onInterceptTouchEvent(MotionEvent ev) {  
  3.         Log.e(TouchDemoActivity.TAG, "MyLayout onInterceptTouchEvent.");  
  4.         Log.e(TouchDemoActivity.TAG,"MyLayout onInterceptTouchEvent default return "   
  5.         + super.onInterceptTouchEvent(ev));  
  6.         return true;  
  7.     }  


运行工程,继续点击红色区域,查看logcat,发现MyView的onTouch事件没有被调用,也就是被拦截了如下图所示:

让我们继续实验,让onInterceptTouchEvent的返回值继续为false,将MyView里的onTouchEvent的返回值修改为false,即MyView里的onTouchEvent修改如下:

  1. @Override  
  2. public boolean onTouchEvent(MotionEvent event) {  
  3.     Log.e(TouchDemoActivity.TAG, "MyView onTouchEvent.");  
  4.     Log.e(TouchDemoActivity.TAG,"MyView onTouchEvent default return "   
  5.     + super.onTouchEvent(event));  
  6.     return false;  
  7. }  


运行工程,继续点击红色区域,查看logcat,如下图:

根据上图,我们可以看出MyView里的OnTouchEvent只消费了一次点击事件也就是ACTION_DOWN,还没有执行ACTION_UP,然后跑到MyLayout里又去执行了OnTouchEvent事件。

所以根据上面的内容总结如下:

ViewGroup里的onInterceptTouchEvent默认值是false这样才能把事件传给View里的onTouchEvent.

ViewGroup里的onTouchEvent默认值是false。

View里的onTouchEvent返回默认值是true.这样才能执行多次touch事件。

好了以上就是今天分享的内容,谢谢大家!大家有什么不明白的,请留言

!!!另外文章标题俩分钟只是个虚头,如果你超过俩分钟才明白,不要喷我啊,哈哈哈!!!

================

执行顺序是:
Activity.dispatchTouchEvent()
ViewGroup.dispatchTouchEvent()
View.dispatchTouchEvent()
View.onTouchEvent()
ViewGroup.onTouchEvent()
Activity.onTouchEvent()

onInterceptTouchEvent 返回false,则后续再来的事件(比如ACTION_UP)会继续传递给子view的ontouchEvent ,
onInterceptTouchEvent 返回true,则后续再来的事件(比如ACTION_UP)就不会传递给子view.

view的onTouchEvent返回true,则表示事件已经消化干净,viewgroup的onTouchEvent将不会被调用,否则相反.



附上官方对于onInterceptTouchEvent 的解释;
3、For as long as you return false from this function, each following event (up to and including the final up) will be delivered first here and then to the target's onTouchEvent().

4、If you return true from here, you will not receive any following events: the target view will receive the same event but with the action ACTION_CANCEL, and all further events will be delivered to your onTouchEvent() method and no longer appear here.

=======

 

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics