`

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

阅读更多

android中的事件类型分为按键事件和屏幕触摸事件,Touch事件是屏幕触摸事件的基础事件,有必要对它进行深入的了解。 

一个最简单的屏幕触摸动作触发了一系列Touch事件:ACTION_DOWN->ACTION_MOVE->ACTION_MOVE->ACTION_MOVE...->ACTION_MOVE->ACTION_UP

当屏幕中包含一个ViewGroup,而这个ViewGroup又包含一个子view,这个时候android系统如何处理Touch事件呢?到底是ViewGroup来处理Touch事件,还是子view来处理Touch事件呢?我只能很肯定的对你说不一定。呵呵,为什么呢?看看下面我的调查结果你就明白了。

android系统中的每个View的子类都具有下面三个和TouchEvent处理密切相关的方法:

1)public boolean dispatchTouchEvent(MotionEvent ev)  这个方法用来分发TouchEvent

2)public boolean onInterceptTouchEvent(MotionEvent ev) 这个方法用来拦截TouchEvent

3)public boolean onTouchEvent(MotionEvent ev) 这个方法用来处理TouchEvent

 

当TouchEvent发生时,首先Activity将TouchEvent传递给最顶层的View, 

TouchEvent最先到达最顶层 view 的 dispatchTouchEvent ,然后由  dispatchTouchEvent 方法进行分发,

如果dispatchTouchEvent返回true ,则交给这个view的onTouchEvent处理,

如果dispatchTouchEvent返回 false ,则交给这个 view 的 interceptTouchEvent 方法来决定是否要拦截这个事件,

如果 interceptTouchEvent 返回 true ,也就是拦截掉了,则交给它的 onTouchEvent 来处理,

如果 interceptTouchEvent 返回 false ,那么就传递给子 view ,由子 view 的 dispatchTouchEvent 再来开始这个事件的分发。

如果事件传递到某一层的子 view 的 onTouchEvent 上了,这个方法返回了 false ,那么这个事件会从这个 view 往上传递,都是 onTouchEvent 来接收。

而如果传递到最上面的 onTouchEvent 也返回 false 的话,这个事件就会“消失”,而且接收不到下一次事件。

总结:事件是从activity开始传到ViewGroup再传到子View的,一般上层的activity和ViewGrou不会消费掉事件。而是把事件继续传到下一层。当传到最低层时事件还没被消费掉,事件就会开始往回传递,一层一层往上传递。若事件传递到最上层还没被消费掉,则这个View以后也不会再接收事件了。事件具有一次性消费的特性。即事件一旦被某个View处理了(onTouchEvent 返回true),则该事件不会再继续传递。

private LayoutInflater inflater;

Java代码   收藏代码
  1. public View fristView;  
  2. public View secondView;  
  3. private MyViewPager myViewPager;  
  4. public ViewPagerAdapter mViewPagerAdapter;  
  5. private List<View> views;  
  6. public Gallery mGallery;  
  7.    @Override  
  8.    public void onCreate(Bundle savedInstanceState) {  
  9.        super.onCreate(savedInstanceState);  
  10.        setContentView(R.layout.main);  
  11.        inflater = getLayoutInflater();  
  12.        fristView = inflater.inflate(R.layout.main1, null);  
  13.        secondView = inflater.inflate(R.layout.main2, null);  
  14.        views = new ArrayList<View>();  
  15.        views.add(fristView);  
  16.        views.add(secondView);  
  17.        mGallery = (Gallery) fristView.findViewById(R.id.gallery);  
  18.        mGallery.setAdapter(new ImageAdapter(this));  
  19.        myViewPager = (MyViewPager) findViewById(R.id.pager);  
  20.        mViewPagerAdapter = new ViewPagerAdapter(views);  
  21.        myViewPager.setAdapter(mViewPagerAdapter);  
  22.   }  
  23.   
  24.   
  25.   
  26. //界面列表  
  27.    private List<View> views;  
  28.      
  29.    public ViewPagerAdapter (List<View> views){  
  30.        this.views = views;  
  31.    }  
  32.   
  33.    //销毁arg1位置的界面  
  34.    @Override  
  35.    public void destroyItem(View arg0, int arg1, Object arg2) {  
  36.        ((ViewPager) arg0).removeView(views.get(arg1));          
  37.    }  
  38.   
  39.    @Override  
  40.    public void finishUpdate(View arg0) {  
  41.        // TODO Auto-generated method stub  
  42.          
  43.    }  
  44.   
  45.    //获得当前界面数  
  46.    @Override  
  47.    public int getCount() {  
  48.        if (views != null)  
  49.        {  
  50.            return views.size();  
  51.        }  
  52.          
  53.        return 0;  
  54.    }  
  55.      
  56.   
  57.    //初始化arg1位置的界面  
  58.    @Override  
  59.    public Object instantiateItem(View arg0, int arg1) {  
  60.          
  61.        ((ViewPager) arg0).addView(views.get(arg1), 0);  
  62.          
  63.        return views.get(arg1);  
  64.    }  
  65.   
  66.    //判断是否由对象生成界面  
  67.    @Override  
  68.    public boolean isViewFromObject(View arg0, Object arg1) {  
  69.        return (arg0 == arg1);  
  70.    }  
  71.   
  72.    @Override  
  73.    public void restoreState(Parcelable arg0, ClassLoader arg1) {  
  74.        // TODO Auto-generated method stub  
  75.          
  76.    }  
  77.   
  78.    @Override  
  79.    public Parcelable saveState() {  
  80.        // TODO Auto-generated method stub  
  81.        return null;  
  82.    }  
  83.   
  84.    @Override  
  85.    public void startUpdate(View arg0) {  
  86.        // TODO Auto-generated method stub  
  87.          
  88.    }  
分享到:
评论

相关推荐

    Android中父View和子view的点击事件处理问题探讨

    在Android开发中,处理触摸事件是一项关键任务,尤其是在复杂的用户界面设计中,涉及到父ViewGroup和子View交互的时候。本文将深入探讨Android系统如何处理触摸事件,以及如何解决父View和子View之间的点击事件冲突...

    子view突出部分不能响应点击

    3. **覆盖点击事件**:在父View中覆写`onInterceptTouchEvent`和`dispatchTouchEvent`方法,自定义点击事件的处理逻辑,确保即使在`android:clipChildren="false"`的情况下,也能正确捕获到子View的突出部分的点击...

    Android判断touch事件点是否在view范围内的方法

    在Android应用开发中,处理触摸事件...总之,Android提供了丰富的API来处理触摸事件和View的几何属性,使得开发者能够创建出具有高度互动性的用户界面。通过熟练掌握这些技巧,可以让你的Android应用更加生动有趣。

    horizaotalScrollview动态添加子view,设置点击事件

    本示例主要展示了如何在HorizontalScrollView中动态添加子View,并为每个子View设置点击事件,以便用户交互。这一知识点对于开发涉及大量可滑动、可点击元素的应用尤其重要。 首先,我们需要创建一个...

    android view 单击、双击和移动事件处理----TestViewEvent

    本文将深入探讨如何在Android中处理View的单击、双击以及移动事件,以实现更佳的用户体验。我们将通过一个名为"TestViewEvent"的例子来阐述这些概念。 首先,我们要了解Android中的基本事件监听器。在Android中,...

    如何让安卓(Android)子控件超出父控件的范围显示

    在Android开发中,有时我们需要设计布局时,让子控件的部分内容超出其父控件的边界,以实现某些特殊的视觉效果或交互体验。例如,一个底部导航栏可能需要部分悬停在屏幕下方,或者一个按钮需要延伸到屏幕边缘。本篇...

    android超出布局点击失效解决方案附带自定义控件

    1. **设置点击监听器**:为父布局设置点击监听器,并在事件处理中通过`event.getX()`和`event.getY()`获取点击位置,判断是否落在子View的范围内。如果是,则手动调用子View的`performClick()`方法。 2. **自定义...

    Android View 事件传递与消费

    Android View 事件传递与消费,用于view事件传递的理解。

    Android 通过NavigationView+DrawerLayout来实现侧滑菜单的功能和点击事件+弹出对话框

    在Android中,我们通常使用`NavigationView`和`DrawerLayout`这两个组件来实现这种功能。`NavigationView`用于创建菜单项,而`DrawerLayout`则作为容器,管理抽屉的滑入滑出行为。接下来,我们将详细探讨如何使用这...

    Android自定义View的事件分发机制(一)

    在自定义View时,有时我们需要改变默认的事件分发行为,例如,可能需要让父View捕获原本属于子View的事件,这时就需要重写 `onInterceptTouchEvent()`。同时,`onTouchEvent()` 也需要进行适当的处理,以确保事件的...

    Android中自定义View

    在Android开发中,自定义View是一项重要的技能,它允许开发者根据需求创建独特的用户界面元素,以实现更加丰富和个性化的交互体验。自定义View通常涉及到绘制、事件处理、动画以及性能优化等多个方面。下面我们将...

    Android中View绘制流程

    在这个阶段,View会根据其测量尺寸和父View的布局规则确定每个子View的位置。对于ViewGroup来说,它需要遍历所有子View并调用它们的`layout()`方法来设定位置。 3. **onDraw()**:最后是绘制阶段,这是实际在屏幕上...

    Android 全局悬浮View实现,以及悬浮view点击事件

    下面将详细介绍如何实现Android全局悬浮View,并讨论其点击事件的处理。 首先,我们需要创建悬浮按钮的布局。在布局文件中,我们可以使用`android.support.design.widget.FloatingActionButton`组件来创建悬浮按钮...

    Android 自定义view模板并实现点击事件的回调

    在Android开发中,自定义View是提升应用个性化和功能扩展性的重要手段。本文将深入探讨如何创建一个自定义View模板,并实现点击事件的回调。这个模板以老版QQ的顶部栏为例,它通常包含左右两个按钮和一个中间的标题...

    Android 中触摸事件与点击事件分析

    本篇文章将深入探讨Android中的View事件传递机制、事件消费以及触摸事件和点击事件的区别。 首先,让我们来了解Android的事件传递机制。在Android中,事件(如触摸事件)是从根View开始,沿着View树向下传递的。这...

    android 动画被父布局遮盖问题解决1

    然而,需要注意的是,`bringToFront()`只能改变View在当前父容器内的层级,并不能影响到父容器自身的层级,所以它不能替代设置`android:clipChildren="false"`和`android:clipToPadding="false"`。 在实际开发中,...

    android触屏事件之activity,view,viewgroup

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

    android获取根View的方法

    在Android开发中,根View(Root View)是布局文件中最高层次的视图容器,它包含并管理着应用界面中的所有子视图。根View通常是LinearLayout、RelativeLayout、FrameLayout等布局组件,它决定了子视图的排列方式和...

Global site tag (gtag.js) - Google Analytics