android触控 MotionEvent (转)

简介: 原文:http://my.oschina.net/banxi/blog/56421 (1)  MotionEvent 中getAction()与getActionMasked()的区别:   /** * Bit mask of the parts of the action code that are the action itself.

 

原文:http://my.oschina.net/banxi/blog/56421

(1)  MotionEvent 中getAction()与getActionMasked()的区别:

 

/**
     * Bit mask of the parts of the action code that are the action itself.
     */
    public static final int ACTION_MASK             = 0xff;

/**
     * Return the kind of action being performed.
     * Consider using {<A class=referer href="http://my.oschina.net/link1212" target=_blank>@link</A>  #getActionMasked} and {<A class=referer href="http://my.oschina.net/link1212" target=_blank>@link</A>  #getActionIndex} to retrieve
     * the separate masked action and pointer index.
     * @return The action, such as {<A class=referer href="http://my.oschina.net/link1212" target=_blank>@link</A>  #ACTION_DOWN} or
     * the combination of {<A class=referer href="http://my.oschina.net/link1212" target=_blank>@link</A>  #ACTION_POINTER_DOWN} with a shifted pointer index.
     */
    public final int getAction() {
        return mAction;
    }

  /**
     * Return the masked action being performed, without pointer index information.
     * Use {<A class=referer href="http://my.oschina.net/link1212" target=_blank>@link</A>  #getActionIndex} to return the index associated with pointer actions.
     * @return The action, such as {<A class=referer href="http://my.oschina.net/link1212" target=_blank>@link</A>  #ACTION_DOWN} or {<A class=referer href="http://my.oschina.net/link1212" target=_blank>@link</A>  #ACTION_POINTER_DOWN}.
     */
    public final int getActionMasked() {
        return mAction & ACTION_MASK;
    }

上面的代码是基于android2.2的,注释是android4.X中最新的.

 

初看一下从技术角度来说通过getActionMasked()和通过getAction()返回的结果应该是一样的,

因为ACTION_MASK 的值 是0xff, 

那他们区别是什么呢?先看关于这两个方法注释:

 

 /**
     * action码的位掩码部分就是action本身
     */
    public static final int ACTION_MASK             = 0xff;

/**
  返回action的类型,考虑使用getActionMasked()和getActionIndex()来获得单独的经过掩码的action和触控点的索引.
 @return action例如ACTION_DOWN或者ACTION_POINTER_DOWN与转换的触控点索引的合成值
     */
    public final int getAction() {
        return mAction;
    }

  /**
   返回经过掩码的action,没有触控点索引信息.
   通过getActionIndex()来得到触控操作点的索引.
@return action,例如ACTION_DOWN,ACTION_POINTER_DOWN

 
     */
    public final int getActionMasked() {
        return mAction & ACTION_MASK;
    }

在上面的两个方法中注释出现差异的地方是对于ACTION_POINTER_DOWN的描述:

通过getAction()返回的ACTION_POINTER_DOWN的是与转换触控点索引的合成值.

而getActionMasked()则就是一个ACTION_POINTER_DOWN的值:

 

这么来看我们知道一个action的代码值还包含了action是那个触控点的索引值:

现在我们对比来看看ACTION_MASK和ACTION_POINTER_INDEX_MASK

 public static final int ACTION_MASK             = 0xff;
public static final int ACTION_POINTER_INDEX_MASK  = 0xff00;

还没有看出来什么吗?

 

您把ACTION_MASK看成是0x00ff

就知道了吧.

也就是说,一个MotionEvent中的action代码,

前8位是实实在在包含表示哪一个动作常量.

后八位呢就是包含了触控点的索引信息.

因为ACTION_MASK = 0x00ff所以,经过ACTION_MASK掩码过后的action码就没有索引信息了.

如何得索引值呢?

原理:

先将action跟0xff00相与清除前8位用于存储动作常量的信息,

然后将action右移8位就可以得到索引值了.

 

我们就可以自己想办法得到索引信息了.

即先对action用ACTION_POINTER_INDEX_MASK进行掩码处理,

即  maskedIndex = action&ACTION_POINTER_INDEX_MASK = action&0xff00

