0


前端实现签字效果+合同展示

文章目录

最近菜鸟公司要做一个这样的功能,后端返回一个合同的整体html,前端进行签字,以下是一些重要思路!

注:本文章是给自己看的,读者酌情考虑!

获取一个高度会变的元素的高度

script 代码

let bigBoxHeight =ref(0);// 获取到元素let bigBox = document.querySelector(".bigBox");// 设置高度为 auto
bigBox.style.height ="auto";// 获取 offsetHeightconst height = bigBox.offsetHeight;// 设置值
bigBoxHeight.value = height;

注:
offsetHeight:返回一个元素的高度,包括其padding和border,但不包括其margin。

template 代码

<divclass="bigBox":style="{ height: bigBoxHeight + 'px' }"><divclass="contractBox"><divv-html="printData"></div></div><!-- 遮罩层,返回的printData里设置了可编辑,但是这里只是展示用,且修改了也不会有影响,所以就简单的加个遮罩就行了 --><divclass="markBox":style="{ height: bigBoxHeight + 'px' }"></div></div>

获取元素设置的 transform

感谢:原生js获取元素transform的scale和rotate

// 获取设置了transform的元素let contractBox = document.querySelector(".contractBox");// 获取浏览器计算后的结果let st = window.getComputedStyle(contractBox,null);// 从结算后的结果中找到 transform,也可以直接 st.transformvar tr = st.getPropertyValue("transform");if(tr ==="none"){// 为none表示未设置
  bigBox.style.height ="auto";const height = bigBox.offsetHeight +50;
  bigBoxHeight.value = height;}else{
  bigBox.style.height ="auto";// 缩放需要 * 缩放比例 + 边距(margin/padding)const height = bigBox.offsetHeight *0.5+50;
  bigBoxHeight.value = height;}

getComputedStyle 可以学习我的博客:看 Javascript实战详解 收获一

适配手机

上面设置transform是因为返回的html文档不是那么的自适应,所以菜鸟就在手机端,让其渲染700px,但是再缩小0.5倍去展示,即可解决!

css 代码

@media screen and(max-width: 690px){.contractBox{width: 700px !important;transform:scale(0.5);
    // 防止延中心点缩放而导致上面留白很多(合同很长,7000px左右)
    transform-origin: 5% 0;}}.bigBox{position: relative;
  // 设置是因为 scale 缩放了但是元素还是占本身那么大,所以要超出隐藏
  overflow: hidden;.markBox{width: 100%;position: absolute;left: 0;bottom: 0;top: 0;bottom: 0;}}.contractBox{width: 70%;margin: 50px auto 0px;overflow: hidden;}

transform-origin: 5% 0; 的原因

这里设置 5% 是为了居中,因为这里有个问题就是不能设置bigBox为display:flex,不然里面的内容就是按照width:100%然后缩放0.5,而不是width:700px来缩放的!

是flex搞的鬼,菜鸟这里就用了个简单办法。

其实正统做法应该是获取宽度,再用窗口宽度减去获取的宽度 / 2,然后通过该值设置margin!

修改后

菜鸟既然想到了上面的居中方式,那就直接实现了,这里给上代码!

script 代码

// 是否缩放,来确定margin-left取值let isScale =ref(false);let bigBoxmargin =ref(0);let bigBox = document.querySelector(".bigBox");let contractBox = document.querySelector(".contractBox");let st = window.getComputedStyle(contractBox,null);var tr = st.getPropertyValue("transform");if(tr ==="none"){
  isScale.value =false;
  bigBox.style.height ="auto";const height = bigBox.offsetHeight +50;
  bigBoxHeight.value = height;}else{
  isScale.value =true;
  bigBox.style.height ="auto";// 缩放需要 * 缩放比例 + 边距(margin/padding)const height = bigBox.offsetHeight *0.5+50;// 不用 st.witdh 是因为 st.witdh 获取的值是 700px,不能直接运算,这里菜鸟就偷懒了,不想处理了
  bigBoxmargin.value =(window.innerWidth -700*0.5)/2;
  bigBoxHeight.value = height;}

template 代码

<divclass="bigBox":style="{ height: bigBoxHeight + 'px' }"><divclass="contractBox":style="{ marginLeft: isScale ? bigBoxmargin + 'px' : 'auto' }"><divv-html="printData"></div></div><divclass="markBox":style="{ height: bigBoxHeight + 'px' }"></div></div>

签字效果

这里签字效果,菜鸟是使用 el-dialog 实现的,el-dialog 的使用方式见:element plus 使用细节

