Android 触摸事件监听(Activity层,ViewGroup层,View层)详细介绍

Android不同层次的触摸事件监听

APP开发中,经常会遇到有关手势处理的操作,比如向右滑动返回上一个页面。关于触摸事件的处理,我们可以大概处理在不同的层次上。

Activity层:可以看做触摸事件获取的最顶层
ViewGroup层:ViewGroup层可以自主控制是否让子View获取触摸事件
View层:可以决定自己是否真正的消费触摸事件,如果不消费抛给上层ViewGroup

Activity级别的手势监听:(右滑动返回上层界面)

Activity层手势监听的使用场景:一般用于当前页面中没有过多的手势需要处理的时候,至多存在点击事件。对于右滑返回上层界面这种需求,可以将其定义在一个BaseActivity中,子Activity如果需要实现,通过某个开关打开即可。

注意事项 :

1、Activity层,用dispatch可以抓取所有的事件 。

2、对于滑动,要设定一个距离阈值mDistanceGat,用于标记手势是否有效,并且注意往回滑动的处理。

3、如果底层存在点击Item,为了防止滑动过程中变色,可以适时地屏蔽触摸事件:手动构造Cancle事件主动下发,这是为了兼容最基本的点击效果,不过,满足点击的手势判定前, Move事件要正常下发。具体实现如下:

@Override public boolean dispatchTouchEvent(MotionEvent event) { case MotionEvent.ACTION_MOVE: if (Math.abs(event.getX() - down_X) > 10 && flagDirection == MotionDirection.HORIZION) { MotionEvent e = MotionEvent.obtain(event.getEventTime(), event.getEventTime(), MotionEvent.ACTION_CANCEL, event.getX(), event.getY(), 0); super.dispatchTouchEvent(e); } else { super.dispatchTouchEvent(event);//不符合条件正常下发 }

4、防止手势的往回滑动,最好利用GestureDectetor来判断,如果存在往回滑动,则手势无效,使用方式如下:

mDetector = new GestureDetector(this, new GestureDetector.SimpleOnGestureListener() { @Override public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) { if (!slideReturnFlag && distanceX > 5) { slideReturnFlag = true; }}

5、如何处理Up事件:dispatch是否往下派发。具体的做法是,根据手势是否有效,如果手势无效,那么Up肯定是需要往下派发的。如果有效,根据后续操作进行,因为有时候为了防止子View获取到不必要的点击事件。具体实现如下

@Override public boolean dispatchTouchEvent(MotionEvent event) { case MotionEvent.ACTION_UP: if (mGestureListener != null && !slideReturnFlag && flagDirection == MotionDirection.HORIZION) { if (stateMotion == CurrentMotionState.SlideRight) { mGestureListener.onSlideRight(); } } else { super.dispatchTouchEvent(event); //无效的手势 } flagDirection = MotionDirection.NONE; stateMotion = CurrentMotionState.NONE; slideReturnFlag=false; break;

6、在disPatch中最好记录down_X、down_Y ,为了后面的处理与判断,因为dispatch中最能保证你获取到该事件。同时要保证Dispatch事件的下发,

第二:父容器级别的手势监听

注意事项:容器级别的监听至少要使得当前容器强制获取手势的焦点,至于如何获取焦点,可以自己编写onTouch事件,并且reture true。不过我们把判断处理放在dispatch里面,这样能够保证事件完全获取。因为,如果底层消费了事件,onTouch是无法完整获取事件的,但是我们有足够的能力保证dispatch获取完整的事件。无论在本层onTouch消费,还是底层消费,dispatch是用于不会漏掉的。对于手势的容器,最好用padding,而不采用Magin,为什么呢,因为Margin不在容器内部。

1、父容器监听的使用场景

容器中,子View是否存在交互事件,是否存在滑动 上层容器是否存在拦截事件的可能,比如SrollView

2、实现

子View不存在交互事件:

这类容器可以采用Dispatch来实现,不过需要强制获取焦点,同时也要适时的释放焦点。具体实现如下:
如何保证本层一定接收到Down后续事件。dispatch的Down事件能够返回True即可。

如何保证本层不被偶然的屏蔽,使用 getParent().requestDisallowInterceptTouchEvent(true)即可。当然,有强制获取也要适时的释放,当手势判定为无效的时候就要释放,具体实现如下:

@Override public boolean dispatchTouchEvent(MotionEvent ev) { getParent().requestDisallowInterceptTouchEvent(true);</strong></span> mGestureDetector.onTouchEvent(ev); switch (ev.getActionMasked()) { case MotionEvent.ACTION_DOWN: down_X = ev.getX(); down_Y = ev.getY(); slideReturnFlag = false; break; case MotionEvent.ACTION_CANCEL: case MotionEvent.ACTION_MOVE: if (Math.abs(down_X - ev.getX()) < Math.abs(down_Y - ev.getY()) && Math.abs(ev.getY() - down_Y) > mDistanceGate / 2) { getParent().requestDisallowInterceptTouchEvent(false);</span></strong> } default: break; } return super.dispatchTouchEvent(ev); }

子View存在交互事件:子View存在交互事件,就要通过dispatch与onTouch的配合使用,dispatch为了判断手势的有效性,同时既然从容器层开始,强制获取焦点是必须的,底层如何强制获取焦点,不关心。这里如果没有消费Down,则说明底层View消费了。同时要兼容无效手势强制焦点获取的释放,防止上传滚动View,具体实现如下:

@Override public boolean dispatchTouchEvent(MotionEvent ev) { mGestureDetector.onTouchEvent(ev); switch (ev.getActionMasked()) { case MotionEvent.ACTION_DOWN: down_X = ev.getX(); down_Y = ev.getY(); slideReturnFlag = false; break; default: break; } return super.dispatchTouchEvent(ev); }

onTouch中处理响应事件,主要是为了防止底层获取后,上层还处理

// ACTION_CANCEL 嵌套如其他scrowView 可能屏蔽 @Override public boolean onTouchEvent(MotionEvent ev) { switch (ev.getActionMasked()) { case MotionEvent.ACTION_DOWN:

// ACTION_CANCEL 嵌套如其他scrowView 可能屏蔽 @Override public boolean onTouchEvent(MotionEvent ev) { switch (ev.getActionMasked()) { case MotionEvent.ACTION_DOWN: getParent().requestDisallowInterceptTouchEvent(true); return true; case MotionEvent.ACTION_CANCEL: return true; case MotionEvent.ACTION_UP: if (Math.abs(down_X - ev.getX()) > Math.abs(down_Y - ev.getY()) && !slideReturnFlag && ev.getX() - down_X > mDistanceGate) { // 返回上个Activity,也有可能是返回上一个Fragment FragmentActivity mContext = null; if (getContext() instanceof FragmentActivity) { mContext = (FragmentActivity)getContext(); FragmentManager fm = mContext.getSupportFragmentManager(); if (fm.getBackStackEntryCount() > 0) { fm.popBackStack(); } else { mContext.finish(); } } } return true; case MotionEvent.ACTION_MOVE: if (Math.abs(down_X - ev.getX()) < Math.abs(down_Y - ev.getY()) && Math.abs(ev.getY() - down_Y) > mDistanceGate / 2) { getParent().requestDisallowInterceptTouchEvent(false); } return true; default: break; } return super.onTouchEvent(ev); }

3、父容器手势的拦截,有些时候,子View具有点击事件,点击变颜色。给予一定容错空间后,强制拦截事件。dispatch返回true保证事件下传,不必担心

@Override public boolean onInterceptTouchEvent(MotionEvent ev) { if (ev.getActionMasked() == MotionEvent.ACTION_MOVE && Math.abs(down_X - ev.getX()) > 20) return true; return super.onInterceptTouchEvent(ev); }

第四:HorizontalScrollView边缘状态下,滑动手势的监听,具体实现如下,主要是边缘处的手势判断。

@Override public boolean dispatchTouchEvent(MotionEvent ev) { getParent().requestDisallowInterceptTouchEvent(true); mGestureDetector.onTouchEvent(ev); switch (ev.getActionMasked()) { case MotionEvent.ACTION_DOWN: slideReturnFlag = false; down_X = ev.getX(); down_Y = ev.getY(); oldScrollX = getScrollX(); break; case MotionEvent.ACTION_UP: if (Math.abs(down_X - ev.getX()) > Math.abs(down_Y - ev.getY()) && ev.getX() - down_X > mDistanceGate && !slideReturnFlag && oldScrollX == 0) { // 返回上个Activity,也有可能是返回上一个Fragment FragmentActivity mContext = null; if (getContext() instanceof FragmentActivity) { mContext = (FragmentActivity)getContext(); FragmentManager fm = mContext.getSupportFragmentManager(); if (fm.getBackStackEntryCount() > 0) { fm.popBackStack(); } else { mContext.finish(); } } } break; case MotionEvent.ACTION_MOVE: if (Math.abs(down_X - ev.getX()) < Math.abs(down_Y - ev.getY()) && Math.abs(ev.getY() - down_Y) > mDistanceGate / 2) { getParent().requestDisallowInterceptTouchEvent(false); } default: break; } return super.dispatchTouchEvent(ev); }

第五:防止垂直滚动的ScrollView过早的屏蔽事件:重写拦截函数即可:

@Override public boolean onInterceptTouchEvent(MotionEvent ev) { if (Math.abs(ev.getY() - down_Y) < getResources().getDimensionPixelSize(R.dimen.slide_gesture_vertical_gate)) { super.onInterceptTouchEvent(ev); return false; } return super.onInterceptTouchEvent(ev); } @Override public boolean dispatchTouchEvent(MotionEvent ev) { switch (ev.getAction()) { case MotionEvent.ACTION_DOWN: down_X = ev.getX(); down_Y = ev.getY(); break;

第六:Viewpager第一页滑动手势;

1、防止过早拦击

@Override public boolean dispatchTouchEvent(MotionEvent ev) { getParent().requestDisallowInterceptTouchEvent(true); mGestureDetector.onTouchEvent(ev); switch (ev.getActionMasked()) { case MotionEvent.ACTION_DOWN: down_X = ev.getX(); down_Y=ev.getY(); slideReturnFlag=false; break; case MotionEvent.ACTION_MOVE: if (Math.abs(down_X - ev.getX()) < Math.abs(down_Y - ev.getY()) && Math.abs(ev.getY() - down_Y) > mDistanceGate / 2) { getParent().requestDisallowInterceptTouchEvent(false); } break; default: break; } return super.dispatchTouchEvent(ev); }

2、防止往回滑动等

/* * 触摸事件的处理,要判断是否是ViewPager不可滑动的时候 */ @Override public boolean onTouchEvent(MotionEvent arg0) { // 防止跳动 boolean ret = super.onTouchEvent(arg0); switch (arg0.getActionMasked()) { case MotionEvent.ACTION_DOWN: Log.v("lishang", "down"); break; case MotionEvent.ACTION_CANCEL: case MotionEvent.ACTION_UP: Log.v("lishang", "up"); if (slideDirection == SlideDirection.RIGHT) { if (slideReturnFlag || getCurrentItem() != 0 || arg0.getX() - down_X < mDistanceGate || mPercent > 0.01f) break; } else if (slideDirection == SlideDirection.LEFT) { if (getAdapter() != null) { if (slideReturnFlag||getCurrentItem() != getAdapter().getCount() - 1 || down_X - arg0.getX() < mDistanceGate || mPercent > 0.01f) break; } } else {

第七:getParent().requestDisallowInterceptTouchEvent

这个函数的的作用仅仅能够保证事件不被屏蔽,但是倘若本层dispatch在down的时候返回false,那么事件的处理就无效了,就算强制获取焦点

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

时间: 2024-09-15 06:42:28

Android 触摸事件监听(Activity层,ViewGroup层,View层)详细介绍的相关文章

android截图事件监听的原理与实现

Android系统没有对用户截屏行为提供回调的api,所以我们只能走野路子来获取用户是否截屏了.一般大家都会采用如下两种方法 1.监听截屏图片所在目录变化(FileObserver) 2.监听媒体库的变化(ContentObserver) 上面两种方法均不是万能的,需要结合使用才能达到良好的效果,首先看看如何监控目录 在android中,我们可以通过FileObserver来监听目录变化,先来看看如何使用 private static final File DIRECTORY_PICTURES

android-Android 写事件监听 真机调试出现闪退(我是新手,刚学Android,求大神指教)

问题描述 Android 写事件监听 真机调试出现闪退(我是新手,刚学Android,求大神指教) 这是我的程序: ImageView imageView = (ImageView) findViewById(R.id.login_image); imageView.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { } }); 这是错误日志: 03-08 18:10:43

Android开发-之监听button点击事件的多种方法_Android

 在Android下,事件的发生是在监听器下进行,android系统可以响应按键事件和触摸屏事件,本文主要介绍了button点击事件的方法 一.实现button点击事件的方法 实现button点击事件的监听方法有很多种,这里总结了常用的四种方法: 1.匿名内部类 2.外部类(独立类) 3.实现OnClickListener接口 4.添加XML属性 每一种方法都有它的优点也有它的不足,那么接下来就来详细的讲解这四个实现方法  二.具体实现 1.匿名内部类: 在Android开发中我们会经常看到各种

Android开发之button事件监听简单实例_Android

本文实例讲述了Android开发之button事件监听用法.分享给大家供大家参考.具体如下: 事件监听的listener,有以下几种方式: 1.声明一个普通的class,实现OnClickListener接口,然后在button的setOnClickListener中new该类的一个对象. 2.使用匿名内部类,直接 btn.setOnClickListener(new OnClickListener() { @Override public void onClick(View arg0) { S

Android编程双重单选对话框布局实现与事件监听方法示例

本文实例讲述了Android编程双重单选对话框布局实现与事件监听方法.分享给大家供大家参考,具体如下: 首先是自定义XML布局代码: <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_pare

android.widget.Switch- 开/关事件监听

问题描述 android.widget.Switch- 开/关事件监听 我想实现一个开关按钮事件android.widget.Switch (适用于 API v.14) <Switch android:id="@+id/switch1" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="Switch"

android开发-一个简单的单击事件监听代码哪里错了?

问题描述 一个简单的单击事件监听代码哪里错了? MainActivity.java package com.example.a007;import android.support.v7.app.ActionBarActivity;import android.os.Bundle;import android.view.Menu;import android.view.MenuItem;import android.view.View;import android.view.View.OnClic

android开发,监听事件不触发

问题描述 android开发,监听事件不触发 登入界面在TextView中输入密码,触发TextView的监听事件(继承TextWatcher),当程序再次返回登入界面时,在TextView控件中输入内容,监听事件并不触发? 监听返回按钮的源代码: public boolean onKeyDown(int keyCode,KeyEvent event){ if(keyCode == KeyEvent.KEYCODE_BACK && event.getAction() == KeyEvent

android开发鼠标监听事件运行错误

问题描述 android开发鼠标监听事件运行错误 解决方案 mainactivity 中27行 空指针了,....写的很明显了.. 解决方案二: 看一下MainActivity.java的第27行代码,有对象为null,所以报空指针异常. 解决方案三: 亲,学会看logcat报异常哦,有时候异常原因写在中间