Vue3 实现拖拽小图片覆盖大图片并下载合并后的图片

体验地址

技术栈

  • Vue3: 用于构建用户界面。
  • HTML5 Canvas: 用于绘制和合并图片。
  • JavaScript: 处理拖拽和下载逻辑。

功能概述

  1. 拖拽小图片: 用户可以通过鼠标拖拽小图片到大图片的任意位置。
  2. 边界限制: 确保小图片不会超出大图片的边界。
  3. 下载合并后的图片: 用户可以点击按钮下载合并后的图片。

实现步骤

1. 创建 Vue 组件

<template><div class="image-container" @mousemove="handleMouseMove" @mouseup="handleMouseUp"><img:src="largeImageUrl"alt="Large Image"class="large-image"ref="largeImageRef"crossorigin="anonymous"/><img:src="smallImageUrl"alt="Small Image"class="small-image"ref="smallImageRef"width="60":style="{ left: smallImagePosition.x + 'px', top: smallImagePosition.y + 'px' }"crossorigin="anonymous"@mousedown="handleMouseDown"/></div>
</template><script>
import { ref, onMounted } from "vue";
import { uploadFileFun } from "@/utils/minio.js";
import { dataURLtoFile } from "@/utils/methods";export default {setup() {const largeImageUrl ="https://";const smallImageUrl ="https:/";const smallImagePosition = ref({ x: 0, y: 0 });const largeImageRef = ref(null);const smallImageRef = ref(null);let isDragging = false;let dragStart = { x: 0, y: 0 };const handleMouseDown = (event) => {isDragging = true;dragStart = {x: event.clientX - smallImagePosition.value.x,y: event.clientY - smallImagePosition.value.y,};};const handleMouseMove = (event) => {if (!isDragging) return;const largeImage = largeImageRef.value;const smallImage = smallImageRef.value;const smallImageWidth = smallImage.width;const smallImageHeight = smallImage.height;let newX = event.clientX - dragStart.x;let newY = event.clientY - dragStart.y;// 边界限制,考虑小图片的大小newX = Math.max(0, Math.min(newX, largeImage.width - smallImageWidth));newY = Math.max(0, Math.min(newY, largeImage.height - smallImageHeight));smallImagePosition.value = { x: newX, y: newY };};const handleMouseUp = () => {isDragging = false;};const saveAsImage = async () => {const largeImage = largeImageRef.value;const smallImage = smallImageRef.value;// 获取大图片的缩放比例const scaleX = largeImage.naturalWidth / largeImage.width;const scaleY = largeImage.naturalHeight / largeImage.height;const canvas = document.createElement("canvas");canvas.width = largeImage.naturalWidth;canvas.height = largeImage.naturalHeight;const ctx = canvas.getContext("2d");// 绘制大图片ctx.drawImage(largeImage, 0, 0);// 计算小图片在Canvas上的缩放后位置const scaledX = smallImagePosition.value.x * scaleX;const scaledY = smallImagePosition.value.y * scaleY;// 绘制小图片ctx.drawImage(smallImage,scaledX,scaledY,smallImage.width * scaleX, // 小图片的缩放宽度smallImage.height * scaleY // 小图片的缩放高度);const dataUrl = canvas.toDataURL("image/png");const link = document.createElement("a");link.href = dataUrl;link.download = "combined-image.png";link.click();};onMounted(() => {const saveButton = document.createElement("button");saveButton.innerText = "下载";saveButton.onclick = saveAsImage;document.body.appendChild(saveButton);});return {largeImageUrl,smallImageUrl,smallImagePosition,largeImageRef,smallImageRef,handleMouseDown,handleMouseMove,handleMouseUp,};},
};
</script><style scoped>
.image-container {position: relative;width: 350px;height: 450px;
}
.large-image {width: 100%;height: 100%;
}
.small-image {position: absolute;cursor: move;
}
</style>

2. 拖拽小图片

使用 ref 来引用大图片和小图片,并通过监听鼠标事件来实现拖拽功能。

  • handleMouseDown: 当用户按下鼠标左键时,记录拖拽起点的位置。
  • handleMouseMove: 当用户移动鼠标时,更新小图片的位置,并限制其在大图片范围内移动。
  • handleMouseUp: 当用户释放鼠标左键时,停止拖拽。

3. 下载合并后的图片

使用 HTML5 Canvas 来绘制合并后的图片,并提供下载功能。

  • saveAsImage: 创建一个 Canvas 元素,绘制大图片和小图片,然后将 Canvas 内容转换为 Data URL,并生成一个下载链接。

4. 添加下载按钮

在组件挂载完成后,动态创建一个下载按钮并添加到页面中。

onMounted(() => {const saveButton = document.createElement("button");saveButton.innerText = "下载";saveButton.onclick = saveAsImage;document.body.appendChild(saveButton);
});

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

基于SpringBoot+Vue音乐播放和推荐系统【提供源码+答辩PPT+参考文档+项目部署】

作者简介&#xff1a;✌CSDN新星计划导师、Java领域优质创作者、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和学生毕业项目实战,高校老师/讲师/同行前辈交流。✌ 主要内容&#xff1a;&#x1f31f;Java项目、Python项目、前端项目、PHP、ASP.NET、人工智能…

震撼!通过双重异步,Excel 10万行数据导入从191秒优化到2秒!

震撼&#xff01;通过双重异步&#xff0c;Excel 10万行数据导入从191秒优化到2秒&#xff01; 在现代的企业级应用开发中&#xff0c;海量数据的处理效率和并发性能优化是一个非常重要的课题。无论是大规模数据导入、文件解析&#xff0c;还是在分布式系统中处理高并发任务&a…

Linux编程:用于调试 C、C++ 和其他编程语言编写的程序的调试工具GDB的使用

目录 一、概述 二、 安装GDB 三、准备程序 四、使用GDB 1、启动GDB 2、获取帮助 五、 常用GDB命令 六、示例调试会话 七、其他事项 一、概述 GDB&#xff08;GNU Debugger&#xff09;是一个非常强大的调试工具&#xff0c;广泛用于调试 C、C 和其他编程语言编写的程序…

书生实战营第四期-基础岛第五关-XTuner 微调个人小助手认知

基础任务 使用 XTuner 微调 InternLM2-Chat-7B 实现自己的小助手认知 一、环境配置与数据准备 1.构建虚拟环境 cd ~ #git clone 本repo git clone https://github.com/InternLM/Tutorial.git -b camp4 mkdir -p /root/finetune && cd /root/finetune conda create -…

java day04-面向对象基础(内存 封装 继承 修饰符 工具类 )

1.对象内存图 1.1 Java 内存分配 1.2 堆和栈 栈:所有局部变量都会在栈内存中创建 局部变量&#xff1a;定义在方法中的变量或者方法声明上的变量 方法执行都会加载到栈中进行 -----------------------------------------------------------------------------------------…

【C++练习】二进制到十进制的转换器

题目&#xff1a;二进制到十进制的转换器 描述 编写一个程序&#xff0c;将用户输入的8位二进制数转换成对应的十进制数并输出。如果用户输入的二进制数不是8位&#xff0c;则程序应提示用户输入无效&#xff0c;并终止运行。 要求 程序应首先提示用户输入一个8位二进制数。…

Pytorch学习--神经网络--网络模型的保存与读取

一、网络模型的保存与读取方式1 方法讲解 保存模型 import torch import torchvision model torchvision.models.vgg16(weightsDEFAULT) #保存模型和参数 torch.save(model,"save_method1.pth")读取模型 import torch model torch.load("save_method1.pth&…

凸优化理论,凸二次规划问题,对偶问题及KKT条件

凸优化理论 ​ 研究凸优化之前我们不妨提出几个小问题&#xff1a; 什么是优化问题&#xff1f;优化问题的解是什么&#xff1f;什么是凸优化问题&#xff1f;凸优化问题的解决方案是什么&#xff1f; 1.1 优化问题 ​ 理解优化问题其实很简单&#xff0c;我们其实从高中事…

智能的编织:C++中auto的编织艺术

在C的世界里&#xff0c;auto这个关键字就像是一个聪明的助手&#xff0c;它能够自动帮你识别变量的类型&#xff0c;让你的代码更加简洁和清晰。下面&#xff0c;我们就来聊聊auto这个关键字的前世今生&#xff0c;以及它在C11标准中的新用法。 auto的前世 在C11之前&#x…

The Rank-then-Encipher Approach

原始观点 Format-Preserving Encryption 4 The Rank-then-Encipher Approach 引用1 Hybrid diffusion-based visual image encryption for secure cloud storage 2.2 Sum-preserving encryption Bellare introduced the concept of format-preserving encryption (FPE)…

江西省补贴性线上职业技能培训管理平台(刷课系统)

江西省补贴性线上职业技能培训管理平台(刷课系统) 目的是为了刷这个网课 此系统有两个版本一个是脚本运行&#xff0c;另外一个是可视化界面运行 可视化运行 技术栈:flask、vue3 原理: 通过分析网站接口&#xff0c;对某些接口加密的参数进行逆向破解&#xff0c;从而修改请求…

Linux基础4-进程5(程序地址空间详解)

上篇文章:Linux基础4-进程4&#xff08;环境变量&#xff0c;命令行参数详解&#xff09;-CSDN博客 本章重点&#xff1a; 1 重新理解c/c地址空间 2 虚拟地址空间 一. c/c地址空间 地址空间布局图: 运行下列代码&#xff0c;进行观察 #include <stdio.h> #include <…

动态规划-背包问题——[模版]01背包(背包母题)

1.题目解析 题目来源 [模版]01背包_牛客题霸_牛客网 测试用例 2.算法原理 1.状态表示 第一小问&#xff1a;求最大价值 第二小问&#xff1a;求充满时的价值 2.状态转移方程 第一小问&#xff1a;求最大价值 第二小问&#xff1a;求充满时的价值 3.初始化 第一小问&#xff1a…

JavaWeb之会话跟踪技术

前言 这一节主要讲会话跟踪技术 1.补充 为了提交Gitee我修改了模块的目录&#xff0c;就是移动了模块&#xff0c;导致模块不是Maven了&#xff0c;可以在右边的Maven小工具&#xff0c;点加号&#xff0c;把模块重新添加为Maven 2. 概述 3. Cookie 3.1 基本使用 //发送coo…

第二十周周报:回顾篇

目录 摘要 Abstract 1 深度学习基础知识 1.1 学习率 1.1.1 自适应学习率 1.1.2 学习率调度 1.2 归一化 1.2.1 批量归一化 1.2.2 特征归一化 1.3 激活函数 1.3.1 Sigmoid函数 1.3.2 Tanh函数 1.3.3 ReLU函数 1.3.4 Leak ReLU函数 1.3.5 PReLU函数 1.3.6 ELU函数…

智能化SCRM方案助力企业高效管理与营销转型

内容概要 现代企业面临着复杂多变的市场环境&#xff0c;传统的管理与营销方式常常无法满足日益增长的需求。这时&#xff0c;智能化SCRM方案便应运而生&#xff0c;为企业带来了新的机遇与挑战。智能化SCRM方案不仅仅是一个单一的工具&#xff0c;它更像是一个全面的解决方案…

PRD2012学习笔记

图例位置&#xff1a; 使用 loc‘upper left’ 指定图例的基本位置为左上角。 使用 bbox_to_anchor(0.1, 0.9) 来进行自定义位置调整&#xff0c;其中 (0.1, 0.9) 指定图例相对于图形区域的坐标 (x, y)。 0.1 表示距离左边界的比例位置&#xff0c;0.9 表示距离上边界的比例位置…

【01课_初识算法与数据结构】

一、理解算法 1、算法的概念 算法&#xff0c;个人理解就是计算一段逻辑&#xff0c;最简化&#xff0c;最快速的方式、方法 每个函数&#xff0c;就包含了一定的算法&#xff0c;执行一定的计算逻辑 算法是一系列程序指令&#xff0c;用于解决特定的运算和逻辑问题 2、衡…

《⼆叉搜索树》

《⼆叉搜索树》 1. ⼆叉搜索树的概念2. ⼆叉搜索树的性能分析3 二叉树的功能说明及实现3.1 ⼆叉搜索树的插⼊3.2 ⼆叉搜索树的查找3.3 ⼆叉搜索树的删除 4二叉搜索树的实现代码5 ⼆叉搜索树key和key/value使⽤场景5.1 key搜索场景&#xff1a;5.2 key/value搜索场景&#xff1a…

stm32 踩坑笔记

串口问题&#xff1a; 问题&#xff1a;会改变接收缓冲的下一个字节 串口的初始化如下&#xff0c;位长度选择了9位。因为要奇偶校验&#xff0c;要选择9位。但是接收有用数据只用到1个字节。 问题原因&#xff1a; 所以串口接收时会把下一个数据更改