`

手势识别兼容Android 1.x和2.x的代码(转载)

 
阅读更多
  我们大家都知道,由于Android 2.x开始很多API变动比较大新增了一些比如多点触控的支持,对于屏幕触控手势识别中我们需要考虑更多的实现方法,下面是一段兼容Android 1.x和2.x的代码,可以让我们的程序兼容几乎99%的Android手机。

       一、首先新建一个抽象类判断SDK版本问题

Java代码:
public abstract class VersionedGestureDetector {
private static final String TAG = "VersionedGestureDetector";

OnGestureListener mListener;

public static VersionedGestureDetector newInstance(Context context,
OnGestureListener listener) {
/*设计实例化构造方法,这里Androideoe提示大家目前有3种API的实现方法,我们需要逐一考虑最优的解决方法,以满足高平台更多的功能实现。*/

final int sdkVersion = Integer.parseInt(Build.VERSION.SDK);
//使用android.os.Build判断API Level,但需要将字符串转换为整形
VersionedGestureDetector detector = null;
if (sdkVersion < Build.VERSION_CODES.ECLAIR) {
//如果版本小于2.0则使用1.5版本的API,可以兼容1.5和1.6
detector = new CupcakeDetector();
} else if (sdkVersion < Build.VERSION_CODES.FROYO) {
//如果版本小于2.1则使用2.0版本的API,可以兼容2.0,2.0.1和2.1这三个版本
detector = new EclairDetector();
} else {
//否则使用2.2开始的新的触控API

detector = new FroyoDetector(context);
}

Log.d(TAG, "Created new " + detector.getClass()); //判断最终选择的到底是哪个版本的类
detector.mListener = listener;

return detector;
}

public abstract boolean onTouchEvent(MotionEvent ev);
//我们需要根据版本决定onTouchEvent的实现

public interface OnGestureListener { //手势判断接口主要是实现两个方法
public void onDrag(float dx, float dy); //拖拽
public void onScale(float scaleFactor); //缩放
}

private static class CupcakeDetector extends VersionedGestureDetector {
//针对Android 1.5和1.6设计的兼容方式
float mLastTouchX;
float mLastTouchY;

float getActiveX(MotionEvent ev) { //获得当前X坐标
return ev.getX();
}

float getActiveY(MotionEvent ev) { //获得当前Y坐标
return ev.getY();
}

boolean shouldDrag() { //是否是拖拽中或者说移动中
return true;
}

@Override
public boolean onTouchEvent(MotionEvent ev) { //重写onTouchEvent方法
switch (ev.getAction()) {
case MotionEvent.ACTION_DOWN: { //向下
mLastTouchX = getActiveX(ev);
mLastTouchY = getActiveY(ev);
break;
}
case MotionEvent.ACTION_MOVE: {

/*Androideoe提醒大家,由于1.x时代的API比较简单,很多手势没有封装,我们只能从ACTION_MOVE中根据坐标变化判断手势样式*/


final float x = getActiveX(ev);
final float y = getActiveY(ev);

if (shouldDrag()) {
mListener.onDrag(x - mLastTouchX, y - mLastTouchY); //处理拖拽移动
}

mLastTouchX = x;
mLastTouchY = y;
break;
}
}
return true;
}
}

private static class EclairDetector extends CupcakeDetector {
//这个是针对Android 2.0,2.0.1和2.1提供的解决方法,可以看到有很多多点触控相关API出现
private static final int INVALID_POINTER_ID = -1;
private int mActivePointerId = INVALID_POINTER_ID;
private int mActivePointerIndex = 0;

@Override
float getActiveX(MotionEvent ev) {
return ev.getX(mActivePointerIndex);
}

@Override
float getActiveY(MotionEvent ev) {
return ev.getY(mActivePointerIndex);
}

@Override
public boolean onTouchEvent(MotionEvent ev) {
final int action = ev.getAction();


switch (action & MotionEvent.ACTION_MASK) {
case MotionEvent.ACTION_DOWN:
mActivePointerId = ev.getPointerId(0);
break;


case MotionEvent.ACTION_CANCEL:
case MotionEvent.ACTION_UP:
mActivePointerId = INVALID_POINTER_ID;

break;


case MotionEvent.ACTION_POINTER_UP: //有个点松开
final int pointerIndex = (ev.getAction() & MotionEvent.ACTION_POINTER_INDEX_MASK)
>> MotionEvent.ACTION_POINTER_INDEX_SHIFT;
final int pointerId = ev.getPointerId(pointerIndex); //获取第几个点


if (pointerId == mActivePointerId) {
final int newPointerIndex = pointerIndex == 0 ? 1 : 0;
mActivePointerId = ev.getPointerId(newPointerIndex);
mLastTouchX = ev.getX(newPointerIndex); //处理第newPointerIndex个点的x位置
mLastTouchY = ev.getY(newPointerIndex);
}
break;
}

mActivePointerIndex = ev.findPointerIndex(mActivePointerId);
return super.onTouchEvent(ev);
}
}

private static class FroyoDetector extends EclairDetector {
//从Android 2.2开始可以很好的处理多点触控的缩放问题
private ScaleGestureDetector mDetector;

public FroyoDetector(Context context) {
mDetector = new ScaleGestureDetector(context,
new ScaleGestureDetector.SimpleOnScaleGestureListener() {
@Override public boolean onScale(ScaleGestureDetector detector) {
mListener.onScale(detector.getScaleFactor()); //根据 ScaleGestureDetector.SimpleOnScaleGestureListener这个系统类处理缩放情况通过onScale方法
return true;
}
});
}

@Override
boolean shouldDrag() {
return !mDetector.isInProgress();
}

@Override
public boolean onTouchEvent(MotionEvent ev) {
mDetector.onTouchEvent(ev);
return super.onTouchEvent(ev);
}
}
}
复制代码
有关调用方法,我们可以自定义一个View,取名为TouchExampleView类,这里来处理触控相关的问题

Java代码:

public class TouchExampleView extends View {
private Drawable mIcon; //我们以一个图片为参照物,根据手势控制
private float mPosX;
private float mPosY;

private VersionedGestureDetector mDetector;
private float mScaleFactor = 1.f; //原始缩放比例为1.0

public TouchExampleView(Context context) {
this(context, null, 0);
}

public TouchExampleView(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}

public TouchExampleView(Context context, AttributeSet attrs, int defStyle) { //实现我们自定义View的构造
super(context, attrs, defStyle);
mIcon = context.getResources().getDrawable(R.drawable.icon);
mIcon.setBounds(0, 0, mIcon.getIntrinsicWidth(), mIcon.getIntrinsicHeight());

mDetector = VersionedGestureDetector.newInstance(context, new GestureCallback());
//实例化刚才的版本自适应手势控制类
}

@Override
public boolean onTouchEvent(MotionEvent ev) {
//重写onTouchEvent方法,使用VersionedGestureDetector类得出的数据。
mDetector.onTouchEvent(ev);
return true;
}

@Override
public void onDraw(Canvas canvas) { //处理自定义View绘制方法
super.onDraw(canvas);

canvas.save();
canvas.translate(mPosX, mPosY); //进行平移操作,根据mPosX和mPosY坐标
canvas.scale(mScaleFactor, mScaleFactor); //进行缩放操作,参数就是刚才定义的float类型的缩放比例
mIcon.draw(canvas); //直接绘制图片变化到画布中
canvas.restore();
}

private class GestureCallback implements VersionedGestureDetector.OnGestureListener {
public void onDrag(float dx, float dy) {
//这里Android123提示大家在2.2中这个回调方法将可以支持拖拽的坐标处理
mPosX += dx;
mPosY += dy;
invalidate();
}

public void onScale(float scaleFactor) {
mScaleFactor *= scaleFactor; //缩放控制

mScaleFactor = Math.max(0.1f, Math.min(mScaleFactor, 5.0f));
//限制最小缩放比例为1.0最大为5.0倍数

invalidate();
}
}
}
复制代码
有关调用我们的自定义的TouchExampleView可以在Activity的onCreate方法中加入以下代码

Java代码:

TouchExampleView view = new TouchExampleView(this);
view.setLayoutParams(new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,
ViewGroup.LayoutParams.MATCH_PARENT));

setContentView(view); //替换掉原始的res.layout.main
复制代码


  最后Androideoe需要给大家说明的是使用本例子,直接使用Android 2.2的SDK创建工程,即API Level为8,发布时在androidmanifest.xml中加入uses-sdk android:minSdkVersion="3" android:targetSdkVersion="8"这句可以兼容从Android 1.5到2.2的版本,有关2.3中新增的一个可以处理5个或5个以上的多点触控增强类Android开发网将在以后的时间中介绍。


转载出自:http://www.apkbus.com/forum.php?mod=viewthread&tid=848
分享到:
评论

相关推荐

    手势识别 实现对手势的识别

    手势识别实现对手gesture的识别 手势识别是指通过分析用户的手势来识别用户的意图,实现对手势的监控和识别。...文章提供了一种兼容 Android 1.x 和 2.x 的代码实现,解决了手势识别在不同版本之间的兼容性问题。

    cocos2d 手势识别

    7. **多平台兼容**:Cocos2d-x支持多平台,确保手势识别代码在iOS、Android、Windows等平台上都能正常工作。 通过以上步骤,开发者可以在Cocos2d-x中实现高效且自定义的手势识别功能,提升游戏的交互性和趣味性。在...

    Android开发 多语言、指纹登录、手势登录.zip

    1. **手势识别库**:Android官方并未提供内置的手势识别,但有许多第三方库,如`GestureOverlayView`或`GestureLock`等,用于实现手势解锁功能。 2. **手势绘制**:用户在屏幕上绘制预设的手势,将其转换为一系列...

    奥比中光openNI(Android).rar

    这些数据流可用于实时的3D重建、物体识别和手势识别等应用。 3. **上下文感知**:OpenNI支持创建上下文节点,使得应用程序可以根据环境和用户行为做出智能反应。例如,它可以识别用户的手势,然后触发相应的操作。 ...

    Android代码-输入法手势程序源码.zip

    7. **兼容性测试**:由于Android设备的多样性,源码可能包含了针对不同屏幕尺寸、分辨率和Android版本的适配代码。 通过研究这个输入法手势程序的源码,开发者不仅可以学习到Android系统的事件处理机制,还可以掌握...

    Android通过手势实现的缩放处理.zip源码资源下载

    在Android开发中,手势识别是实现用户交互的重要一环,特别是在移动设备上,手势操作能够为用户提供更加直观和便捷的体验。本资源“Android通过手势实现的缩放处理.zip”包含了一个示例项目,用于演示如何在Android...

    MapboxGesturesforAndroid库可以轻松检测和处理Android设备上的用户手势_Java_Kotli.zip

    1. **手势识别** Mapbox Gestures for Android库提供了多种常见手势的识别,如单击、双击、长按、滑动、旋转、捏合等。这些手势可以帮助用户在无须明确按钮或菜单的情况下与应用进行交互,提升操作的直观性。 2. *...

    Android程序研发源码Android 输入法手势程序源码.rar

    10. **测试和调试**:源码中可能包含单元测试和集成测试,以验证手势识别的准确性和稳定性,以及对不同设备和Android版本的兼容性。 通过研究GestureProject,开发者不仅能掌握手势输入法的实现原理,还能提升对...

    GestureRecognitionV1_QT_qt手势识别_qt手势_动作识别_against1l6_源码.rar.rar

    该压缩包文件“GestureRecognitionV1_QT_qt手势识别_qt手势_动作识别_against1l6_源码.rar.rar”显然包含了基于Qt框架的手势识别和动作识别的源代码。Qt是一个跨平台的应用程序开发框架,广泛用于创建桌面、移动和...

    反应本机的智能手势_JavaScript_下载.zip

    6. **平台兼容性**:React Native项目需要考虑跨平台兼容性,手势识别可能需要针对iOS和Android的差异进行调整。 7. **自定义手势**:项目可能还实现了自定义的手势识别逻辑,以便根据应用需求扩展基本的手势库。 ...

    Android输入法手势程序源码.zip

    1. **手势识别技术** 手势输入法的核心在于手势识别。在Android源码中,这通常涉及到识别用户在触控屏上的一系列连续点。这些点的数据通过MotionEvent事件传递,通过算法分析点的轨迹和速度来识别手势。常见的手势...

    触摸手势识别

    1. Android系统:在Android平台上,可以通过监听MotionEvent事件来实现手势识别。例如,重写onTouchEvent()方法,根据ACTION_DOWN、ACTION_UP、ACTION_MOVE等动作来判断手势类型。 2. iOS系统:iOS使用UIResponder...

    android开发GestureDetector手势识别调节音量亮度快进和后退完整源码

    在Android开发中,手势识别是提升用户体验的重要一环。`GestureDetector`是Android SDK提供的一种用于处理常见触摸事件...记得在实际开发中,一定要测试不同设备和屏幕尺寸下的兼容性,确保手势识别的稳定性和一致性。

    gesture-refresh-layout,Android手势刷新布局.zip

    1. **手势识别**:手势刷新布局首先需要识别用户的下拉手势。这通常通过监听触摸事件并计算手指在屏幕上的移动来实现。Android系统提供了一系列的MotionEvent类和ViewGroup的onTouchEvent()方法,开发者可以通过这些...

    android多手势视频播放器

    综上所述,"android多手势视频播放器"是一个结合了Android多媒体框架、手势识别技术和用户界面设计的项目。它展示了如何通过创新的手势控制提升移动设备上的视频播放体验。YaoPlayer可能是这个项目的名字,可能包含...

    引入“手势”,这是iOS和Android设备上前端手势识别的概念验证。在移动Safari(iPhone、iPad等)以及.zip

    手势识别在现代移动设备上已经成为一种重要的交互方式,特别是在iOS和Android系统的智能手机和平板电脑上。这个概念验证项目,名为“Mobile-Web-based-Gesture-Recognition-master”,旨在探索如何在移动版Safari...

    Android View进行手势识别详解

    首先,Android系统提供了两种主要的手势识别类:`android.view.GestureDetector`和`android.gesture.Gesture`. `GestureDetector`自Android 1.6开始引入,适用于大部分设备,而`android.gesture.Gesture`则在更早的...

    GT1X Driver Porting Guide_for_Android_20150710_Rev.02.pdf

    文档提供了如何识别和配置GT1X传感器ID的说明。 #### 4.2 固件和配置 介绍了固件文件的结构以及如何进行配置。 #### 4.3 基于SLOT的坐标报告 说明了如何根据SLOT信息来报告坐标数据。 #### 4.4 ESD保护机制 电子...

    Android游戏开发 自定义手势--输入法手势技术

    在Android游戏开发中,自定义手势是一种增强用户体验和交互性的关键技术。本文主要探讨的是输入法手势技术,这是一种允许...在实践中,不断测试和优化手势识别的效率和准确性,将有助于打造更加流畅、易用的游戏体验。

    Android-Swipper-Android库可滑动手势来控制音量亮度和搜索

    在Android应用开发中,创新和用户体验是至关重要的因素。Swipper库就是为了提供这样...在Swipper-master这个压缩包中,包含了Swipper库的源代码和示例,开发者可以通过阅读和研究这些代码,更深入地理解和运用这个库。

Global site tag (gtag.js) - Google Analytics