Android Touch事件传递机制 二:单纯的(伪生命周期)

转载于:http://blog.csdn.net/yuanzeyao/article/details/38025165

在前一篇文章中,我主要讲解了Android源码中的Touch事件的传递过程,现在我想使用一个demo以及一个实例来学习一下Andorid中的Touch事件处理过程。

在Android系统中,和Touch事件分发和处理紧密相关的三个函数如下:
(1) public boolean dispatchTouchEvent(MotionEvent ev)
(2) public boolean onInterceptTouchEvent(MotionEvent ev)
(3) public boolean onTouchEvent(MotionEvent event)

这三个方法我在前一篇文章中都对他们的源码进行了分析:方法1主要是对Touch事件进行分发,方法2主要是对Touch事件进行拦截,方法3是对Touch事件进行处理

这三个方法主要存在于ViewGroup,View,Activity中,具体情况如下图:

  ViewGroup View  Activity
dispatchTouchEvent
onInterceptTouchEvent
onTouchEvent

下面我们就使用一个demo来看看这些方法的执行流程:
自定义一个类:MyLayoutFirst.java

  1. public class MyLayoutFirst extends LinearLayout
  2. {
  3. private static final String TAG = "MyLayoutFirst";
  4. public MyLayoutFirst(Context context, AttributeSet attrs)
  5. {
  6. super(context, attrs);
  7. }
  8. @Override
  9. public boolean onInterceptTouchEvent(MotionEvent ev)
  10. {
  11. Log.w("yzy", "MyLayoutFirst->onInterceptTouchEvent->"+MyUtils.getActionName(ev));
  12. return super.onInterceptTouchEvent(ev);
  13. }
  14. @Override
  15. public boolean onTouchEvent(MotionEvent event)
  16. {
  17. Log.e("yzy", "MyLayoutFirst->onTouchEvent->"+MyUtils.getActionName(event));
  18. return super.onTouchEvent(event);
  19. }
  20. @Override
  21. public boolean dispatchTouchEvent(MotionEvent ev)
  22. {
  23. Log.i("yzy", "MyLayoutFirst->dispatchTouchEvent->"+MyUtils.getActionName(ev));
  24. return super.dispatchTouchEvent(ev);
  25. }
  26. }

自定义一个类;MyLayoutSecond.java

  1. public class MyLayoutSecond extends LinearLayout
  2. {
  3. private static final String TAG = "MyLayoutSecond";
  4. public MyLayoutSecond(Context context, AttributeSet attrs)
  5. {
  6. super(context, attrs);
  7. }
  8. @Override
  9. public boolean onTouchEvent(MotionEvent event)
  10. {
  11. Log.e("yzy", "MyLayoutSecond->MyLayoutSecond->"+MyUtils.getActionName(event));
  12. return super.onTouchEvent(event);
  13. }
  14. @Override
  15. public boolean onInterceptTouchEvent(MotionEvent ev)
  16. {
  17. Log.w("yzy", "MyLayoutSecond->onInterceptTouchEvent->"+MyUtils.getActionName(ev));
  18. return super.onInterceptTouchEvent(ev);
  19. }
  20. @Override
  21. public boolean dispatchTouchEvent(MotionEvent ev)
  22. {
  23. Log.i("yzy", "MyLayoutSecond->dispatchTouchEvent->"+MyUtils.getActionName(ev));
  24. return super.dispatchTouchEvent(ev);
  25. }
  26. }

加入到main_layout.xml中

  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. xmlns:tools="http://schemas.android.com/tools"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. >
  6. <com.event.demo.MyLayoutFirst
  7. android:id="@+id/layout_first"
  8. android:layout_width="match_parent"
  9. android:layout_height="match_parent"
  10. android:background="#FF0000"
  11. >
  12. <com.event.demo.MyLayoutSecond
  13. android:id="@+id/layout_second"
  14. android:layout_width="320dip"
  15. android:layout_height="120dip"
  16. android:layout_gravity="center"
  17. android:background="#0000FF"
  18. >
  19. </com.event.demo.MyLayoutSecond>
  20. </com.event.demo.MyLayoutFirst>
  21. </RelativeLayout>

