首页 > 其他分享 >【Vue】 签名组件

【Vue】 签名组件

时间:2024-03-28 09:35:57浏览次数:21  
标签:Vue const default fileList return 签名 组件 false type

一、需求背景:

检查业务,检查完成后,执行人需要签字证明检查完成

 

二、实现效果:

 

三、技术实现

通过canvas转换成blob对象,可以上传到文件服务,或者是下载另存为到本地磁盘

注意重点,canvas的样式的宽高和dom对象宽高一定要一致才可以,否则无法在面板绘制线条!

<template>
  <el-dialog title="签名面板" :close-on-click-modal="false" append-to-body :visible.sync="visible" class="JNPF-dialog JNPF-dialog_center" lock-scroll width="600px">
    <canvas id="signatureCanvas" style="width: 500px; height: 300px;"></canvas>
    <span slot="footer" class="dialog-footer">
      <el-button @click="clearCanvas"> 清 除</el-button>
      <el-button @click="visible = false"> 取 消</el-button>
      <el-button type="primary" @click="dataFormSubmit()" :loading="btnLoading"> 确 定</el-button>
    </span>
  </el-dialog>
</template>

<script>
import request from '@/utils/request'

export default {
  name: 'SignPanel',
  components: {},
  props: {
    pathType: {
      type: String,
      require: true,
      default: 'annexpic'
    },
    apiData: {
      type: Object,
      require: true,
      default: () => {}
    }
  },
  data() {
    return {
      visible: false,
      btnLoading: false,
      canvasInstance: null,
      canvasContext: null,
      drawing: false,
      lastX: false,
      lastY: false
    }
  },
  mounted() {

  },
  methods: {
    init() {
      this.visible = true
      this.$nextTick(() => {
        this.initialCanvas()
      })
    },
    initialCanvas() {
      // 获取画布元素和上下文对象
      const canvas = document.getElementById('signatureCanvas')
      const ctx = canvas.getContext('2d');
      this.canvasInstance = canvas
      this.canvasContext = ctx

// 设置 canvas 的宽度和高度
      canvas.width = 500; // 根据需要设置
      canvas.height = 300; // 根据需要设置

      // // 初始化变量
      this.drawing = false
      this.lastX = 0
      this.lastY = 0
      const _that = this;


      // 处理鼠标按下事件
      canvas.addEventListener('mousedown', (e) => {
        _that.lastX = e.offsetX
        _that.lastY = e.offsetY
        _that.drawing=true;
      })

      // 处理鼠标移动事件
      canvas.addEventListener('mousemove', (e) => {
        if (!_that.drawing) return
        ctx.beginPath()
        ctx.moveTo(_that.lastX, _that.lastY)
        ctx.lineTo(e.offsetX , e.offsetY)
        ctx.stroke()
        ctx.closePath()
        _that.lastX = e.offsetX
        _that.lastY = e.offsetY
      })

      // 处理鼠标松开事件
      canvas.addEventListener('mouseup', (e) => {
        _that.drawing = false
      })

      // 处理鼠标离开事件
      canvas.addEventListener('mouseout', () => {
        _that.drawing = false
      })
    },
    dataFormSubmit() {
      this.toUploadSignPic()
    },
    dataURLtoBlob(dataUrl) {
      const arr = dataUrl.split(','), mime = arr[0].match(/:(.*?)/)[1]
      const bstr = atob(arr[1])
      let n = bstr.length
      const u8arr = new Uint8Array(n)
      while(n--){
        u8arr[n] = bstr.charCodeAt(n)
      }
      return new Blob([u8arr], {type:mime})
    },
    toUploadSignPic() {
      const dataUrl = this.canvasInstance.toDataURL('image/png')
      const blobData = this.dataURLtoBlob(dataUrl)
      // 创建 File 对象
      const fileName = 'image.png'
      const file = new File([blobData], fileName, {type: 'image/png'})
      const formData = new FormData()
      formData.append('file', file)
      this.uploadSignApi(formData).then(res => {
        if (res.code !== 200) return
        this.$emit('whenSuccess', res, file)
        this.visible = false
      }).catch(err => {
        this.$emit('whenError', err, file)
        this.visible = false
      })
    },
    toDownloadSignPic() {
      const dataUrl = this.canvasInstance.toDataURL('image/png')
      const link = document.createElement('a')
      link.href = dataUrl
      link.download = 'signature.png'
      link.click()
    },
    clearCanvas() {
      this.canvasContext.clearRect(0, 0,  this.canvasInstance.width,  this.canvasInstance.height)
    },
    uploadSignApi(formData) {
      const apiPath = `/api/file/Uploader/${this.pathType}`
      const param = this.apiData
      Object.keys(param).forEach(key => ( formData.append(key, param[key]) ))
      return request({
        url: apiPath,
        method: 'POST',
        headers: { 'Content-Type': 'multipart/form-data' },
        data: formData
      })
    }
  }
}
</script>

