You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

132 lines
4.6 KiB

<template>
<div id="signature-pad" class="signature-pad">
<div class="signature-pad--body" style="min-height: 300px; width: 100%;">
<canvas v-show="!signImageUrl" class="canvasId" style="border: 2px dashed #f7f7f7;" />
<img v-if="signImageUrl" :src="signImageUrl">
</div>
<p class="desc-text">请在上面区域完成签名 然后点击确"确认"按钮</p>
<div v-if="!signImageUrl">
<el-button plain size="mini" type="danger" @click="clear">清除</el-button>
<el-button plain size="mini" type="warning" @click="undo">回撤</el-button>
<el-button size="mini" type="success" @click="savePng">确认</el-button>
</div>
<div v-if="signImageUrl">
<el-button size="mini" type="primary" @click="reSign">重签</el-button>
</div>
</div>
</template>
<script>
import SignaturePad from 'signature_pad'
export default {
name: 'SignPad',
components: {},
props: {
value: {
type: String,
default: ''
}
},
data() {
return {
signaturePad: null,
canvas: null,
signImageUrl: '',
config: {
minWidth: 1,
maxWidth: 3,
penColor: 'rgb(66, 133, 244)'
}
}
},
mounted() {
this.init()
window.onresize = () => {
return (() => {
this.resizeCanvas()
})()
}
this.resizeCanvas()
},
methods: {
resizeCanvas() {
console.log('resizeCanvas')
let canvas = this.canvas
// When zoomed out to less than 100%, for some very strange reason,
// some browsers report devicePixelRatio as less than 1
// and only part of the canvas is cleared then.
let ratio = Math.max(window.devicePixelRatio || 1, 1)
// This part causes the canvas to be cleared
canvas.width = canvas.offsetWidth * ratio
canvas.height = canvas.offsetHeight * ratio
canvas.getContext('2d').scale(ratio, ratio)
// This library does not listen for canvas changes, so after the canvas is automatically
// cleared by the browser, SignaturePad#isEmpty might still return false, even though the
// canvas looks empty, because the internal data of this library wasn't cleared. To make sure
// that the state of this library is consistent with visual state of the canvas, you
// have to clear it manually.
this.signaturePad.clear()
},
init() {
this.canvas = document.querySelector('.canvasId')
let canvas = this.canvas
this.signaturePad = new SignaturePad(canvas, this.config)
// canvas.height = 300
// canvas.width = 500
},
clear() {
this.signaturePad.clear()
},
undo() {
let data = this.signaturePad.toData()
if (data) {
data.pop() // remove the last dot or line
this.signaturePad.fromData(data)
}
},
savePng() {
if (this.signaturePad.isEmpty()) {
this.msgError('请先填写')
return
}
let data = this.signaturePad.toDataURL('image/png')
let file = this.dataURLtoFile(data, `${new Date().getTime()}.png`)
let param = new FormData() // 创建form对象
param.append('file', file) // 通过append向form对象添加数据
let config = {
headers: {'Content-Type': 'multipart/form-data'}
}
this.$api.post('/project/file/upload/77f1648542af4caf98deb8345a3d0406', param, config).then(res => {
this.signImageUrl = res.data
this.$emit('input', res.data)
})
},
reSign() {
this.signImageUrl = ''
this.signaturePad.clear()
},
dataURLtoFile(dataurl, filename) {
// 获取到base64编码
const arr = dataurl.split(',')
// 将base64编码转为字符串
const bstr = window.atob(arr[1])
let n = bstr.length
const u8arr = new Uint8Array(n) // 创建初始化为0的,包含length个元素的无符号整型数组
while (n--) {
u8arr[n] = bstr.charCodeAt(n)
}
return new File([u8arr], filename, {
type: 'image/png'
})
}
}
}
</script>
<style scoped>
@import "index.css";
.sign-pad-container {
padding: 1px;
}
</style>