MainActivity中加入onTouchEvent方法

  1. public class MainActivity extends Activity
  2. {
  3. @Override
  4. protected void onCreate(Bundle savedInstanceState)
  5. {
  6. super.onCreate(savedInstanceState);
  7. setContentView(R.layout.activity_main);
  8. }
  9. @Override
  10. public boolean onCreateOptionsMenu(Menu menu)
  11. {
  12. // Inflate the menu; this adds items to the action bar if it is present.
  13. getMenuInflater().inflate(R.menu.main, menu);
  14. return true;
  15. }
  16. @Override
  17. public boolean dispatchTouchEvent(MotionEvent ev)
  18. {
  19. Log.i("yzy", "MainActivity->dispatchTouchEvent->"+MyUtils.getActionName(ev));
  20. return super.dispatchTouchEvent(ev);
  21. }
  22. @Override
  23. public boolean onTouchEvent(MotionEvent event)
  24. {
  25. Log.e("yzy", "MainActivity->onTouchEvent->"+MyUtils.getActionName(event));
  26. return super.onTouchEvent(event);
  27. }
  28. }

最后就一个工具类,用来将事件id转换为字符串。

  1. public class MyUtils
  2. {
  3. private static final String TAG = "MyUtils";
  4. public static String getActionName(MotionEvent event)
  5. {
  6. String name="";
  7. switch(event.getAction())
  8. {
  9. case MotionEvent.ACTION_DOWN:
  10. name="ACTION_DOWN";
  11. break;
  12. case MotionEvent.ACTION_MOVE:
  13. name="ACTION_MOVE";
  14. break;
  15. case MotionEvent.ACTION_UP:
  16. name="ACTION_UP";
  17. break;
  18. }
  19. return name;
  20. }
  21. }

运行效果如图:

第一中情况:

 
MainActivity


MyLayoutFirst


MyLayoutSecond


dispatchTouchEvent


super.dispatchTouchEvent


super.dispatchTouchEvent


super.dispatchTouchEvent


onInterceptTouchEvent


--


super.onInterceptTouchEvent(ev)


super.onInterceptTouchEvent(ev)


onTouchEvent


super.onTouchEvent


super.onTouchEven


super.onTouchEvent

运行结果:

其中蓝色部分是MyLayoutSecond.Java ,红色部分是MyLayoutFirst.java
现在我点击一下蓝色部分:运行结果如图:

从图中可以看出,事件最先被Activity捕获,然后分发给 MyLayoutFirst,MyLayoutFirst首先调用自身的onInterceptTouchEvent判断是否将该事件拦截,由于默认返回是false,所以没有拦截,从而事件分发给了MyLayoutSecond,MyLayoutSecond同样通过dispatchTouchEvent分发出去,分发出去之前同样检查是否被拦截,默认都是没有被拦截的,但是由于MyLayoutSecond是没有子视图的,所有最终事件有自己处理,调用自身的onTouchEvent方法,由于该方法默认返回的是false,所以认为此事件是没有被消费掉的,继续传递到了MyLayoutFirst中,同样也没有消费这个事件,最终传递到了Mainactivity,继续往后看发现后面的ACTION_MOVE和ACTION_UP并没有传入MyLayoutFirst和MyLayoutSecond,这是因为一旦某一个事件没有被处理,后面的事件是不会被分发的。所以ACTION_MOVE和ACTION_UP直接被MainActivity处理掉了。

下面再看第二种情况:

 
MainActivity


MyLayoutFirst


MyLayoutSecond


dispatchTouchEvent


super.dispatchTouchEvent


super.dispatchTouchEvent


super.dispatchTouchEvent


onInterceptTouchEvent


--


true


super.onInterceptTouchEvent(ev)


onTouchEvent


super.onTouchEvent


super.onTouchEvent


super.onTouchEvent

运行结果如下:

从图中可以看出,事件传递到了MyLayoutFirst后没有分发到MyLayoutSecond,直接调用自身的onTouchEvent,由于返回的是false,导致事件没有消费,最终传递给了MainActivity,
而且后续事件也没有传递到MyLayoutFirst和MyLayoutSecond,直接被MainActivity处理

第三种情况:

 
MainActivity


MyLayoutFirst


MyLayoutSecond


dispatchTouchEvent


super.dispatchTouchEvent


super.dispatchTouchEvent


super.dispatchTouchEvent


onInterceptTouchEvent


--


true


super.onInterceptTouchEvent(ev)


onTouchEvent


super.onTouchEvent


true


super.onTouchEvent

运行结果:

和情况二不同的是MyLayoutFirst的onTouchEvent返回了true,也就是说MyLayoutFirst消费了此事件,所以ACTION_DOWN也没有再传给MainActivity,并且ACTION_MOVE和ACTION_UP
均传给了MyLayoutFirst

第四中情况:

 
MainActivity


MyLayoutFirst


MyLayoutSecond


dispatchTouchEvent


super.dispatchTouchEvent


super.dispatchTouchEvent


super.dispatchTouchEvent


onInterceptTouchEvent


--


super.onInterceptTouchEvent(ev)


super.onInterceptTouchEvent(ev)


onTouchEvent


super.onTouchEvent


super.onTouchEven


true

运行结果:

发现所有的事件都是传递到了MyLayoutSecond后被消费了

其实还有很多其他组合方式,大家如果又兴趣可以自己尝试改变每个函数的返回值,查看打印结果,这里我就不一一列举了。。。。。

最后我会提供一个小demo演示如何解决滑动冲突,背景如下:
一个ViewPager里面包含两个Framgent,有一个Fragment里面有一个HorizontalListView ,如何滑动冲突?
我就贴出关键代码吧

  1. horizontal=(HorizontalListView)view.findViewById(R.id.hscroll);
  2. horizontal.setOnTouchListener(new OnTouchListener()
  3. {
  4. @Override
  5. public boolean onTouch(View arg0, MotionEvent event)
  6. {
  7. if(event.getAction()==MotionEvent.ACTION_DOWN)
  8. {
  9. parent.requestDisallowInterceptTouchEvent(true);
  10. }else if(event.getAction()==MotionEvent.ACTION_UP)
  11. {
  12. parent.requestDisallowInterceptTouchEvent(false);
  13. }
  14. return false;
  15. }
  16. });

加入这段代码就可以避免滑动冲突了,至于为什么大家可以参考我的前以前文章《Android Touch 事件传递机制详解 上》 这两个demo的例子我均会上传下载的

时间: 2024-07-31 23:14:06

Android Touch事件传递机制 二:单纯的(伪生命周期)的相关文章

Android touch 事件传递机制

前言: (1)在自定义view的时候经常会遇到事件拦截处理,比如在侧滑菜单的时候,我们希望在侧滑菜单里面有listview控件,但是我们希望既能左右滑动又能上下滑动,这个时候就需要对触摸的touch事件进行拦截.这个时候我们就需要明白android touch 事件传递机制, (2)以前很多时候比较模糊,也许是网上看到也有很多事件传递的相关文章,但我看着头晕,解释不彻底,有的说得一半,总算不满足不满意,于是据我自己的理解来彻底的来整理下具体的是怎么个传递方式,分享给大家,希望大家看到有什么不对的

【转】Android Touch事件传递机制解析

原文地址:http://www.cnblogs.com/runssnail/p/4250549.html 说明:本文在原文地址上有所改动 一.小故事 在讲正题之前我们讲一段有关任务传递的小故事,抛砖迎玉下: 话说一家软件公司,来一个任务,分派给了开发经理去完成 开发经理拿到,看了一下,感觉好简单,于是 开发经理:分派给了开发组长 开发组长:分派给了自己组员(程序员) 程序员:分派给了自己带的实习生. 实习生:好苦逼,无法分派,怎么办啊?只能自己干了 但是实习生能不能做好,有两种情况了. 情况一:

