在 Vue 3 中实现电子签名组件

news2024/10/23 10:57:14

在 Vue 3 中实现一个简单的电子签名组件,并解决一个常见问题:当签名组件放在弹窗内时,鼠标绘制会出现偏移的问题。

项目环境:
  • Vue 3:前端框架
  • Element Plus:UI 组件库

电子签名组件功能

  1. 画布绘制:用户可以在画布上使用鼠标进行签名。
  2. 画笔设置:支持调整画笔的颜色和大小。
  3. 提交签名:将画布内容提交,生成图片并回传给父组件。
  4. 清除签名:可以清除当前画布内容。

源代码实现

下面是完整的 Vue 3 电子签名组件代码:

1. template 模板部分
<template>
  <!-- 电子签名组件 -->
  <div>
    <!-- 画布对象 -->
    <div
      ref="grapDiv"
      style="width: 800px; height: 400px; border: 2px dotted #ddd; border-radius: 10px;"
      v-show="!isCommit"
    >
      <canvas
        ref="grapCvs"
        id="container"
        @mousedown="mousedown"
        @mousemove="mousemove"
      />
    </div>

    <!-- 设置面板 -->
    <div
      ref="setControlDiv"
      style="width: 280px; height: 200px; background-color: #474747; border: 1px solid #ddd; border-radius: 10px; margin-top: -203px; margin-left: 1px; position: absolute;"
      v-show="ifSetController"
    >
      <div style="width: 100%; height: 30%; margin-top: 10px">
        <span style="width: 100%; height: 30%">
          <label style="float: left; color: white; margin-left: 14px; margin-top: 15px;">字体大小</label>
          <label style="float: right; color: white; margin-right: 14px; margin-top: 15px;">{{ fontSize }}</label>
        </span>
        <el-slider v-model="fontSize" style="width: 89%; margin-top: 20px; margin-left: 17px" :min="1" :max="10" />
      </div>
      <div style="width: 100%; height: 45%; margin-top: 15px; padding: 7px">
        <li
          @click="setImgColor"
          v-for="(item, index) in fontColorArray"
          :key="index"
          :style="'list-style: none;width: 38px;height: 38px;float: left;background: ' + item"
        ></li>
      </div>
    </div>

    <!-- 提交的画布对象 -->
    <div style="width: 800px; height: 400px; border: 2px solid green; border-radius: 10px;" v-show="isCommit">
      <img :src="content" />
    </div>

    <!-- 操作按钮 -->
    <div style="width: 800px; padding-top: 10px">
      <el-button @click="set()" v-show="!isCommit">画笔设置</el-button>
      <el-button @click="clear()" v-show="!isCommit">清除</el-button>
      <el-button type="primary" @click="commit()" v-show="!isCommit">提交</el-button>
      <el-button @click="goback()" v-show="isCommit">返回</el-button>
    </div>
  </div>
</template>
2. script 部分
<script setup>
import { ref, onMounted } from "vue";
import { ElMessage, ElMessageBox } from "element-plus";
const emit = defineEmits(["commitDatas"]);

// 定义变量
const canvas = ref(null);
const graphics = ref(null);
const isDrawing = ref(false);
const curMouseX = ref(null);
const curMouseY = ref(null);
const isCommit = ref(false);
const content = ref(null);
const ifGraph = ref(false);
const ifSetController = ref(false);
const fontSize = ref(1);
const fontColorArray = ref(["#F59999", "#E86262", "#AA4446", "#6B4849", "#34231E", "#435772", "#2DA4A8", "#EFDCD3", "#FEAA3A", "#FD6041", "#CF2257", "#404040", "#92BEE2", "#2286D8"]);
const fontColor = ref("#000000");

// 选择画笔颜色
const setImgColor = (curIndex) => {
  let liArray = curIndex.currentTarget.parentElement.children;
  for (let child of liArray) {
    child.className = "";
  }
  curIndex.currentTarget.className = "activeteLi";
  fontColor.value = curIndex.currentTarget.style.background;
};

