实现自定义的移动端双指缩放

原理:

DOM上绑定双指触控相关的事件,当双指触控时,保存初始距离,当双指移动时,计算两触控点的距离,根据移动中的距离与初始距离调节缩放比例,再根据缩放比例改变元素样式即可实现缩放

效果演示:

在这里插入图片描述

vue3 + ts代码如下:

<script setup lang="ts">
import {ref, onMounted} from 'vue';

const scale = ref(1)
let startScale = scale.value

// 缩放灵敏度
const sensitivity = 0.5

// 触控开始的距离
let startDistance: number = 0

/**
 * 计算距离
 * @param touch1
 * @param touch2
 */
function getDistance(touch1: Touch, touch2: Touch): number {
  return Math.sqrt((touch1.clientX - touch2.clientX) ** 2 + (touch1.clientY - touch2.clientY) ** 2)
}

onMounted(() => {
  document.addEventListener('touchstart', onTouchstart, {
    passive: false
  })

  // 触控开始
  function onTouchstart(e: TouchEvent) {
    e.preventDefault()
    const touches = e.touches
    if (touches.length !== 2) {
      return
    }
    startDistance = getDistance(touches[0], touches[1]);
    startScale = scale.value
    document.addEventListener('touchmove', onTouchmove)
    document.addEventListener('touchend', onTouchend)
  }

  // 触控移动
  function onTouchmove(e: TouchEvent) {
    const touches = e.touches
    if (touches.length !== 2) {
      return
    }
    let movingDistance: number = getDistance(touches[0], touches[1]);
    scale.value = Number((startScale * (1 + (movingDistance - startDistance) / startDistance * sensitivity)).toFixed(2))
  }

  // 触控结束
  function onTouchend(e: TouchEvent) {
    document.removeEventListener('touchmove', onTouchmove)
    document.removeEventListener('touchend', onTouchend)
  }
})
</script>

<template>
  <div class="HomeView" :style="{fontSize: `${14 * scale}px`}">
    <div>尝试双指缩放</div>
    <div class="box" :style="{width: `${100 * scale}px`, height: `${100 * scale}px`,}"></div>
    <div>缩放比例:{{ scale }}</div>
  </div>
</template>

<style scoped>
.HomeView {
  transform-origin: left top;
}

.box {
  background-color: red;
}
</style>
  • 12
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,针对您的问题,我可以给您一些思路和代码实现的指导。 首先,我们需要在自定义的 TextureView 中监听手势事件,这里我们可以使用 GestureDetector 和 ScaleGestureDetector 来分别处理单指移动和双指缩放。以下是一个实现的示例代码: ```java public class MyTextureView extends TextureView { private GestureDetector mGestureDetector; private ScaleGestureDetector mScaleGestureDetector; private float mScaleFactor = 1.0f; public MyTextureView(Context context) { super(context); init(); } public MyTextureView(Context context, AttributeSet attrs) { super(context, attrs); init(); } public MyTextureView(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); init(); } private void init() { mGestureDetector = new GestureDetector(getContext(), new GestureDetector.SimpleOnGestureListener() { @Override public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) { // 处理单指移动事件 float dx = e2.getX() - e1.getX(); float dy = e2.getY() - e1.getY(); Matrix matrix = new Matrix(getMatrix()); matrix.postTranslate(-dx, -dy); setTransform(matrix); return true; } }); mScaleGestureDetector = new ScaleGestureDetector(getContext(), new ScaleGestureDetector.SimpleOnScaleGestureListener() { @Override public boolean onScale(ScaleGestureDetector detector) { // 处理双指缩放事件 mScaleFactor *= detector.getScaleFactor(); mScaleFactor = Math.max(0.1f, Math.min(mScaleFactor, 10.0f)); Matrix matrix = new Matrix(getMatrix()); matrix.setScale(mScaleFactor, mScaleFactor, detector.getFocusX(), detector.getFocusY()); setTransform(matrix); return true; } }); } @Override public boolean onTouchEvent(MotionEvent event) { // 将手势事件交给 GestureDetector 和 ScaleGestureDetector 处理 mGestureDetector.onTouchEvent(event); mScaleGestureDetector.onTouchEvent(event); return true; } } ``` 在上面的代码中,我们通过 GestureDetector 处理单指移动事件,并在 onTouchEvent 中调用 mGestureDetector.onTouchEvent(event) 来传递手势事件。而双指缩放事件则通过 ScaleGestureDetector 处理,并在 onTouchEvent 中调用 mScaleGestureDetector.onTouchEvent(event) 来传递手势事件。 在处理单指移动事件和双指缩放事件时,我们都需要使用 Matrix 来对 TextureView 进行变换。在这里,我们使用 setTransform(matrix) 方法来设置变换矩阵。 希望以上代码能够帮助您实现自定义 TextureView 的双指缩放和单指移动功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值