vue3 使用 konva

1:安装 npm install vue-konva konva --save
        在main.ts 里面引入
import VueKonva from 'vue-konva';
app.use(VueKonva);
2:效果图

3:直接粘贴复制就能用你 (重要的地方做了备注)

<template><div style="display: flex; height: 100vh;"><!-- 左侧 el-tree 树形结构 --><div style="width: 300px; padding: 20px; background: #f0f0f0; border-right: 2px solid #ccc;"><h3>选择图片</h3><el-tree:data="treeData":props="treeProps"draggabledefault-expand-all@node-drag-start="onDragStart"@node-drag-end="onDragEnd"style="width: 100%;"><template #default="{ node, data }"><span>{{ data.label }}</span></template></el-tree></div><!-- 右侧 Konva 画布区域 --><div ref="stageContainer" @dragover.prevent="onDragOver" @drop.prevent="onDrop"style="flex: 1; height: 100%; background: #fafafa; padding: 20px; position: relative;"><button @click="saveCanvas">保存</button><div id="konva-container" style="width: 100%; height: 100%;background:#f0f0f0;"></div></div></div>
</template><script>
import { onMounted, ref, nextTick } from 'vue';
import Konva from 'konva';
import { ElTree } from 'element-plus';export default {components: {ElTree,},setup() {const stage = ref(null);const layer = ref(null);const draggedNode = ref(null);// 树形数据结构const treeData = ref([{id: 1,label: 'Lion',src: 'https://konvajs.org/assets/lion.png',},]);// 树的配置const treeProps = {label: 'label',children: 'children',};// 初始化 Konva stageonMounted(() => {nextTick(() => {// 这里的宽高,可以改成某个容器的宽高// stageContainer.value.clientWidth// stageContainer.value.clientHeightstage.value = new Konva.Stage({container: 'konva-container',width: window.innerWidth - 350,height: window.innerHeight,});layer.value = new Konva.Layer();stage.value.add(layer.value);// 添加右键菜单事件处理stage.value.on('contextmenu', function (e) {e.evt.preventDefault(); // 阻止默认的右键菜单const clickedItem = e.target;            if (clickedItem.hasName('image')) {console.log(clickedItem)alert("可以做一些提示,比如是否要删除,目前右键直接删除了")// 确认删除// proxy.$modal.confirm('是否确认删除该设备?').then(() => {clickedItem.destroy(); // 删除图片// 同时删除可能存在的变换器stage.value.find('Transformer').forEach((tr) => tr.destroy());layer.value.draw(); // 重绘画布// }).catch(() => {});}});// 点击事件处理stage.value.on('click', function (e) {const clickedItem = e.target;          // 移除所有现有的变换器stage.value.find('Transformer').forEach((tr) => tr.destroy());          // 如果点击的是图片,添加变换器if (clickedItem.hasName('image')) {const transformer = new Konva.Transformer({nodes: [clickedItem],boundBoxFunc: (oldBox, newBox) => {// 限制缩放和移动范围if (newBox.width < 50 || newBox.height < 50) {return oldBox;}return newBox;}});layer.value.add(transformer);}          layer.value.draw();});});});// 拖拽开始时记录节点信息const onDragStart = (event, nodeData) => {draggedNode.value = nodeData;};// 拖拽结束const onDragEnd = () => {draggedNode.value = null;};// 拖拽悬停处理const onDragOver = (event) => {event.preventDefault();};// 拖放处理const onDrop = (event) => {event.preventDefault();if (draggedNode.value) {// 获取相对于 Konva 容器的准确坐标const container = document.getElementById('konva-container');const containerRect = container.getBoundingClientRect();// 这里减50是因为宽是100,减去50为了让图片在鼠标中间// 高同理const x = event.clientX - containerRect.left-50;const y = event.clientY - containerRect.top-50;addImageToCanvas("https://konvajs.org/assets/lion.png", x, y);}};// 将图片添加到 Konva 画布const addImageToCanvas = (imageUrl, x, y) => {const imageObj = new Image();imageObj.src = imageUrl;imageObj.onload = () => {const konvaImage = new Konva.Image({image: imageObj,x: x,y: y,// 自定义字段attrs: { device: "2", adad: "2131", yrr: "2313" },// 默认的宽高width: 100,height: 100,draggable: true,name: 'image',dragBoundFunc: function(pos) {const container = stage.value.container();const containerRect = container.getBoundingClientRect();const minX = 0;const minY = 0;const maxX = containerRect.width - 100;  // 减去默认设置的宽const maxY = containerRect.height - 100; // 减去默认设置的高return {x: Math.max(minX, Math.min(pos.x, maxX)),y: Math.max(minY, Math.min(pos.y, maxY))};}});layer.value.add(konvaImage);layer.value.draw();};};const saveCanvas =async () => {if (!stage.value) return;const images = layer.value.find('Image');const layoutData = images.map(image => {return { pointx: image.x(),pointy: image.y(),width: image.scaleX() * image.width(),height: image.scaleY() * image.height(),rotation: image.rotation()};});console.log(layoutData)try {alert('保存成功');} catch (error) {alert('保存失败');}};return {treeData,treeProps,onDragStart,onDragEnd,onDragOver,onDrop,saveCanvas};},
};
</script><style scoped>
#konva-container {border: 2px dashed #ccc;
}
</style>

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring Boot集成Knife4j文档工具