// 鼠标按下事件处理
const mousedown = (e) => {
  const rect = canvas.value.getBoundingClientRect();
  isDrawing.value = true;
  curMouseX.value = e.clientX - rect.left;
  curMouseY.value = e.clientY - rect.top;
  graphics.value.beginPath();
  graphics.value.moveTo(curMouseX.value, curMouseY.value);
};

// 鼠标移动事件处理
const mousemove = (e) => {
  if (isDrawing.value) {
    const rect = canvas.value.getBoundingClientRect();
    graphics.value.strokeStyle = fontColor.value;
    graphics.value.lineWidth = fontSize.value;
    curMouseX.value = e.clientX - rect.left;
    curMouseY.value = e.clientY - rect.top;
    graphics.value.lineTo(curMouseX.value, curMouseY.value);
    graphics.value.stroke();
    ifGraph.value = true;
  }
};

// 清除画布
const clear = () => {
  ifGraph.value = false;
  canvas.value.width = canvas.value.width;
};

// 将 DataURL 转换为文件
const dataURLtoFile = (dataurl, filename) => {
  const arr = dataurl.split(",");
  const bstr = window.atob(arr[1]);
  let n = bstr.length;
  const u8arr = new Uint8Array(n);
  while (n--) {
    u8arr[n] = bstr.charCodeAt(n);
  }
  return new File([u8arr], filename, { type: "image/png" });
};

// 提交签名
const commit = () => {
  if (!ifGraph.value) {
    ElMessage({ message: "没有可提交的内容!", type: "error", duration: 2000 });
    return;
  }
  ElMessageBox({
    title: "操作提示",
    message: "确定提交?",
    confirmButtonText: "确定",
    cancelButtonText: "取消",
    showCancelButton: true,
    closeOnClickModal: false,
    type: "warning",
  }).then(() => {
    isCommit.value = true;
    content.value = canvas.value.toDataURL();
    emit("commitDatas", dataURLtoFile(content.value, "签名"));
  });
};

// 返回编辑模式
const goback = () => {
  ifGraph.value = false;
  canvas.value.width = canvas.value.width;
  isCommit.value = false;
};

// 初始化画布
onMounted(() => {
  const cvs = document.getElementById("container");
  cvs.width = 800;
  cvs.height = 400;
  graphics.value = cvs.getContext("2d");
  canvas.value = cvs;

  document.addEventListener("mouseup", () => {
    isDrawing.value = false;
    graphics.value.closePath();
  });
});
</script>
3. 样式部分
<style>
.activeteLi {
  box-shadow: 0 0 3px rgb(0 0 0 / 95%);
  transform: scale(1.2);
}
</style>

解决鼠标和画笔偏移问题

当我们将签名组件放到 el-dialog 弹窗中时,出现鼠标点击位置与实际绘制位置不符的偏移问题。这是由于 canvas 相对于窗口的位置发生了变化。为了解决这个问题,我们使用了 getBoundingClientRect() 来动态计算 canvas 在页面中的位置,从而调整鼠标绘制的准确性。

const rect = canvas.value.getBoundingClientRect();
curMouseX.value = e.clientX - rect.left;
curMouseY.value = e.clientY - rect.top;

getBoundingClientRect() 方法返回元素的大小及其相对于视口的位置,因此可以精确计算出鼠

标相对于 canvas 的位置,确保绘制效果不会偏移。

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

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

相关文章

数据结构修炼——常见的排序算法:插入/希尔/选择/堆排/冒泡/快排/归并/计数

目录 一、常见的排序算法二、常见排序算法的实现2.1 排序算法回顾2.1.1 冒泡排序2.1.2 堆排序 2.2 直接插入排序2.3 希尔排序2.4 选择排序2.5 快速排序2.5.1 快速排序&#xff08;霍尔法&#xff09;2.5.2 快速排序&#xff08;挖坑法&#xff09;2.5.3 快速排序&#xff08;前…

