vue2 Canvas 多边形区域绘制组件封装

效果预览:

CanvasBox组件

<!-- 区域设置canvas -->
<template><div class="all" ref="divideBox"><!-- <div><button @click="test">清空</button></div> --><img id="img" v-if="imgUrl" :src="imgUrl"><!-- width="700" height="450" --><canvas id="mycanvas" ref="mycanvas" :width="canvasWidth" :height="canvasHeight" @mousedown="canvasDown($event)"@mousemove="canvasMove($event)" @mouseup="canvasUp($event)" @dblclick="doubleclick()">浏览器不支持canvas</canvas></div>
</template><script>
export default {props: {// canvas宽度canvasWidth: {type: Number,default: 0},// canvas高度canvasHeight: {type: Number,default: 0},// 时间戳timeStamp: {type: Number,default: 0},// 图片imgUrl: {type: String,default: ""},// 是否可编辑 detail 不可编辑type: {type: String,default: ""},// 多边形区域aiDataRegionalInfoList: {type: Array,default: []},// 摄像头id// cameraId: {//     default: null// }},watch: {// 清空画布timeStamp() {this.test();},// 接收到多边形区域数据aiDataRegionalInfoList(val) {// console.log("接收到多边形区域数据", val);let newArray = []if (val && val.length > 0) {val.forEach(item => {newArray.push({"cor_x": item.x,"cor_y": item.y})})}if (newArray && newArray.length === 0) return;this.drawPolygon(newArray)// this.draw(val); // this.drawPolygon([//     {//         "cor_x": 443,//         "cor_y": 223//     },//     {//         "cor_x": 672,//         "cor_y": 197//     },//     {//         "cor_x": 562,//         "cor_y": 368//     }// ])}},data() {return {isMultiple: false, // 是否支持多个多边形绘制observer: null,x: null,y: null,isdraw: false, //是否在画图形ctx: null, //canvas对象coordinates: [], //一个多边形的坐标信息 cor_index: 0, //当前多边形的索引endtip: false, //是否结束一个多边形的绘制all_coordinates: [], //所有多边形的信息isdrag: false, //是否正在拖动drag_index: [-1, -1], //当前拖动的多边形坐标}},mounted() {// 监听画布尺寸变化// this.initObserver();setTimeout(() => {//初始化画布对象this.initDraw()}, 500)},// beforeDestroy() {//     // 销毁监听器//     this.cleanupObserver();// },methods: {// 初始化监听器// initObserver() {//     this.observer = new MutationObserver(mutations => {//         mutations.forEach(mutation => {//             if (mutation.type === 'attributes' && mutation.attributeName === 'style') {//                 this.handleResize();//             }//         });//     });//     this.observer.observe(this.$refs.divideBox, { attributes: true });//     this.handleResize(); // 初始化时也调用一次// },// 销毁监听器// cleanupObserver() {//     if (this.observer) {//         this.observer.disconnect();//     }// },// 监听到尺寸变化// handleResize() {//     const divideBox = this.$refs.divideBox;//     const canvas = this.$refs.myCanvas;//     if (divideBox && canvas) {//         const newWidth = divideBox.clientWidth;//         const newHeight = divideBox.clientHeight;//         // console.log('Divide box width:', newWidth);//         // console.log('Divide box height:', newHeight);//         // 更新 canvas 的尺寸//         canvas.width = newWidth;//         canvas.height = newHeight;//         // 进行其他初始化操作//     } else {//         console.error('未找到分割框或画布元素');//     }// },// 监听到尺寸变化// updateBoxWidth(width) {//     let widthVal = width//     let heightVal = width / 1920 * 1080//     // console.log("width", widthVal)//     // console.log("height", heightVal) //     let pointList = JSON.parse(JSON.stringify(this.oldPointList))//     pointList.forEach(item => {//         item.x = item.x / 1920 * widthVal//         item.y = item.y / 1080 * heightVal//         item.type = item.type//         item.isShow = item.isShow//     })//     this.pointList = pointList//     // console.log("更新位置", this.pointList); // },// updateCanvasSize() {//     const canvas = this.$refs.mycanvas;//     this.canvasWidth = canvas.clientWidth;//     this.canvasHeight = canvas.clientHeight;// },// 清空画布test() {this.all_coordinates = [];this.coordinates = [];this.isdraw = false;this.endtip = false;// this.ctx.clearRect(0, 0, 700, 450);this.ctx.clearRect(0, 0, this.canvasWidth, this.canvasHeight);},// 填充区域fillarea() {this.ctx.fillStyle = 'rgba(255, 255, 0,0.4)';for (let i = 0; i < this.all_coordinates.length; i++) {let cors = this.all_coordinates[i];let x0 = cors[0].cor_x;let y0 = cors[0].cor_y;this.ctx.beginPath();this.ctx.moveTo(x0, y0);for (let j = 1; j < cors.length; j++) {let x = cors[j].cor_x;let y = cors[j].cor_y;this.ctx.lineTo(x, y);}this.ctx.fill();this.ctx.closePath();}},//初始化画布对象initDraw() {const canvas = document.querySelector("#mycanvas");this.ctx = canvas.getContext("2d");// this.ctx.strokeStyle = 'rgb(255, 255, 0)';this.ctx.strokeStyle = 'yellow';},// 判断是否是拖拽点isdragpoint(x, y) {if (this.all_coordinates.length == 0) {return false;}for (let i = 0; i < this.all_coordinates.length; i++) {for (let j = 0; j < this.all_coordinates[i].length; j++) {let px = this.all_coordinates[i][j].cor_x;let py = this.all_coordinates[i][j].cor_y;if (Math.abs(x - px) <= 5 && Math.abs(y - py) <= 5) {this.drag_index[0] = i;this.drag_index[1] = j;return true;}}}return false;},// 画布中鼠标按下canvasDown(e) {if (this.type === 'detail') return;// console.log(123, e.offsetX, e.offsetY);// const cameraId = this.cameraId// if (cameraId == "" || cameraId == undefined || cameraId == null) {//     this.$message({//         message: '请先选择摄像头',//         type: 'warning'//     });//     return;// }// console.log("鼠标按下", this.coordinates);// console.log("鼠标按下", this.all_coordinates);let x = e.offsetX;let y = e.offsetY;if (this.isdragpoint(x, y)) {this.isdrag = true;return 0;}//画布中鼠标按下if (this.endtip) {//已经结束了上个多边形的绘制,把上个多边形的坐标放入数组,同时清空单个多边形数组信息this.endtip = false;}if (this.all_coordinates && this.all_coordinates.length == 1) {this.$message({message: '最多只能绘制一个区域',type: 'warning'});this.coordinates = []return;}//获取鼠标按下的坐标,放入数组中this.coordinates.push({ cor_x: x, cor_y: y });this.isdraw = true; //正在画多边形},// 画布中鼠标移动drawlines() {//把所有多边形画出来for (let i = 0; i < this.all_coordinates.length; i++) {let cors = this.all_coordinates[i];//前后坐标连线for (let j = 0; j < cors.length - 1; j++) {this.ctx.beginPath();let x0 = cors[j].cor_x;let y0 = cors[j].cor_y;let x1 = cors[j + 1].cor_x;let y1 = cors[j + 1].cor_y;this.ctx.moveTo(x0, y0);this.ctx.lineTo(x1, y1);this.ctx.stroke();this.ctx.closePath();}//最后一个与第一个连线let begin_x = cors[0].cor_x;let begin_y = cors[0].cor_y;let end_x = cors[cors.length - 1].cor_x;let end_y = cors[cors.length - 1].cor_y;this.ctx.beginPath();this.ctx.moveTo(begin_x, begin_y);this.ctx.lineTo(end_x, end_y);this.ctx.stroke();this.ctx.closePath();}},//把当前绘制的多边形之前的坐标线段绘制出来drawline() {for (let i = 0; i < this.coordinates.length - 1; i++) {this.ctx.beginPath();let x0 = this.coordinates[i].cor_x;let y0 = this.coordinates[i].cor_y;let x1 = this.coordinates[i + 1].cor_x;let y1 = this.coordinates[i + 1].cor_y;this.ctx.moveTo(x0, y0);this.ctx.lineTo(x1, y1);this.ctx.stroke();this.ctx.closePath();}},// 针对的是单个多边形的所有顶点drawcircle() {//为当前的多边形端点画圆this.ctx.fillStyle = 'rgb(255, 255, 0)';for (let i = 0; i < this.coordinates.length; i++) {let x = this.coordinates[i].cor_x;let y = this.coordinates[i].cor_y;this.ctx.beginPath();this.ctx.moveTo(x, y);this.ctx.arc(x, y, 5, 0, Math.PI * 2);this.ctx.fill();this.ctx.closePath();}},// 用来在画布上标示出每个多边形顶点的位置,并且为这些顶点画出小圆圈drawcircles() {//为所有的多边形端点画圆this.ctx.fillStyle = 'rgb(255, 255, 0)';for (let i = 0; i < this.all_coordinates.length; i++) {let cors = this.all_coordinates[i];for (let j = 0; j < cors.length; j++) {let x = cors[j].cor_x;let y = cors[j].cor_y;this.ctx.beginPath();this.ctx.moveTo(x, y);this.ctx.arc(x, y, 5, 0, Math.PI * 2);this.ctx.fill();this.ctx.closePath();}}},// 画布中鼠标抬起canvasUp(e) {if (this.isdrag) {this.isdrag = false;// 打印所有多边形的顶点坐标this.all_coordinates.forEach((polygon, index) => {// console.log(`多边形 ${index + 1} 的顶点坐标:`);polygon.forEach((vertex, vertexIndex) => {// console.log(`  顶点 ${vertexIndex + 1}: (${vertex.cor_x}, ${vertex.cor_y})`);});});// console.log("拖拽点后抬起", this.all_coordinates);this.$emit("all_coordinates", this.all_coordinates[0]);}this.drag_index = [-1, -1];// this.drag_index = [e.layerX, e.layerY];// console.log("鼠标抬起", e);},// 画布中点击后线条鼠标移动中canvasMove(e) {//画布中鼠标移动//没开始画或者结束画之后不进行操作let x = e.offsetX;let y = e.offsetY;if (this.isdrag) {this.ctx.clearRect(0, 0, this.canvasWidth, this.canvasHeight);// this.ctx.clearRect(0, 0, 700, 450);this.all_coordinates[this.drag_index[0]][this.drag_index[1]].cor_x = x;this.all_coordinates[this.drag_index[0]][this.drag_index[1]].cor_y = y;this.drawlines();this.drawcircles();this.fillarea();}if (this.coordinates.length == 0 || !this.isdraw || this.endtip) {return 0;}//获取上一个点let last_x = this.coordinates[this.coordinates.length - 1].cor_x;let last_y = this.coordinates[this.coordinates.length - 1].cor_y;// this.ctx.clearRect(0, 0, 700, 450); //清空画布this.ctx.clearRect(0, 0, this.canvasWidth, this.canvasHeight);this.drawline();//把之前的点连线this.drawcircle();// 如果不止一个多边形,把多边形们画出来if (this.all_coordinates.length != 0) {this.drawlines();this.drawcircles();this.fillarea();}//获取鼠标移动时的点,画线,实现线段跟踪效果。this.ctx.beginPath();this.ctx.moveTo(last_x, last_y);this.ctx.lineTo(x, y);this.ctx.strokeStyle = 'yellow';this.ctx.stroke();this.ctx.closePath();},// 双击画布结束连线doubleclick() {if (this.type === 'detail') return;//双击画布,在最后一个点的时候双击,自动连线第一个点,同时宣告画结束let x0 = this.coordinates[0].cor_x;let y0 = this.coordinates[0].cor_y;let x1 = this.coordinates[this.coordinates.length - 1].cor_x;let y1 = this.coordinates[this.coordinates.length - 1].cor_y;this.ctx.beginPath();this.ctx.moveTo(x0, y0);this.ctx.lineTo(x1, y1);this.ctx.stroke();this.ctx.closePath();this.isdraw = false;this.endtip = truethis.drawcircle();this.coordinates.pop();this.all_coordinates.push(this.coordinates);this.ctx.fillStyle = 'rgba(255, 255, 0, 0.4)';let bx = this.coordinates[0].cor_x;let by = this.coordinates[0].cor_y;this.ctx.beginPath();this.ctx.moveTo(bx, by);for (let k = 1; k < this.coordinates.length; k++) {let x = this.coordinates[k].cor_x;let y = this.coordinates[k].cor_y;this.ctx.lineTo(x, y)}this.ctx.fill();this.ctx.closePath();// console.log("绘制完毕获得数据", this.coordinates)// console.log("绘制完毕获得全部数据", this.all_coordinates)this.$emit("all_coordinates", this.all_coordinates[0]);// console.log(666, this.all_coordinates[0]);this.coordinates = [];},// 接收多边形数据并绘制// receivePolygonData(polygonData) {//     if (Array.isArray(polygonData)) {//         // 清空画布//         this.test();//         // 绘制每个多边形//         polygonData.forEach(cors => {//             console.log(111, cors); //             this.drawPolygon(cors);//         });//     }// },// 在画布中绘制多边形并添加可拖拽的顶点drawPolygon(coordinates) {// 清空画布this.test();if (coordinates.length >= 2) {this.ctx.beginPath();let x0 = coordinates[0].cor_x;let y0 = coordinates[0].cor_y;this.ctx.moveTo(x0, y0);// 绘制多边形的边for (let j = 1; j < coordinates.length; j++) {let x = coordinates[j].cor_x;let y = coordinates[j].cor_y;this.ctx.lineTo(x, y);}// 最后一个点回到第一个点闭合多边形this.ctx.lineTo(x0, y0);this.ctx.closePath();this.ctx.stroke();// 填充多边形this.ctx.fillStyle = 'rgba(255, 255, 0, 0.4)';this.ctx.fill();// 绘制多边形顶点的小圆圈if (this.type != "detail") {this.ctx.fillStyle = 'rgb(255, 255, 0)';for (let j = 0; j < coordinates.length; j++) {let x = coordinates[j].cor_x;let y = coordinates[j].cor_y;this.ctx.beginPath();this.ctx.arc(x, y, 5, 0, Math.PI * 2);this.ctx.fill();this.ctx.closePath();}// 打印所有顶点的坐标// console.log('绘制完成的多边形顶点坐标:', coordinates);this.$emit("all_coordinates", coordinates);}let newArray = []newArray.push(coordinates)this.all_coordinates = newArray// console.log(666, this.all_coordinates, newArray);}}},
};
</script><style lang="scss" scoped>
.all {position: relative;width: 100%;height: 100%;// border: 2px solid red;border: 1px solid yellow;
}#mycanvas {// border: 1px solid red;// width: 100%;// height: 100%;// width: 700px;// height: 45px;position: absolute;top: 0;bottom: 0;left: 0;right: 0;// border: 2px solid red;
}#img {width: 100%;height: 100%;user-select: none;
}
</style>
<CanvasBox v-if="canvasIsShow" :imgUrl="imgUrl" :aiDataRegionalInfoList="aiDataRegionalInfoList":canvasWidth="canvasWidth" :canvasHeight="canvasHeight" :timeStamp="timeStamp"@all_coordinates="getCoordinates" :type="type" />
      // 默认隐藏canvascanvasIsShow: false,// 子组件背景图片imgUrl: "",// 子组件多边形区域aiDataRegionalInfoList: [],// 子组件canvas宽度canvasWidth: 0,// 子组件canvas高度canvasHeight: 0,timeStamp: 0,// 记录新增->add  修改->edit  查看->detailtype: "",// 接收到绘制后多边形数据getCoordinates(val) {// console.log("接收到canvas区域数据", val);let newArray = []if (val && val.length > 0) {val.forEach(item => {newArray.push({x: this.autoPageSizeToDefault(item.cor_x, item.cor_y).x,y: this.autoPageSizeToDefault(item.cor_x, item.cor_y).y,})})}this.form.aiDataRegionalInfoList = newArray},// 清空画布clearCanvas() {this.timeStamp = Date.now();this.aiDataRegionalInfoList = []this.form.aiDataRegionalInfoList = []},

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

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