Knife4j 搭建 Knife4j环境的的搭建和Swagger一样都比较简单&#xff0c;只需要极简的配置即可。 maven依赖 我使用的是较高版本的基于openapi规范的依赖包&#xff0c;OpenAPI2(Swagger)规范是Knife4j之前一直提供支持的版本&#xff0c;底层依赖框架为Springfox。 此次在4…

keil5搜索框还有左侧文件状态栏不见的问题

点击上面的window&#xff0c;弹出 reset view to default &#xff0c;然后点击&#xff0c;再点击reset&#xff0c;就ok了

基于Mybatis,MybatisPlus实现数据库查询分页功能

基于Mybatis&#xff0c;MybatisPlus实现数据库查询分页功能 目录 基于Mybatis&#xff0c;MybatisPlus实现数据库查询分页功能使用Mybatis插件实现分页数据库准备分页插件配置和使用常用数据&#xff1a; 使用MybatisPlus插件实现分页数据库准备分页插件配置和使用自定义分页查…

HNU_多传感器(专选)_作业4(构建单层感知器实现分类)

1. (论述题)&#xff08;共1题&#xff0c;100分&#xff09; 假设平面坐标系上有四个点&#xff0c;要求构建单层感知器实现分类。 (3,3),(4,3) 两个点的标签为1&#xff1b; (1,1),(0,2) 两个点的标签为-1。 思路&#xff1a;要分类的数据是2维数据&#xff0c;需要2个输入…

内圆弧转子泵绘制工具开发

接着上期的Gerotor 泵的话题继续。最近有小伙伴找我开发一个内圆弧摆线泵的计算绘制工具&#xff0c;也就是把上次计算绘制的过程做成一个桌面应用工具&#xff0c;这样用起来会更方便、效率更高。那究竟是什么样的工具呢&#xff1f;一起来看看&#xff1a; 前面不是已经有了上…

面试技术点之安卓篇

一、基础 二、高级 三、组件 Android中SurfaceView和TextureView有什么区别&#xff1f; 参考 Android中SurfaceView和TextureView有什么区别&#xff1f; 四、三方框架 五、系统源码 六、性能优化

在Ubuntu 2404上使用最新的PicGo

在转向Ubuntu之后&#xff0c;果断下载了今年最新的Ubuntu2404,但是随之而来的是底层组件的更新&#xff0c;很多以前可以畅快使用的软件&#xff0c;因为需要老版本的组件而不能正确运行&#xff0c;PicGo就是如此 我们从这里打开Release列表 其中Ubuntu可用的只有这个AppIma…

用ue5打开网址链接

需要用到 Launch URL 这个函数 字面意思就是打开填写的链接网页 这里填写的是百度&#xff0c;按下Tab键后就会打开百度的网页

ISP(Image Signal Processor)——HDR技术总结

传统多帧融合技术 拍摄一系列不同曝光时长的图像帧&#xff08;LDR&#xff09;&#xff0c;然后使用融合算法进行融合成HDR图像。 融合算法可以分为两种 基于照度图估计的融合 基于照度估计需要拟合相机响应函数&#xff0c;详细可以参考如下论文&#xff1a; Recovering H…

