前言

本文将用一个简单的例子详细讲解如何用原生JS一步步实现完整的任意位置缩放图片功能,无任何第三方依赖,指针事件

进行多端统一的事件监听,干货满满。

完整代码

为提升阅读体验,正文中代码展示有部分省略处理,完整代码可以在码上掘金查看:

https://code.juejin.cn/pen/7356824634160840730

可在任意位置缩放、移动图片

实现原理

实现图片放大的关键点在于 CSS3 中的 transform

变换,该属性应用于元素在2D或3D上的旋转,缩放,移动,倾斜等等变换,通过设置 translate(x,y) 即可偏移元素位置,设置 scale

即可缩放元素,当然你也可以只设置 matrix 来完成上述所有操作,这涉及到矩阵变换的知识,本文使用的均是CSS提供的语法糖进行变换操作。

PC上的点击、移动,H5的手势操作,都离不开DOM事件监听。例如鼠标移动事件对应 mousemove

,移动端因为没有鼠标则对应 touchmove,而本文将介绍如何仅通过指针事件来进行多端统一

的事件监听。在监听事件中我们可以通过 event 对象获取各种属性,例如常用的 offsetXoffsetY

相对偏移量,clientXclientY 距离窗口的横坐标和纵坐标等。

理解transform-origin值

官方文档解释为:**transform-origin**CSS 属性让你更改一个元素变形的原点。

https://developer.mozilla.org/zh-CN/docs/Web/CSS/transform-origin

我们可以简单的理解为图片缩放起点,这个值默认为图片的正中心,所以我们进行放大或缩小都是依图片中心来缩放。

css缩放图片有两种方法

一、修改transform-origin值进行缩放

优点:简单快捷,容易理解

缺点:频繁修改transform-origin值会抖动,需要计算修正量

将鼠标当前的偏移量offsetX、offsetY 的值改变 transform-origin

来动态设置缩放的原点,再进行缩放,那么最终效果就是依照最新的transform-origin值来进行缩放。

比如修改transform-origin值为90% 90%。再设置放大倍数scale为1.1。效果就是下图的样子

二、不修改transform-origin值,设置偏移量translate(x,y)进行缩放

我们利用滚轮事件监听并改变 scale 值。重点是利用 deltaY 值的正负来判断滚轮是朝上还是朝下:

let scale = 1;
image.addEventListener("wheel", function (e) {
//d值永远是正的0.1或者负的0.1,代表每次缩放的倍数
let d = e.deltaY < 0 ? 0.1 : -0.1;
scale = scale * (1 + d);
...
});

如上图,怎么样从左图变成右图呢?由于我们未修改transform-origin,所以缩放始终都以图片中心

为原点进行缩放,这显然不符合我们的操作习惯,我们需要的是以鼠标点为中心对图片进行缩放。

可以清晰的看到,我们先将图片放大0.1倍,再设置偏移量translate(x,y)就可以实现任意位置缩放

需要计算的translate(x,y)值,实际上就是放大后的图片中心点与原始图片中心点的差值


每次可偏移translate(x,y)最大值 = 长宽 * 放大倍数 / 2

例如:长400,宽为800的图片。放大0.1倍之后,图片的长为440,宽为880。如果是在中心点放大,那么不需要移动图片,xy等于0。如果在左上角放大,那么图片整体向右下角偏移

x = (440 - 400) / 2 = 20,y = (880 - 800) / 2 = 40.

let d = e.deltaY < 0 ? 0.1 : -0.1;
const max = {
x: (d * rect.width) / 2,
y: (d * rect.height) / 2,
};

鼠标点的座标减去图片的在页面中的位置可以得到鼠标在图片中的位置

rect = image.getBoundingClientRect()
const mouseOffset = {
x: e.clientX - rect.x,
y: e.clientY - rect.y
}

