Canvas
Canvas签名
提供原生js版本和vue版本
1、原生JavaScript版本
利用canvas画布实现签名效果
/**
* Canvas 签名
*
* 使用:
* new drawCanvas({
* saveCallback: function (imgBase64, clearCallback) {
* document.getElementById('signImage').src = imgBase64
* // 清空canvas
* // clearCallback()
* }
* })
* <String> el:canvas容器id,默认:'canvas'
* <String> saveEl:保存按钮id,默认:'saveBtn'
* <String> clearEl:清除按钮id,默认:'clearBtn'
* <Number> width:容器宽度,默认:浏览器宽度
* <Number> height:容器高度,默认:200
* <String> color:画线颜色,默认:'#000'
* <String> backgroundColor:容器背景颜色颜色,默认:'rgba(255, 255, 255, 0)'
* <Number> borderWidth:容器边框宽度,默认:1
* <String> borderColor:容器边框颜色颜色,默认:'#333'
* <String> imageType:图片类型,默认:image/png(推荐),可选:image/jpeg(注意修改background;不推荐:清空画布再次绘制可能无法正常生成base64),image/webp(Chrome支持),其他类型均为image/png
* <Number> imageQual:图片质量,当imageType为image/jpeg时生效,默认值:0.92,可选0~1之间,超出范围使用默认0.92
* <Function> saveCallback(imgBase64, clearCanvas):保存回调(base64图片,清空画布)
* <Function> clearCallback():清空画布回调
*/
function drawCanvas(args) {
this.el = 'canvas'
this.saveEl = 'saveBtn'
this.clearEl = 'clearBtn'
this.width = document.documentElement.clientWidth || document.body.clientWidth
this.height = 200
this.lineWidth = 2
this.color = '#000'
this.backgroundColor = 'rgba(255, 255, 255, 0)'
this.borderWidth = 1
this.borderColor = '#333'
this.imageType = 'image/png'
this.imageQual = 0.92
this.saveCallback = function (imgBase64, clearCanvas) { }
this.clearCallback = function () { }
for (var k in args) {
this[k] = args[k]
}
// 获取canvas
var canvas = document.getElementById(this.el)
if (!canvas) {
return
}
// 设置边框
if (this.borderWidth > 0 && this.width > this.borderWidth * 2) {
this.width = this.width - this.borderWidth * 2
canvas.style.border = this.borderWidth + 'px solid' + this.borderColor
}
// 设置宽高
canvas.width = this.width
canvas.height = this.height
// 获取canvas context
var cxt = canvas.getContext('2d')
// canvas context相关设置
cxt.fillStyle = this.backgroundColor
cxt.fillRect(0, 0, this.width, this.height)
cxt.strokeStyle = this.color
cxt.lineWidth = this.lineWidth
cxt.lineCap = 'round' // 线条末端添加圆形线帽,减少线条的生硬感
cxt.lineJoin = 'round' // 线条交汇时为原型边角
// 利用阴影,消除锯齿
cxt.shadowBlur = 1
cxt.shadowColor = this.color
var isDraw = false // 是否绘制中,用于mousemove
// 开始绘制
canvas.ontouchstart = function (e) {
var touch = e.changedTouches[0]
if (touch.target.tagName === 'CANVAS') {
var rect = canvas.getBoundingClientRect()
var offsetTop = rect.top
var offsetLeft = rect.left
cxt.beginPath()
cxt.moveTo(touch.clientX - offsetLeft, touch.clientY - offsetTop)
}
}
canvas.onmousedown = function (e) {
if (e.target.tagName === 'CANVAS') {
var rect = canvas.getBoundingClientRect()
var offsetTop = rect.top
var offsetLeft = rect.left
isDraw = true
cxt.beginPath()
cxt.moveTo(e.clientX - offsetLeft, e.clientY - offsetTop)
}
}
// 绘制中
canvas.ontouchmove = function (e) {
e.stopPropagation()
e.preventDefault()
var touch = e.changedTouches[0]
if (touch.target.tagName === 'CANVAS') {
var rect = canvas.getBoundingClientRect()
var offsetTop = rect.top
var offsetLeft = rect.left
cxt.lineTo(touch.clientX - offsetLeft, touch.clientY - offsetTop)
cxt.stroke()
}
}
canvas.onmousemove = function (e) {
e.stopPropagation()
e.preventDefault()
if (isDraw) {
if (e.target.tagName === 'CANVAS') {
var rect = canvas.getBoundingClientRect()
var offsetTop = rect.top
var offsetLeft = rect.left
cxt.lineTo(e.clientX - offsetLeft, e.clientY - offsetTop)
cxt.stroke()
if (e.clientX - offsetLeft <= this.borderWidth
|| e.clientX - offsetLeft >= canvas.width - this.borderWidth * 2
|| e.clientY - offsetTop <= this.borderWidth
|| e.clientY - offsetTop >= canvas.height - this.borderWidth * 2) {
isDraw = false
}
}
}
}.bind(this)
// 结束绘制
canvas.ontouchend = function () {
cxt.closePath()
}
canvas.onmouseup = function mouseupHandle () {
isDraw = false
cxt.closePath()
}
// 清除画布
var clearBtn = document.getElementById(this.clearEl)
if (clearBtn) {
clearBtn.onclick = function () {
cxt.clearRect(0, 0, canvas.width, canvas.height)
cxt.fillStyle = this.backgroundColor
cxt.fillRect(0, 0, this.width, this.height)
this.clearCallback()
}.bind(this)
}
// 保存图片
var saveBtn = document.getElementById(this.saveEl)
if (saveBtn) {
saveBtn.onclick = function () {
var imgBase64
if (this.imageType === 'image/jpeg') {
imgBase64 = canvas.toDataURL('image/jpeg', this.imageQual)
} else {
imgBase64 = canvas.toDataURL()
}
this.saveCallback(imgBase64, function () {
cxt.clearRect(0, 0, canvas.width, canvas.height)
})
}.bind(this)
}
}
使用
<canvas id="canvasBox"></canvas>
<img id="signImage" width="300" height="200" src="" alt="生成的图片">
<div style="display: flex; justify-content: space-around; padding: 0 20%; font-size: 12px">
<button id="saveBtn">生成图片</button>
<button id="clearBtn">清空画布</button>
</div>
<script>
window.onload = function () {
new drawCanvas({
el: 'canvasBox',
saveEl: 'saveBtn',
clearEl: 'clearBtn',
width: 300,
height: 200,
saveCallback: function (imgBase64, clearCanvas) {
document.getElementById('signImage').src = imgBase64
// 清空canvas
// clearCanvas()
},
clearCallback: function () {
document.getElementById('signImage').src = ''
}
})
}
</script>
2、Vue.js版本
vue组件
直接使用npm组件yarn add vue-canvas-sign
ornpm i vue-canvas-sign
输出base64图片,如需转为file文件,可参照