极客wordpress模板

这是一个展示WordPress主题的网页设计。页面顶部有一个导航栏&#xff0c;包含多个选项&#xff0c;如“关于我们”、“产品中心”、“案例展示”、“新闻动态”、“联系我们”和“技术支持”。页面中间部分展示了多个产品&#xff0c;每个产品都有一个图片和简短的描述。页面下…

【Linux】冯诺依曼体系结构 OS的概念

&#x1fa90;&#x1fa90;&#x1fa90;欢迎来到程序员餐厅&#x1f4ab;&#x1f4ab;&#x1f4ab; 主厨&#xff1a;邪王真眼 主厨的主页&#xff1a;Chef‘s blog 所属专栏&#xff1a;青果大战linux 总有光环在陨落&#xff0c;总有新星在闪烁 前言废话&#xff1a…

动态链接过程分析

目录 一、前言二、示例程序三、动态库的加载过程1、动态链接器加载动态库2、动态库的加载地址 四、符号重定位1、全局符号表2、全局偏移表 GOT3、liba.so 动态库文件的布局4、liba.so 动态库的虚拟地址5、GOT 表的内部结构6、反汇编 liba.so 代码 五、补充1、延迟绑定 plt 上文…

【ARM】ARM架构参考手册_Part B 内存和系统架构(5)

目录 5.1关于缓存和写缓冲区 5.2 Cache 组织 5.2.1 集联性&#xff08;Set-associativity&#xff09; 5.2.2 缓存大小 5.3 缓存类型 5.3.1 统一缓存或分离缓存 5.3.2 写通过&#xff08;Write-through&#xff09;或写回&#xff08;Write-back&#xff09;缓存 5.3.3…

基于R语言机器学习遥感数据处理与模型空间预测技术及实际项目案例分析

随机森林作为一种集成学习方法&#xff0c;在处理复杂数据分析任务中特别是遥感数据分析中表现出色。通过构建大量的决策树并引入随机性&#xff0c;随机森林在降低模型方差和过拟合风险方面具有显著优势。在训练过程中&#xff0c;使用Bootstrap抽样生成不同的训练集&#xff…

Linux环境配置(学生适用)

1.挑选最便宜的云服务器 如腾讯云服务器&#xff0c;华为云服务器&#xff0c;百度云服务器等等…… 2.找到你的云服务器实例&#xff0c;然后找到你的公网IP。 3.云服务器实例 ---更多 --- 重置root密码 (一定要重置&#xff09; 4. 下载并安装 xshell 或者其他登陆软件 xshel…

12. 命令行

Hyperf 的命令行默认由 hyperf/command 组件提供&#xff0c;而该组件本身也是基于 symfony/console 的抽象。 一、安装 通常来说该组件会默认存在&#xff0c;但如果您希望用于非 Hyperf 项目&#xff0c;也可通过下面的命令依赖 hyperf/command 组件。 composer require hype…

告别ELK,APO提供基于ClickHouse开箱即用的高效日志方案——APO 0.6.0发布

ELK一直是日志领域的主流产品&#xff0c;但是ElasticSearch的成本很高&#xff0c;查询效果随着数据量的增加越来越慢。业界已经有很多公司&#xff0c;比如滴滴、B站、Uber、Cloudflare都已经使用ClickHose作为ElasticSearch的替代品&#xff0c;都取得了不错的效果&#xff…

【Golang】Go语言中如何创建Cron定时任务

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…

MySQL【知识改变命运】11

联合查询 6. ⼦查询6.1 语法6.2 单⾏⼦查询6.3 多⾏⼦查询6.4 多列⼦查询6.5 在from⼦句中使⽤⼦查询 7. 合并查询7.1 创建新表并初始化数据7.2 Union7.3 Union all 8. 插⼊查询结果8.1 语法8.2 ⽰例 6. ⼦查询 ⼦查询是把⼀个SELECT语句的结果当做别⼀个SELECT语句的条件&…

