Android手势识别GestureDetector和ScaleGestureDetector介绍与使用,以自定义一个可拖拽拉伸的ImageView为例

一、GestureDetector

1. 简介

GestureDetector主要用于检测单指手势,例如单击、长按、滑动等,不支持多指手势。

2. SimpleOnGestureListener 内部类

GestureDetector.SimpleOnGestureListener 是用于处理手势事件的辅助类,它包含了一系列回调方法,用于处理不同类型的单指手势事件。下面是对每个回调方法的简要介绍:

  • onDown(MotionEvent e): 当用户按下(Down)手指时触发。这个方法返回 true 表示事件被消费了,false 表示未被消费。

  • onShowPress(MotionEvent e): 当用户按下并保持按压一段时间时触发。它表示按下动作已被识别,但尚未发生其它任何行为。

  • onSingleTapUp(MotionEvent e): 当用户轻击屏幕时触发。这个方法返回 true 表示事件被消费了,false 表示未被消费。

  • onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY): 当用户在屏幕上滚动时触发。它提供了滚动开始和结束时的事件信息,以及在X和Y方向上的距离差。

  • onLongPress(MotionEvent e): 当用户长按屏幕时触发。

  • onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY): 当用户迅速滑动手指并松开时触发。它提供了滑动开始和结束时的事件信息,以及在X和Y方向上的速度。

  • onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY): 当用户在屏幕上滚动时触发。与第四个回调方法不同,这个方法在滚动过程中持续触发,而不仅仅是在滚动结束时触发。

  • onDoubleTap(MotionEvent e): 当用户双击屏幕时触发。

  • onDoubleTapEvent(MotionEvent e): 当双击事件包含按下、移动和抬起动作时触发。通常与 onDoubleTap() 结合使用,以处理更复杂的双击手势。

  • onSingleTapConfirmed(MotionEvent e): 当确认发生了单击事件时触发。与 onSingleTapUp() 不同的是,这个方法确保了事件是单击事件而不是双击事件。

这些回调方法提供了处理各种类型手势事件的灵活性,可以根据需求选择实现相应的方法来处理手势事件。

3. 示例