<style scoped lang="scss">
canvas {
  border: 1px solid #DCDFE6;
  cursor: crosshair;
  border-radius: 5px;
}
</style>

 

因为兼容现有系统的组件,我而外将框架自带的图片上传改造成签名上传组件

图片上传组件点击时一定会选取本地文件,为了解决这个问题我是选择直接隐藏了上传组件

改为追加了一个签名面板按钮,面板确认时,发射器回调到组件上传成功的回调

因为签名只存在一份,所以文件数量限制1即可,通过上传成功的回调就能拦截处理

<template>
  <div class="UploadFile-container">
    <el-button @click="openSignPanel" style="margin-right: 5px;">打开签名</el-button>
    <template v-if="fileList.length">
      <transition-group class="el-upload-list el-upload-list--picture-card" tag="ul" name="el-list">
        <li class="el-upload-list__item is-success" v-for="(file,index) in fileList"
            :key="file.fileId">
          <el-image :src="define.comUrl+file.url" class="el-upload-list__item-thumbnail" :preview-src-list="getImgList(fileList)" :z-index="10000" :ref="'image'+index">
          </el-image>
          <span class="el-upload-list__item-actions">
            <span class="el-upload-list__item-preview" @click="handlePictureCardPreview(index)">
              <i class="el-icon-zoom-in"></i>
            </span>
            <span v-if="!disabled" class="el-upload-list__item-delete" @click="handleRemove(index)">
              <i class="el-icon-delete"></i>
            </span>
          </span>
        </li>
      </transition-group>
    </template>
    <template v-if="!detailed">
      <el-upload
        v-show="false"
        :action="define.comUploadUrl+'/'+type"
        :headers="uploadHeaders" :data="params"
        ref="elUpload"
        :on-success="handleSuccess"
        :multiple="limit!==1"
        :show-file-list="false"
        accept="image/*"
        :before-upload="beforeUpload"
        :disabled="disabled"
        list-type="picture-card"
        :auto-upload="false"
        class="upload-btn">
        <i slot="default" class="el-icon-plus" disabled></i>
      </el-upload>
    </template>
    <template>
      <div class="el-upload__tip" slot="tip" v-if="tipText">{{ tipText }}</div>
    </template>

    <sign-panel
      :visible.sync="signPaneVisible"
      :path-type="type"
      :api-data="params"
      ref="signForm"
      @whenSuccess="signUploadSuccess"
      @whenError="signUploadError"
    />
  </div>
</template>