最后用mouseOffset减去已偏移的xy,再剩以放大倍数d,再减去max,就是当前的translate(x,y)

  1. 因为mouseOffset的值,基本可以看作为一个常量,如果鼠标不移动位置,那么mouseOffset的值不会变,所以需要减去已偏移的xy,才能累计偏移量
  2. 再剩以放大倍数,得到的值,还不能用于偏移
  3. 第二步得到的这个值已经超过我们的最大偏移值max,所以需要减去max,最终就是当前的偏移量
// 计算每次的偏移量
x -= d * (mouseOffset.x - x) - max.x;
y -= d * (mouseOffset.y - y) - max.y;
image.style.transform = "translate3d(" + x + "px, " + y + "px, 0) scale(" + scale + ")";

移动图片

移动图片的实现是比较简单的,在每次指针按下时我们记录 clientXclientY 为初始值,移动时计算当前的值与初始点位的差值加到 translate 偏移量中即可另外当抬起动作结束时,会触发 click 事件,所以注意加入全局变量标记以及定时器进行一些判断处理。

image.addEventListener("pointerdown", function (e) {
isPointerdown = true;
lastPointermove = {x: e.clientX, y: e.clientY};
});
image.addEventListener("pointermove", function (e) {
if (isPointerdown) {
const current = {x: e.clientX, y: e.clientY};
x += current.x - lastPointermove.x;
y += current.y - lastPointermove.y;
lastPointermove = current;
image.style.transform = "translate3d(" + x + "px, " + y + "px, 0) scale(" + scale + ")";
}
e.preventDefault();
});
image.addEventListener("pointerup", function (e) {
isPointerdown = false;
});

结束

以上就是文章的全部内容,感谢看到这里,希望对你有所帮助或启发!创作不易,如果觉得文章写得不错,可以点赞收藏支持一下,也欢迎关注,我会更新更多实用的前端知识与技巧,期待与你共同成长~

