vue使用Fabric和pdfjs完成合同签章及批注
合同签章及批注模块涉及以下核心难点和复杂点,按功能模块归类如下:
效果展示
使用拖拽的方式将签名放入pdf文件,再点击文字批注对文件进行添加批注
难点及功能点描述
一、PDF渲染与交互
多页动态渲染
使用pdfjs-dist库实现PDF分页渲染
动态创建Canvas元素并根据页面尺寸自动调整布局
处理PDF缩放时的全量重新渲染(sliderChange方法)
滚动定位
通过canvasLayoutTopList记录每页位置
滚动事件监听实现精准页码定位(outViewRun方法)
滚动节流处理防止性能问题
二、电子签章系统
图形交互
使用Fabric.js实现可拖拽签章
自定义控制点图标(删除/缩放按钮)
边界检测防止拖出画布(object:moving事件处理)
状态同步
通过coordinateList维护签章坐标/尺寸/角度等状态
实时同步画布操作与数据模型(getSignatureJson方法)
唯一标识符cacheKey管理对象身份
缩放控制
自定义缩放算法(handleScaling方法)
最小缩放限制防止过度缩小
等比缩放保持图形比例
三、文字批注系统
富文本交互
动态创建可编辑文本框(fabric.IText)
字体缩放时自动调整字号(baseFontSize跟踪)
文本框边界检测与自动换行
复合对象管理
每页仅允许单个批注的限制逻辑
文字与签章的状态混合存储(通过name字段区分)
四、文件处理
PDF文件下载
Blob对象处理二进制流
文件名编码解码处理(content-disposition解析)
内存管理(revokeObjectURL)
签章图片上传
自定义上传组件(el-upload封装)
服务端响应处理(二进制流/JSON判断)
Token鉴权处理
五、性能优化
画布渲染优化
局部渲染(canvas.requestRenderAll())
对象缓存(cacheKey机制)
事件委托减少重复绑定
内存管理
Canvas对象销毁逻辑
Blob URL及时释放
六、业务逻辑复杂性
状态验证
签章必填校验(提交前coordinateList检查)
每页签章/批注数量限制
数据转换
坐标系统转换(屏幕坐标与PDF坐标)
服务端数据结构适配(processSignsData方法)
七、浏览器兼容性
Blob API兼容
多浏览器下载兼容处理(Chrome/Firefox/Edge)
大文件下载内存管理
触摸事件支持
移动端拖拽/缩放适配(未完全实现)
前期准备
引入插件
在文件public下增加一个pdfjs文件夹,且再增加一个build文件夹,其中放pdf.js和pdf.worker.js文件
文末放有pdf.js和pdf.worker.js文件源码
// pdf插件
import { fabric } from "fabric";
import * as pdfjsLib from "pdfjs-dist/build/pdf";
import workerSrc from "pdfjs-dist/build/pdf.worker.entry";
import * as pdfjsViewer from "pdfjs-dist/web/pdf_viewer";
主要代码,
这是大佬的地址 vue里面使用pdfjs-dist+fabric实现pdf电子签章!!!
在借鉴了另外一篇文章后进行修改的如下,就是做了点修改(ps:目前印章的位置和坐标保存,使用的得本地缓存,便于调试,后期会保存到接口里面!)
<!-- //?模块说明 => 合同签章模块 addToTab-->
<template><div class="contract-signature-view"><div class="section-box"><!-- 签章图片 --><!-- <aside class="signature-img"><div class="info"><h3 class="name">印章</h3><p class="text">将示例印章标识拖到文件相应区域即可获取签章位置</p></div></aside> --><!-- 主体区域 --><div class="main-layout" :class="{ 'is-first': isFirst }"><!-- 操作 --><div class="title-operation"><div class="operation"><el-buttonv-if="$route.query.formst==31"class="searchbutton"type="primary"@click="optionSignature('addSubSign')"><svg-icon icon-class="check" />呈签</el-button><el-buttonv-elseclass="searchbutton"type="primary"@click="optionSignature('agree')"><svg-icon icon-class="check" />同意</el-button><!-- <el-buttonclass="searchbutton"icon="el-icon-close"@click="optionSignature('return')">回退</el-button> --><!-- <el-button class="searchbutton" @click="download()"><svg-icon icon-class="download" />下载签核文件</el-button> --><el-button class="searchbutton" @click="returnSignature"><svg-icon icon-class="return" />返回</el-button><el-popover placement="bottom" trigger="click"><template #default><div class="signatureimg" v-if="vissign == true"><!-- 拖拽 --><draggablev-model="mainImagelist":group="{ name: 'itext', pull: 'clone' }":sort="false"@end="end"@move="onMove"><transition-group type="transition"><liv-for="item in mainImagelist":key="item.img"class="item"style="text-align: center"><authImg:authSrc="item.img"alt=""class="img"></authImg><!-- <img ref="img" :src="item.img" width="100%;" height="100%" class="img" /> --></li></transition-group></draggable><div class="opera"><span>请拖拽签名至目标位置</span><span><el-buttonclass="delbut"icon="el-icon-delete"@click="deleteSignature">删除</el-button></span></div></div><div v-else class="upimg"><!-- <div style="color: #3f7afa;font-size: 14px;"><svg-icon icon-class="plus" />添加签名</div> --><el-uploadclass="upload-demo"action="#":http-request="customRequest":on-preview="handlePreview":on-remove="handleRemove":before-remove="beforeRemove":limit="1"accept=".png, .jpg":on-exceed="handleExceed"><el-button size="small" class="upload-button"><svg-icon icon-class="plus" /> 添加签名</el-button><divslot="tip"style="font-size: 10px;padding-top: 10px;color:#00000066;">支持上传png图片,大小不超过500kb</div></el-upload></div></template><el-button slot="reference" class="button"><svg-icon icon-class="Esign" /> 电子签名</el-button></el-popover><!-- <el-button class="button" @click="addTextobjectHandle($event)" > <svg-icon icon-class="Esign" /> 电子签名</el-button> --><!-- <el-button type="danger" @click="removeSignature">删除签章</el-button>--><!-- <el-button type="primary" @click="submitSignature">提交签章</el-button> --><el-button class="button" @click="addTextobjectHandle($event)"><svg-icon icon-class="font-size" /> 文字批注</el-button><!-- <el-button type="danger" @click="clearSignature">清空签章</el-button> --></div></div><div class="operate-box"><div class="pageNo-change"><i class="icon el-icon-arrow-left" @click="prevPage" /><el-inputclass="input-box"v-model.number="pageNum":max="defaultNumPages"@change="cutover"/><span class="default-text">/{{ defaultNumPages }}</span><i class="icon el-icon-arrow-right" @click="nextPage" /></div></div><!-- <div class="container_cont" style="margin-top:1%;position: relative;display: flex;align-items: center;justify-content: center;"><button id="prevpage" style=" width: 120px;height: 30px;background: none;border: 1px solid #b1afaf;border-radius: 5px;font-size: 12px;font-weight: 1000;color: #384240;cursor: pointer;outline: none;margin: 0 0.5%">上一页</button><button id="nextpage" style=" width: 120px;height: 30px;background: none;border: 1px solid #b1afaf;border-radius: 5px;font-size: 12px;font-weight: 1000;color: #384240;cursor: pointer;outline: none; margin: 0 0.5%">下一页</button></div> --><!-- 画图 --><div class="out-view" :class="{ 'is-show': isShowPdf }"><div class="canvas-layout" v-for="item in numPages" :key="item"><!-- pdf部分 --><canvas class="the-canvas" id="box" /><!-- 签名部分 --><canvas class="ele-canvas" id="canvas"></canvas><!-- <div id="menu" class="menu-x"><div class="menu-li" @click="removeSignature">删除</div></div> --><!-- <canvas class="text-canvas" id="canvas" ></canvas> --></div></div><i class="loading" v-loading="!isShowPdf" /></div><!-- 位置信息 --></div><Dialog:isVisible="isVisible":id="id":scale="scale":title="title"@closeVisible="closeVisible":signlist="signlist"></Dialog></div>
</template><script>
import { getToken } from "@/utils/auth";
import { uploadSignture, removeSignture } from "@/api/create/index";
import userApi from "@/api/user";
import Dialog from "./components/Dialog.vue";
import authImg from "@/components/authImg.vue";
import { downloadsignoffApi } from "@/api/common";
// 拖拽插件
import draggable from "vuedraggable";
// pdf插件
import { fabric } from "fabric";
import * as pdfjsLib from "pdfjs-dist/build/pdf";
import workerSrc from "pdfjs-dist/build/pdf.worker.entry";
import * as pdfjsViewer from "pdfjs-dist/web/pdf_viewer";
import { fa, it } from "element-plus/es/locales.mjs";
pdfjsLib.GlobalWorkerOptions.workerSrc = workerSrc;export default {components: { draggable, Dialog, authImg },data() {return {// pdf地址pdfUrl: "",// 左侧签章列表mainImagelist: [],// 右侧坐标数据coordinateList: [],// 总页数numPages: 1,defaultNumPages: 1,// 当前页pageNum: 1,// 缩放比例scale: 1,// pdf是否显示isFirst: true,isShowPdf: false,// pdf最外层的out-viewoutViewDom: null,// 各页pdf的canvas-layoutcanvasLayoutTopList: [],currentDragPosition: {x: "",y: ""},// 用来签章的canvas数组canvasEle: [],// 绘图区域的宽高whDatas: null,whlist: [],// pdf渲染的canvas数组canvas: [],// pdf渲染的canvas的ctx数组ctx: [],// pdf渲染的canvas的宽高pdfDoc: null,// 隐藏的input,用来提交数据shadowInputValue: "",points: [],activeEl: null,activeobjectData: {type: "textbox",text: "文字批注示例",fontSize: 20,left: 200,top: 400,width: 100,fill: "#000000",textBackgroundcolor: "rgba(0,0,0,0)",opacity: 1,stroke: "#ffffff",strokeWidth: 0,background: "#7ED321",scaleX: 1,scaleY: 1,fontFamily: 'Microsoft YaHei',underline: false,linethrough: false,overline: false,textAlign: "left",lineHeight: 1,charSpacing: 1,cornerColor: "#3B82F6",cornerStyle: "circle",borderScaleFactor: 2,transparentCorners: false,rotate: 0,selectable: true},isVisible: false,title: "",deleteIcon:"",samllIcon:"",largeIcon:"",fileList: [],vissign: false,signlist: [],id: "",sts: [],baseIp: "",userInfo: {}};},created() {this.getsignimg();this.setPdfArea();},watch: {coordinateList: {handler(val) {},deep: true},vissign: {handler(val) {},deep: true}},mounted() {},methods: {getsignimg() {//初始化判断是否个人信息有绑定自己的签名章,userApi.getInfo().then(res => {this.userInfo = res.data.data;//如果没有则上传后获取图片地址,如果有就直接获取if (this.userInfo.signture != null) {this.vissign = true;const hostname = window.location.hostname;this.baseIp ="https://xxx.com/mis/api-docs/chairmansoffice/download";// /2025/2/F1339892_212757768635974367.pngif (hostname != "localhost") {var origin = window.location.origin;this.baseIp = origin + "/mis/api-docs/chairmansoffice/download";}this.mainImagelist = [{ name: "印章", img: this.baseIp + this.userInfo.signture }];} else {this.vissign = false;}});},/*** pdf相关部分*/// 设置PDF地址setPdfArea() {// 1. 获取地址栏//z这里的地址根据用户需求自己更改,不是真实可用地址// 获取pdf地址,这里区分是否开发环境和测试环境var fileurl = this.$route.query.fileurl;const hostname = window.location.hostname;var baseIp ="https://test-xxxx.com/mis/download";if (hostname != "localhost") {var origin = window.location.origin;baseIp = origin + "/mis/download";}this.baseIp = baseIp;this.pdfUrl = baseIp + fileurl;// this.pdfUrl ="https://test-xxxx.com/mis/download/2025/2/430956676170020900__2674826190452557870.pdf"//可以先定一个固定pdf地址进行测试this.showpdf(this.pdfUrl); // 接口返回的应该还有签名信息,不只是pdf},showsts() {if (this.$route.query.sts?.length > 0) {this.sts = this.$route.query.sts;this.sts.forEach(element => {const targetCanvas = this.canvasEle[element.pageNo - 1];if (!targetCanvas) return;// 创建文本的逻辑封装const createAndAddText = () => {if (!element.opinion) return;const scale = element.opscale || 1;const fontSize = element.fontsize || 14;const textConfig = {left: element.yopinion,top: element.xopinion,width: element.wopinion / scale,fontSize: fontSize,fill: "#000000",scaleX: scale,scaleY: scale,selectable: false,hasControls: false,splitByGrapheme: true,textBackgroundColor: "rgba(0,0,0,0)",fontWeight: (element.fontweight || "normal").toLowerCase()};const text = new fabric.Textbox(element.opinion, textConfig);targetCanvas.add(text);targetCanvas.renderAll();console.log("Created text:", text, "Config:", textConfig);};// 创建图片的公共逻辑const createAndAddImage = () => {fabric.Image.fromURL(this.baseIp + element.fn, img => {img.set({left: element.ysign,top: element.xsign,scaleX: element.scale || 1,scaleY: element.scale || 1,selectable: false});targetCanvas.add(img);});};// 执行创建操作createAndAddText(); // 有opinion才创建createAndAddImage(); // 无论有无opinion都创建});}},showpdf(pdfUrl) {this.canvas = document.querySelectorAll(".the-canvas");let currentPage = 1;// console.log(getToken(), "-------------");pdfjsLib.getDocument({url: pdfUrl,httpHeaders: { Token: getToken() },rangeChunkSize: 65536,disableAutoFetch: false}).promise.then(pdfDoc_ => {this.pdfDoc = pdfDoc_;this.numPages = this.pdfDoc.numPages;this.defaultNumPages = this.pdfDoc.numPages;this.$nextTick(() => {this.canvas = document.querySelectorAll(".the-canvas");this.canvas.forEach(item => {this.ctx.push(item.getContext("2d"));});// 循环渲染pdffor (let i = 1; i <= this.numPages; i++) {this.renderPage(i).then(() => {this.renderPdf({width: this.canvas[i - 1].width,height: this.canvas[i - 1].height});});}setTimeout(() => {this.renderFabric();this.canvasEvents();}, 1000);});});},// 设置pdf宽高,缩放比例,渲染pdfrenderPage(num) {return this.pdfDoc.getPage(num).then(pageNo => {const viewport = pageNo.getViewport({ scale: this.scale }); // 设置视口大小this.canvas[num - 1].height = viewport.height;this.canvas[num - 1].width = viewport.width;// Render PDF pageNo into canvas contextconst renderContext = {canvasContext: this.ctx[num - 1],viewport: viewport};pageNo.render(renderContext);});},// 设置绘图区域宽高renderPdf(data) {this.whlist.push(data);this.whDatas = data;// document.querySelector(".elesign").style.width = data.width + "px";},// 生成绘图区域renderFabric() {// 1. 拿到全部的canvas-layoutconst canvasLayoutDom = document.querySelectorAll(".canvas-layout");// 2. 循环遍历canvasLayoutDom.forEach((item, index) => {this.canvasLayoutTopList.push({ obj: item, top: item.offsetTop });// 3. 设置宽高和居中,根据存放绘制宽高遍历获取// item.style.width = this.whDatas.width + "px";// item.style.height = this.whDatas.height + "px";item.style.width = this.whlist[index].width + "px";item.style.height = this.whlist[index].height + "px";item.style.margin = "0 auto 18px";item.style.boxShadow = "4px 4px 4px #e9e9e9";// 4. 拿到签名canvasconst canvasEle = item.querySelector(".ele-canvas");// 5. 拿到pdf的canvasconst pCenter = item.querySelector(".the-canvas");// 6. 设置签名canvas的宽高canvasEle.width = pCenter.clientWidth;canvasEle.height = this.whlist[index].height;// 7. 创建fabric对象并存储const canvas = new fabric.Canvas(canvasEle);// canvas.on('mouse:up', (e) => {// // 画布鼠标按下事件// this.getSignatureJson();// })s// .on('object:scaling', (e) => {// // 图形缩放时触发;// console.log(e.transform);// })this.canvasEle.push(canvas);// 按下鼠标const containers = item.querySelectorAll(".canvas-container");// 8. 设置签名和text文本输入的canvas的样式// containers.forEach(div => {// div.style.position = "absolute";// div.style.left = "50%";// div.style.transform = "translateX(-50%)";// div.style.top = "0px";// // console.dir(div);// 找到和class相关的属性// })const container = item.querySelector(".canvas-container");container.style.position = "absolute";container.style.left = "50%";container.style.transform = "translateX(-50%)";container.style.top = "0px";});// 现形this.isFirst = false;this.isShowPdf = true;this.outViewDom = document.querySelector(".out-view");// 开启监听窗口滚动this.outViewScroll();this.showsts();},// 开启监听窗口滚动outViewScroll() {this.outViewDom.addEventListener("scroll", this.outViewRun);},// 关闭监听窗口滚动outViewScrollClose() {this.outViewDom.removeEventListener("scroll", this.outViewRun);},// 窗口滚动outViewRun() {const scrollTop = this.outViewDom.scrollTop;const topList = this.canvasLayoutTopList.map(item => item.top);// 增加一个最大值topList.push(Number.MAX_SAFE_INTEGER);for (let index = 0; index < topList.length; index++) {const element = topList[index];if (element <= scrollTop && scrollTop < topList[index + 1]) {this.pageNum = index + 1;break;}}},// scale滑块,重新渲染整个pdfsliderChange() {this.pageNum = 1;this.numPages = 0;this.canvasLayoutTopList = [];this.canvasEle = [];this.ctx = [];this.canvas = [];this.isShowPdf = false;// this.outViewScrollClose();this.whDatas = null;this.coordinateList = [];this.getSignatureJson();setTimeout(() => {this.numPages = this.pdfDoc.numPages;this.$nextTick(() => {this.canvas = document.querySelectorAll(".the-canvas");this.canvas.forEach(item => {this.ctx.push(item.getContext("2d"));});// 循环渲染pdffor (let i = 1; i <= this.numPages; i++) {this.renderPage(i).then(() => {this.renderPdf({width: this.canvas[i - 1].width,height: this.canvas[i - 1].height});});}setTimeout(() => {this.renderFabric();this.canvasEvents();this.showsts();}, 1000);});}, 1000);},/*** 签章相关部分*/// 签章拖拽边界处理,不能将图片拖拽到绘图区域外// canvasEvents() {// this.canvasEle.forEach(item => {// item.on("object:moving", e => {// const obj = e.target;// if(e.target!=null){// const top = obj.top;// const left = obj.left;// // if object is too big ignore// if (// obj.currentHeight > obj.canvas.height ||// obj.currentWidth > obj.canvas.width// ) {// return;// }// obj.setCoords();// // top-left corner// if (obj.getBoundingRect().top < 0 || obj.getBoundingRect().left < 0) {// obj.top = Math.max(obj.top, obj.top - obj.getBoundingRect().top);// obj.left = Math.max(// obj.left,// obj.left - obj.getBoundingRect().left// );// }// // bot-right corner// if (// obj.getBoundingRect().top + obj.getBoundingRect().height >// obj.canvas.height ||// obj.getBoundingRect().left + obj.getBoundingRect().width >// obj.canvas.width// ) {// obj.top = Math.min(// obj.top,// obj.canvas.height -// obj.getBoundingRect().height +// obj.top -// obj.getBoundingRect().top// );// obj.left = Math.min(// obj.left,// obj.canvas.width -// obj.getBoundingRect().width +// obj.left -// obj.getBoundingRect().left// );// }// var findIndex= 0// this.coordinateList.forEach((element,index) => {// if(element.cacheKey == e.target.cacheKey){// findIndex = index// }// });// console.log( this.coordinateList[findIndex ],'------------------ this.coordinateList[findIndex + 1]')// // const findIndex = this.coordinateList// // .slice(1)// // .findIndex(coord => coord.cacheKey == obj.cacheKey);// const keys = [// "width",// "height",// "top",// "left",// "angle",// "scaleX",// "scaleY"// ];// keys.forEach(item => {// this.coordinateList[findIndex ][item] = Math.ceil(// obj[item] * obj["scaleX"]// );// });// if(this.coordinateList[findIndex + 1].name == '批注'){// this.coordinateList[findIndex + 1].xopinion= JSON.parse(JSON.stringify(top))// this.coordinateList[findIndex + 1].opinion= JSON.parse(JSON.stringify(obj.text))// this.coordinateList[findIndex + 1].fontsize= JSON.parse(JSON.stringify(obj.fontsize))// this.coordinateList[findIndex + 1].yopinion= JSON.parse(JSON.stringify(left))// this.coordinateList[findIndex + 1].wopinion= JSON.parse(JSON.stringify(obj.width))// this.coordinateList[findIndex + 1].hopinion= JSON.parse(JSON.stringify(obj.height))// this.coordinateList[findIndex + 1].opscale= JSON.parse(JSON.stringify(obj.scaleX))// }else{// this.coordinateList[findIndex + 1].name = '印章'// this.coordinateList[findIndex + 1].xsign= JSON.parse(JSON.stringify(top))// this.coordinateList[findIndex + 1].ysign= JSON.parse(JSON.stringify(left))// this.coordinateList[findIndex + 1].wsign= JSON.parse(JSON.stringify(obj.width))// this.coordinateList[findIndex + 1].hsign= JSON.parse(JSON.stringify(obj.height))// this.coordinateList[findIndex + 1].scale= JSON.parse(JSON.stringify(obj.scaleX))// }// this.coordinateList[findIndex + 1].scaleX= JSON.parse(JSON.stringify(obj.scaleX))// this.coordinateList[findIndex + 1].scaleY= JSON.parse(JSON.stringify(obj.scaleY))// // console.log( this.coordinateList[findIndex + 1].top, this.coordinateList[findIndex + 1].left,'----------item')// this.getSignatureJson();// }// });// // item.on('mouse:down', e => {// item.on("mouse:move", e => {// if(e.target!=null){// const obj =JSON.parse(JSON.stringify( e.target));// const top = obj.top;// const left = obj.left;// var findIndex= 0// this.coordinateList.forEach((element,index) => {// if(element.cacheKey == e.target.cacheKey){// findIndex = index// }// });// console.log( findIndex,this.coordinateList,e,'------------------ this.coordinateList[findIndex + 1]')// const keys = [// "width",// "height",// "top",// "left",// "angle",// "scaleX",// "scaleY"// ];// keys.forEach(item => {// // console.log(item, this.coordinateList[findIndex + 1][item],'-------------item')// this.coordinateList[findIndex][item] = Math.ceil(// obj[item] * obj["scaleX"]// );// });// // this.coordinateList[findIndex + 1].width= JSON.parse(JSON.stringify(obj.width))// // this.coordinateList[findIndex + 1].height= JSON.parse(JSON.stringify(obj.height))// if(this.coordinateList[findIndex].name == '批注'){// this.coordinateList[findIndex].xopinion= JSON.parse(JSON.stringify(top))// this.coordinateList[findIndex].yopinion= JSON.parse(JSON.stringify(left))// this.coordinateList[findIndex].fontsize= JSON.parse(JSON.stringify(obj.fontsize))// this.coordinateList[findIndex].opinion= JSON.parse(JSON.stringify(obj.text))// this.coordinateList[findIndex].wopinion= JSON.parse(JSON.stringify(obj.width))// this.coordinateList[findIndex].hopinion= JSON.parse(JSON.stringify(obj.height))// this.coordinateList[findIndex].opscale= JSON.parse(JSON.stringify(obj.scaleX))// }else{// this.coordinateList[findIndex].name = '印章'// this.coordinateList[findIndex].xsign= JSON.parse(JSON.stringify(top))// this.coordinateList[findIndex].ysign= JSON.parse(JSON.stringify(left))// this.coordinateList[findIndex].wsign= JSON.parse(JSON.stringify(obj.width))// this.coordinateList[findIndex].hsign= JSON.parse(JSON.stringify(obj.height))// this.coordinateList[findIndex].scale= JSON.parse(JSON.stringify(obj.scaleX))// }// this.coordinateList[findIndex ].scaleX= JSON.parse(JSON.stringify(obj.scaleX))// this.coordinateList[findIndex].scaleY= JSON.parse(JSON.stringify(obj.scaleY))// // console.log(obj.fontSize, top,left,'--obj.fontsize--------item')// console.log( this.coordinateList[findIndex ],'----------coordinate33List')// this.getSignatureJson();// }// });// });// },canvasEvents() {this.canvasEle.forEach(canvas => {canvas.on("object:moving", e => {const obj = e.target;if (!obj) return;// 边界处理obj.setCoords(); // 更新控制点坐标const canvasWidth = canvas.getWidth();const canvasHeight = canvas.getHeight();// 限制移动范围(考虑缩放)const boundingRect = obj.getBoundingRect();const scale = this.scale;// 计算有效移动范围const minX = 0 - (boundingRect.width * (1 - scale)) / 2;const minY = 0 - (boundingRect.height * (1 - scale)) / 2;const maxX = canvasWidth - (boundingRect.width * (1 + scale)) / 2;const maxY = canvasHeight - (boundingRect.height * (1 + scale)) / 2;// 应用位置限制obj.left = Math.min(Math.max(obj.left, minX), maxX);obj.top = Math.min(Math.max(obj.top, minY), maxY);// 更新坐标数据const findIndex = this.coordinateList.findIndex(item => item.cacheKey === obj.cacheKey);if (findIndex === -1) return;// 计算实际字体大小(核心修改部分)const baseFontSize = obj.baseFontSize || obj.fontSize; // 获取基准字号const scaleFactor = Math.max(obj.scaleX, obj.scaleY); // 取最大缩放值const actualFontSize = Math.round(baseFontSize * scaleFactor);// 通用属性更新const coordinate = this.coordinateList[findIndex];const updateProperty = (key, value) => {coordinate[key] = Math.round(value / this.scale);};updateProperty("left", obj.left);updateProperty("top", obj.top);updateProperty("width", obj.getScaledWidth());updateProperty("height", obj.getScaledHeight());coordinate.angle = Math.round(obj.angle);coordinate.scaleX = obj.scaleX;coordinate.scaleY = obj.scaleY;// 特定类型处理if (coordinate.name === "批注") {coordinate.xopinion = coordinate.top;coordinate.yopinion = coordinate.left;coordinate.opinion = obj.text;coordinate.fontsize = obj.fontSize;coordinate.wopinion = coordinate.width;coordinate.hopinion = coordinate.height;coordinate.opscale = obj.scaleX;coordinate.fontsize = actualFontSize;obj.set("fontSize", actualFontSize); // 更新画布显示字号// 重置缩放比例避免双重缩放obj.set({scaleX: 1,scaleY: 1,baseFontSize: actualFontSize // 更新基准字号});} else {coordinate.xsign = coordinate.top;coordinate.ysign = coordinate.left;coordinate.wsign = coordinate.width;coordinate.hsign = coordinate.height;coordinate.scale = obj.scaleX;}canvas.requestRenderAll();this.getSignatureJson();});canvas.on("mouse:move", e => {const obj = e.target;if (!obj) return;// 边界处理obj.setCoords(); // 更新控制点坐标const canvasWidth = canvas.getWidth();const canvasHeight = canvas.getHeight();// 限制移动范围(考虑缩放)const boundingRect = obj.getBoundingRect();const scale = this.scale;// 计算有效移动范围const minX = 0 - (boundingRect.width * (1 - scale)) / 2;const minY = 0 - (boundingRect.height * (1 - scale)) / 2;const maxX = canvasWidth - (boundingRect.width * (1 + scale)) / 2;const maxY = canvasHeight - (boundingRect.height * (1 + scale)) / 2;// 应用位置限制obj.left = Math.min(Math.max(obj.left, minX), maxX);obj.top = Math.min(Math.max(obj.top, minY), maxY);// 更新坐标数据const findIndex = this.coordinateList.findIndex(item => item.cacheKey === obj.cacheKey);if (findIndex === -1) return;// 计算实际字体大小(核心修改部分)const baseFontSize = obj.baseFontSize || obj.fontSize; // 获取基准字号const scaleFactor = Math.max(obj.scaleX, obj.scaleY); // 取最大缩放值const actualFontSize = Math.round(baseFontSize * scaleFactor);// 通用属性更新const coordinate = this.coordinateList[findIndex];const updateProperty = (key, value) => {coordinate[key] = Math.round(value / this.scale);};updateProperty("left", obj.left);updateProperty("top", obj.top);updateProperty("width", obj.getScaledWidth());updateProperty("height", obj.getScaledHeight());coordinate.angle = Math.round(obj.angle);coordinate.scaleX = obj.scaleX;coordinate.scaleY = obj.scaleY;// 特定类型处理if (coordinate.name === "批注") {coordinate.xopinion = coordinate.top;coordinate.yopinion = coordinate.left;coordinate.opinion = obj.text;coordinate.fontsize = obj.fontSize;coordinate.wopinion = coordinate.width;coordinate.hopinion = coordinate.height;coordinate.opscale = obj.scaleX;coordinate.fontsize = actualFontSize;obj.set("fontSize", actualFontSize); // 更新画布显示字号// 重置缩放比例避免双重缩放obj.set({scaleX: 1,scaleY: 1,baseFontSize: actualFontSize // 更新基准字号});} else {coordinate.xsign = coordinate.top;coordinate.ysign = coordinate.left;coordinate.wsign = coordinate.width;coordinate.hsign = coordinate.height;coordinate.scale = obj.scaleX;}canvas.requestRenderAll();this.getSignatureJson();});});},onMove(evt) {// 记录当前拖拽位置this.currentDragPosition = {x: evt.draggedContext.futureIndex,y: evt.draggedContext.index};console.log(this.currentDragPosition,"----------currentDragPosition----");},// 拖拽结束end(e) {// 找到当前拖拽到哪一个canvas-layout上const currentCanvasLayout =e.originalEvent.target.parentElement.parentElement;const findIndex = this.canvasLayoutTopList.findIndex(item => item.obj == currentCanvasLayout);if (findIndex == -1) return false;// 取整// console.log("e", e, findIndex);const left =e.originalEvent.offsetX < 0? 0: Math.ceil(e.originalEvent.offsetX / this.scale);const top =e.originalEvent.offsetY < 0? 0: Math.ceil(e.originalEvent.offsetY / this.scale);// console.log('e', e, findIndex,left,top,this.scale,);this.addSeal({sealUrl: this.mainImagelist[e.newDraggableIndex].img,left,top,index: e.newDraggableIndex,pageNum: findIndex});},// 注意图片的,onload是异步的,如果要封装成工具函数,需要用promise包装一下// 引入项目中的图片// 添加公章addSeal({ sealUrl, left, top, index, pageNum }) {const hasDuplicate = this.coordinateList.some(item => item.sealUrl === sealUrl && item.pageNo === pageNum + 1);if (hasDuplicate) {return this.$message.error("每页只能添加一个电子签名");}// 生成唯一 cacheKeyconst cacheKey = `seal_${Date.now()}_${pageNum}_${index}`;// const deleteIcon = "@assets/icon/del.svg";var deleteImg = document.createElement("img");deleteImg.src = this.deleteIcon;var samllImg = document.createElement("img");samllImg.src = this.samllIcon;var largeImg = document.createElement("img");largeImg.src = this.largeIcon;fabric.Image.fromURL(sealUrl, oImg => {oImg.set({left: left,top: top,cacheKey: cacheKey,cornerColor: "#3B82F6",cornerStyle: "circle",borderScaleFactor: 2,transparentCorners: false,// 角度// angle: 10,// 缩放比例,需要乘以scalescaleX: 1 * this.scale,scaleY: 1 * this.scale,// 设置当点击了该控制点,鼠标弹起是执行的动作处理方法index// 禁止缩放// lockScalingX: true,// lockScalingY: true,// 禁止旋转// lockRotation: true,});oImg.setControlsVisibility({mtr: false,mt: false,ml: false,mb: false,mr: false});oImg.controls.mtControl = new fabric.Control({visible: true, // 控制角的显隐x: -0.5,y: -0.5,offsetY: -16,offsetX: 30,cursorStyle: "pointer",//removeSignature// mouseDownHandler: (eventData, transform) => createLineDown(eventData, transform),mouseUpHandler: (eventData, transform) =>this.largeObject(eventData, transform),render: function(ctx, left, top, styleOverride, fabricObject) {// 渲染一个粉红色的正方形var size = this.cornerSize;ctx.save();ctx.translate(left, top);ctx.rotate(fabric.util.degreesToRadians(fabricObject.angle));ctx.drawImage(largeImg, -size / 2, -size / 2, size, size);ctx.restore();},cornerSize: 26});// // 左上oImg.controls.tlControl = new fabric.Control({visible: true, // 控制角的显隐x: -0.5,y: -0.5,offsetY: -16,offsetX: 0,cursorStyle: "pointer",//removeSignature// mouseDownHandler: (eventData, transform) => createLineDown(eventData, transform),mouseUpHandler: (eventData, transform) =>this.smallObject(eventData, transform),render: function(ctx, left, top, styleOverride, fabricObject) {// 渲染一个粉红色的正方形var size = this.cornerSize;ctx.save();ctx.translate(left, top);ctx.rotate(fabric.util.degreesToRadians(fabricObject.angle));ctx.drawImage(samllImg, -size / 2, -size / 2, size, size);ctx.restore();},cornerSize: 26});// 左上 删除oImg.controls.trControl = new fabric.Control({visible: true, // 控制角的显隐x: -0.5,y: -0.5,offsetY: -16,offsetX: 60,cursorStyle: "pointer",//removeSignature// mouseDownHandler: (eventData, transform) => createLineDown(eventData, transform),mouseUpHandler: (eventData, transform) =>this.deleteObject(eventData, transform),render: function(ctx, left, top, styleOverride, fabricObject) {// 渲染一个粉红色的正方形var size = this.cornerSize;ctx.save();ctx.translate(left, top);ctx.rotate(fabric.util.degreesToRadians(fabricObject.angle));ctx.drawImage(deleteImg, -size / 2, -size / 2, size, size);ctx.restore();},cornerSize: 26});this.canvasEle[pageNum].add(oImg);// 保存签章信息this.saveSignature({ pageNum, index, sealUrl,cacheKey });});this.removeActive();},//控制器删除对象deleteObject(eventData, transform) {let target = transform.target;let canvas = target.canvas;// 拿到选中的文本的var activeObj = canvas.getActiveObject();const findIndex = this.coordinateList.findIndex(item =>item.cacheKey == activeObj.cacheKey && item.pageNo == activeObj.pageNo);// console.log(findIndex,this.coordinateList,activeObj,'-----------activeObjS')// 删除选中的文本canvas.remove(target); // 删除元素// 删除选中的文本的信息this.coordinateList.splice(findIndex, 1);this.getSignatureJson();},// 放大操作处理largeObject(eventData, transform) {this.handleScaling(transform, 0.1); // +10%缩放},// 缩小操作处理smallObject(eventData, transform) {this.handleScaling(transform, -0.1); // -10%缩放},// 统一缩放处理方法handleScaling(transform, scaleStep) {const target = transform.target;const canvas = target.canvas;const activeObj = canvas.getActiveObject();// 校验有效性if (!activeObj || !activeObj.cacheKey) {console.error("操作对象无效或缺少cacheKey");return;}console.log(this.coordinateList,'--------')// 精确查找坐标项const findIndex = this.coordinateList.findIndex(item => item.cacheKey === activeObj.cacheKey);if (findIndex === -1) {console.warn("未找到对应坐标项,cacheKey:", activeObj.cacheKey);return;}// 计算新缩放比例(带最小限制)const minScale = 0.2;const newScaleX = Math.max(minScale, activeObj.scaleX + scaleStep);const newScaleY = Math.max(minScale, activeObj.scaleY + scaleStep);// 更新对象属性activeObj.set({scaleX: newScaleX,scaleY: newScaleY}).setCoords();// 同步到数据存储this.coordinateList[findIndex] = {...this.coordinateList[findIndex],scaleX: newScaleX,scaleY: newScaleY,width: activeObj.width * newScaleX,height: activeObj.height * newScaleY};// 渲染优化(避免全局重绘)canvas.renderAll();console.log("缩放操作完成:", this.coordinateList[findIndex]);},// 保存签章saveSignature({ pageNum, index, sealUrl ,cacheKey}) {// 1. 拿到当前签章的信息let length = 0;let pageConfig = this.coordinateList.filter(item => item.pageNo - 1 == pageNum);if (pageConfig) length = pageConfig.length;const currentSignInfo = this.canvasEle[pageNum].getObjects()[length];// 2. 拼接数据const keys = ["width","height","top","left","angle","scaleX","scaleY","xsign",'cacheKey',"ysign"];const obj = {};keys.forEach(item => {obj[item] = Math.ceil(currentSignInfo[item] / this.scale);});obj.cacheKey = cacheKey;obj.sealUrl = sealUrl;obj.index = index;obj.xsign = obj.top;obj.ysign = obj.left;obj.scale = obj.scaleX;obj.wsign = obj.width;obj.hsign = obj.height;obj.name = "印章";obj.pageNo = pageNum + 1;// currentSignInfo.set("cacheKey", obj.cacheKey); // 关键:将 cacheKey 存入 Fabric 对象// currentSignInfo.set("name", obj.name); // 关键:将 cacheKey 存入 Fabric 对象this.coordinateList.push(obj);console.log(obj,'-------cacheKey')this.getSignatureJson();},// 签章生成json字符串getSignatureJson() {// 1. 判断是否有签章if (this.coordinateList.length <= 1) return (this.shadowInputValue = "");// 2. 拿到签章的信息,去除第一条const signatureList = this.coordinateList;// 3. 拼接数据,只要left和top和pageconst keys = ["pageNo","left","top","width","height","xsign","ysign","xopinion","text","yopinion"];const arr = [];signatureList.forEach(item => {const obj = {};keys.forEach(key => {obj[key] = item[key];});arr.push(obj);});// 4. 转成json字符串this.shadowInputValue = JSON.stringify(arr);},/*** 操作相关部分*/// 上一页prevPage() {if (this.pageNum <= 1) return;this.pageNum--;// 滚动到指定位置this.outViewDom.scrollTop = this.canvasLayoutTopList[this.pageNum - 1].top;},// 下一页nextPage() {if (this.pageNum >= this.numPages) return;this.pageNum++;// 滚动到指定位置this.outViewDom.scrollTop = this.canvasLayoutTopList[this.pageNum - 1].top;},// 切换页码cutover() {this.outViewScrollClose();if (this.pageNum < 1) {this.pageNum = 1;} else if (this.pageNum > this.numPages) {this.pageNum = this.numPages;}// 滚动到指定位置this.outViewDom.scrollTop = this.canvasLayoutTopList[this.pageNum - 1].top;setTimeout(() => {this.outViewScroll();}, 500);},// 删除所有的签章选中状态removeActive() {this.canvasEle.forEach(item => {item.discardActiveObject().renderAll();});},// 删除签章removeSignature() {// 1. 判断是否有选中的签章const findItem = this.canvasEle.filter(item => item.getActiveObject());// 2. 判断选中签章的个数if (findItem.length == 0)return this.$message.error("请选择要删除的签章");// 3. 判断选中签章的个数是否大于1if (findItem.length > 1) {this.removeActive();return this.$message.error("只能选择删除一个签章,请重新选择");}// 4. 拿到选中的签章的cacheKeyconst activeObj = findItem[0].getActiveObject();const findIndex = this.coordinateList.findIndex(item =>item.cacheKey == activeObj.cacheKey && item.pageNo == activeObj.pageNo);// 5. 删除选中的签章findItem[0].remove(activeObj);// 6. 删除选中的签章的信息this.coordinateList.splice(findIndex, 1);this.getSignatureJson();},// 清空签章clearSignature() {this.canvasEle.forEach(item => {item.clear();});this.coordinateList = [];this.getSignatureJson();},processSignsData(originalData) {// 按pageNo分组const grouped = originalData.reduce((acc, item) => {const pageNo = item.pageNo;if (!acc[pageNo]) acc[pageNo] = [];acc[pageNo].push(item);return acc;}, {});// 生成目标数据格式const signs = Object.keys(grouped).map(Number).sort((a, b) => a - b).map(pageNo => {const items = grouped[pageNo];const signItem = {xsign: "",ysign: "",wsign: "",hsign: "",xopinion: "",yopinion: "",wopinion: "",hopinion: "",scale: "1",opscale: "1",sealUrl: "",opinion: "",pageNo: "",fontsize: ""};items.forEach(item => {if (item.name === "印章") {signItem.xsign = String(item.xsign ?? "");signItem.ysign = String(item.ysign ?? "");signItem.wsign = String(item.wsign ?? "");signItem.hsign = String(item.hsign ?? "");signItem.scale = String(item.scaleX ?? "");signItem.sealUrl = String(item.sealUrl ?? "");} else if (item.name === "批注") {signItem.xopinion = String(item.xopinion ?? "");signItem.yopinion = String(item.yopinion ?? "");signItem.wopinion = String(item.wopinion ?? "");signItem.hopinion = String(item.hopinion ?? "");signItem.opscale = String(item.scaleX ?? "");signItem.opinion = String(item.opinion ?? "");signItem.fontsize = String(item.fontsize ?? "");}signItem.pageNo = String(item.pageNo ?? "");});return signItem;});return { signs };},//操作单据optionSignature(val) {this.id = this.$route.query.id;var includesign = this.coordinateList.some(item =>item.sealUrl?.includes("/chairmansoffice/download/"));console.log(this.coordinateList,"------------------------this.coordinateList ");if (val == "agree" ||val == "addSubSign") {if (includesign &&this.coordinateList != [] &&this.coordinateList.length > 0) {this.isVisible = true;this.signlist = this.processSignsData(this.coordinateList)["signs"];// console.log(// this.signlist,// "------------------------this.coordinateList "// );} else {this.$message.warning(`请先对文件进行签章后再提交`);}this.title = val;} else {this.title = "return";this.isVisible = true;}// console.log("this.coordinateList", this.coordinateList);},closeVisible(value) {this.isVisible = value;},//返回returnSignature() {this.$router.go(-1);},// 提交数据submitSignature() {console.log("this.coordinateList", this.coordinateList);},//添加文字addTextobjectHandle(e) {var deleteImg = document.createElement("img");deleteImg.src = this.deleteIcon;var samllImg = document.createElement("img");samllImg.src = this.samllIcon;var largeImg = document.createElement("img");largeImg.src = this.largeIcon;let Shape;let currentoptionCss;var findIndex = this.pageNum - 1;const cacheKey = `text_${Date.now()}_${findIndex}`;currentoptionCss = this.activeobjectData;//通过最大行高计算高度,并删除多余文字,多出文字..表示,三个会换行Shape = new fabric.IText(currentoptionCss.text || "", currentoptionCss);Shape.set({cacheKey: cacheKey // 为文本对象设置唯一标识});Shape.setControlVisible("mtr", false);Shape.setControlVisible("mt", false);Shape.setControlVisible("ml", false);Shape.setControlVisible("mb", false);Shape.setControlVisible("mr", false);Shape.controls.mtControl = new fabric.Control({visible: true, // 控制角的显隐x: -0.5,y: -0.5,offsetY: -16,offsetX: 30,cursorStyle: "pointer",//removeSignature// mouseDownHandler: (eventData, transform) => createLineDown(eventData, transform),mouseUpHandler: (eventData, transform) =>this.largeObject(eventData, transform),render: function(ctx, left, top, styleOverride, fabricObject) {var size = this.cornerSize;ctx.save();ctx.translate(left, top);ctx.rotate(fabric.util.degreesToRadians(fabricObject.angle));ctx.drawImage(largeImg, -size / 2, -size / 2, size, size);ctx.restore();},cornerSize: 26});Shape.controls.tlControl = new fabric.Control({visible: true, // 控制角的显隐x: -0.5,y: -0.5,offsetY: -16,offsetX: 0,cursorStyle: "pointer",//removeSignature// mouseDownHandler: (eventData, transform) => createLineDown(eventData, transform),mouseUpHandler: (eventData, transform) =>this.smallObject(eventData, transform),render: function(ctx, left, top, styleOverride, fabricObject) {var size = this.cornerSize;ctx.save();ctx.translate(left, top);ctx.rotate(fabric.util.degreesToRadians(fabricObject.angle));ctx.drawImage(samllImg, -size / 2, -size / 2, size, size);ctx.restore();},cornerSize: 26});// 左上 删除Shape.controls.trControl = new fabric.Control({visible: true, // 控制角的显隐x: -0.5,y: -0.5,offsetY: -16,offsetX: 60,cursorStyle: "pointer",//removeSignature// mouseDownHandler: (eventData, transform) => createLineDown(eventData, transform),mouseUpHandler: (eventData, transform) =>this.deleteObject(eventData, transform),render: function(ctx, left, top, styleOverride, fabricObject) {// 渲染一个粉红色的正方形var size = this.cornerSize;ctx.save();ctx.translate(left, top);ctx.rotate(fabric.util.degreesToRadians(fabricObject.angle));ctx.drawImage(deleteImg, -size / 2, -size / 2, size, size);ctx.restore();},cornerSize: 26});Shape.splitByGrapheme = true;var index = 0,index = index + 1;var pageNum = findIndex;var sealUrl = this.activeobjectData.text;// 获取当前页面信息const currentPage = this.pageNum;const canvasIndex = currentPage - 1;// 验证当前页是否已有批注const hasExistingAnnotation = this.coordinateList.some(item => item.pageNo === currentPage && item.name === "批注");if (hasExistingAnnotation) {this.$message.error("每页只能添加一个文字批注");return;}this.canvasEle[findIndex].add(Shape).setActiveObject(Shape);this.saveText({ pageNum, index, sealUrl,cacheKey });},saveText({ pageNum, index, sealUrl,cacheKey }) {// 1. 拿到当前批注的信息let length = 0;let pageConfig = this.coordinateList.filter(item => item.pageNo - 1 == pageNum);if (pageConfig) length = pageConfig.length;const currentSignInfo = this.canvasEle[pageNum].getObjects()[length];const keys = ["width","height","cacheKey","top","left","angle","scaleX","scaleY"];const obj = {};keys.forEach(item => {obj[item] = Math.ceil(currentSignInfo[item] / this.scale);});obj.cacheKey = cacheKey;obj.opinion = obj.text;obj.fontsize = obj.fontsize;obj.xopinion = obj.top;obj.yopinion = obj.left;obj.wopinion = obj.width;obj.hopinion = obj.height;obj.opscale = obj.scaleX;obj.index = index;obj.name = "批注";obj.pageNo = pageNum + 1;// currentSignInfo.set("cacheKey", obj.cacheKey); // 关键:将 cacheKey 存入 Fabric 对象// currentSignInfo.set("name", obj.name); // 关键:将 cacheKey 存入 Fabric 对象this.coordinateList.push(obj);},//下载文件download() {// downloadApi(this.id)// .then(res => {// const fileNameEncode = res.headers["content-disposition"]// .split(";")[1]// .split("filename=")[1];// let contentDisposition = decodeURIComponent(fileNameEncode);// this.downloadBinaryFile(res.data, contentDisposition);// this.loading = false;// })// .catch();},async downloadBinaryFile(binFile,fileName,blobType = "application/octet-stream") {// 处理二进制数据并创建 Blob 对象const blobObj = new Blob([binFile], { type: blobType });// 创建一个链接并设置下载属性const downloadLink = document.createElement("a");let url = window.URL || window.webkitURL || window.moxURL; // 兼容不同浏览器的 URL 对象url = url.createObjectURL(blobObj);downloadLink.href = url;downloadLink.download = fileName; // 设置下载的文件名// 将链接添加到 DOM 中,模拟点击document.body.appendChild(downloadLink);downloadLink.click();// 移除创建的链接和释放 URL 对象document.body.removeChild(downloadLink);window.URL.revokeObjectURL(url);},customRequest(param) {this.uploadAction(param.file);},//上传文件async uploadAction(file) {this.dayLoading = true;const param = new FormData();param.append("file", file);uploadSignture(param).then(res => {if (res.data.type == "application/json") {//转换步骤const file = new FileReader();file.readAsText(res.data, "utf-8");file.onload = e => {const obj = JSON.parse(file.result);res.data = obj;if (obj.code == "100") {this.$message({ message: "上传成功", type: "success" });this.userInfo.signture = obj.data.fn;// this.form.fn = obj.data[0].fn;this.getsignimg();// this.form.sourceName = obj.data[0].sourceName;// console.log(this.userInfo.signture);} else {this.$message({ message: obj.message, type: "error" });}};} else {this.$message.warning(`上传失败`);}}).catch(err => {this.$message.error(res.$message);});// this.dialogVisible = false;// this.$emit("dialogVisible", this.dialogVisible);},// 超出上传文件个数的钩子handleExceed(file, fileList) {this.$message.warning(`只能上传一个文件`);},beforeRemove(file, fileList) {return this.$confirm(`确定移除 ${file.name}?`);},handleRemove(file, fileList) {console.log(file, fileList);},handlePreview(file) {console.log(file);},//删除绑定个人的签章文件deleteSignature() {removeSignture().then(res => {if (res.data.code == 100) {this.$message({ message: res.data.message, type: "success" });this.getsignimg();this.vissign = false;} else {this.$message({ message: res.data.message, type: "error" });}});}// imgsrc(val){// let token = getToken()// Object.defineProperty(Image.prototype, 'authsrc', {// writable: true,// enumerable: true,// configurable: true// })// let img = val// let request = new XMLHttpRequest();// request.responseType = 'blob';// request.open('get', this.authSrc, true);// request.setRequestHeader('token', token);// request.onreadystatechange = e => {// if (request.readyState == XMLHttpRequest.DONE && request.status == 200) {// img.src = URL.createObjectURL(request.response);// img.onload = () => {// URL.revokeObjectURL(img.src);// }// }// };// request.send(null);// },}
};
</script>
<style lang="scss" scoped>
.contract-signature-view {/*pdf部分*/.ele-canvas {overflow: hidden;}.text-canvas {overflow: hidden;}.title-operation {background: rgba(255, 255, 255, 1);height: 80px;padding: 20px 40px;display: flex;align-items: center;justify-content: space-between;.operation {.searchbutton {margin-right: 10px;min-width: 96px;height: 40px;border: 1px 0px 0px 0px;gap: 8px;border-radius: 4px;padding-top: 8px;padding-right: 20px;padding-bottom: 8px;padding-left: 18px;font-size: 12px;}.button {margin-right: 10px;min-width: 96px;height: 40px;border: none;gap: 8px;border-radius: 4px;padding-top: 8px;padding-right: 20px;padding-bottom: 8px;padding-left: 18px;font-size: 12px;}}.title {font-size: 20px;font-weight: 600;}border-bottom: 1px solid #e4e4e4;}.section-box {position: relative;display: flex;height: calc(100vh - 60px);.main-layout {// flex: 1;width: 100%;background-color: #f7f8fa;position: relative;&.is-first {.operate-box {opacity: 0;}}.operate-box {opacity: 1;position: absolute;top: 80px;left: 0;width: 100%;height: 40px;background-color: #fff;border-bottom: 1px solid #e4e4e4;display: flex;justify-content: center;align-items: center;.slider-box {width: 230px;display: flex;justify-content: center;align-items: center;border-left: 1px solid #e4e4e4;border-right: 1px solid #e4e4e4;.slider {width: 120px;}.scale-value {margin-left: 24px;font-size: 16px;color: #000000;line-height: 22px;}}.pageNo-change {display: flex;align-items: center;margin-left: 30px;.icon {cursor: pointer;padding: 0 5px;color: #c1c1c1;}.input-box {border: none;.el-input__inner {width: 34px;height: 20px;border: none;padding: 0;text-align: center;border-bottom: 1px solid #e4e4e4;}}.default-text {display: flex;line-height: 22px;margin-right: 5px;}}}.out-view {height: calc(100vh - 185px);margin: 40px auto;overflow-x: auto;overflow-y: auto;padding-top: 20px;text-align: center;opacity: 0;transition: all 0.5s;&.is-show {opacity: 1;}.canvas-layout {position: relative;text-align: center;margin: 0 auto 18px;#box {position: relative;}.menu-x {visibility: hidden;z-index: -100;position: absolute;top: 0;left: 0;box-sizing: border-box;border-radius: 4px;box-shadow: 0 0 4px rgba(0, 0, 0, 0.3);background-color: #fff;}.menu-li {box-sizing: border-box;padding: 4px 8px;border-bottom: 1px solid #ccc;cursor: pointer;}.menu-li:hover {background-color: antiquewhite;}.menu-li:first-child {border-top-left-radius: 4px;border-top-right-radius: 4px;}.menu-li:last-child {border-bottom: none;border-bottom-left-radius: 4px;border-bottom-right-radius: 4px;}}}.loading {width: 20px;height: 20px;position: absolute;top: 50%;left: 50%;transform: translate(-50%, -50%);z-index: 999;.el-loading-mask {background-color: transparent;}}}.position-info {width: 355px;min-width: 355px;border-left: 1px solid #e4e4e4;background-color: #fff;padding: 14px 15px;.title {font-size: 14px;font-weight: 400;color: #000000;line-height: 20px;padding-bottom: 18px;}.nav {display: flex;flex-direction: column;.item {display: flex;justify-content: space-between;padding: 10px 0;border-bottom: 1px solid #eee;&:first-child {background-color: #f7f8fa;}span {flex: 1;text-align: center;font-size: 12px;color: #000000;line-height: 20px;}}}}}
}.signatureimg {min-width: 240px !important;background-color: #fff;//padding-bottom: 20px;.name {font-size: 18px;font-weight: 600;color: #000000;line-height: 25px;margin-bottom: 20px;}.text {font-size: 14px;color: #000000;line-height: 20px;}.item {// margin: 10p;// padding: 10px;border: 1px dashed rgba(0, 0, 0, 0.3);&:not(:last-child) {margin-bottom: 10px;}.img {vertical-align: middle;background-repeat: no-repeat;}}.opera {bottom: 0;position: absolute;justify-content: space-between;background: rgba(243, 243, 243, 1);left: 0;width: 100%;display: flex;align-items: center;.delbut {cursor: pointer;background: rgba(243, 243, 243, 1);border: none;color: #000000;}:hover {color: #409eff;border: #c6e2ff;// background-color: #ecf5ff;}}
}.upimg {.upload-demo {padding: 10px 20px;text-align: center;.upload-button {padding: 30px 0 0 0;color: #3f7afa;font-size: 14px;border: none;&:hover,&.is-active {background: transparent;}}}
}
</style>