C++打造局域网聊天室第七课: Socket编程初步2

文章目录 前言一、Socket的API函数二、服务端建立Socket步骤总结 前言 C打造局域网聊天室第七课&#xff1a; Socket编程初步2 一、Socket的API函数 接着上一课的内容&#xff0c;我们在chartroom.cpp中找到如下位置 插入断点&#xff0c;运行 运行到断点处后&#xff0c;按…

【razor】echo搭配relay功能分析

echo 要搭配relay 实现作者说relay在linux上跑,可以模拟丢包、延迟目前没看到如何模拟。relay监听9200,有俩作用 echopeer1 发relay,replay 把peer1的包给peer2 ,实现p2p能力。 接收端:采集后发送发给relay的 接收端的地址就是自己,的地址就是本地的9200,因此是让relay接…

Wallpaper壁纸制作学习记录11

操控变形动画介绍 操控变形是使用Wallpaper Engine为角色和某些对象创建复杂动画的一种高级方法。操控变形是一个多步骤过程&#xff0c;要求您有一个单独的图像图层&#xff0c;其中包含要制作动画的角色或对象的剪切图。 操控变形功能相对复杂&#xff0c;您也可以创建仅包含…

黑马程序员Java项目实战《苍穹外卖》Day12

苍穹外卖-day12 课程内容 工作台Apache POI导出运营数据Excel报表 功能实现&#xff1a;工作台、数据导出 工作台效果图&#xff1a; 数据导出效果图&#xff1a; 在数据统计页面点击数据导出&#xff1a;生成Excel报表 1. 工作台 1.1 需求分析和设计 1.1.1 产品原…

windows下Qt5自动编译配置QtMqtt环境

原文链接&#xff1a;windows下Qt5自动编译配置QtMqtt环境&#xff08;11&#xff09;-CSDN博客 1、概述 Qt默认是不包含mqtt库的&#xff0c;如果需要使用到mqtt库就只能自己编译配置&#xff1b;网络所有的QtMqtt配置的文章都是编译完成手动复制&#xff0c;非常麻烦&#x…

阿里云服务器Linux(centos)系统安装nginx1.20.2

阿里云服务器Linux(centos)系统安装nginx1.20.2 1.安装依赖包 一共要安装4种依赖&#xff08;基于c语言&#xff09; yum -y install gcc zlib zlib-devel pcre-devel openssl openssl-devel2.下载nginx安装包并解压安装包 nginx官网下载&#xff1a;http://nginx.org/en/do…

【C++】三角形校验和算法优化的深入分析

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C 文章目录 &#x1f4af;前言&#x1f4af;题目描述&#x1f4af;老师的解法实现代码解法分析优势缺点 &#x1f4af;我的解法实现代码解法分析优势缺点 &#x1f4af;数学解释&#xff1a;为什么新增条件是冗余的&a…

【C++软件调试技术】dump文件类型与dump文件生成方法详解

目录 1、概述 2、dump文件的分类 2.1、dump按大小分类 2.2、查看dump文件中函数调用堆栈中变量的值 3、调用SetUnhandledExceptionFilter设置异常处理回调函数&#xff0c;然后调用MiniDumpWriteDump生成dump文件 4、使用Google开源库CrashRpt捕获异常&#xff0c;并自动…

持有CSPM证书可以用于评职称吗?

CSPM的诞生背景 CSPM&#xff08;Certified Strategic Project Manager&#xff09;即项目管理专业人员能力评价等级证书&#xff0c;是由中国标准化协会&#xff08;CAS&#xff09;根据国标《项目管理专业人员能力评价要求》&#xff08;GB/T 41831-2022&#xff09;推出的项…

SpringBoot左脚进门之常用注解

类级别注解 SpringBootApplication Configuration //表明这是一个配置类 EnableAutoConfiguration //开启自动配置 ComponentScan() //开启组件扫描1、Configuration&#xff1a; 当一个类被 Configuration 注解…

刷题日志【4】

目录 1、猜数字大小 1、猜数字大小 题意有点抽象&#xff0c;我大概讲一下&#xff0c;就是在1——n里面会有一个目标数&#xff0c;我们通过猜数字的方式逼近这个数字&#xff0c;直到解出这个数&#xff0c;之前我们是用二分法求最快达到求解的问题&#xff0c;这道题多了每…