合同签章及批注模块涉及以下核心难点和复杂点,按功能模块归类如下:
效果展示
使用拖拽的方式将签名放入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-button
v-if="$route.query.formst==31"
class="searchbutton"
type="primary"
@click="optionSignature('addSubSign')"
>
<svg-icon icon-class="check" />呈签</el-button
>
<el-button
v-else
class="searchbutton"
type="primary"
@click="optionSignature('agree')"
>
<svg-icon icon-class="check" />同意</el-button
>
<!-- <el-button
class="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">
<!-- 拖拽 -->
<draggable
v-model="mainImagelist"
:group="{ name: 'itext', pull: 'clone' }"
:sort="false"
@end="end"
@move="onMove"
>
<transition-group type="transition">
<li
v-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-button
class="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-upload
class="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>
<div
slot="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-input
class="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-view
outViewDom: null,
// 各页pdf的canvas-layout
canvasLayoutTopList: [],
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.png
if (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"));
});
// 循环渲染pdf
for (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宽高,缩放比例,渲染pdf
renderPage(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 context
const 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-layout
const 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. 拿到签名canvas
const canvasEle = item.querySelector(".ele-canvas");
// 5. 拿到pdf的canvas
const 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滑块,重新渲染整个pdf
sliderChange() {
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"));
});
// 循环渲染pdf
for (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("每页只能添加一个电子签名");
}
// 生成唯一 cacheKey
const 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,
// 缩放比例,需要乘以scale
scaleX: 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和page
const 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. 判断选中签章的个数是否大于1
if (findItem.length > 1) {
this.removeActive();
return this.$message.error("只能选择删除一个签章,请重新选择");
}
// 4. 拿到选中的签章的cacheKey
const 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>