10.22 MySQL

存储过程 存储函数 存储函数是有返回值的存储过程&#xff0c;存储函数的参数只能是in类型的。具体语法如下&#xff1a; characteristic 特性 练习&#xff1a; 从1到n的累加 ​​​​​​ create function fun1(n int) returns int deterministic begindeclare total i…

制氮机分子筛的作用

制氮机作为一种重要的工业设备&#xff0c;广泛应用于食品、饮料、化学、石油、电子和医疗保健等多个行业。其核心组件之一——分子筛。本文将详细探讨制氮机分子筛的作用及其重要性。 一、分子筛的基本概念 分子筛是一种具有均匀孔径的多孔材料&#xff0c;常用于气体分离和纯…

Elasticsearch 中的高效按位匹配

作者&#xff1a;来自 Elastic Alexander Marquardt 探索在 Elasticsearch 中编码和匹配二进制数据的六种方法&#xff0c;包括术语编码&#xff08;我喜欢的方法&#xff09;、布尔编码、稀疏位位置编码、具有精确匹配的整数编码、具有脚本按位匹配的整数编码以及使用 ESQL 进…

基于vue框架的的二手车交易系统的设计与实现thx7v(程序+源码+数据库+调试部署+开发环境)系统界面在最后面。

系统程序文件列表 项目功能&#xff1a;用户,卖家,车辆类型,二手车,在线留言,订单信息 开题报告内容 基于Vue框架的二手车交易系统的设计与实现开题报告 一、课题背景及意义 随着汽车消费市场的日益成熟与消费者换车频率的增加&#xff0c;二手车交易市场逐渐成为汽车市场的…

pycharm配置git版本控制

今天记录一下如何在pycharm工具中配置git版本控制&#xff0c;主要分以下步骤&#xff1a; 1、安装git 首先需要有git环境&#xff0c;去git官网下载git安装包&#xff0c;下一步下一步执行安装完成即可 2、在pycharm中配置git路径 下载git后&#xff0c;在pycharm的 setti…

「AIGC」n8n AI Agent开源的工作流自动化工具

n8n AI Agent 是一个利用大型语言模型(LLMs)来设计和构建智能体(agents)的工具,这些智能体能够执行一系列复杂的任务,如理解指令、模仿类人推理,以及从用户命令中理解隐含意图。n8n AI Agent 的核心在于构建一系列提示(prompts),使 LLM 能够模拟自主行为。 传送门→ …

GAMES104:17 游戏引擎的玩法系统:高级AI-学习笔记

文章目录 课前QA一&#xff0c;层次任务网络&#xff08;Hierarchical Tasks Network&#xff0c;HTN&#xff09;1.1 HTN Framework1.2 HTN Task Types1.2.1 Primitive Task基本任务1.2.2 Compound Task符合任务 1.3 Planning1.4 Replan1.5 总结 二&#xff0c;目标导向行为规…

在ECS实例上搭建WordPress博客平台

WordPress是使用PHP语言开发的博客平台&#xff0c;在支持PHP和MySQL数据库的服务器上&#xff0c;您可以用WordPress搭建自己的网站&#xff0c;也可以用作内容管理系统&#xff08;CMS&#xff09;。本教程介绍如何在不同操作系统的ECS实例上&#xff0c;手动搭建WordPress网…

SonarQube快速实践

SonarQube快速实践 1. 简介 SonarQube 是一个本地部署的代码分析工具&#xff0c;旨在检测30多种编程语言、框架和基础设施即代码&#xff08;IaC&#xff09;平台中的代码问题。通过直接集成到您的持续集成&#xff08;CI&#xff09;流水线中或在我们支持的DevOps平台之一上…