Android触屏事件和MotionEvent详解

Android屏幕操作

屏幕是用户和Android设备交互的主要媒介,屏幕分为触屏和非触屏。Android设备目前有四种类型:Android Phone,Android Tablet,Android Wear和Android TV。Android TV大都使用非触屏,其他三类设备则大都使用触屏。对非触屏设备,用户可以通过键盘鼠标或遥控器在屏幕上操作。对触屏设备,用户主要通过手指或触控笔等工具在屏幕上操作,当然也可以通过外接的键盘,鼠标和轨迹球等工具来操作。

Android屏幕交互事件

用户在设备屏幕上的所有操作都会转换为各类屏幕交互事件。Android屏幕交互事件主要有如下几种类型。

  • key event 键盘、遥控器按键,鼠标点击会生成按键事件(key event)
  • hover event 鼠标在屏幕上的停留、滑动会产生hover event
  • scroll event 鼠标滚轮的滚动会生成scroll event
  • touch event 对触屏设备,当用户用手指或触控笔在设备屏幕上操作时会产生触屏事件(touch event)。

为了方便理解和简化描述,后文在介绍时会统一用手指操作来代指所有的触屏操作。例如“当手指接触屏幕时产生此事件”,并不表示只能用手指接触屏幕才会产生此事件,而是需要理解为“当手指,触控笔等工具接触屏幕时都会产生此事件”。

触屏事件类型

按照动作来分,可以将触屏事件可以分为以下三类

  1. 手指按到屏幕上
  2. 手指在屏幕上移动
  3. 手指离开屏幕

其中手指按到屏幕上和手指离开屏幕一定是成对出现的,在这中间会出现不定次数的手指在屏幕上移动的事件。

触屏事件序列

在Android系统中,从手指按到屏幕上开始,到手指离开屏幕,这个过程中产生的一系列触屏事件构成了一个事件序列(也可以称为事件流)。对多点触屏事件,则是从第一个手指按到屏幕上开始,到最后一个手指离开屏幕为止。

一个触屏事件序列第一个事件一定是手指按到屏幕上,最后一个事件一定是手指离开屏幕。用户在设备屏幕上的所有触屏操作最终都会转换为若干个这样的事件序列。

理解触屏事件序列的概念非常重要,Android中对触屏事件的处理很多时候需要以事件序列为单位进行考察。

Android触屏事件在代码中的表示

在Android系统中使用MotionEvent对象来表示一个触屏事件,当用户用手指在屏幕上操作时,会产生一系列的MotionEvent对象。但是需要注意的是,产生了一个MotionEvent对象并不表示这一定是一个触屏操作,MotionEvent不仅可以用来表示touch event,还可以表示hover event,scroll event。也就是说,除了key event之外的其他屏幕交互事件都用MotionEvent来表示(key event用KeyEvent对象表示)。

在MotionEvent类中将产生此次事件的动作称为motion,将产生此动作的主体(如手指,鼠标等)称为pointer。一个MotionEvent对象中可以包含一个或多个pointer,每个pointer都包含id,index,位置,大小,方向等属性。在一个触屏事件序列的多个事件中,同一个pointer拥有相同的id,但是index可以不同。

这里只讨论MotionEvent中关于touch event的部分。在MotionEvent对象中主要包含了如下信息:

1.操作类型(action code)

MotionEvent提供了getActionMasked()方法来获取此次操作的类型,它是一个int型数值。除了getActionMasked()外还有一个getAction()方法,它和getActionMasked()的区别会在后面介绍。

在MotionEvent类中定义了一系列的int常量来表示各种预定义的操作类型。列举如下。