这里主要粘贴签字的代码

<script setup>import{ ref, onMounted, nextTick }from"vue";// eslint-disable-next-lineconst props =defineProps({dialogVisible:{type: Boolean,default:false,},});// eslint-disable-next-lineconst emit =defineEmits(["closeEvent"]);// 关闭弹窗functionhandleClose(){emit("closeEvent",false);// 解除禁止页面滚动
  document.body.removeEventListener("touchmove", preventDefault);}const dialogBox =ref();functioncloseDialog(){
  dialogBox.value.resetFields();}// 禁止页面滚动functionpreventDefault(e){
  e.preventDefault();}
document.body.addEventListener("touchmove", preventDefault,{passive:false});// 签名// 配置内容const config ={width: window.innerWidth,// 宽度height: window.innerHeight -300,// 高度,减300是为了给dialog的footer一点空间显示lineWidth:5,// 线宽strokeStyle:"red",// 线条颜色lineCap:"round",// 设置线条两端圆角lineJoin:"round",// 线条交汇处圆角};let canvas =null;let ctx =null;onMounted(async()=>{awaitnextTick();// 获取canvas 实例
  canvas = document.querySelector(".canvas");// 设置宽高
  canvas.width = config.width;
  canvas.height = config.height;// 设置一个边框
  canvas.style.border ="1px solid #000";// 创建上下文
  ctx = canvas.getContext("2d");// 设置填充背景色
  ctx.fillStyle ="transparent";// 绘制填充矩形
  ctx.fillRect(0,// x 轴起始绘制位置0,// y 轴起始绘制位置
    config.width,// 宽度
    config.height // 高度);});// 保存上次绘制的 坐标及偏移量const client ={offsetX:0,// 偏移量offsetY:0,endX:0,// 坐标endY:0,};// 判断是否为移动端const mobileStatus =/Mobile|Android|iPhone/i.test(navigator.userAgent);// 初始化constinit=(event)=>{// 获取偏移量及坐标const{ offsetX, offsetY, pageX, pageY }= mobileStatus ? event.changedTouches[0]: event;// 修改上次的偏移量及坐标
  client.offsetX = offsetX;
  client.offsetY = offsetY;
  client.endX = pageX;
  client.endY = pageY;// 清除以上一次 beginPath 之后的所有路径,进行绘制
  ctx.beginPath();// 根据配置文件设置相应配置
  ctx.lineWidth = config.lineWidth;
  ctx.strokeStyle = config.strokeStyle;
  ctx.lineCap = config.lineCap;
  ctx.lineJoin = config.lineJoin;// 设置画线起始点位
  ctx.moveTo(client.endX, client.endY);// 监听 鼠标移动或手势移动
  window.addEventListener(mobileStatus ?"touchmove":"mousemove", draw);};// 绘制constdraw=(event)=>{
  console.log(event);// 获取当前坐标点位const{ pageX, pageY }= mobileStatus ? event.changedTouches[0]: event;// 超出范围不监听if(pageY > config.height){return;}// 修改最后一次绘制的坐标点
  client.endX = pageX;
  client.endY = pageY;// 根据坐标点位移动添加线条
  ctx.lineTo(pageX, pageY);// 绘制
  ctx.stroke();};// 结束绘制constcloaseDraw=()=>{// 结束绘制
  ctx.closePath();// 移除鼠标移动或手势移动监听器
  window.removeEventListener("mousemove", draw);};// 创建鼠标/手势按下监听器
window.addEventListener(mobileStatus ?"touchstart":"mousedown", init);// 创建鼠标/手势 弹起/离开 监听器
window.addEventListener(mobileStatus ?"touchend":"mouseup", cloaseDraw);// 取消-清空画布constcancel=()=>{// 清空当前画布上的所有绘制内容
  ctx.clearRect(0,0, config.width, config.height);};// 保存-将画布内容保存为图片constsave=()=>{// 将canvas上的内容转成blob流
  canvas.toBlob((blob)=>{// 获取当前时间并转成字符串,用来当做文件名const date = Date.now().toString();// 创建一个 a 标签const a = document.createElement("a");// 设置 a 标签的下载文件名
    a.download =`${date}.png`;// 设置 a 标签的跳转路径为 文件流地址
    a.href =URL.createObjectURL(blob);// 手动触发 a 标签的点击事件
    a.click();// 移除 a 标签
    a.remove();});handleClose();};</script><template><div><el-dialog
      title="签字"
      ref="dialogBox":modelValue="dialogVisible":before-close="handleClose"
      @close="closeDialog":close-on-click-modal="false":destroy-on-close="true"
      top="0"
      width="100%"><canvas class="canvas"></canvas><template #footer><div><el-button type="primary" @click="save">保存</el-button><el-button @click="cancel">清除</el-button><el-button @click="handleClose">关闭</el-button></div></template></el-dialog></div></template><style lang="scss">.el-dialog__header {display: none;}.el-dialog__body {padding:0!important;}</style>