相关文章

Ubuntu中MySQL远程登录设置

mysql单独放在一台Ubuntu服务器上&#xff0c;我远程连接不上。可能是安装的时候忘记设置远程登录了。事后补救措施如下&#xff1a; MySQL 绑定地址配置问题 MySQL 可能只绑定了 localhost&#xff0c;无法接受来自外部主机的连接。你需要检查 MySQL 的配置文件 /etc/mysql/…

使用Vscode配置ftp连接远程服务器(上传本地文件)

1.安装插件 扩展商店搜sftp,点击进行安装。 2.配置json文件 crtl+shift+p 输入ftp配置命令 sftp:config {"name": "My Server", //设置名字"host": "localhost"</

腐蚀膨胀预处理

腐蚀&#xff1a;通过减少前景对象&#xff08;例如白色字符&#xff09;的边缘&#xff0c;腐蚀可以用来减小或消除细小的干扰线。如果干扰线较细&#xff0c;腐蚀可以有效地“消除”这些线条&#xff0c;同时保留较粗的字符。 膨胀&#xff1a;在腐蚀之后&#xff0c;膨胀可…

使用 Logistic Regression 识别手写数字-PyTorch

逻辑回归是一种非常常用的统计方法,它允许我们从一组自变量预测二进制输出。本文之前已经介绍了 Logistic 回归的各种属性及其 Python 实现 。现在,我们将了解如何在 PyTorch 中实现这一点,PyTorch 是由 Facebook 开发的一个非常流行的深度学习库。 现在,我们将了解如何使用…

上门按摩系统开发方案源码搭建

上门按摩系统开发方案 一、项目概述 上门按摩系统是一个连接按摩技师和客户的平台&#xff0c;旨在提供便捷、高效的上门按摩服务。通过该系统&#xff0c;客户可以轻松预约合适的按摩技师&#xff0c;并享受个性化的按摩服务。 二、系统功能模块 用户管理模块&#xff1a;…

TCP——Socket

应用进程只借助Socket API发和收但是不关心他是怎么进行传和收的 数据结构 图示Socket连接 捆绑属于隐式捆绑

Redis非关系型数据库操作命令大全

以下是 Redis 的常用操作命令大全&#xff0c;涵盖了键值操作、字符串、哈希、列表、集合、有序集合、发布/订阅、事务等多个方面的操作。 1. 通用键命令 命令说明SET key value设置指定 key 的值GET key获取指定 key 的值DEL key删除指定的 keyEXISTS key检查 key 是否存在E…

数据驱动时代:五款免费报表工具深度解析

在当今数据驱动的时代&#xff0c;报表工具已经成为各类企业进行决策和管理的重要工具。无论是大中型企业还是小微企业&#xff0c;能够快速、高效地生成可视化报表&#xff0c;洞察业务运营情况&#xff0c;已经成为提升竞争力的关键。今天为大家挑选了5款非常优秀的报表软件&…

Autosar软件组件概述

AUTOSAR中的应用软件设计为独立的软件单元&#xff0c;即软件组件类型&#xff08;SwComponentTypes&#xff09;&#xff0c;封装了相关功能和行为的实现&#xff0c;并通过端口原型&#xff08;PortPrototypes&#xff09;&#xff0c;向外部暴露自己。 AUTOSAR 标准应用了设…

std::setw中文不能对齐的问题

从别的博主那 copy过来 记录下 #include <codecvt> #include <locale>class chs_codecvt : public std::codecvt_byname<wchar_t, char, std::mbstate_t> {public:chs_codecvt() : codecvt_byname("zh_CN.GBK") {} };int encoding_diff(string …

【VUE】封装用户树形选择和部门树形选择控件

用vue实现封装用户树形选择和部门树形选择控件&#xff0c;采用el-tree。方便各个功能模块的使用和以后的开发。 一、封装用户树形选择控件&#xff08;userTree.vue&#xff09; <template><div style"padding: 10px;"><!-- 选择人员对话框 -->…

Java 类,变量,对象

类的声明 1、类的权限访问修饰符&#xff0c;public 缺省 public修涉的&#xff0c;在同包中可以进行访问&#xff0c;在不同包中也可以进行访问&#xff0c;缺省的修饰符只能在同包中进行访问 2、类的成员&#xff1a;属性&#xff0c;属性如何调用&#xff1f;对象.属性 1、权…

日常随笔1--MySQL中添加Redo日志文件的步骤详解

Redo日志是MySQL中重要的事务日志&#xff0c;用于记录事务对数据库进行的修改操作&#xff0c;确保数据库的持久性和恢复能力。本文将详细介绍在MySQL中添加Redo日志文件的步骤&#xff0c;包括设置Redo日志文件大小、位置和数量等。 1. 确认当前Redo日志文件状态 在添加新的…

xtu oj 原根

文章目录 回顾杂思路c 语言代码 回顾 AB III问题 H: 三角数问题 G: 3个数等式 数组下标查询&#xff0c;降低时间复杂度1405 问题 E: 世界杯xtu 数码串xtu oj 神经网络xtu oj 1167 逆序数&#xff08;大数据&#xff09; 杂 有一些题可能是往年的程设的题&#xff0c;现在搬到…

20255 - 中医方剂学 - 考研 - 执业

第1章 总论 1.我国现存最早的记载方剂的医书是&#xff08;&#xff09;( ) [单选] A.《太平圣惠方》 B.《黄帝内经》 C.《五十二病方》 D.《千金要方》 E.《外台秘要》 正确答案: C 2.我国最早的中医经典理论著作是&#xff08;&#xff09;( ) [单选] A.《伤寒杂病论…

Higress 重磅更新:AI 能力全面开源,云原生能力再升级

作者&#xff1a;澄潭、钰诚 新版本简介 Higress 最新的 1.4 版本基于为通义千问&#xff0c;以及多家云上 AGI 厂商客户提供 AI 网关的积累沉淀&#xff0c;开源了大量 AI 原生的网关能力。同时也在 Ingress、可观测、流控等云原生能力上做了全方位升级&#xff1a; AI 能力…

VS code部署Vue项目Demo

在之前已经在IDEA中部署过vue项目demo。本次在上次基础上进行。 IDEA中Vue的安装和使用【window10】_idea安装vue-CSDN博客 步骤一、安装VSCode 双击安装即可 步骤二&#xff1a;检查npm是否安装 步骤三&#xff1a;检查vue是否安装 &#xff08;vue create 项目名 只要在v…

9.指针与链表

一、指针 1.基本介绍 在程序中&#xff0c;我们的数据都有其存储的地址。在程序每次的实际运行过程中&#xff0c;变量在物理内存中的存储位置不尽相同。不过&#xff0c;我们仍能够在编程时&#xff0c;通过一定的语句&#xff0c;来取得数据在内存中的地址。地址也是数据。…

Leetcode 单词规律

即判断给定的模式字符串&#xff08;pattern&#xff09;和单词字符串&#xff08;s&#xff09;是否遵循相同的对应规则。具体来说&#xff0c;就是要判断 pattern 中的字符与 s 中的单词是否存在一一对应的关系&#xff0c;即双射&#xff08;bijection&#xff09;。 算法思…

安卓无障碍获取录屏权限

每次需要录屏操作时&#xff0c;都会弹窗需要用户手动授权比较麻烦&#xff0c;可以通过无障碍模拟用户点击进行授权。 方法主要有三种&#xff1a; 1.查找“立即开始”&#xff08;华为手机是“允许”&#xff09;节点&#xff0c;模拟点击&#xff1b; Overridepublic void…