事件类型常量 含义说明
ACTION_DOWN 当手指接触屏幕时产生此事件,在多点触摸时,只有第一个手指接触屏幕时才会产生此事件,中间其他手指接触屏幕不会产生此事件。它表示一个触屏事件序列的开始。
ACTION_UP 当手指离开屏幕时产生此事件,在多点触摸时,只有最后一个手指(这个手指并不一定是产生ACTION_DOWN事件的那个手指)离开屏幕时才会产生此事件,中间其他手指离开屏幕不会产生此事件。它表示一个触屏事件序列的结束。
ACTION_MOVE 当手指在屏幕上滑动时产生此事件, 在多点触摸时,每个手指的滑动都会产生一个此事件
ACTION_POINTER_DOWN 只有在多点触摸时才会产生此事件,在一个触屏事件序列中,除第一个接触屏幕的手指外,其他手指接触屏幕时会产生此事件。
ACTION_POINTER_UP 同样只有在多点触摸时才会产生此事件,在一个触屏事件序列中,除最后一个离开屏幕的手指外,其他手指离开屏幕时会产生此事件。
ACTION_CANCEL 这个事件比较特殊,它和上述事件都不一样,上述事件都是由用户在屏幕上操作所触发的,但是这个事件是由系统自动产生的。当一个事件序列需要提前终止的时候由系统自动产生此事件。正常来说,一个事件序列应该以最后一个手指离开屏幕,也就是ACTION_UP作为结束,但是在某些情况下,事件序列需要被提前终止。这通常是因为处理这个事件序列的View对象的Parent对象在事件序列结束之前主动拦截了后续的事件。此外,如果处理这个事件序列的View对象从窗口中被移除了,它也会收到ACTION_CANCEL事件。例如处理这个事件序列的View对象所在的Activty被finish(),所在的Dialog被dismiss(),或者被其Parent View Remove了。在这些情况下,虽然这时手指还停留在屏幕上,但View对象将无法再接收到后续的触屏事件,这时它会收到ACTION_CANCEL事件,表示事件序列由于外在原因需要提前终止。

结合上面触屏事件序列的描述可以知道,一个正常的触屏事件序列一定是以ACTION_DOWN为开始,以ACTION_UP为结束,中间可以有0个或多个ACTION_MOVE, 如果是多点触摸,中间还会有若干次的ACTION_POINTER_DOWN和ACTION_POINTER_UP。ACTION_POINTER_DOWN和ACTION_POINTER_UP一定是数量相对的。

一个提前终止的触屏事件序列一定是以ACTION_DOWN为开始,以ACTION_CANCEL为结束,中间可以有0个或多个ACTION_MOVE, 如果是多点触摸,中间还会有若干次的ACTION_POINTER_DOWN和ACTION_POINTER_UP。ACTION_POINTER_DOWN和ACTION_POINTER_UP的数量可能不同。

getAction()和getActionMasked()的区别:对ACTION_POINTER_DOWN和ACTION_POINTER_UP之外的事件,getAction()返回值和getActionMasked()是相同的。对ACTION_POINTER_DOWN和ACTION_POINTER_UP,getAction()返回值和getActionMasked()返回值稍有不同。getAction()返回值包含了操作类型和产生此事件的pointer对应的pointer index两个信息,其中低8位代表操作类型,高8位代表pointer index 。

2.pointer信息

  1. 通过getPointerCount()方法获取此事件产生时pointer的个数,它一定是大于等于1的。例如有两个手指接触在屏幕上,则getPointerCount()为2。
  2. 通过getPointerId(int pointerIndex)获取pointerIndex对应的pointer id。
  3. 通过findPointerIndex(int pointerId)获取pointerId对应的pointer index。
  4. 通过getX(int pointerIndex),getY(int pointerIndex)方法来获取此事件产生时pointerIndex对应的pointer在屏幕上的相对位置。
  5. 通过getRawX(),getRawY()方法来获取此事件产生时pointerIndex对应的pointer在屏幕上的绝对位置。不带参数的重载方法表示获取pointerIndex为0的pointer在屏幕上的位置。

除此之外,还有getToolMajor(),getToolMinor(),getTouchMajor(),getTouchMinor(),getOrientation()等方法获取pointer的区域大小,方向等信息。由于实际使用的较少,这里就不做介绍了。

3.操作时间

可以通过MotionEvent类的getEventTime()方法来获取此事件产生的时间。

4.事件序列的历史数据

在MotionEvent对象中还会保存其所在的事件序列的一些历史事件的信息,可以通过getHistorySize()获取历史事件记录的条数,通过一系列的getHistoricalXXX()方法获取历史事件的信息。由于ACTION_DOWN 是一个事件序列的开始,所以ACTION_DOWN对应的事件对象中是不会有历史事件记录的,在这之后的事件对应的MotionEvent对象中会有0到多个的历史事件信息的记录,具体记录的个数并不固定,总的数量也不会太多。

在上述信息中,使用比较多的是前两条,也就是事件的类型和事件产生时pointer的相关信息。

如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

(0)

