`
meohao
  • 浏览: 97765 次
  • 性别: Icon_minigender_1
  • 来自: 成都
社区版块
存档分类
最新评论

写在20110915:ViewGroup和View的触屏事件传递

 
阅读更多
原生的Android系统中,主菜单页面是不支持图标的编辑功能的,在Launcher模块中有这样一个需求,需要实现主菜单的编辑功能,也就是要实现主菜单中的应用程序图标能自定义位置,并能自由拖拽,通过点击可以直接卸载某应用程序等等,大家都知道,主菜单中的应用程序图标会响应点击事件,点击应用程序图标之后会进入对应的应用程序,那么要实现主菜单的编辑功能,必须先了解Android事件触发机制,最终实现图标的拖拽和位置的自定义排序。
首先,先了解由于触摸(Touch)而触发的事件。 Android的事件:onClick, onScroll, onFling等等,都是由许多个Touch组成的。但是他们有一些主要的特征,其中Touch的第一个状态肯定是ACTION_DOWN, 表示按下了屏幕。之后,touch将会有后续事件,可能是:
ACTION_MOVE //表示为移动手势
ACTION_UP //表示为离开屏幕
ACTION_CANCEL //表示取消手势,不会由用户产生,而是由程序产生的
一个Action_DOWN, n个ACTION_MOVE, 1个ACTION_UP,就构成了Android中众多的事件。
在Android中,有一类控件是中还可以包含其他的子控件,这类控件是继承于ViewGroup类,例如:ListView, Gallery, GridView。 还有一类控件是不能再包含子控件,例如:TextView。
本文的主要讨论对象就是ViewGroup类的控件嵌套时事件触发情况。
对于ViewGroup类的控件,有一个很重要的方法,就是onInterceptTouchEvent(),用于处理事件并改变事件的传递方向,它的返回值是一个布尔值,决定了Touch事件是否要向它包含的子View继续传递,这个方法是从父View向子View传递。
而方法onTouchEvent(),用于接收事件并处理,它的返回值也是一个布尔值,决定了事件及后续事件是否继续向上传递,这个方法是从子View向父View传递。
touch事件在 onInterceptTouchEvent()和onTouchEvent以及各个childView间的传递机制完全取决于onInterceptTouchEvent()和onTouchEvent()的返回值。返回值为true表示事件被正确接收和处理了,返回值为false表示事件没有被处理,将继续传递下去。
ACTION_DOWN事件会传到某个ViewGroup类的onInterceptTouchEvent,如果返回false,则DOWN事件继续向子ViewGroup类的onInterceptTouchEvent传递,如果子View不是ViewGroup类的控件,则传递给它的onTouchEvent。 如果onInterceptTouchEvent返回了true,则DOWN事件传递给它的onTouchEvent,不再继续传递,并且之后的后续事件也都传递给它的onTouchEvent。如果某View的onTouchEvent返回了false,则DOWN事件继续向其父ViewGroup类的onTouchEvent传递;如果返回了true,则后续事件会直接传递给其onTouchEvent继续处理。(后续事件只会传递给对于必要事件ACTION_DOWN返回了true的onTouchEvent)
总结一下就是:onInterceptTouchEvent可以接受到所有的Touch事件,而onTouchEvent则不一定。
    上面的描述可能看得有些糊涂了,下面再仔细看看整个MotionEvent事件在onInterceptTouchEvent()、onTouchEvent()中的传递顺序 。
    onInterceptTouchEvent()用于处理事件并改变事件的传递方向。处理事件这个不用说了,你在函数内部编写代码处理就可以了。而决定传递方向的是返回值,返回为false时事件会传递给子控件的onInterceptTouchEvent();返回值为true时事件会传递给当前控件的onTouchEvent(),而不在传递给子控件,这就是所谓的Intercept(截断)。
onTouchEvent() 用于处理事件,返回值决定当前控件是否消费(consume)了这个事件。可能你要问是否消费了又区别吗,反正我已经针对事件编写了处理代码?答案是有区别!比如ACTION_MOVE或者ACTION_UP发生的前提是一定曾经发生了ACTION_DOWN,如果你没有消费ACTION_DOWN,那么系统会认为ACTION_DOWN没有发生过,所以ACTION_MOVE或者ACTION_UP就不能被捕获。
看看下面的实例就更加清楚了。
<?xml version="1.0" encoding="utf-8"?>
<com.touchstudy.LayoutView1 xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent" >
    <com.touchstudy.LayoutView2
        android:orientation="vertical"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        android:gravity="center">
       <com.touchstudy.MyTextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:id="@+id/tv"
            android:text="TEST"
            android:textSize="40sp"
            android:textStyle="bold"
            android:background="#FFFFFF"
            android:textColor="#0000FF"/>
   </com.touchstudy.LayoutView2>
</com.touchstudy.LayoutView1>
在没有重写onInterceptTouchEvent()和onTouchEvent()的情况下(他们的返回值都是false), 对上面这个布局,MotionEvent事件的传递顺序如下:

LayoutView1(onInterceptTouchEvent())->LayoutView2(onInterceptTouchEvent())->MyTextView(onTouchEvent())->LayoutView2(onTouchEvent())->LayoutView1(onTouchEvent())

当某个控件的onInterceptTouchEvent()返回值为true时,就会发生截断,事件被传到当前控件的onTouchEvent()。如我们将LayoutView2的onInterceptTouchEvent()返回值为true,则传递流程变成:
LayoutView1(onInterceptTouchEvent())->LayoutView2(onInterceptTouchEvent())->LayoutView2(onTouchEvent())->LayoutView1(onTouchEvent())


如果我们同时将LayoutView2的onInterceptTouchEvent()和onTouchEvent()设置成true,那么LayoutView2将消费被传递的事件,同时后续事件(如跟着ACTION_DOWN的ACTION_MOVE或者ACTION_UP)会直接传给LayoutView2的onTouchEvent(),不传给其他任何控件的任何函数。同时传递给子控件一个ACTION_CANCEL事件。传递流程变成(图中没有画出ACTION_CANCEL事件):
LayoutView1(onInterceptTouchEvent())->LayoutView2(onInterceptTouchEvent())->LayoutView2(onTouchEvent())
           
由于onInterceptTouchEvent()的机制比较复杂,上面的说明写的也比较复杂,总结一下,基本的规则是:
1.  down事件首先会传递到onInterceptTouchEvent()方法
2.  如果该ViewGroup的onInterceptTouchEvent()在接收到down事件处理完成之后return false,那么后续的move, up等事件将继续会先传递给该ViewGroup,之后才和down事件一样传递给最终的目标view的onTouchEvent()处理。
3.  如果该ViewGroup的onInterceptTouchEvent()在接收到down事件处理完成之后return true,那么后续的move, up等事件将不再传递给onInterceptTouchEvent(),而是和down事件一样传递给该ViewGroup的onTouchEvent()处理,注意,目标view将接收不到任何事件。
4.  如果最终需要处理事件的view的onTouchEvent()返回了false,那么该事件将被传递至其上一层次的view的onTouchEvent()处理。
5.  如果最终需要处理事件的view 的onTouchEvent()返回了true,那么后续事件将可以继续传递给该view的onTouchEvent()处理。
下面附上该事件传递流程的源代码,源代码其实很简单,如下。
主Activity: InterceptTouchStudyActivity.java:
public class InterceptTouchStudyActivity extends Activity {
    static final String TAG = "ITSActivity";
    TextView tv;
 
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.layers_touch_pass_test);
     }
}

LayoutView1.java:
      public class LayoutView1 extends LinearLayout {
    private final String TAG = "LayoutView1";
      public LayoutView1(Context context, AttributeSet attrs) {
         super(context, attrs);
         Log.d(TAG,TAG);
     }

     @Override
     public boolean onInterceptTouchEvent(MotionEvent ev) {
         int action = ev.getAction();
         switch(action){
         case MotionEvent.ACTION_DOWN:
              Log.d(TAG,"onInterceptTouchEvent action:ACTION_DOWN");
//            return true;
              break;
         case MotionEvent.ACTION_MOVE:
              Log.d(TAG,"onInterceptTouchEvent action:ACTION_MOVE");
              break;
         case MotionEvent.ACTION_UP:
              Log.d(TAG,"onInterceptTouchEvent action:ACTION_UP");
              break;
         case MotionEvent.ACTION_CANCEL:
              Log.d(TAG,"onInterceptTouchEvent action:ACTION_CANCEL");
              break;
         }
         return false;
     }

     @Override
     public boolean onTouchEvent(MotionEvent ev) {
         int action = ev.getAction();
         switch(action){
         case MotionEvent.ACTION_DOWN:
              Log.d(TAG,"onTouchEvent action:ACTION_DOWN");
              break;
         case MotionEvent.ACTION_MOVE:
              Log.d(TAG,"onTouchEvent action:ACTION_MOVE");
              break;
         case MotionEvent.ACTION_UP:
              Log.d(TAG,"onTouchEvent action:ACTION_UP");
              break;
         case MotionEvent.ACTION_CANCEL:
              Log.d(TAG,"onTouchEvent action:ACTION_CANCEL");
              break;
         }
         return true;
     }

     @Override
     protected void onLayout(boolean changed, int l, int t, int r, int b) {
         // TODO Auto-generated method stub
         super.onLayout(changed, l, t, r, b);
     }

     @Override
     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
         // TODO Auto-generated method stub
         super.onMeasure(widthMeasureSpec, heightMeasureSpec);
     }
}

LayoutView2.java:
public class LayoutView2 extends LinearLayout {
    private final String TAG = "LayoutView2";
    public LayoutView2(Context context, AttributeSet attrs) {
       super(context, attrs);
       Log.d(TAG,TAG);
}

    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
       int action = ev.getAction();
       switch(action){
       case MotionEvent.ACTION_DOWN:
           Log.d(TAG,"onInterceptTouchEvent action:ACTION_DOWN");
           break;
       case MotionEvent.ACTION_MOVE:
           Log.d(TAG,"onInterceptTouchEvent action:ACTION_MOVE");
           break;
       case MotionEvent.ACTION_UP:
           Log.d(TAG,"onInterceptTouchEvent action:ACTION_UP");
           break;
       case MotionEvent.ACTION_CANCEL:
           Log.d(TAG,"onInterceptTouchEvent action:ACTION_CANCEL");
           break;
       }
       return false;
    }

    @Override
    public boolean onTouchEvent(MotionEvent ev) {
       int action = ev.getAction();
       switch(action){
       case MotionEvent.ACTION_DOWN:
           Log.d(TAG,"onTouchEvent action:ACTION_DOWN");
           break;
       case MotionEvent.ACTION_MOVE:
           Log.d(TAG,"onTouchEvent action:ACTION_MOVE");
           break;
       case MotionEvent.ACTION_UP:
           Log.d(TAG,"onTouchEvent action:ACTION_UP");
           break;
       case MotionEvent.ACTION_CANCEL:
           Log.d(TAG,"onTouchEvent action:ACTION_CANCEL");
           break;
       }
       return true;
    }
}

MyTextView.java:
public class MyTextView extends TextView {
    private final String TAG = "MyTextView";
    public MyTextView(Context context, AttributeSet attrs) {
       super(context, attrs);
       Log.d(TAG,TAG);
    }
    @Override
    public boolean onTouchEvent(MotionEvent ev) {
       int action = ev.getAction();
       switch(action){
       case MotionEvent.ACTION_DOWN:
           Log.d(TAG,"onTouchEvent action:ACTION_DOWN");
           break;
       case MotionEvent.ACTION_MOVE:
           Log.d(TAG,"onTouchEvent action:ACTION_MOVE");
           break;
       case MotionEvent.ACTION_UP:
           Log.d(TAG,"onTouchEvent action:ACTION_UP");
           break;
       case MotionEvent.ACTION_CANCEL:
           Log.d(TAG,"onTouchEvent action:ACTION_CANCEL");
           break;
       }
       return false;
    }
    public void onClick(View v) {
       Log.d(TAG, "onClick");
    }
    public boolean onLongClick(View v) {
       Log.d(TAG, "onLongClick");
       return false;
    }
}
分享到:
评论

相关推荐

    Android事件处理机制ViewGroup的事件传递

    Android事件处理机制主要涉及三类对象:事件(Event)、事件源(Event Source)和事件监听器(EventListener)。在本主题中,我们将深入探讨`ViewGroup`的事件传递,即事件分发过程。 1. **事件模型** Android事件...

    android触屏事件之activity,view,viewgroup

    在Android开发中,触屏事件处理是用户交互的核心部分,涉及到Activity、View和ViewGroup这三个关键组件。本文将深入探讨这些组件在触屏事件处理中的角色和机制。 首先,我们来理解Activity。Activity是Android应用...

    View事件传递机制Demo源码

    在Android开发中,View事件传递机制是理解和优化用户交互界面不可或缺的部分。本Demo源码着重展示了这一机制,旨在帮助开发者深入理解并应用到实际项目中。以下是对这一主题的详细阐述。 首先,我们要知道Android中...

    事件传递机制Demo

    在Android开发中,事件传递机制是用户界面交互的核心部分,它决定了用户在屏幕上触摸或点击时,事件如何在各个View之间进行分发。本Demo,"事件传递机制Demo",着重展示了这一机制的工作原理和常见应用场景。我们...

    事件传递机制和原理

    ### 事件传递机制和原理 在Android开发过程中,理解事件传递机制对于实现高效的用户交互界面至关重要。本篇文章基于一个关于“事件传递机制和原理”的流程图进行深入解析,旨在帮助开发者更好地掌握事件传递的核心...

    Android点击事件传递分析Demo

    在"TouchEventDemo"中,我们可以通过创建自定义View和ViewGroup来观察并控制事件的传递。通常,`onTouchEvent()`方法用于处理事件,当返回true表示事件已被当前View消费,否则事件将继续传递给父View或下一级子View...

    Android事件传递机制

    在Android UI体系中,所有的视图(View)和布局(ViewGroup)构成了事件处理的基础。View是基本的交互元素,而ViewGroup是用于组织和布局多个View的容器。Button、TextView等常见控件继承自View,RelativeLayout、...

    Android事件传递机制Demo.zip

    2. **事件分发**:事件分发主要涉及`dispatchTouchEvent()`、`onInterceptTouchEvent()`和`onTouchEvent()`这三个方法。 - **dispatchTouchEvent()**:Activity或ViewGroup接收到事件后,首先调用此方法进行分发。...

    Android Touch事件传递机制解析

    在Android系统中,触摸事件(Touch Event)是用户与设备交互的重要方式,它涉及到了UI控件的响应和处理。本文将深入解析Android的Touch事件传递机制,帮助开发者理解这一核心概念。 首先,我们了解下Android中的...

    View的事件分发测试例子

    在TestEventDemo这个测试项目中,我们可以通过设置断点和打印日志来观察事件在View和ViewGroup之间的传递路径。通常,我们会创建一个自定义的`ViewGroup`和`View`,重写`onInterceptTouchEvent()`和`onTouchEvent()`...

    Android中父View和子view的点击事件

    本文将深入探讨Android中父View( ViewGroup)与子View的点击事件处理机制,以及如何在实际应用中进行有效管理。 一、点击事件的基础知识 在Android中,点击事件是由MotionEvent类来表示的,主要包括ACTION_DOWN、...

    Android TouchEvent事件传递

    在Android开发中,触摸事件(TouchEvent)是用户与设备交互的主要方式之一,它涉及到Activity、View及ViewGroup之间的事件分发。理解Android TouchEvent事件传递机制对于构建具有良好用户体验的应用至关重要。本文将...

    View事件分发

    在Android开发中,View事件分发是用户交互的核心机制,它涉及到触摸事件的捕获、处理和传递。事件分发主要包括三个步骤:dispatchTouchEvent、onTouchEvent和onInterceptTouchEvent。理解这一过程对于优化用户界面和...

    android事件传递测试

    - **Down事件**:事件首先由Activity的Window捕获,然后传递给根布局ViewGroup(如LinearLayout或RelativeLayout),接着沿着Z轴顺序逐级向下分发。 - **ACTION_DOWN**:在ViewGroup中,如果设置了`onTouchEvent()...

    Android中的View与ViewGroup绘制过程,手势监听顺序与使用

    例如,可以自定义一个ViewGroup,使其在特定条件下拦截并处理子View的触摸事件,或者创建一个View,改变默认的触摸反馈行为。 了解了这些基础知识后,我们可以利用它们来实现各种界面效果,如滑动菜单、自定义动画...

    Android onTouch事件传递机制

    - **MOVE事件**:在`ACTION_DOWN`后,所有`ACTION_MOVE`事件都会直接传递给当前拥有焦点的View,即使该View不在触摸点的正下方。 - **UP事件**:最后的`ACTION_UP`事件会沿着与`ACTION_DOWN`相反的方向,从当前...

    android事件传递机制demo代码

    在Android开发中,事件传递机制是用户界面交互的基础,它涉及到Activity、View和ViewGroup之间的协同工作。本示例代码“android事件传递机制demo代码”着重于ViewGroup的事件处理流程,通过`testTouchEvent`这个文件...

    AndroidView的事件分发机制和滑动冲突解决方案.docx

    - **`dispatchTouchEvent(MotionEvent ev)`**:这是事件分发的第一个入口,用于将触摸事件传递给具体的View进行处理。该方法的主要职责是根据当前View的状态决定是否继续传递事件。如果返回`true`,表示当前View已经...

Global site tag (gtag.js) - Google Analytics