取消el-dialog的头部+边距

因为这里的 client 设置的偏移量都是 0,菜鸟不会改(感觉应该加上el-dialog的头部+边框的偏移量),如果不取消的话,就是错位着写的!

为什么禁止界面滚动

这里禁止是因为手机端,签名时写 “竖” 操作时,容易触发下拉整个界面的事件!导致写字中断,体验感极差,所以弹窗弹出时阻止事件,关闭后移除!

这里函数 preventDefault 必须提出,不然会取消不掉!

vue3 使用 nextTick

获取元素必须在 onMounted 中,但是 el-dialog 即使写在 onMounted 里面也不行,需要加上 nextTick !

实现效果

在这里插入图片描述

签字判断是横是竖

今天菜鸟又遇见了大麻烦,就是这个签字不能知道别人是横屏横着写的还是竖屏横着写的,eg:

在这里插入图片描述
在这里插入图片描述
菜鸟的思路就是获取到签字部分,然后如果横着签字就直接截取那部分设置样式,如果竖着签字就设置样式旋转 -90deg,那如何获取签字部分的大小呢?

canvas 去掉空白部分

canvas 去掉空白部分

修改上面 save 中代码

constsave=()=>{// 将canvas上的内容转成blob流var imgData = ctx.getImageData(0,0, canvas.width, canvas.height).data;var lOffset = canvas.width,
      rOffset =0,
      tOffset = canvas.height,
      bOffset =0;for(var i =0; i < canvas.width; i++){for(var j =0; j < canvas.height; j++){var pos =(i + canvas.width * j)*4;if(imgData[pos]>0|| imgData[pos +1]>0|| imgData[pos +2]|| imgData[pos +3]>0){// 说第j行第i列的像素不是透明的// 楼主貌似底图是有背景色的,所以具体判断RGBA的值可以根据是否等于背景色的值来判断
          bOffset = Math.max(j, bOffset);// 找到有色彩的最底部的纵坐标
          rOffset = Math.max(i, rOffset);// 找到有色彩的最右端
    
          tOffset = Math.min(j, tOffset);// 找到有色彩的最上端
          lOffset = Math.min(i, lOffset);// 找到有色彩的最左端}}}}

canvas.getContext(‘2d’).getImageData(0, 0, 宽, 高) 会返回一个当前 canvas 的图像数据对象,其中有一个data属性,是一个一维数组,这个一维数组,每4个下标分别代表了一个像素点的 R,G,B,A 的值,只需要遍历这些值就能找到边界了。

感谢:canvas 裁剪空白区域

canvas裁剪图片

canvas裁剪图片

但是获取到了区域并不行,因为我还需要将其截取,然后转成图片传给后端,且还要让后端知道到底是横着放 html 模板里还是竖着放,思来想去,感觉直接返回 base64 的 img 元素给后端更好,因为我就可以直接设置style,后端只需要放到对应的地方就行,所以save继续修改为

constsave=()=>{// 将canvas上的内容转成blob流var imgData = ctx.getImageData(0,0, canvas.width, canvas.height).data;var lOffset = canvas.width,
    rOffset =0,
    tOffset = canvas.height,
    bOffset =0;for(var i =0; i < canvas.width; i++){for(var j =0; j < canvas.height; j++){var pos =(i + canvas.width * j)*4;if(imgData[pos]>0|| imgData[pos +1]>0|| imgData[pos +2]|| imgData[pos +3]>0){// 说第j行第i列的像素不是透明的// 楼主貌似底图是有背景色的,所以具体判断RGBA的值可以根据是否等于背景色的值来判断
        bOffset = Math.max(j, bOffset);// 找到有色彩的最底部的纵坐标
        rOffset = Math.max(i, rOffset);// 找到有色彩的最右端

        tOffset = Math.min(j, tOffset);// 找到有色彩的最上端
        lOffset = Math.min(i, lOffset);// 找到有色彩的最左端}}}// 重新创建一个canvas,将之前的canvas上的图片,按照获取到的大小去截取const trimmedWidth = rOffset - lOffset +1;const trimmedHeight = bOffset - tOffset +1;const trimmedCanvas = document.createElement("canvas");
  trimmedCanvas.width = trimmedWidth;
  trimmedCanvas.height = trimmedHeight;const trimmedContext = trimmedCanvas.getContext("2d");
  trimmedContext.putImageData(
    ctx.getImageData(lOffset, tOffset, trimmedWidth, trimmedHeight),0,0);// 将截取后的生成图片,并设置样式
  console.log(trimmedWidth);
  console.log(trimmedHeight);var newUrl = trimmedCanvas.toDataURL();var newImage =newImage();
  newImage.src = newUrl;
  console.log(trimmedWidth < trimmedHeight);if(trimmedWidth < trimmedHeight){
    newImage.style.height ="100px";
    newImage.style.transform ="rotate(-"+90+"deg)";}else{
    newImage.style.width ="100px";}

  console.log(newImage);handleClose();};

newImage 打印出来是一个元素:

<imgsrc=""style="transform:rotate(-90deg);">

至此算是完成了整个签字功能!

最终完善代码,可以直接使用

其实这里还有一个问题,就是不知道横屏后的用户到底是哪边横屏,可能要旋转-90deg,也可能是正90deg,这里菜鸟想的是一个简单办法,就是给个示例文字,让用户根据示例文字进行签字!

template

<template><div><el-dialogtitle="签字"ref="dialogBox":modelValue="dialogVisible":before-close="handleClose"@close="closeDialog":close-on-click-modal="false":destroy-on-close="true"top="0"width="100%"><divclass="canvasBox":style="{ height: config.height + 'px' }"><canvasclass="canvas"></canvas><divclass="tipTxt"v-show="showTip"><p>示例文字!</p><pclass="redTxt">请按示例文字方向,正楷清晰书写。谢谢!</p></div></div><template#footer><div><el-buttontype="primary"@click="save">保存</el-button><el-button@click="cancel">清除</el-button><el-button@click="handleClose(false)">关闭</el-button></div></template></el-dialog></div></template><stylelang="scss">.el-dialog__header{display: none;}.el-dialog__body{padding: 0 !important;}.canvasBox{position: relative;}.canvas,
.tipTxt{position: absolute;top: 0;left: 0;bottom: 0;right: 0;}.tipTxt{display: flex;flex-direction: column;align-items: center;justify-content: center;}.tipTxt p{color: #999;font-size: 80px;}.tipTxt .redTxt{color:rgba(255, 0, 0, 0.5);font-size: 50px;}@media screen and(max-width: 690px){.tipTxt{transform:rotate(90deg);}.tipTxt p{font-size: 50px;}.tipTxt .redTxt{font-size: 18px;}}</style>

js 控制 showTip 的展示

<script setup>import{ ref, onMounted, nextTick }from"vue";// eslint-disable-next-lineconst props =defineProps({dialogVisible:{type: Boolean,default:false,},});// eslint-disable-next-lineconst emit =defineEmits(["closeEvent"]);// 关闭弹窗functionhandleClose(imgEl){
  console.log(imgEl);emit("closeEvent", imgEl);// 禁止页面滚动
  document.body.removeEventListener("touchmove", preventDefault);}const dialogBox =ref();functioncloseDialog(){
  dialogBox.value.resetFields();}// 禁止页面滚动functionpreventDefault(e){
  e.preventDefault();}
document.body.addEventListener("touchmove", preventDefault,{passive:false});// 签名// 配置内容const config ={width: window.innerWidth,// 宽度height: window.innerHeight -150,// 高度lineWidth:5,// 线宽strokeStyle:"red",// 线条颜色lineCap:"round",// 设置线条两端圆角lineJoin:"round",// 线条交汇处圆角};let canvas =null;let ctx =null;onMounted(async()=>{awaitnextTick();// 获取canvas 实例
  canvas = document.querySelector(".canvas");
  console.log(canvas);// 设置宽高
  canvas.width = config.width;
  canvas.height = config.height;// 设置一个边框
  canvas.style.border ="1px solid #000";// 创建上下文
  ctx = canvas.getContext("2d");// 设置填充背景色
  ctx.fillStyle ="transparent";// 绘制填充矩形
  ctx.fillRect(0,// x 轴起始绘制位置0,// y 轴起始绘制位置
    config.width,// 宽度
    config.height // 高度);});// 保存上次绘制的 坐标及偏移量const client ={offsetX:0,// 偏移量offsetY:0,endX:0,// 坐标endY:0,};// 判断是否为移动端const mobileStatus =/Mobile|Android|iPhone/i.test(navigator.userAgent);// 初始化constinit=(event)=>{// 获取偏移量及坐标const{ offsetX, offsetY, pageX, pageY }= mobileStatus ? event.changedTouches[0]: event;// 修改上次的偏移量及坐标
  client.offsetX = offsetX;
  client.offsetY = offsetY;
  client.endX = pageX;
  client.endY = pageY;// 清除以上一次 beginPath 之后的所有路径,进行绘制
  ctx.beginPath();// 根据配置文件设置相应配置
  ctx.lineWidth = config.lineWidth;
  ctx.strokeStyle = config.strokeStyle;
  ctx.lineCap = config.lineCap;
  ctx.lineJoin = config.lineJoin;// 设置画线起始点位
  ctx.moveTo(client.endX, client.endY);// 监听 鼠标移动或手势移动
  window.addEventListener(mobileStatus ?"touchmove":"mousemove", draw);};// 绘制constdraw=(event)=>{// 获取当前坐标点位const{ pageX, pageY }= mobileStatus ? event.changedTouches[0]: event;// 超出范围不监听if(pageY > config.height){return;}
  showTip.value =false;// 修改最后一次绘制的坐标点
  client.endX = pageX;
  client.endY = pageY;// 根据坐标点位移动添加线条
  ctx.lineTo(pageX, pageY);// 绘制
  ctx.stroke();};// 结束绘制constcloaseDraw=()=>{// 结束绘制
  ctx.closePath();// 移除鼠标移动或手势移动监听器
  window.removeEventListener("mousemove", draw);};// 创建鼠标/手势按下监听器
window.addEventListener(mobileStatus ?"touchstart":"mousedown", init);// 创建鼠标/手势 弹起/离开 监听器
window.addEventListener(mobileStatus ?"touchend":"mouseup", cloaseDraw);// 取消-清空画布constcancel=()=>{// 清空当前画布上的所有绘制内容
  ctx.clearRect(0,0, config.width, config.height);
  showTip.value =true;};// 保存-将画布内容保存为图片constsave=()=>{// 将canvas上的内容转成blob流var imgData = ctx.getImageData(0,0, canvas.width, canvas.height).data;var lOffset = canvas.width,
    rOffset =0,
    tOffset = canvas.height,
    bOffset =0;for(var i =0; i < canvas.width; i++){for(var j =0; j < canvas.height; j++){var pos =(i + canvas.width * j)*4;if(imgData[pos]>0|| imgData[pos +1]>0|| imgData[pos +2]|| imgData[pos +3]>0){// 说第j行第i列的像素不是透明的// 楼主貌似底图是有背景色的,所以具体判断RGBA的值可以根据是否等于背景色的值来判断
        bOffset = Math.max(j, bOffset);// 找到有色彩的最底部的纵坐标
        rOffset = Math.max(i, rOffset);// 找到有色彩的最右端

        tOffset = Math.min(j, tOffset);// 找到有色彩的最上端
        lOffset = Math.min(i, lOffset);// 找到有色彩的最左端}}}if(lOffset === config.width && rOffset ===0&& tOffset === config.height && bOffset ===0){// eslint-disable-next-lineElMessage({message:"请签名后保存!",type:"warning",});return;}const trimmedWidth = rOffset - lOffset +1;const trimmedHeight = bOffset - tOffset +1;const trimmedCanvas = document.createElement("canvas");
  trimmedCanvas.width = trimmedWidth;
  trimmedCanvas.height = trimmedHeight;const trimmedContext = trimmedCanvas.getContext("2d");
  trimmedContext.putImageData(
    ctx.getImageData(lOffset, tOffset, trimmedWidth, trimmedHeight),0,0);const newUrl = trimmedCanvas.toDataURL();const newImage =newImage();
  newImage.src = newUrl;
  console.log(trimmedWidth < trimmedHeight);if(trimmedWidth < trimmedHeight){
    newImage.style.height ="100px";
    newImage.style.transform ="rotate(-"+90+"deg)";}else{
    newImage.style.width ="100px";}// console.log(newImage.outerHTML + "</img>");handleClose(newImage.outerHTML +"</img>");};// 示例文字let showTip =ref(true);</script>

效果:
在这里插入图片描述
后续其实还有更多需求,但是菜鸟感觉再写就太复杂了,增加了读者的负担,而且也不一定用得上,所以就不再写了!


本文转载自: https://blog.csdn.net/langwang_100/article/details/140872616
版权归原作者 PBitW 所有, 如有侵权,请联系我们删除。

“前端实现签字效果+合同展示”的评论:

还没有评论