| <template> | |
| 	<view class="page"> | |
| 		<navbar title="编辑模板" leftClick @leftClick="$utils.navigateBack" /> | |
| 		 | |
| 		<view class="title"> | |
| 			<input type="text" placeholder="请输入标题" v-model="form.title"/> | |
| 		</view> | |
|  | |
| 		<view style="padding: 25rpx;background-color: #fff;"> | |
| 			<!-- 没有模板时的提示 --> | |
| 			<view v-if="!form.template || imageArr.length === 0" class="no-template-tip"> | |
| 				<text>请先点击"导入合同"按钮上传PDF合同模板</text> | |
| 			</view> | |
| 			 | |
| 			<canvas-drag ref="canvasRef" | |
| 			id="canvas-drag" :graph="graph"  | |
| 			:width="width" :height="height" | |
| 			@onDrawArrChange="onDrawArrChange" | |
| 			enableUndo="true"></canvas-drag> | |
| 		</view> | |
|  | |
| 		<view class="btn-list" v-if="imageArr.length > 0"> | |
| 			<view class="uni-color-btn" @tap="pro">上一页</view> | |
| 			<view>{{ imagesIndex + 1 }}/{{ imageArr.length }}</view> | |
| 			<view class="uni-color-btn" @tap="next">下一页</view> | |
| 		</view> | |
| 		<view class="btn-list"> | |
| 			<!-- <view class="uni-color-btn" @tap="onAddImage">新建签名</view> --> | |
| 			<view class="uni-color-btn" @tap="onChangeBgImage">导入合同</view> | |
| 			<view class="uni-color-btn" @tap="onUndo">后退</view> | |
| 			<!-- <view class="uni-color-btn" @tap="onClearCanvas">清空</view> --> | |
| 			<view class="uni-color-btn" @tap="submit">保存</view> | |
| 			<!-- <view class="uni-color-btn" @tap="onExportJSON">导出模板</view> --> | |
| 			<!-- <view class="uni-color-btn" @tap="onAddText">添加文字</view> --> | |
| 			<view class="uni-color-btn" @tap="previewContract" v-if="form.template">预览合同</view> | |
| 			 | |
| 		</view> | |
| 	</view> | |
| </template> | |
|  | |
| <script> | |
| 	import canvasDrag from "../components/canvas-drag/index"; | |
| 	import setData from "../components/canvas-drag/setData.js"; | |
| 	import wxBase64src from '@/utils/wxBase64src.js' | |
| 	export default { | |
| 		components: { | |
| 			canvasDrag | |
| 		}, | |
| 		mixins : [setData], | |
| 		data() { | |
| 			return { | |
| 				height: 750, | |
| 				width: 700, | |
| 				drawArr : {}, | |
| 				form : { | |
| 					template : '', | |
| 					employeePage : 0,//求职者(乙方)签字页码 | |
| 					employeePosition : '',//求职者(乙方)签字坐标 | |
| 					bossPage : 0,//招聘者(甲方)签字页码 | |
| 					bossPosition : '',//招聘者(甲方)签字坐标 | |
| 					title : '', | |
| 				}, | |
| 				imagesIndex : 0, | |
| 				imageArr : [], | |
| 				id : 0, | |
| 			} | |
| 		}, | |
| 		onLoad({id}) { | |
| 			// uni.request({ | |
| 			// 	url : 'http://192.168.1.5:5173/arr', | |
| 			// 	method: 'GET', | |
| 			// 	success : res => { | |
| 			// 		this.imageArr = res.data | |
| 			// 		this.onChangeBgImage() | |
| 			// 	} | |
| 			// }) | |
| 			 | |
| 			this.id = id | |
| 			this.getDetail() | |
| 		}, | |
| 		mounted() { | |
| 			console.log("mounted mounted mounted"); | |
| 		}, | |
| 		methods: { | |
| 			// async pro(){ | |
| 			// 	if(this.imagesIndex == 0){ | |
| 			// 		uni.showToast({ | |
| 			// 			title: '已经是第一页了' | |
| 			// 		}) | |
| 			// 		return | |
| 			// 	} | |
| 			// 	this.imagesIndex-- | |
| 			// 	await wxBase64src.unlink() | |
| 			// 	this.changeBgImage() | |
| 			// }, | |
| 			// async next(){ | |
| 			// 	if(this.imagesIndex == this.imageArr.length - 1){ | |
| 			// 		uni.showToast({ | |
| 			// 			title: '已经是最后一页了' | |
| 			// 		}) | |
| 			// 		return | |
| 			// 	} | |
| 			// 	this.imagesIndex++ | |
| 			// 	await wxBase64src.unlink() | |
| 			// 	this.changeBgImage() | |
| 			// }, | |
| 			async pro(){ | |
| 				if(this.imagesIndex == 0){ | |
| 					uni.showToast({ | |
| 						title: '已经是第一页了' | |
| 					}) | |
| 					return | |
| 				} | |
| 				this.imagesIndex-- | |
| 				await wxBase64src.unlink() | |
| 				// 强制重新渲染canvas | |
| 				this.$nextTick(async () => { | |
| 					await this.changeBgImage() | |
| 				}) | |
| 			}, | |
| 			async next(){ | |
| 				if(this.imagesIndex == this.imageArr.length - 1){ | |
| 					uni.showToast({ | |
| 						title: '已经是最后一页了' | |
| 					}) | |
| 					return | |
| 				} | |
| 				this.imagesIndex++ | |
| 				await wxBase64src.unlink() | |
| 				// 强制重新渲染canvas | |
| 				this.$nextTick(async () => { | |
| 					await this.changeBgImage() | |
| 				}) | |
| 			}, | |
| 			// 预览合同 | |
| 			previewContract(){ | |
| 				uni.downloadFile({ | |
| 					url : this.form.template, | |
| 					success : res => { | |
| 						uni.openDocument({ | |
| 							filePath: res.tempFilePath, | |
| 						}) | |
| 					} | |
| 				}) | |
| 			}, | |
| 			// 签名位置更新时 | |
| 			onDrawArrChange(arr){ | |
| 				console.log(arr); | |
| 				this.drawArr = arr | |
| 			}, | |
| 			/** | |
| 			 * 添加签名位置 | |
| 			 */ | |
| 			onAddImage() { | |
| 				// wx.chooseImage({ | |
| 				// 	success: res => { | |
| 				// 		this.setData({ | |
| 				// 			graph: { | |
| 				// 				w: 100, | |
| 				// 				h: 50, | |
| 				// 				type: 'image', | |
| 				// 				url: res.tempFilePaths[0] | |
| 				// 			} | |
| 				// 		}); | |
| 				// 	} | |
| 				// }); | |
| 				 | |
| 				 | |
| 				let a = {} | |
| 				let b = {} | |
| 				 | |
| 				if(this.id){ | |
| 					a = JSON.parse(this.form.bossPosition) | |
| 					b = JSON.parse(this.form.employeePosition) | |
| 					 | |
| 					for(let ak in a){ | |
| 						a[ak] = this.$utils.screenSystemInfoInt(a[ak]) - 0 | |
| 					} | |
| 					for(let bk in b){ | |
| 						b[bk] = this.$utils.screenSystemInfoInt(b[bk]) - 0 | |
| 					} | |
| 					 | |
| 					console.log(a, b); | |
| 				} | |
| 				 | |
| 				uni.getImageInfo({ | |
| 					src: this.configList.config_sign_a, | |
| 					// src: 'https://img.teyizhao.com/2025-02-07/3d837c0e-91f4-4682-82d9-a04c2af737e8.png', | |
| 					success : res => { | |
| 						this.setData({ | |
| 							graph: { | |
| 								id : 'a', | |
| 								w: a.w || 100, | |
| 								h: a.h || 50, | |
| 								x: a.x || 30, | |
| 								y: a.y || 30, | |
| 								type: 'image', | |
| 								url : res.path | |
| 							} | |
| 						}); | |
| 					}, | |
| 					fail(e) { | |
| 						console.log(e); | |
| 					} | |
| 				}) | |
| 				 | |
| 				uni.getImageInfo({ | |
| 					src: this.configList.config_sign_b, | |
| 					// src: 'https://img.teyizhao.com/2025-02-07/af4786c0-7b62-4a86-a090-9f28eca3674a.png', | |
| 					success : res => { | |
| 						this.setData({ | |
| 							graph: { | |
| 								id : 'b', | |
| 								w: b.w || 100, | |
| 								h: b.h || 50, | |
| 								x: b.x || 160, | |
| 								y: b.y || 30, | |
| 								type: 'image', | |
| 								url : res.path | |
| 							} | |
| 						}); | |
| 					}, | |
| 					fail(e) { | |
| 						console.log(e); | |
| 					} | |
| 				}) | |
| 				 | |
| 			}, | |
|  | |
| 			/** | |
| 			 * 导入合同 | |
| 			 */ | |
| 			async onChangeBgImage() { | |
| 				let self = this | |
| 				 | |
| 				let pdfUrl = await self.loadFDP() | |
| 				 | |
| 				this.form.template = pdfUrl | |
| 				 | |
| 				await this.pdf2imagebase64() | |
| 				 | |
| 				this.changeBgImage() | |
| 				 | |
| 			}, | |
| 			// async changeBgImage(){ | |
| 			// 	let self = this | |
| 				 | |
| 			// 	let src = await wxBase64src.base64src(this.imageArr[this.imagesIndex]) | |
| 				 | |
| 			// 	uni.getImageInfo({ | |
| 			// 		src, | |
| 			// 		success: image => { | |
| 			// 			console.log(image); | |
| 				 | |
| 			// 			let allwh = image.height + image.width | |
| 			// 			let imgw = (image.width / allwh).toFixed(2) | |
| 			// 			let imgh = (image.height / allwh).toFixed(2) | |
| 				 | |
| 			// 			self.height = imgh * Math.ceil(this.width / imgw) | |
| 				 | |
| 			// 			self.$nextTick(() => { | |
| 			// 				uni.hideLoading() | |
| 			// 				this.$refs.canvasRef.changeBgImage(image.path) | |
| 			// 			}) | |
| 			// 		}, | |
| 			// 		fail(err) { | |
| 			// 			wxBase64src.unlink() | |
| 			// 			console.log(err); | |
| 			// 		} | |
| 			// 	}) | |
| 			// }, | |
| 			async changeBgImage(){ | |
|                 let self = this | |
|                  | |
|                 try { | |
|                     uni.showLoading({ | |
|                         title: '加载中...' | |
|                     }) | |
|                      | |
|                     let src = await wxBase64src.base64src(this.imageArr[this.imagesIndex]) | |
|                      | |
|                     const imageInfo = await new Promise((resolve, reject) => { | |
|                         uni.getImageInfo({ | |
|                             src, | |
|                             success: resolve, | |
|                             fail: reject | |
|                         }) | |
|                     }) | |
|                      | |
|                     let allwh = imageInfo.height + imageInfo.width | |
|                     let imgw = (imageInfo.width / allwh).toFixed(2) | |
|                     let imgh = (imageInfo.height / allwh).toFixed(2) | |
|                      | |
|                     self.height = imgh * Math.ceil(this.width / imgw) | |
|                      | |
|                     // 确保DOM更新后再更新canvas | |
|                     await this.$nextTick() | |
|                     // 先清空画布再设置新图片 | |
|                     // await this.$refs.canvasRef.clearCanvas() | |
|                     await this.$refs.canvasRef.changeBgImage(imageInfo.path) | |
|                      | |
|                     // 重新添加签名位置 | |
|                     // this.onAddImage() | |
|                      | |
|                 } catch(err) { | |
|                     console.error('changeBgImage error:', err) | |
|                     wxBase64src.unlink() | |
|                 } finally { | |
|                     uni.hideLoading() | |
|                 } | |
|             }, | |
| 			loadFDP(){ | |
| 				let that = this | |
| 				return new Promise((success, error) => { | |
| 					uni.chooseMessageFile({ | |
| 						count: 1, //一次可以上传多少个 | |
| 						type: 'file', | |
| 						// 移除extension限制,改为后续检查文件类型 | |
| 						success(res) { | |
| 							const file = res.tempFiles[0] | |
| 							const fileName = file.name || '' | |
| 							const fileExtension = fileName.toLowerCase().split('.').pop() | |
| 							 | |
| 							// 检查文件类型 | |
| 							if (fileExtension !== 'pdf') { | |
| 								uni.showToast({ | |
| 									title: '只能上传PDF文件,请重新选择', | |
| 									icon: 'none', | |
| 									duration: 3000 | |
| 								}) | |
| 								error(new Error('文件类型不正确')) | |
| 								return | |
| 							} | |
| 							 | |
| 							// 文件类型正确,继续上传 | |
| 							success(that.$Oss.ossUpload(file.path)) | |
| 						}, | |
| 						fail: (err) => { | |
| 							error(err) | |
| 							console.log(err, 'err'); | |
| 						} | |
| 					}); | |
| 				}) | |
| 			}, | |
| 			// 后退 | |
| 			onUndo(event) { | |
| 				this.$refs.canvasRef.undo(); | |
| 			}, | |
| 			/** | |
| 			 * 导出当前画布为模板 | |
| 			 */ | |
| 			onExportJSON() { | |
| 				this.$refs.canvasRef.exportFun().then(imgArr => { | |
| 					console.log(imgArr); | |
| 					uni.previewImage({ | |
| 						urls: [imgArr], | |
| 						current: 0, | |
| 					}) | |
| 				}).catch(e => { | |
| 					console.error(e); | |
| 				}); | |
| 			}, | |
| 			/** | |
| 			 * 添加文本 | |
| 			 */ | |
| 			onAddText() { | |
| 				this.setData({ | |
| 					graph: { | |
| 						type: 'text', | |
| 						text: 'helloworld' | |
| 					} | |
| 				}); | |
| 			}, | |
|  | |
| 			onClearCanvas(event) { | |
| 				let _this = this; | |
| 				_this.setData({ | |
| 					canvasBg: null | |
| 				}); | |
| 				this.$refs.canvasRef.clearCanvas(); | |
| 			}, | |
| 			async submit() { | |
| 				 | |
| 				if(this.$utils.verificationAll(this.form, { | |
| 					title : '请输入标题', | |
| 				})){ | |
| 					return | |
| 				} | |
| 				 | |
| 				let bossPosition = { | |
| 					// ...this.drawArr.a, | |
| 					x : this.$utils.rpxSystemInfoInt(this.drawArr.a.x), | |
| 					y : this.$utils.rpxSystemInfoInt(this.drawArr.a.y), | |
| 					w : this.$utils.rpxSystemInfoInt(this.drawArr.a.w), | |
| 					h : this.$utils.rpxSystemInfoInt(this.drawArr.a.h), | |
| 				} | |
| 				let employeePosition = { | |
| 					// ...this.drawArr.b, | |
| 					x : this.$utils.rpxSystemInfoInt(this.drawArr.b.x), | |
| 					y : this.$utils.rpxSystemInfoInt(this.drawArr.b.y), | |
| 					w : this.$utils.rpxSystemInfoInt(this.drawArr.b.w), | |
| 					h : this.$utils.rpxSystemInfoInt(this.drawArr.b.h), | |
| 				} | |
| 				 | |
| 				this.form.bossPosition = JSON.stringify(bossPosition) | |
| 				this.form.employeePosition = JSON.stringify(employeePosition) | |
| 				 | |
| 				this.form.bossPage = this.imagesIndex | |
| 				this.form.employeePage = this.imagesIndex | |
| 				 | |
| 				this.$api(this.id ? 'updateContractTemplate' :  | |
| 				'addContractTemplate', this.form).then(res => { | |
| 					if(res.code == 200){ | |
| 						uni.navigateBack(-1) | |
| 					} | |
| 				}) | |
| 			}, | |
| 			// pdf转图片base64 | |
| 			async pdf2imagebase64(){ | |
| 				uni.showLoading({ | |
| 					title: '解析文档中...' | |
| 				}) | |
| 				let res = await this.$api('pdf2imagebase64', { | |
| 					pdfPath :  this.form.template | |
| 				}) | |
| 				this.imageArr = [] | |
| 				for(let key in res.result){ | |
| 					this.imageArr.push(res.result[key]) | |
| 				} | |
| 			}, | |
| 			getDetail(){ | |
| 				if(!this.id){ | |
| 					this.onAddImage() | |
| 					return | |
| 				} | |
| 				this.$api('queryContractTemplateById', { | |
| 					contractTemplateId :  this.id | |
| 				}).then(async (res) => { | |
| 					if(res.code == 200){ | |
| 						this.form = res.result | |
| 						 | |
| 						this.imagesIndex = this.form.employeePage | |
| 						 | |
| 						delete this.form.createBy | |
| 						delete this.form.createTime | |
| 						delete this.form.updateBy | |
| 						delete this.form.updateTime | |
| 						 | |
| 						this.onAddImage() | |
| 						 | |
| 						await this.pdf2imagebase64() | |
| 						 | |
| 						this.changeBgImage() | |
| 					} | |
| 				}) | |
| 			}, | |
| 		} | |
| 	} | |
| </script> | |
|  | |
| <style scoped lang="scss"> | |
| 	.page { | |
| 		padding-bottom: 100rpx; | |
| 		.title{ | |
| 			padding: 10rpx 30rpx; | |
| 			background-color: #fff; | |
| 			margin-bottom: 10rpx; | |
| 			input{ | |
| 				padding: 20rpx; | |
| 				background-color: rgba($uni-color, 0.1); | |
| 				color: $uni-color; | |
| 				border: 1rpx solid $uni-color; | |
| 				border-radius: 10rpx; | |
| 			} | |
| 		} | |
| 		.no-template-tip { | |
| 			display: flex; | |
| 			align-items: center; | |
| 			justify-content: center; | |
| 			height: 300rpx; | |
| 			background-color: #f8f9fa; | |
| 			border: 2rpx dashed #ddd; | |
| 			border-radius: 10rpx; | |
| 			margin-bottom: 20rpx; | |
| 			text { | |
| 				color: #999; | |
| 				font-size: 28rpx; | |
| 				text-align: center; | |
| 			} | |
| 		} | |
| 		.btn-list { | |
| 			display: flex; | |
| 			flex-wrap: wrap; | |
| 			align-items: center; | |
| 			padding: 20rpx; | |
| 			gap: 30rpx; | |
| 			justify-content: center; | |
| 			.uni-color-btn { | |
| 				margin: 0; | |
| 				border-radius: 10rpx; | |
| 			} | |
| 		} | |
| 	} | |
| </style> |