<script>
import emitter from 'element-ui/src/mixins/emitter'
import SignPanel from '@/components/Generator/components/Upload/SignPanel.vue'
import BigForm from '@/views/dp-mng/scr-se-check/big-form.vue'
let { methods: { dispatch } } = emitter
const units = {
  KB: 1024,
  MB: 1024 * 1024,
  GB: 1024 * 1024 * 1024
}
export default {
  name: 'UploadSign',
  components: { BigForm, SignPanel },
  props: {
    value: {
      type: Array,
      default: () => []
    },
    type: {
      type: String,
      default: 'annexpic'
    },
    disabled: {
      type: Boolean,
      default: false
    },
    detailed: {
      type: Boolean,
      default: false
    },
    showTip: {
      type: Boolean,
      default: false
    },
    limit: {
      type: Number,
      default: 0
    },
    accept: {
      type: String,
      default: 'image/*'
    },
    sizeUnit: {
      type: String,
      default: 'MB'
    },
    pathType: {
      type: String,
      default: 'defaultPath'
    },
    isAccount: {
      type: Number,
      default: 0
    },
    folder: {
      type: String,
      default: ''
    },
    fileSize: {
      default: 10
    },
    tipText: {
      type: String,
      default: ''
    }
  },
  data() {
    return {
      signPaneVisible: false,
      fileList: [],
      uploadHeaders: { Authorization: this.$store.getters.token },
    }
  },
  watch: {
    value: {
      immediate: true,
      handler(val) {
        this.fileList = Array.isArray(val) ? val : []
      }
    }
  },
  computed: {
    params() {
      return {
        pathType: this.pathType,
        isAccount: this.isAccount,
        folder: this.folder
      }
    }
  },
  methods: {
    signUploadSuccess(result, file) {
      this.handleSuccess(result, file, this.$refs.elUpload.fileList)
    },
    signUploadError() {
      console.log(result)
    },
    openSignPanel() {
      this.signPaneVisible = true
      this.$refs.signForm.init()
    },
    beforeUpload(file) {
      if (this.fileList.length >= this.limit) {
        this.handleExceed()
        return false
      }
      const unitNum = units[this.sizeUnit];
      if (!this.fileSize) return true
      let isRightSize = file.size / unitNum < this.fileSize
      if (!isRightSize) {
        this.$message.error(`图片大小超过${this.fileSize}${this.sizeUnit}`)
        return isRightSize;
      }
      let isAccept = new RegExp('image/*').test(file.type)
      if (!isAccept) {
        this.$message.error(`请上传图片`)
        return isAccept;
      }
      return isRightSize && isAccept;
    },
    handleSuccess(res, file, fileList) {
      if (this.fileList.length >= this.limit) return this.handleExceed()
      if (res.code == 200) {
        this.fileList.push({
          name: file.name,
          fileId: res.data.name,
          url: res.data.url
        })
        this.$emit('input', this.fileList)
        this.$emit('change', this.fileList)
        dispatch.call(this, 'ElFormItem', 'el.form.change', this.fileList)
      } else {
        this.$refs.elUpload.uploadFiles.splice(fileList.length - 1, 1)
        fileList.filter(o => o.uid != file.uid)
        this.$emit('input', this.fileList)
        this.$emit('change', this.fileList)
        dispatch.call(this, 'ElFormItem', 'el.form.change', this.fileList)
        this.$message({ message: res.msg, type: 'error', duration: 1500 })
      }
    },
    handleExceed(files, fileList) {
      this.$message.warning(`当前限制最多可以上传${this.limit}张图片`)
    },
    handlePictureCardPreview(index) {
      this.$refs['image' + index][0].clickHandler()
    },
    handleRemove(index) {
      this.fileList.splice(index, 1)
      this.$refs.elUpload.uploadFiles.splice(index, 1)
      this.$emit("input", this.fileList)
      this.$emit('change', this.fileList)
      dispatch.call(this, 'ElFormItem', 'el.form.change', this.fileList)
    },
    getImgList(list) {
      const newList = list.map(o => this.define.comUrl + o.url)
      return newList
    }
  }
}
</script>
<style lang="scss" scoped>
>>> .el-upload-list--picture-card .el-upload-list__item {
  width: 120px;
  height: 120px;
}
>>> .el-upload--picture-card {
  width: 120px;
  height: 120px;
  line-height: 120px;
}
.upload-btn {
  display: inline-block;
}
.el-upload__tip {
  color: #a5a5a5;
  word-break: break-all;
  line-height: 1.3;
  margin-top: 5px;
}
// .el-upload-list--picture-card {
//   display: inline-block;
//   height: 0;
// }
</style>

  

