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 or npm i vue-canvas-sign
输出base64图片,如需转为file文件,可参照

© 2024 www.wdg.pub all right reserved Last modified: 2024-06-14

results matching ""

    No results matching ""