这各掩码也就是将action这个数的前8位清零.

然后再将maskedIndex向右移8位就能够得到索引值了.

 

再看看android真实是怎么做的吧,

用于右移8位的常量.

/**
   * Bit shift for the action bits holding the pointer index as
   * defined by {<A class=referer href="http://my.oschina.net/link1212" target=_blank>@link</A>  #ACTION_POINTER_INDEX_MASK}.
   */
  public static final int ACTION_POINTER_INDEX_SHIFT = 8;

得到索引值方法源代码,如下:

public final int getActionIndex() {
    return (mAction & ACTION_POINTER_INDEX_MASK) >> ACTION_POINTER_INDEX_SHIFT;
}

为什么要有索引信息?

 因为,这样说吧,android中,当有触摸事件发生时(假设已经注册了事件监听器),调用你注册监听器中的方法onTouch(,MotionEvent ev);传递了一个MotionEvent的对象过来.

但是,想想,上面只传递进来一个MotionEvent过来,如果只是单点触控那是没有问题.

问题就是当你多个手指触控的时候也是只传递这一个MotionEvent进来,

这个时候,你当然想知道每个手指的所对应的触控点数据信息啦.

所以MotionEvent中有就要索引信息了.

事件是你可以很容易通过API看到,MotionEvent还包含了移动操作中其它历史移动数据.

方便处理触控的移动操作.

android sdk对于这个类的描述中就有这么一句:

 

For efficiency, motion events with ACTION_MOVE may batch together multiple movement samples within a single object.

 

我翻译下:"出于效率的考虑,事件代码为ACTION_MOVE的Motion,会在一个MotionEvent对象中包含多个移动数据采样."

 

 

现在我们对于MotionEvent有了初步的了解了.

 

PS:

 

我发现android4中MotionEvent中的代码大多变成了原生代码了:

 

例如如getX(int)在2.2中是这样的:

public final float getX(int pointerIndex) {
    return mDataSamples[(pointerIndex*NUM_SAMPLE_DATA) + SAMPLE_X];
}

 

但到了4.x是这样的了:

public final float getX(int pointerIndex) {
     return nativeGetAxisValue(mNativePtr, AXIS_X, pointerIndex, HISTORY_CURRENT);
 }

 

 

 

 

 

 

 

 

 

 

 

 

目录
相关文章
|
8月前
|
Android开发
[Android]视图的控触操作-MotionEvent
[Android]视图的控触操作-MotionEvent
76 0
|
Android开发
Android中级第十一讲之MotionEvent的分发、拦截机制分析
最件看到事件分发机制,一方面在看内核剖析,一方面找测试小例子,最终找到,内容如下,——下载地址 咱们就借这个小例子来讲一些问题 一开始只关注了onTouch事件,应用于Window层,用来判断一些操作;后来研究到手势解锁...
1035 0
|
Android开发 索引
Android MotionEvent详解
 在前边几篇博文中(《图解Android事件传递之ViewGroup篇》,《图解Android事件传递之View篇》),我们已经了解了android触摸事件传递机制,接着我们再来研究一下与触摸事件传递相关的几个比较重要的类,比如MotionEvent。
|
3天前
|
缓存 前端开发 Android开发
【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
|
7天前
|
Dart 前端开发 Android开发
【02】写一个注册页面以及配置打包选项打包安卓apk测试—开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
【02】写一个注册页面以及配置打包选项打包安卓apk测试—开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
【02】写一个注册页面以及配置打包选项打包安卓apk测试—开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
|
1月前
|
搜索推荐 前端开发 API
探索安卓开发中的自定义视图:打造个性化用户界面
在安卓应用开发的广阔天地中,自定义视图是一块神奇的画布,让开发者能够突破标准控件的限制,绘制出独一无二的用户界面。本文将带你走进自定义视图的世界,从基础概念到实战技巧,逐步揭示如何在安卓平台上创建和运用自定义视图来提升用户体验。无论你是初学者还是有一定经验的开发者,这篇文章都将为你打开新的视野,让你的应用在众多同质化产品中脱颖而出。
70 19

热门文章

最新文章