表单效果:

 

标签:Vue,const,default,fileList,return,签名,组件,false,type
From: https://www.cnblogs.com/mindzone/p/18100767

相关文章

  • 第四章(页面组件)总结
    4.1一组件的定义及属性组件是页面视图层(WXML)的基本组成单元,组件组合可以构建功能强大的页面结构。小程序框架为开发者提供了容器视图、基础内容、表单、导航、多媒体、地图、画布、开放能力等8类(30多个)基础组件。每一个组件都由一对标签组成,有开始标签和结束标签,内容放置在开......
  • 基于PHP+vue的社区居民互助交流系统016gm
    。开发环境开发语言:php后端框架:Thinkphp/Laravel前端框架:vue.js服务器:apache数据库:mysql运行环境:phpstudy/wamp/xammp等社区互助平台的功能分为管理员和用户两个部分,系统的主要功能包括首页,个人中心,用户管理,租房信息管理,失物招领管理,宠物代遛管理,停车位出租管理,其他管......
  • 基于SpringBoot+Vue的在线家具商城毕业设计
    文章目录前言详细视频演示具体实现截图技术栈后端框架SpringBoot前端框架Vue持久层框架MyBaitsPlus系统测试系统测试目的系统功能测试系统测试结论为什么选择我代码参考数据库参考源码获取前言......
  • 基于SpringBoot+Vue的新冠病毒密接者跟踪系统毕业设计
    文章目录前言详细视频演示具体实现截图技术栈后端框架SpringBoot前端框架Vue持久层框架MyBaitsPlus系统测试系统测试目的系统功能测试系统测试结论为什么选择我代码参考数据库参考源码获取前言......
  • vue3背景下,el-input嵌套在弹出框中,自动聚焦“失效”?如何实现自动聚焦
    情景:在一个弹出框中,有一个el-input输入框,想要实现当弹出框出现时,input会自动聚焦。使用input的原生属性autofocus去自动获取焦点失效;使用ref获取el-input元素也显示undefined!!!<el-dialogv-model="dialogFormVisible"width="300">.......<el-input......
  • Vue 与 React:前端框架对比分析
    ......
  • 使用可迭代对象作为 React 组件
    源码最近看React源码的时候发现其在处理数组的同时也对可迭代协议做了处理,因此React组件可以是一个返回可迭代对象的函数。源码如下:reconcileChildFibersImplif(isArray(newChild)){returnreconcileChildrenArray(returnFiber,currentFirstChild,newChild,lanes,......
  • 深入理解 Vue 3.0 宏函数:提升组件代码的工程化与可维护性
    Vue3.0宏函数详解:defineProps、defineEmits、defineExpose、defineSlots和defineOptions在Vue3.0中,为了更好地组织和维护组件代码,引入了几个新的宏函数。这些宏函数包括defineProps、defineEmits、defineExpose、defineSlots和defineOptions。本文将详细介绍这五......
  • 基于java+springboot+vue实现的超市货品信息管理系统(文末源码+Lw+ppt)23-355
    摘 要随着世界经济信息化、全球化的到来和互联网的飞速发展,推动了各行业的改革。若想达到安全,快捷的目的,就需要拥有信息化的组织和管理模式,建立一套合理、动态的、交互友好的、高效的超市货品信息管理系统。当前的信息管理存在工作效率低,工作繁杂等问题,基于信息化的超市货品......
  • 基于java+springboot+vue实现的校园二手交易系统(文末源码+Lw+ppt)23-336
    摘 要自从新冠疫情爆发以来,各个线下实体越来越难做,线下购物的人也越来越少,随之带来的是一些不必要的浪费,尤其是即将毕业的大学生,各种用品不方便携带走导致被遗弃,造成大量的浪费。本系统目的就是让毕业生的二手物品有一定的价值,并且在疫情环境下做到零接触买卖,更加安全。在......