Android Touch事件传递机制详解 下

尊重原创:http://blog.csdn.net/yuanzeyao/article/details/38025165 资源下载:http://download.csdn.net/detail/yuanzeyao2008/7660997 在前一篇文章中,我主要讲解了Android源码中的Touch事件的传递过程,现在我想使用一个demo以及一个实例来学习一下Andorid中的Touch事件处理过程. 在Android系统中,和Touch事件分发和处理紧密相关的三个函数如下:(1) public

Android Touch事件传递机制通俗讲解

在讲正题之前我们讲一段有关任务传递的小故事,抛砖迎玉下: 话说一家软件公司,来一个任务,分派给了开发经理去完成: 开发经理拿到,看了一下,感觉好简单,于是 开发经理:分派给了开发组长 开发组长:分派给了自己组员(程序员) 程序员:分派给了自己带的实习生. 实习生:好苦逼,无法分派,怎么办啊?只能自己干了 但是实习生能不能做好,有两种情况了. 情况一: 实习生:经过一段时间的研究,琢磨,熬夜,奋斗,死敲,皇天不负有心人啊,完成了. 后来又来一个类似的任务,也按着这样传递下去了(开发经理->开发组长

Android Touch事件传递机制详解 上

尊重原创:http://blog.csdn.net/yuanzeyao/article/details/37961997 最近总是遇到关于Android Touch事件的问题,如:滑动冲突的问题,以前也花时间学习过Android Touch事件的传递机制,可以每次用起来的时候总是忘记了,索性自己总结一下写篇文章避免以后忘记了,其实网上关于Touch事件的传递的文章真的很多,但是很少有系统性的,都是写了一个简单的demo运行了一下,对于我们了解Android Touch事件基本上没有任何帮助. 今

Android Touch事件传递机制引发的血案

尊重原创:http://blog.csdn.net/yuanzeyao/article/details/38942135 关于Android Touch事件传递机制我之前也写过两篇文章,自觉得对Touche事件还是理解得比較清楚的,可是近期遇到的一个问题,让我再次对Android Touche事件进行一次学习. 我的关于Android Touche事件传递机制的文章例如以下: http://blog.csdn.net/yuanzeyao/article/details/37961997 http

菜鸟进阶之Android Touch事件传递(二)

这是touch事件传递系列博客的第二篇,如果想了解touch和click的那些事,请浏览投产事件传递系列的第一篇.http://blog.csdn.net/bingospunky/article/details/43603397 理理思路,我发现touch传递这部分的内容很多,所以每篇博客介绍一个方面比较好.这篇博客主要介绍touch事件传递的现象,一个简单的demo,让大家可以看到touch一步一步传递的过程.下篇博客中在研究源码是怎么实现的.再后面的博客会试图改变这篇文章看到的touch的传

(转)Android Touch事件传递机制

-----来源:http://www.trinea.cn/android/touch-event-delivery-mechanism/ 介绍Android Touch事件的传递机制. 不少朋友私信问到这个问题,那就推荐一篇我看到的对传递机制介绍最清楚的国外文章吧.本文略作翻译. 1.基础知识 (1) 所有Touch事件都被封装成了MotionEvent对象,包括Touch的位置.时间.历史记录以及第几个手指(多指触摸)等. (2) 事件类型分为ACTION_DOWN, ACTION_UP, A

Android Touch事件传递机制全面解析(从WMS到View树)

转眼间近一年没更新博客了,工作一忙起来.非常难有时间来写博客了,因为如今也在从事Android开发相关的工作,因此以后的博文也会很多其它地专注于这一块. 这篇文章准备从源代码层面为大家带来Touch事件的传递机制.我这里分析的源代码时Android4.4的. 说到分析源代码,光看肯定是不行的,一定要亲自去跟,而且要边跟边思考,所以在下一篇中.会有一个Demo来为大家详细分析源代码的走向. 以下进入正题,先来看下Android中事件的分类: 1.键盘事件:主要是指按下虚拟键盘的某个按键.或者机身的