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

PBitW 2024-08-11 08:03:01 阅读 50

文章目录

获取一个高度会变的元素的高度获取元素设置的 transform适配手机transform-origin: 5% 0; 的原因修改后

签字效果取消el-dialog的头部+边距为什么禁止界面滚动vue3 使用 nextTick实现效果

签字判断是横是竖canvas 去掉空白部分canvas裁剪图片最终完善代码,可以直接使用

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

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

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

script 代码

<code>let bigBoxHeight = ref(0);

// 获取到元素

let bigBox = document.querySelector(".bigBox");

// 设置高度为 auto

bigBox.style.height = "auto";

// 获取 offsetHeight

const height = bigBox.offsetHeight;

// 设置值

bigBoxHeight.value = height;

注:

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

template 代码

<div class="bigBox" :style="{ height: bigBoxHeight + 'px' }">code>

<div class="contractBox">code>

<div v-html="printData"></div>code>

</div>

<!-- 遮罩层,返回的printData里设置了可编辑,但是这里只是展示用,且修改了也不会有影响,所以就简单的加个遮罩就行了 -->

<div class="markBox" :style="{ height: bigBoxHeight + 'px' }"></div>code>

</div>

获取元素设置的 transform

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

// 获取设置了transform的元素

let contractBox = document.querySelector(".contractBox");

// 获取浏览器计算后的结果

let st = window.getComputedStyle(contractBox, null);

// 从结算后的结果中找到 transform,也可以直接 st.transform

var 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 代码

<div class="bigBox" :style="{ height: bigBoxHeight + 'px' }">code>

<div class="contractBox" :style="{ marginLeft: isScale ? bigBoxmargin + 'px' : 'auto' }">code>

<div v-html="printData"></div>code>

</div>

<div class="markBox" :style="{ height: bigBoxHeight + 'px' }"></div>code>

</div>

签字效果

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

这里主要粘贴签字的代码

<script setup>

import { -- --> ref, onMounted, nextTick } from "vue";

// eslint-disable-next-line

const props = defineProps({

dialogVisible: {

type: Boolean,

default: false,

},

});

// eslint-disable-next-line

const emit = defineEmits(["closeEvent"]);

// 关闭弹窗

function handleClose() {

emit("closeEvent", false);

// 解除禁止页面滚动

document.body.removeEventListener("touchmove", preventDefault);

}

const dialogBox = ref();

function closeDialog() {

dialogBox.value.resetFields();

}

// 禁止页面滚动

function preventDefault(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 () => {

await nextTick();

// 获取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);

// 初始化

const init = (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);

};

// 绘制

const draw = (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();

};

// 结束绘制

const cloaseDraw = () => {

// 结束绘制

ctx.closePath();

// 移除鼠标移动或手势移动监听器

window.removeEventListener("mousemove", draw);

};

// 创建鼠标/手势按下监听器

window.addEventListener(mobileStatus ? "touchstart" : "mousedown", init);

// 创建鼠标/手势 弹起/离开 监听器

window.addEventListener(mobileStatus ? "touchend" : "mouseup", cloaseDraw);

// 取消-清空画布

const cancel = () => {

// 清空当前画布上的所有绘制内容

ctx.clearRect(0, 0, config.width, config.height);

};

// 保存-将画布内容保存为图片

const save = () => {

// 将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="签字"code>

ref="dialogBox"code>

:modelValue="dialogVisible"code>

:before-close="handleClose"code>

@close="closeDialog"code>

:close-on-click-modal="false"code>

:destroy-on-close="true"code>

top="0"code>

width="100%"code>

>

<canvas class="canvas"></canvas>code>

<template #footer>

<div>

<el-button type="primary" @click="save">保存</el-button>code>

<el-button @click="cancel">清除</el-button>code>

<el-button @click="handleClose">关闭</el-button>code>

</div>

</template>

</el-dialog>

</div>

</template>

<style lang="scss">code>

.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 中代码

<code>const save = () => { -- -->

// 将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继续修改为

const save = () => {

// 将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 = new Image();

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 打印出来是一个元素:

<img src="" style="transform: rotate(-90deg);">code>

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

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

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

template

<template>

<div>

<el-dialog

title="签字"code>

ref="dialogBox"code>

:modelValue="dialogVisible"code>

:before-close="handleClose"code>

@close="closeDialog"code>

:close-on-click-modal="false"code>

:destroy-on-close="true"code>

top="0"code>

width="100%"code>

>

<div class="canvasBox" :style="{ height: config.height + 'px' }">code>

<canvas class="canvas"></canvas>code>

<div class="tipTxt" v-show="showTip">code>

<p>示例文字!</p>

<p class="redTxt">请按示例文字方向,正楷清晰书写。谢谢!</p>code>

</div>

</div>

<template #footer>

<div>

<el-button type="primary" @click="save">保存</el-button>code>

<el-button @click="cancel">清除</el-button>code>

<el-button @click="handleClose(false)">关闭</el-button>code>

</div>

</template>

</el-dialog>

</div>

</template>

<style lang="scss">code>

.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-line

const props = defineProps({

dialogVisible: {

type: Boolean,

default: false,

},

});

// eslint-disable-next-line

const emit = defineEmits(["closeEvent"]);

// 关闭弹窗

function handleClose(imgEl) {

console.log(imgEl);

emit("closeEvent", imgEl);

// 禁止页面滚动

document.body.removeEventListener("touchmove", preventDefault);

}

const dialogBox = ref();

function closeDialog() {

dialogBox.value.resetFields();

}

// 禁止页面滚动

function preventDefault(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 () => {

await nextTick();

// 获取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);

// 初始化

const init = (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);

};

// 绘制

const draw = (event) => {

showTip.value = false;

// 获取当前坐标点位

const { pageX, pageY } = mobileStatus ? event.changedTouches[0] : event;

// 修改最后一次绘制的坐标点

client.endX = pageX;

client.endY = pageY;

// 根据坐标点位移动添加线条

ctx.lineTo(pageX, pageY);

// 绘制

ctx.stroke();

};

// 结束绘制

const cloaseDraw = () => {

// 结束绘制

ctx.closePath();

// 移除鼠标移动或手势移动监听器

window.removeEventListener("mousemove", draw);

};

// 创建鼠标/手势按下监听器

window.addEventListener(mobileStatus ? "touchstart" : "mousedown", init);

// 创建鼠标/手势 弹起/离开 监听器

window.addEventListener(mobileStatus ? "touchend" : "mouseup", cloaseDraw);

// 取消-清空画布

const cancel = () => {

// 清空当前画布上的所有绘制内容

ctx.clearRect(0, 0, config.width, config.height);

showTip.value = true;

};

// 保存-将画布内容保存为图片

const save = () => {

// 将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-line

ElMessage({

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 = new Image();

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>

效果:

在这里插入图片描述



声明

本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。