以鼠标位置为中心进行滚动缩放

需求:在图片上进行手动标注,当目标较小时需要放大后再进行框选标注。默认状态下,图片是以左上角为中心进行缩放的,如果对图片设置了居中则是以中点为中心进行缩放的。现在我们需要实现以目标(即鼠标位置)为中心进行滚动缩放。

1.原理

以鼠标位置为中心进行缩放其实就是先进行缩放然后再进行平移操作,将缩放完成的图形平移到鼠标位置,使得缩放前后鼠标指向的图形位置不变。

那么我们最重要的就是计算出偏移量,具体怎么算不再赘述,详情点击偏移量计算原理

2.套公式

首先我们需要先确定我们的图形是以什么位置进行缩放,这点很重要,因为不同的缩放位置进行平移的时候算法不同

  • 以图片左上角缩放
javascript 复制代码
deltaX -= (1 - n) * (width / 2 - e.clientX + x1)
deltaY -= (1 - n) * (height / 2 - e.clientY + y1)
  • 以图片中点缩放
javascript 复制代码
deltaX -= (e.clientX - x1) * (n - 1)
deltaY -= (e.clientY - y1) * (n - 1)

以上两种方式只有 deltaX 和 deltaY的计算有区别,其他逻辑是一样的,即缩放再平移

3.绑定缩放事件

以下为以图片左上角缩放的例子

javascript 复制代码
<template>
    <div class="img-wrap" @mousewheel.prevent="rollImg($event)">
	    <img class="imgDiv" ref="imgDiv" :src="imgSrc" alt="" />
    <div>
</template>

<script>
const imgDiv = ref(null)
const deltaX = ref(0) // 需要平移的x轴距离
const deltaY = ref(0) // 需要平移的y轴距离
const preScale = ref(1) // 记录图片缩放前的比例
// 图片参数
const params = ref({
	zoomVal: 1, // 缩放比例
	deg: 0,
	offsetLeft: 0,
	offsetTop: 0,
})

// 以鼠标位置为中心进行缩放
function rollImg(e) {
	params.value.zoomVal += e.wheelDelta / 1200
	const n = params.value.zoomVal / preScale.value
	const { x: x1, y: y1, width, height } = imgDiv.value.getBoundingClientRect()
	deltaX.value -= (1 - n) * (width / 2 - e.clientX + x1)
	deltaY.value -= (1 - n) * (height / 2 - e.clientY + y1)
	if (params.value.zoomVal >= 0.2) {
		img.style.transform = `translate(${deltaX.value}px, ${deltaY.value}px) scale(${params.value.zoomVal})`
	} else {
		params.value.zoomVal = 0.2
		img.style.transform = `translate(${deltaX.value}px, ${deltaY.value}px) scale(${params.value.zoomVal})`
		return false
	}
	preScale.value = params.value.zoomVal
}
</script>

以图片中点缩放只需要替换deltaX 和 deltaY的计算即可,但是中点缩放有一个坑,下面详细说明

4.以图片中点缩放的坑

如果

css 复制代码
.img-wrap {
	position: relative;
	.imgDiv {
		position: absolute;
		left: 50%;
		transform: translateX(-50%);
	}
}
  • 初始状态下图片使用 left: 50% + transform: translate(-50%, 0) 实现水平居中

  • 缩放计算时使用 deltaX/deltaY 的绝对像素位移``transform: `translate({deltaX.value}px, {deltaY.value}px)```

  • 根本原因在于初始状态下的居中定位方式与缩放时的位移 计算逻辑存在冲突,首次缩放时这个偏移量未被纳入 deltaX/deltaY 计算

解决方法:

不再使用CSS的百分比和translate来实现居中,而是用`deltaX`和`deltaY`来模拟居中。这样,在缩放时,位移计算就可以正确叠加

javascript 复制代码
function resetParams(){
	const imgWrapWidth = imgWrap.value.clientWidth
	const imgDivWidth = imgDiv.value.offsetWidth
	deltaX.value = (imgWrapWidth - imgDivWidth ) / 2
	deltaY.value = 0
	preScale.value = 1
	imgDiv.value.style.left = 0
	imgDiv.value.style.top = 0
	imgDiv.value.style.transform = `translate(${deltaX.value}px, ${deltaY.value}}px)`
}
相关推荐
用户57573033462412 分钟前
拒绝“首屏爆炸”:用 React 哨兵模式与懒加载打造丝滑列表
前端
大腕先生41 分钟前
通用分页超详细介绍(附带源代码解析&页面展示效果)
xml·java·linux·服务器·开发语言·前端·idea
睿智的海鸥1 小时前
Markdown 语法大全详解
开发语言·前端·javascript·css·html
Highcharts.js1 小时前
用Highcharts如何动态向一个序列添加点
前端·javascript·react.js·highcharts
HookJames1 小时前
设计Section 09 · Cost & Lead Time Factors 的完整 Block Editor 操作步骤
前端
玖玖passion1 小时前
React 常用 Hooks 函数及使用方法完全指南(useState / useEffect / useRef / useContext / useCallback / useMemo / useReducer)
前端·javascript
Awu12272 小时前
⚡精通Claude第6课-Hooks钩子系统:从前端视角玩转AI自动化工作流
前端·aigc·claude
椰猫子2 小时前
Spring Framework(Bean)
java·前端·spring
道清茗2 小时前
【RH294知识点汇总】第 7 章 《 使用角色和 Ansible 内容集合简化 Playbook 》
java·前端·ansible
前端那点事2 小时前
彻底弄懂async/await!解决回调地狱,Vue异步开发必备(超全实战)
前端·vue.js