相关推荐

  • Android:Field can be converted to a local varible.的解决办法

    Android:Field can be converted to a local varible.的解决办法 前言: 使用 Android Studio 开发 Android 有一段时间了,偶尔会碰到 AS 在一些私有变量上有黄色高亮提示Field can be converted to a local varible,有些强迫症的我还是不希望看到这个黄色的高亮.百度没查到什么有用的信息,还是用谷歌搜到了一些解答. 解析 Field can be converted to a local va

  • 详解Android中获取软键盘状态和软键盘高度

    详解Android中获取软键盘状态和软键盘高度 应用场景 在Android应用中有时会需要获取软键盘的状态(即软键盘是显示还是隐藏)和软键盘的高度.这里列举了一些可能的应用场景. 场景一 当软键盘显示时,按下返回键应当是收起软键盘,而不是回退到上一个界面,但部分机型在返回键处理上有bug,按下返回键后,虽然软键盘会自动收起,但不会消费返回事件,导致Activity还会收到这次返回事件,执行回退操作,这时就需要判断,如果软键盘刚刚由显示变为隐藏状态,就不执行回退操作. 场景二 当软键盘弹出后,会将

  • Android 高仿微信朋友圈拍照上传功能

    模仿微信朋友圈发布动态,输入文字支持文字多少高度自增,有一个最小输入框高度,输入文字有限制,不过这些都很easy! 1. PhotoPicker的使用 这是一个支持选择多张图片,点击图片放大,图片之间左右滑动互相切换的库,同时支持图片删除的库,效果类似微信. (1) 添加PhotoPicker的架包 (2) 使用 选择图片:安卓6.0以后需要在代码中添加读写sd卡和相机的权限 当然清单文件中也需要添加的 PhotoPicker.builder() .setPhotoCount(maxPhoto)

  • Android 自定义标题栏的实例详解

     Android 自定义标题栏的实例详解 开发 Android APP 经常会用到自定义标题栏,而有多级页面的情况下还需要给自定义标题栏传递数据. 本文要点: 自定义标题填充不完整 自定义标题栏返回按钮的点击事件 一.代码 这里先介绍一下流程: 1. 创建一个标题栏布局文件 mytitlebar.xml 2. 在style.xml中创建 mytitlestyle 主题 3. 创建类 CustomTitleBar 4. 在需要自定义标题栏的Activity的OnCreate方法中实例化 Custo

  • Android 中RecyclerView顶部刷新实现详解

    Android 中RecyclerView顶部刷新实现详解 1. RecyclerView顶部刷新的原理 RecyclerView顶部刷新的实现通常都是在RecyclerView外部再包裹一层布局.在这个外层布局中,还包含一个自定义的View,作为顶部刷新时的指示View.也就是说,外层布局中包含两个child,一个顶部刷新View,一个RecyclerView,顶部刷新View默认是隐藏不可见的.在外层布局中对滑动事件进行处理,当RecyclerView滑动到顶部并继续下滑的时候,根据滑动的距

  • Android使用RSA加密和解密的示例代码

    一.公钥加密和私钥解密 /**RSA算法*/ public static final String RSA = "RSA"; /**加密方式,android的*/ // public static final String TRANSFORMATION = "RSA/None/NoPadding"; /**加密方式,标准jdk的*/ public static final String TRANSFORMATION = "RSA/None/PKCS1Pad

  • Android NoSuchFieldError解决办法

    Android NoSuchFieldError解决办法 前几天在开发的时候,把一个library搬到了一个新的工程中,然后在主应用模块中调用library的Activity,发现出现了NoSuchFieldError: 然而,查看代码明明是没有问题的,layout文件存在且id正确,R文件也正常.后来在StackOverFlow上找到了答案,原因是主应用模块和library里面的layout文件重名了,把其中一个名字改了就正常运行.后来写了个小demo重现了错误并分析了一下打包的apk: 主模

  • Android使用URL读取网络资源的方法

    URL(Uniform Resource Locator)是统一资源定位器,它是对可以从互联网上得到的资源的位置和访问方法的一种简洁的表示,是互联网上标准资源的地址.互联网上的每个文件都有一个唯一的URL,它包含的信息指出文件的位置以及浏览器应该怎么处理它.就通常情况而言,URL可以由协议名.主机.端口和资源组成. URL类提供了多个构造器用于创建URL对象,一旦获得了URL对象之后,就可以调用如下常用方法来访问该URL对应的资源了. ->String getFile():获取此URL的资源名:

  • Android触屏事件和MotionEvent详解

    Android屏幕操作 屏幕是用户和Android设备交互的主要媒介,屏幕分为触屏和非触屏.Android设备目前有四种类型:Android Phone,Android Tablet,Android Wear和Android TV.Android TV大都使用非触屏,其他三类设备则大都使用触屏.对非触屏设备,用户可以通过键盘鼠标或遥控器在屏幕上操作.对触屏设备,用户主要通过手指或触控笔等工具在屏幕上操作,当然也可以通过外接的键盘,鼠标和轨迹球等工具来操作. Android屏幕交互事件 用户在设备

  • Android View的事件体系教程详解

    目录 一.什么是View?什么是ViewGroup? 二.View的位置 三.View的触摸事件 1.MotionEvent 2.TouchSlop 3.VelocityTracker 5.Scroller 四.View的滑动 1)使用Scroll 2)通过动画 3)使用延时策略 五.View的事件分发机制 六.View的滑动冲突问题 View的滑动冲突常见可以简单分为三种: 滑动冲突的处理规则 滑动冲突的解决方法 一.什么是View?什么是ViewGroup? View是Android中所有控

  • Android ListView监听滑动事件的方法(详解)

    ListView的主要有两种滑动事件监听方法,OnTouchListener和OnScrollListener 1.OnTouchListener OnTouchListener方法来自View中的监听事件,可以在监听三个Action事件发生时通过MotionEvent的getX()方法或getY()方法获取到当前触摸的坐标值,来对用户的滑动方向进行判断,并可在不同的Action状态中做出相应的处理 mListView.setOnTouchListener(new View.OnTouchLis

  • Android 拦截返回键事件的实例详解

    Android 拦截返回键事件的实例详解 KeyEvent类 Android.View.KeyEvent类中定义了一系列的常量和方法,用来描述Android中的 按键事件和返回键有关的常量和方法有. KeyEvent.KEYCODE_BACK: 表示key类型为返回键 KeyEvent.ACTION_DOWN:表示事件为按下key,如果一直按住不放,则会不停产生此事件. KeyEvent.ACTION_UP:表示事件为为放开key,一次点击key过程只会调用一次. public final in

  • Android实现连续点击多次事件的代码详解

    有时候我们需要实现这样的场景,类似进入开发者模式,即多次点击后执行操作. 首先我们先看一个方法: System提供的一个静态方法arraycopy(),我们可以使用它来实现数组之间的复制. public static void arraycopy(Object src,int srcPos,Object dest,int destPos,int length): src:源数组: srcPos:源数组要复制的起始位置: dest:目的数组: destPos:目的数组放置的起始位置: length

  • Android事件处理的两种方式详解

    安卓提供了两种方式的事件处理:基于回调的事件处理和基于监听的事件处理. 基于监听的事件处理 基于监听的事件处理一般包含三个要素,分别是: Event Source(事件源):事件发生的场所,通常是各个组件 Event(事件):事件封装了界面组件上发生的特定事件(通常就是用户的一次操作) Event Listener(事件监听器):负责监听事件源发生的事件,并对各种事件作出相应的响应 下面使用一个简单的案例介绍按钮事件监听器 布局文件就是简单的线性布局器,上面是一个EditText,下面是一个Bu

  • Android触屏测试实例代码

    本文实例详细描述了Android触屏测试代码,可实现对触屏的点击.移动.离开等事件的处理,对于Android初学者有很好的借鉴价值. 具体功能代码如下: package com.test; import android.app.Activity; import android.os.Bundle; import android.util.DisplayMetrics; import android.util.Log; import android.view.MotionEvent; import

  • Android 滑动监听的实例详解

    Android 滑动监听的实例详解 摘要: ScollBy,ScollTo是对内容的移动,view.ScollyBy是对view的内容的移动 view,ScollTo是对内容的移动(移动到指定位置),view.ScollyBy是对view的内容的移动(移动距离) 在次activity中,当手指点击TextView ,此时是ViewGroup 响应还是TextView响应呢? 代码实践: 在activity中重写onTouchEvent(): public boolean onTouchEvent

  • Android 实现锚点定位思路详解

    相信做前端的都做过页面锚点定位的功能,通过<a href="#head" rel="external nofollow" > 去设置页面内锚点定位跳转. 本篇文章就使用tablayout.scrollview来实现android锚点定位的功能. 效果图: 实现思路 1.监听scrollview滑动到的位置,tablayout切换到对应标签 2.tablayout各标签点击,scrollview可滑动到对应区域 自定义scrollview 因为我们需要监听

  • Android性能优化大图治理示例详解

    目录 引言 1 自定义大图View 1.1 准备工作 1.2 图片宽高适配 1.3 BitmapRegionDecoder 2 大图View的手势事件处理 2.1 GestureDetector 2.2 双击放大效果处理 2.3 手指放大效果处理 引言 在实际的Android项目开发中,图片是必不可少的元素,几乎所有的界面都是由图片构成的:像列表页.查看大图页等,都是需要展示图片,而且这两者是有共同点的,列表展示的Item数量多,如果全部加载进来势必会造成OOM,因此列表页通常采用分页加载,加上

随机推荐