import android.content.Context;
import android.util.AttributeSet;
import android.view.GestureDetector;
import android.view.MotionEvent;
import android.view.View;public class MyView extends View {private GestureDetector gestureDetector;public MyView(Context context, AttributeSet attrs) {super(context, attrs);// 实例化 GestureDetector,并传入 SimpleOnGestureListener 对象gestureDetector = new GestureDetector(context, new MyGestureListener());}@Overridepublic boolean onTouchEvent(MotionEvent event) {// 将触摸事件传递给 GestureDetectorreturn gestureDetector.onTouchEvent(event) || super.onTouchEvent(event);}private class MyGestureListener extends GestureDetector.SimpleOnGestureListener {@Overridepublic boolean onDown(MotionEvent e) {// 用户按下屏幕时触发return true; // 返回 true 表示事件被消费}@Overridepublic boolean onSingleTapConfirmed(MotionEvent e) {// 单击事件确认时触发return true;}@Overridepublic boolean onDoubleTap(MotionEvent e) {// 双击事件时触发return true;}@Overridepublic void onLongPress(MotionEvent e) {// 长按事件时触发}@Overridepublic boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) {// 滚动事件时触发return true;}// 其他手势事件处理方法...}
}

二、ScaleGestureDetector

1. 简介

用于检测缩放手势,即双指捏合或者扩张的手势。它提供了 onScale() 和 onScaleBegin() 等回调方法来处理缩放手势的开始、进行中和结束时的事件。

2. SimpleOnGestureListener 内部类

ScaleGestureDetector.SimpleOnGestureListener 用于处理手势事件的辅助类,它包含了一系列回调方法,用于处理不同类型的双指手势事件。下面是对每个回调方法的简要介绍:

  • onScale(ScaleGestureDetector detector): 当缩放手势进行中时调用。这个方法会在缩放手势进行过程中持续调用,每次缩放都会触发。参数 detector 提供了有关缩放手势的信息,如当前的缩放因子等。

  • onScaleBegin(ScaleGestureDetector detector): 当缩放手势开始时调用。这个方法在缩放手势的第一次触发时调用,可以用来初始化缩放相关的状态。参数 detector 提供了有关缩放手势的信息。

  • onScaleEnd(ScaleGestureDetector detector): 当缩放手势结束时调用。这个方法在缩放手势结束后调用,可以用来清理缩放相关的状态。参数 detector 提供了有关缩放手势的信息。

3. 示例

import android.content.Context;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.ScaleGestureDetector;
import android.view.View;public class MyScaleView extends View {private ScaleGestureDetector scaleGestureDetector;private float scaleFactor = 1.0f;public MyScaleView(Context context, AttributeSet attrs) {super(context, attrs);scaleGestureDetector = new ScaleGestureDetector(context, new ScaleListener());}@Overridepublic boolean onTouchEvent(MotionEvent event) {// 将触摸事件传递给 ScaleGestureDetectorscaleGestureDetector.onTouchEvent(event);return true;}@Overrideprotected void onDraw(Canvas canvas) {super.onDraw(canvas);// 在画布上绘制内容,并根据 scaleFactor 进行缩放canvas.scale(scaleFactor, scaleFactor, getWidth() / 2f, getHeight() / 2f);// 绘制内容...}private class ScaleListener extends ScaleGestureDetector.SimpleOnScaleGestureListener {@Overridepublic boolean onScale(ScaleGestureDetector detector) {// 缩放因子的变化scaleFactor *= detector.getScaleFactor();// 限制缩放因子的范围(可选)scaleFactor = Math.max(0.1f, Math.min(scaleFactor, 5.0f));// 重绘 Viewinvalidate();return true;}}
}

三、自定义一个可拖拽和拉伸的ImageView

1. 思路整理

  • 首先,我们可以直接继承 ImageView,并通过 Matrix 来控制图片的移动和拉伸。
  • 其次,使用 GestureDetector 监听移动的相关事件,使用 ScaleGestureDetector 监听拉伸的相关事件。
  • 最后,我们可能需要控制图片最大和最小缩放的比例。实际应用中还会考虑一些图片边界、双击放大、动画等,可根据需求自行添加。

2. 示例

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.Canvas;
import android.graphics.Matrix;
import android.graphics.drawable.BitmapDrawable;
import android.graphics.drawable.ColorDrawable;
import android.graphics.drawable.Drawable;
import android.util.AttributeSet;
import android.view.GestureDetector;
import android.view.MotionEvent;
import android.view.ScaleGestureDetector;import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.appcompat.widget.AppCompatImageView;public class DragZoomImageView extends AppCompatImageView {private static final int NONE = 0;private static final int DRAG = 1;private static final int ZOOM = 2;private int mMode = NONE;private Matrix mFinalMatrix = new Matrix();private Matrix mSavedMatrix = new Matrix();// 图像以FitXY显示时使用的Scale大小private float mOriginScale = 1.0f;// 图像的最小、最大缩放比例private float mMinScale = 0.5f;private float mMaxScale = 5.0f;private float mCurrentScale = 1.0f;private Bitmap mBitmap;private GestureDetector mGestureDetector;private ScaleGestureDetector mScaleGestureDetector;public DragZoomImageView(@NonNull Context context) {super(context);init();}public DragZoomImageView(@NonNull Context context, @Nullable AttributeSet attrs) {super(context, attrs);init();}public DragZoomImageView(@NonNull Context context, @Nullable AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);init();}private void init() {setScaleType(ScaleType.MATRIX);mGestureDetector = new GestureDetector(getContext(), new GestureListener());mScaleGestureDetector = new ScaleGestureDetector(getContext(), new ScaleListener());}@Overrideprotected void onSizeChanged(int w, int h, int oldw, int oldh) {super.onSizeChanged(w, h, oldw, oldh);Drawable drawable = getDrawable();Bitmap bitmap = drawableToBitmap(drawable);if (bitmap != null) {Matrix matrix = getFitCenterMatrix(bitmap.getWidth(), bitmap.getHeight(), w, h);setImageMatrix(matrix);mOriginScale = getMatrixScaleX(matrix);mBitmap = bitmap;}}@Overridepublic void setImageMatrix(Matrix matrix) {super.setImageMatrix(matrix);mFinalMatrix.set(matrix);}@Overridepublic void setImageBitmap(Bitmap bm) {if (bm == null) {return;}super.setImageBitmap(bm);Matrix matrix = getFitCenterMatrix(bm.getWidth(), bm.getHeight(), getWidth(), getHeight());setImageMatrix(matrix);mOriginScale = getMatrixScaleX(matrix);mBitmap = bm;}@Overridepublic boolean onTouchEvent(MotionEvent event) {if (mBitmap == null) {return false;}switch (event.getAction() & MotionEvent.ACTION_MASK) {case MotionEvent.ACTION_DOWN:  // 单指mMode = DRAG;break;case MotionEvent.ACTION_POINTER_DOWN:  // 多指mMode = ZOOM;break;}if (mMode == DRAG) {mGestureDetector.onTouchEvent(event);}if (mMode == ZOOM) {mScaleGestureDetector.onTouchEvent(event);}return true;}public void handleScale(float scale) {handleScale(scale, getWidth() / 2, getHeight() / 2);}public void handleScale(float scale, float px, float py) {if (scale < mMinScale) {scale = mMinScale;}if (scale > mMaxScale) {scale = mMaxScale;}if (mCurrentScale == scale) {return;}mCurrentScale = scale;  // record scalefloat newScale = scale * mOriginScale;float oldScale = getMatrixScaleX(mFinalMatrix);float postScale = newScale / oldScale;mFinalMatrix.postScale(postScale, postScale, px, py);super.setImageMatrix(mFinalMatrix);}private float getMatrixScaleX(Matrix matrix) {float[] values = new float[9];matrix.getValues(values);return values[Matrix.MSCALE_X];}private Matrix getFitCenterMatrix(int bitmapWidth, int bitmapHeight, int viewWidth, int viewHeight) {Matrix matrix = new Matrix();matrix.reset();float scale;float dx;float dy;scale = Math.min((float) viewWidth / (float) bitmapWidth, (float) viewHeight / (float) bitmapHeight);dx = Math.round((viewWidth - bitmapWidth * scale) * 0.5f);dy = Math.round((viewHeight - bitmapHeight * scale) * 0.5f);matrix.setScale(scale, scale);matrix.postTranslate(dx, dy);return matrix;}private Bitmap drawableToBitmap(Drawable drawable) {if (drawable == null) {return null;}if (drawable instanceof BitmapDrawable) {return ((BitmapDrawable) drawable).getBitmap();} else if (drawable instanceof ColorDrawable) {// 如果 Drawable 是 ColorDrawable,则创建一个相同大小的 BitmapColorDrawable colorDrawable = (ColorDrawable) drawable;int width = colorDrawable.getIntrinsicWidth();int height = colorDrawable.getIntrinsicHeight();Bitmap bitmap = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888);Canvas canvas = new Canvas(bitmap);colorDrawable.setBounds(0, 0, canvas.getWidth(), canvas.getHeight());colorDrawable.draw(canvas);return bitmap;} else {// 如果 Drawable 不是 BitmapDrawable 或 ColorDrawable,则返回空return null;}}private class GestureListener extends GestureDetector.SimpleOnGestureListener {@Overridepublic boolean onDown(MotionEvent e) {mSavedMatrix.set(mFinalMatrix);return super.onDown(e);}@Overridepublic boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) {Matrix matrix = new Matrix(mSavedMatrix);float dx = e2.getX() - e1.getX();float dy = e2.getY() - e1.getY();matrix.postTranslate(dx, dy);setImageMatrix(matrix);return true;}}private class ScaleListener extends ScaleGestureDetector.SimpleOnScaleGestureListener {float px;float py;@Overridepublic boolean onScaleBegin(ScaleGestureDetector detector) {px = detector.getFocusX();py = detector.getFocusY();return true;}@Overridepublic boolean onScale(ScaleGestureDetector detector) {float scale = detector.getScaleFactor() * mCurrentScale;handleScale(scale, px, py);return true;}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/798290.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

华为openEuler-22.03-LTS-SP3配置yum源

先有华为后有天&#xff0c;遥遥领先&#xff01; 1 确定使用的OS版本 # cat /etc/os-release NAME"openEuler" VERSION"22.03 (LTS-SP3)" ID"openEuler" VERSION_ID"22.03" PRETTY_NAME"openEuler 22.03 (LTS-SP3)" ANSI…

编程生活day7--明明的随机数、6翻了、吃火锅

明明的随机数 题目描述 明明想在学校中请一些同学一起做一项问卷调查&#xff0c;为了实验的客观性&#xff0c;他先用计算机生成了N个1到1000之间的随机整数&#xff08;N≤100&#xff09;&#xff0c;对于其中重复的数字&#xff0c;只保留一个&#xff0c;把其余相同的数…

[Qt]解析moc文件

产生moc文件 moc文件是Qt的moc预处理器处理带QOBJECT宏的类是产生的文件。 分析 一个Qt创建的示例工程&#xff0c;加上一个按钮的信号和槽产生的moc文件如下 moc_widget.cpp /**************************************************************************** ** Meta obje…

【NLP】关于BERT模型的一些认知

BERT&#xff08;Bidirectional Encoder Representations from Transformers&#xff09;模型是由Google在2018年提出的预训练Transformer模型&#xff0c;用于自然语言处理任务。 一. BERT模型的架构 1.1 输入表示 / Encoder模块 BERT中的Encoder模块是由三种Embedding&…

GD32F470_ DS18B20温度传感器模块移植

DS18B20温度传感器 DS18B20数字温度传感器提供9位至12位精度的温度测量&#xff0c;并具有非易失性用户可编程上下触发点报警功能。DS18B20通过单总线通信&#xff0c;根据定义&#xff0c;只需要一条数据线(和地线)即可与单片机通信。此外&#xff0c;DS18B20可以直接从数据线…

vue2开发好还是vue3开发好vue3.0开发路线

Vue 2和Vue 3都是流行的前端框架&#xff0c;它们各自有一些特点和优势。选择Vue 2还是Vue 3进行开发&#xff0c;主要取决于你的项目需求、团队的技术栈、以及对新特性的需求等因素。以下是一些关于Vue 2和Vue 3的比较&#xff0c;帮助你做出决策&#xff1a; Vue 2&#xff1…

关于项目中遇到的一些form表单校验

1.model 属性绑定 form 数据对象 const formModel ref({username: ,password: ,repassword: })<el-form :model"formModel" >2.v-model 绑定 form 数据对象的子属性 <el-inputv-model"formModel.username":prefix-icon"User"placeh…

设计模式详解(十三)——享元模式

享元模式简介 享元模式定义 享元模式&#xff08;Flyweight Pattern&#xff09;是一种用于优化性能的软件设计模式。它是一种结构型的设计模式。它的主要目的是通过共享物件来减少内存使用量&#xff0c;同时分享信息给尽可能多的相似物件&#xff0c;从而降低内存消耗和提高…

2024.3.20力扣每日一题——数组元素的最小非零乘积

2024.3.20 题目来源我的题解方法一 贪心 题目来源 力扣每日一题&#xff1b;题序&#xff1a;1969 我的题解 方法一 贪心 采用贪心&#xff0c;使得最终的序列应该是满足除了二进制位全为1的 2 p − 1 2^p-1 2p−1之外&#xff0c;其余的首尾两两配对进行交换&#xff0c;最…

docker安装nacos,单例模式(standalone),使用mysql数据库

文章目录 前言安装创建文件夹"假装"安装一下nacos拷贝文件夹删除“假装”安装的nacos容器生成nacos所需的mysql表获取mysql-schema.sql文件创建一个mysql的schema 重新生成新的nacos容器 制作docker-compose.yaml文件查看网站 前言 此处有本人写得简易版本安装&…

log4j 集成 ELK环境搭建

一、前言 1.需要准备一台linux服务器&#xff08;最好是CentOS7&#xff09;,内存至少4g以上&#xff08;三个组件都比较占用内存&#xff09; 2.需要有docker使用经验 3. 三个软件的版本要一致 二、安装ElasticSearch 这里先创建一个网络&#xff1a;因为我们还需要部署k…

VS2022编译OpenCV库(静态库和动态库)

OpenCV是一个基于Apache2.0许可(开源)发行的跨平台计算机视觉和机器学习软件库,可以运行在Linux、Windows、Android和Mac OS操作系统上。 它轻量级而且高效——由一系列 C 函数和少量 C++ 类构成,同时提供了Python、Ruby、MATLAB等语言的接口,实现了图像处理和计算机视觉方…

品牌定位升级|飞雕开关如何从家庭作坊走上国际之路?

飞雕电器,这个名字在中国开关插座行业中如同一面旗帜,自1987年起就扬帆在电工领域的大海中。它不仅见证了这个行业的起起伏伏,还始终以其创新的姿态站在浪尖之上。 飞雕的产品线丰富多彩,除主营的墙壁开关插座领域外,飞雕电器还涉足了与墙壁开关紧密相关的其它领域,现已推出移…

蓝桥杯第十四届C++C组

目录 三国游戏 填充 翻转 【单调队列优化DP】子矩阵 【快速幂、欧拉函数】互质数的个数 【tire树】异或和之差 【质因数分解】公因数匹配 子树的大小 三国游戏 题目描述 小蓝正在玩一款游戏。游戏中魏蜀吴三个国家各自拥有一定数量的士兵X, Y, Z (一开始可以认为都…

Linux中安装nacos

Linux中安装nacos 一、前言二、准备1、下载2、上传到服务器3、解压 三、配置1、备份配置文件2、导入sql3、修改前4、修改后 四、使用1、启动2、关闭 一、前言 最近国外的docker似乎是出什么问题&#xff0c;试过很多手段都拉不下 nacos 镜像&#xff0c;就打算在服务器装一下 …

windows git bash 报错 bash: gh: command not found,需要安装 github的做桌面版工具 或者换成git 命令

需要安装 github的做桌面版工具 或者换成git 命令 git clone https://.........git 当你在 Windows 的 Git Bash 中遇到错误 bash: gh: command not found 时&#xff0c;这表示 Git Bash 无法识别 gh 命令。gh 命令是 GitHub 命令行工具的一部分&#xff0c;它提供了从命令行与…

能否安全地删除 Mac 资源库中的文件?

在管理Mac电脑存储空间时&#xff0c;用户确实可能考虑对资源库&#xff08;Library&#xff09;文件夹进行清理以释放空间。Mac资源库是一个系统及应用程序存放重要支持文件的地方&#xff0c;其中包括但不限于配置文件、临时文件、缓存、插件、偏好设置、应用程序支持数据等。…

ModuleNotFoundError: No module named ‘xxx.xx‘; ‘xxx‘ is not a package

ModuleNotFoundError: No module named sqllineage.runner; sqllineage is not a package 数据血缘关系使用python3安装了sqllineage&#xff0c;在执行python脚本时始终报错sqllineage is not a package&#xff0c;经过检查python3安装配置没问题&#xff0c;并且sqllineage…

Rustdesk二次编译,新集成AI功能开源Gpt小程序为远程协助助力,全网首发

环境&#xff1a; Rustdesk1.1.9 sciter版 问题描述&#xff1a; Rustdesk二次编译&#xff0c;新集成AI功能开源Gpt小程序为远程协助助力,全网首发 解决方案&#xff1a; Rustdesk二次编译&#xff0c;新集成开源AI功能Gpt小程序&#xff0c;为远程协助助力&#xff0c…

c语言中有哪些格式说明符,在scanf 会在遇到空白字符(空格、制表符或换行符)时停止读取(即scanf会忽略空格)

在C语言的 scanf 函数中&#xff0c;有几个格式说明符在遇到空白字符&#xff08;包括空格、制表符和换行符&#xff09;时会停止读取。这些格式说明符通常用于读取特定类型的输入数据。以下是一些常用的格式说明符&#xff1a; 1.%d 或 %i&#xff1a;用于读取整数。当 scanf …