js函数预览图片:支持鼠标和手势拖拽缩放
对之前的方式改进:原生js实现图片预览控件,支持丝滑拖拽,滚轮放缩,放缩聚焦_js图片预览-CSDN博客
/*** 图片预览函数,调用后自动预览图片* @param {图片地址} imgurl*/
function openImagePreview(imgurl) {if (!imgurl) return;const max = 30;//放缩原图倍数const scale = 0.1;//滚轮一下缩放的变化比例 越大放缩越快const box = document.createElement("div");box.style.cssText = "position: fixed;top:0px;left:0px;z-index: 10000;background-color: rgba(0,0, 0, 0.3);width:100%;height:100%;overflow: hidden;cursor: grab;";const img = document.createElement("img");img.style.cssText = "position: absolute;object-fit: contain;user-select: none;-webkit-user-drag: none;user-select: none;-moz-user-select: none;-webkit-user-select: none;-ms-user-select: none;";img.src = imgurl;img.title = "预览";const close = document.createElement("div");close.innerHTML = `<svg aria-hidden="true" role="img" font-size="20" width="1em" height="1em" preserveAspectRatio="xMidYMid meet" viewBox="0 0 16 16"><g fill="none" fill-rule="evenodd"><path d="M0 0h16v16H0z"></path><path fill="currentColor" d="M13.303 2.697a.5.5 0 010 .707L8.707 8l4.596 4.596a.5.5 0 01-.707.707L8 8.707l-4.596 4.596a.5.5 0 01-.707-.707L7.293 8 2.697 3.404a.5.5 0 01.707-.707L8 7.293l4.596-4.596a.5.5 0 01.707 0z"></path></g></svg>`;close.style.cssText = "position: absolute;top:20px;right:20px;align-items: center;color:white;display: flex;justify-content: center;background: rgba(29,29,37,0.4);border-radius: 50%;width:30px;height:30px;";close.onclick = () => {document.body.removeChild(box);}box.append(img);box.append(close);document.body.append(box);img.onload = function () {const imgRatio = img.naturalWidth / img.naturalHeight; //图片横纵比const boxRatio = box.clientWidth / box.clientHeight; //容器横纵比if (imgRatio > boxRatio) {const scale = box.clientWidth / img.naturalWidth;img.style.width = box.clientWidth + "px"; //长度填充img.style.height = img.naturalHeight * scale + "px"; //高度自适应img.style.top = Math.ceil((box.clientHeight - img.clientHeight) / 2) + "px";//位置居中} else {const scale = box.clientHeight / img.naturalHeight;img.style.height = box.clientHeight + "px";//高度填充img.style.width = img.naturalWidth * scale + "px";//长度自适应img.style.left = Math.ceil((box.clientWidth - img.clientWidth) / 2) + "px";//位置居中}const download = document.createElement("div");download.innerHTML = `<svg style="width: 1em;height: 1em;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" ><path d="M511.3 856.9c-11.3 0-20.4-9.1-20.4-20.4V87c0-11.3 9.1-20.4 20.4-20.4s20.4 9.2 20.4 20.4v749.5c0.1 11.2-9.1 20.4-20.4 20.4z" fill="#FFFFFF" ></path><path d="M511.3 857.3c-20.5 0-39.7-8-54.2-22.4L187.3 565c-8-8-8-20.9 0-28.9s20.9-8 28.9 0L486.1 806c6.7 6.7 15.7 10.5 25.2 10.5s18.5-3.7 25.2-10.5l269.9-269.9c8-8 20.9-8 28.9 0s8 20.9 0 28.9L565.5 834.9c-14.5 14.5-33.7 22.4-54.2 22.4zM170.8 961.2c-11 0-20.1-8.8-20.4-19.9-0.3-11.3 8.6-20.7 19.9-21l681-18.1h0.6c11 0 20.1 8.8 20.4 19.9 0.3 11.3-8.6 20.7-19.9 21l-681 18.1h-0.6z" fill="#FFFFFF" ></path></svg>`;download.style.cssText = "position: absolute;top:20px;right:60px;align-items: center;color:white;display: flex;justify-content: center;background: rgba(29,29,37,0.4);border-radius: 50%;width:30px;height:30px;";download.onclick = () => {let fileName = imgurl;let downloadUrl = imgurl;try {const regex = /([^/\\?&]+(?:\.[a-zA-Z0-9\u4e00-\u9fa5_-]+))(?:[?#]|$)/;const match = imgurl.match(regex);fileName = match ? match[0] : null;if (!fileName) {fileName = new Date().getTime() + ".png";}} catch (e) {}if (fileName.trim().toLowerCase().endsWith(".svg")) {fileName = fileName.substring(0, fileName.lastIndexOf(".svg")) + ".png";// 创建一个 canvas 元素const canvas = document.createElement('canvas');const ctx = canvas.getContext('2d');canvas.width = img.clientWidth;canvas.height = img.clientHeight;ctx.drawImage(img, 0, 0);downloadUrl = canvas.toDataURL('image/png');}let link = document.createElement('a')link.style.display = 'none'link.href = downloadUrllink.target = "_blank";link.setAttribute('download', fileName);document.body.appendChild(link)link.click()document.body.removeChild(link) //下载完成移除元素window.URL.revokeObjectURL(downloadUrl) //释放掉blob对象}box.append(download);};const getRange = (actual, limita, limitb) => {if (actual < -limita) {return -limita;} else if (actual > limitb) {return limitb;}return actual;}const drag = {status: false, lastX: null, lastY: null}const dragStart = (x, y) => {if (drag === true) return;drag.status = truedrag.lastX = xdrag.lastY = y}const dragMove = (x, y) => {if (drag.status) {let mx = x - drag.lastXlet my = y - drag.lastYdrag.lastX = xdrag.lastY = ylet top = img.offsetTop + mylet left = img.offsetLeft + mximg.style.left = getRange(left, img.clientWidth - 10, box.clientWidth - 10) + 'px';img.style.top = getRange(top, img.clientHeight - 10, box.clientHeight - 10) + "px";}}const dragEnd = () => {drag.status = falsedrag.lastX = nulldrag.lastY = null}const zoom = {status: false,lastDistance: null,lastCenter: null,}const getCenter = (x1, y1, x2, y2) => {return {x: (x1 + x2) / 2, y: (y1 + y2) / 2}}const getDistance = (x1, y1, x2, y2) => {const dx = x2 - x1;const dy = y2 - y1;return Math.sqrt(dx * dx + dy * dy); // 返回距离}const zoomStart = (x1, y1, x2, y2) => {zoom.lastDistance = getDistance(x1, y1, x2, y2);zoom.center = getCenter(x1, y1, x2, y2)dragStart(zoom.center.x, zoom.center.y);}const zoomCore = (isChangeBig, center_x, center_y, changeScale) => {const top_d = center_y - img.offsetTop;const left_d = center_x - img.offsetLeft;let modifyHeight = img.clientHeight * changeScale;let modifyWidth = img.clientWidth * changeScale;if (isChangeBig && modifyHeight > img.naturalHeight * max) return;if (!isChangeBig && modifyHeight * max < img.naturalHeight) return;img.style.height = modifyHeight + "px";img.style.width = modifyWidth + "px";img.style.top = (center_y - top_d * changeScale) + "px";img.style.left = (center_x - left_d * changeScale) + "px";}const zoomMove = (x1, y1, x2, y2) => {let newDistance = getDistance(x1, y1, x2, y2);let center = getCenter(x1, y1, x2, y2)if (Math.abs(newDistance - zoom.lastDistance) > 0.1) {zoomCore(zoom.lastDistance < newDistance, center.x, center.y, newDistance / zoom.lastDistance);}zoom.lastDistance = newDistance;}const zoomEnd = () => {zoom.lastDistance = null;zoom.center = null;dragEnd();}box.ontouchstart = (event) => {if (event.touches.length === 1) {dragStart(event.touches[0].pageX, event.touches[0].pageY);} else if (event.touches.length === 2) {zoomStart(event.touches[0].pageX, event.touches[0].pageY, event.touches[1].pageX, event.touches[1].pageY)}box.ontouchmove = (e) => {if (event.touches.length === 1) {dragMove(e.touches[0].pageX, e.touches[0].pageY);} else if (e.touches.length === 2) {zoomMove(e.touches[0].pageX, e.touches[0].pageY, e.touches[1].pageX, e.touches[1].pageY)}}box.ontouchend = () => {zoomEnd();box.ontouchmove = null;box.ontouchend = null;}}box.onmousedown = (event) => {document.body.style.userSelect = 'none';if (event.button === 0) {document.onmousemove = (event) => {dragMove(event.clientX, event.clientY);}document.onmouseup = (event) => {document.body.style.userSelect = '';if (event.button === 0) {dragEnd();document.onmousemove = nulldocument.onmouseup = null}}dragStart(event.clientX, event.clientY);}}box.onwheel = (event) => {event.preventDefault(); //关闭默认事件zoomCore(event.deltaY < 0, event.clientX, event.clientY, event.deltaY > 0 ? (1 - scale) : (1 + scale));}
}
相关文章:

js函数预览图片:支持鼠标和手势拖拽缩放
对之前的方式改进:原生js实现图片预览控件,支持丝滑拖拽,滚轮放缩,放缩聚焦_js图片预览-CSDN博客 /*** 图片预览函数,调用后自动预览图片* param {图片地址} imgurl*/ function openImagePreview(imgurl) {if (!imgurl…...

用QT实现 端口扫描工具1
安装在线QT,尽量是完整地自己进行安装,不然会少包 参考【保姆级图文教程】QT下载、安装、入门、配置VS Qt环境-CSDN博客 临时存储空间不够。 Windows系统通常会使用C盘来存储临时文件。 修改临时文件存储位置 打开系统属性: 右键点击“此电…...

设计模式 结构型 适配器模式(Adapter Pattern)与 常见技术框架应用 解析
适配器模式(Adapter Pattern)是一种结构型设计模式,它允许将一个类的接口转换成客户端所期望的另一个接口,从而使原本因接口不兼容而无法一起工作的类能够协同工作。这种设计模式在软件开发中非常有用,尤其是在需要集成…...

vue 项目集成 electron 和 electron 打包及环境配置
vue electron 开发桌面端应用 安装 electron npm i electron -D记得加上-D,electron 需添加到devDependencies,如果添加到dependencies后面运行可能会报错 根目录创建electron文件夹,在electron文件夹创建main.js(或者backgrou…...

vscode如何离线安装插件
在没有网络的时候,如果要安装插件,就会麻烦一些,需要通过离线安装的方式进行。下面记录如何在vscode离线安装插件。 一、下载离线插件 在一台能联网的电脑中,下载好离线插件,拷贝到无法联网的电脑上。等待安装。 vscode插件商店地址:https://marketplace.visualstudio.co…...
计算机网络常见面试题及解答
以下是计算机网络中常见的面试题及解答,按主题分类: --- ## **一、基础概念** ### **1. OSI 七层模型和 TCP/IP 模型的区别是什么?** **答:** - **OSI 七层模型:** - 应用层、表示层、会话层、传输层、网络层、数…...
举例说明AI模型怎么聚类,最后神经网络怎么保存
举例说明怎么聚类,最后神经网络怎么保存 目录 举例说明怎么聚类,最后神经网络怎么保存K - Means聚类算法实现神经元特征聚类划分成不同专家的原理和过程 特征提取: 首先,需要从神经元中提取有代表性的特征。例如,对于一个多层感知机(MLP)中的神经元,其权重向量可以作为特…...

HarmonyOS NEXT应用开发实战(一):边学边玩,从零开发一款影视APP
引言 学习一项技能,最好也最快的办法就是动手实战。通过自己给自己找项目练习,不仅能够激发兴趣,还能从开发实战中不断总结经验。这种学习方法是最为高效的。今天,我们将通过开发一款名为“爱影家”的影视APP,来学习H…...

STM32G0B1 can Error_Handler 解决方法
问题现象 MCU上电,发送0x13帧数据固定进入 Error_Handler 硬件介绍 MCU :STM32G0B1 can:NSI1042 tx 接TX RX 接RX 折腾了一下午,无解,问题依旧; 对比测试 STM32G431 手头有块G431 官方评估版CAN 模块; 同样的…...
使用 `llama_index` 构建智能问答系统:多种文档切片方法的评估
使用 llama_index 构建智能问答系统:多种文档切片方法的评估 代码优化与解析1. **代码结构优化**2. **日志管理**3. **环境变量管理**4. **模型初始化**5. **提示模板更新**6. **问答函数优化**7. **索引构建与查询引擎**8. **节点解析器测试** 总结 在现代自然语言…...

【大模型】7 天 AI 大模型学习
7 天 AI 大模型学习 Day 2 今天是 7 天AI 大模型学习的第二天 😄,今天我将会学习 Transformer 、Encoder-based and Decoder-Based LLMs 等 。如果有感兴趣的,就和我一起开始吧 ~ 课程链接 :2025年快速吃透AI大模型&am…...
软件工程大复习之(四)——面向对象与UML
4.1 面向对象概述 面向对象(OO)是一种编程范式,它将数据和处理数据的方法封装在对象中。面向对象的主要概念包括: 对象:实例化的数据和方法的集合。类:对象的蓝图或模板。封装:隐藏对象的内部…...

【Linux】shell命令
目录 shell的基本命令 shell - 贝壳 外在保护工具 用户、shell、内核、硬件之间的关系 解析器的分类: shell命令格式 history -历史记录查询 修改环境变量的值: shell中的特殊字符 通配符 管道 | 输入输出重定向 命令置换符 shell的基本命…...

ValuesRAG:以检索增强情境学习强化文化对齐
随着大型语言模型(LLMs)的迅猛发展,其在各个领域展现出强大的能力。然而,训练数据中西方中心主义的倾向,使得 LLMs 在文化价值观一致性方面面临严峻挑战,这一问题在跨文化场景中尤为突出,可能导…...

【机器学习篇】交通革命:机器学习如何引领未来的道路创新
嘿,你知道吗?机器学习正在交通领域掀起一场革命啦!它将如何引领未来道路创新呢 本文有精彩的 C 代码演示、实用的图片解释,还有超多干货,保证让你大开眼界,点赞收藏关注, 开启一场奇妙的探索之…...

DeepSeek-V3 通俗详解:从诞生到优势,以及与 GPT-4o 的对比
1. DeepSeek 的前世今生 1.1 什么是 DeepSeek? DeepSeek 是一家专注于人工智能技术研发的公司,致力于打造高性能、低成本的 AI 模型。它的目标是让 AI 技术更加普惠,让更多人能够用上强大的 AI 工具。 1.2 DeepSeek-V3 的诞生 DeepSeek-V…...

把vue项目或者vue组件发布成npm包或者打包成lib库文件本地使用
将vue项目发布成npm库文件,第三方通过npm依赖安装使用;使用最近公司接了一个项目,这个项目需要集成到第三方页面,在第三方页面点击项目名称,页面变成我们的项目页面;要求以npm库文件提供给他们;…...

【STC库函数】Compare比较器的使用
如果我们需要比较两个点的电压,当A点高于B点的时候我们做一个操作,当B点高于A点的时候做另一个操作。 我们除了加一个运放或者比较器,还可以直接使用STC内部的一个比较器。 正极输入端可以是P37、P50、P51,或者从ADC的十六个通道…...

单片机-独立按键矩阵按键实验
1、按键介绍 按键管脚两端距离长的表示默认是导通状态,距离短的默认是断开状态, 如果按键按下,初始导通状态变为断开,初始断开状态变为导通 我们开发板是采用软件消抖,一般来说一个简单的按键消抖就是先读取按键的状…...
若要把普通表转成分区表,就需要先新建分区表,然后把普通表中的数据导入新建分区表。 具体怎么导入?
将普通表转换为分区表并导入数据是一个常见的数据库管理任务。以下是详细的步骤和示例,帮助你在 GaussDB 中完成这一过程: 1. 创建分区表 首先,你需要创建一个新的分区表,定义好分区键和分区策略。假设你有一个普通表 orders&am…...
内存分配函数malloc kmalloc vmalloc
内存分配函数malloc kmalloc vmalloc malloc实现步骤: 1)请求大小调整:首先,malloc 需要调整用户请求的大小,以适应内部数据结构(例如,可能需要存储额外的元数据)。通常,这包括对齐调整,确保分配的内存地址满足特定硬件要求(如对齐到8字节或16字节边界)。 2)空闲…...
反向工程与模型迁移:打造未来商品详情API的可持续创新体系
在电商行业蓬勃发展的当下,商品详情API作为连接电商平台与开发者、商家及用户的关键纽带,其重要性日益凸显。传统商品详情API主要聚焦于商品基本信息(如名称、价格、库存等)的获取与展示,已难以满足市场对个性化、智能…...

VB.net复制Ntag213卡写入UID
本示例使用的发卡器:https://item.taobao.com/item.htm?ftt&id615391857885 一、读取旧Ntag卡的UID和数据 Private Sub Button15_Click(sender As Object, e As EventArgs) Handles Button15.Click轻松读卡技术支持:网站:Dim i, j As IntegerDim cardidhex, …...

【力扣数据库知识手册笔记】索引
索引 索引的优缺点 优点1. 通过创建唯一性索引,可以保证数据库表中每一行数据的唯一性。2. 可以加快数据的检索速度(创建索引的主要原因)。3. 可以加速表和表之间的连接,实现数据的参考完整性。4. 可以在查询过程中,…...

DIY|Mac 搭建 ESP-IDF 开发环境及编译小智 AI
前一阵子在百度 AI 开发者大会上,看到基于小智 AI DIY 玩具的演示,感觉有点意思,想着自己也来试试。 如果只是想烧录现成的固件,乐鑫官方除了提供了 Windows 版本的 Flash 下载工具 之外,还提供了基于网页版的 ESP LA…...
React---day11
14.4 react-redux第三方库 提供connect、thunk之类的函数 以获取一个banner数据为例子 store: 我们在使用异步的时候理应是要使用中间件的,但是configureStore 已经自动集成了 redux-thunk,注意action里面要返回函数 import { configureS…...

基于TurtleBot3在Gazebo地图实现机器人远程控制
1. TurtleBot3环境配置 # 下载TurtleBot3核心包 mkdir -p ~/catkin_ws/src cd ~/catkin_ws/src git clone -b noetic-devel https://github.com/ROBOTIS-GIT/turtlebot3.git git clone -b noetic https://github.com/ROBOTIS-GIT/turtlebot3_msgs.git git clone -b noetic-dev…...
日常一水C
多态 言简意赅:就是一个对象面对同一事件时做出的不同反应 而之前的继承中说过,当子类和父类的函数名相同时,会隐藏父类的同名函数转而调用子类的同名函数,如果要调用父类的同名函数,那么就需要对父类进行引用&#…...

Linux部署私有文件管理系统MinIO
最近需要用到一个文件管理服务,但是又不想花钱,所以就想着自己搭建一个,刚好我们用的一个开源框架已经集成了MinIO,所以就选了这个 我这边对文件服务性能要求不是太高,单机版就可以 安装非常简单,几个命令就…...
git: early EOF
macOS报错: Initialized empty Git repository in /usr/local/Homebrew/Library/Taps/homebrew/homebrew-core/.git/ remote: Enumerating objects: 2691797, done. remote: Counting objects: 100% (1760/1760), done. remote: Compressing objects: 100% (636/636…...