Js实现任意位置缩放图片,深入理解背后原理的更多相关文章

  1. JS实现等比例缩放图片

    JS实现等比例缩放图片 2014-01-19 21:57 by 龙恩0707, 40 阅读, 0 评论, 收藏, 编辑 JS实现等比例缩放图片 有时候我们前端页面只有500×500像素的宽和高的布局, ...

  2. Magnifier.js - 支持鼠标滚轮缩放的图片放大镜效果

    Magnifier.js 是一个 JavaScript 库,能够帮助你在图像上实现放大镜效果,支持使用鼠标滚轮放大/缩小功能.放大的图像可以显示在镜头本身或它的外部容器中.Magnifier.js 使 ...

  3. JS等比例缩放图片,限定最大宽度和最大高度

    JS等比例缩放图片,限定最大宽度和最大高度 JavaScript //图片按比例缩放 var flag=false; function DrawImage(ImgD,iwidth,iheight){ ...

  4. 如何在html中把一个图片或者表格覆盖在一张已有图片上的任意位置

    如何在html中把一个图片或者表格覆盖在一张已有图片上的任意位置   <div style="position:relative;"> <img src=&quo ...

  5. 【js】【图片显示】js控制html页面显示图片方式

    js控制html页面显示图片方式,只需要引入“jquery-1.11.2.min.js” js: /* 引用 <script src="jquery-1.11.2.min.js&quo ...

  6. Android 图片文件操作、屏幕相关、.9图片的理解

     一:Android图片操作 1.存储bitmap到本地文件系统 public static void bitmapToFile(Bitmap bitmap, String uri) { if(!ex ...

  7. C语言中链表任意位置怎么插入数据?然后写入文件中?

    链表插入示意图:(图是个人所画)因为链表指针指来指去,难以理解,所以辅助画图更加方便. 插入某个学号后面图: 定义的结构体: struct student { ]; //学生学号 ]; //学生姓名 ...

  8. Android拖动和缩放图片

    Android拖动和缩放图片 2014年5月9日 我们在使用应用其中常常须要浏览图片.比方在微信其中.点击图片之后能够对图片进行缩放. 本博客介绍怎样对图片进行拖拽和缩放.这首先要了解Android中 ...

  9. JavaScript等比例缩放图片

    js等比例缩放图片,这个功能非常实用,当网页加载一个尺寸比较大的图片时,往往会把一个网页撑的变形,页面变得很难看,于是我们就想到了用JS去控制超出一定范围的图片,以稳定页面布局,本代码段就是完成了此功 ...

  10. 【全面总结】js获取元素位置大小

    [js获取元素位置+元素大小]全面总结 目录 1.关于offset offsetParent(只读) offsetTop(只读) offsetLeft(只读) offsetHeight(只读) off ...

随机推荐

  1. Pytorch DistributedDataParallel(DDP)教程一:快速入门理论篇

    一. 写在前面 随着深度学习技术的不断发展,模型的训练成本也越来越高.训练一个高效的通用模型,需要大量的训练数据和算力.在很多非大模型相关的常规任务上,往往也需要使用多卡来进行并行训练.在多卡训练中, ...

  2. k8s 深入篇———— docker 镜像是什么[二]

    前言 简单介绍一下docker的镜像. 正文 前面讲到了容器的工作原理了(namespace 限制了时间, cgroup限制了资源),知道docker 历史的也知道,docker 之所以能够称为容器大 ...

  3. redis 简单整理——缓存设计[三十二]

    前言 简单整理一下缓存设计. 正文 缓存的好处: ·加速读写:因为缓存通常都是全内存的(例如Redis.Memcache),而 存储层通常读写性能不够强悍(例如MySQL),通过缓存的使用可以有效 地 ...

  4. spring cloud 学习笔记 基础工程的构建(一)

    前言 学习一下spring cloud,只是过一遍微服务的一些现代化工具,微服务其实一直都存在,去公司的时候发现一个问题,即使有些项目没有用到现代这些什么docker.k8s,其实也是微服务,微服务一 ...

  5. 重走py 之路 ——普通操作与函数(三)

    前言 本节主要介绍函数,但是函数是由操作组成的.那么就分为两部,一部分为操作一部分为函数. 正文 py世界中的操作. 操作 if: 在学习任何一门语言中,关系if,要关系另外一件事,那就是if是否只能 ...

  6. 深度解读《深度探索C++对象模型》之默认构造函数

    接下来我将持续更新"深度解读<深度探索C++对象模型>"系列,敬请期待,欢迎关注!也可以关注公众号:iShare爱分享,主动获得推文. 提到默认构造函数,很多文章和书籍 ...

  7. kong管理界面konga的安装

    kong网关自身的管理界面属于付费的应用,而第三方界面又非常少,konga算是相对比较好的一款了,虽然也有一些问题,但整体的功能还比较全,github仓库为:https://github.com/pa ...

  8. 力扣495(java)-提莫攻击(简单)

    题目: 在<英雄联盟>的世界中,有一个叫 "提莫" 的英雄,他的攻击可以让敌方英雄艾希(编者注:寒冰射手)进入中毒状态.现在,给出提莫对艾希的攻击时间序列和提莫攻击的中 ...

  9. What's new in dubbo-go v1.5.6

    简介: dubbogo 社区近期发布了 dubbogo v1.5.6.该版本和 dubbo 2.7.8 对齐,提供了命令行工具,并提供了多种加载配置的方式. 作者 | 铁城  dubbo-go 社区 ...

  10. dotnet 在 UOS 统信系统上运行 UNO 程序输入时闪烁黑屏问题

    本文记录我在虚拟机内安装了 UOS 统信系统,运行 UNO 的基于 Skia 的 Gtk 应用程序时,在输入的过程中不断窗口闪黑问题 本质上说这个问题和 UNO 毫无关系,这是一个 OpenGL 硬件 ...