taro vue 微信小程序 手写签名 功能实现

目录

需求:

效果:

template:

data:

methods:

手写绘制:

图片保存:

css:


需求:

在指定区域可以手写签名,点击重新签名,清除手写区域

点击提交签名,将手写签名保存为一张图片上传至oss并将地址传给服务端保存

再次进入时,如果已经保存过签名,则显示上次的签名图片

效果:

template:

html 复制代码
<template>
  <view class="container">
    <image class="sign-img" v-if="!isShowSignCanvas" :src="signUrl" mode="aspectFill"></image>
    <canvas v-show="isShowSignCanvas" class="canvas" id="canvas" canvas-id="canvas" disable-scroll="true" @touchstart="canvasStart" @touchmove="canvasMove" @touchend="canvasEnd" @touchcancel="canvasEnd" @error="canvasIdErrorCallback"></canvas>
    <view class="tips">请在框内签字</view>
    <view class='addBtn'>
      <AtButton  @click="cleardraw" class="zs-btn-plain AtButton" type='primary' >重新签名</AtButton>
      <AtButton  @click="getImg" class="zs-btn-primary AtButton" type='primary' >提交签字</AtButton>
    </view>
  </view>
</template>

data:

javascript 复制代码
import Taro from "@tarojs/taro";
import { ref } from "vue";

const signUrl = ref('') // 签名图片url
const isShowSignCanvas = ref(true) // 是否展示手写区域,有图片不展示,无图片展示

let isButtonDown = false;
let arrx = [];
let arry = [];
let arrz = [];
let canvasw = 0;
let canvash = 0;
let signFlag = ref(false)

let context = Taro.createCanvasContext('canvas'); // 使用 wx.createContext 获取绘图 下文 context
context.setFillStyle('#fff')
context.fillRect(0, 0, canvasw, canvash)
context.draw(true)
context.beginPath()
context.setStrokeStyle('#000000');
context.setLineWidth(4);
context.setLineCap('round');
context.setLineJoin('round');

methods:

手写绘制:

javascript 复制代码
const canvasIdErrorCallback = (e) => { console.log(e) }
//开始
const canvasStart = (event) => {
  isButtonDown = true;
  arrz.push(0);
  arrx.push(event.changedTouches[0].x);
  arry.push(event.changedTouches[0].y);
}
//过程
const canvasMove = (event) => {
  if (isButtonDown) {
    arrz.push(1);
    arrx.push(event.changedTouches[0].x);
    arry.push(event.changedTouches[0].y);
  };

  signFlag.value = true

  for (var i = 0; i < arrx.length; i++) {
    if (arrz[i] == 0) {
      context.moveTo(arrx[i], arry[i])
    } else {
      context.lineTo(arrx[i], arry[i])
    };

  };

  context.setStrokeStyle('#000000');
  context.setLineWidth(4);
  context.setLineCap('round');
  context.setLineJoin('round');
  context.stroke();
  context.draw(true);
}

const canvasEnd = (event) => {
  isButtonDown = false;
}
const cleardraw = () => {
  isShowSignCanvas.value = true
  //清除画布
  arrx = [];
  arry = [];
  arrz = [];
  context.clearRect(0, 0, canvasw, canvash);
  context.draw(true);
}

图片保存:

javascript 复制代码
const getImg = () => {
    if (!isShowSignCanvas.value) {
        Taro.navigateBack();
        return false
    }
    if (arrx.length == 0 || !signFlag.value) {
        Taro.showToast({
            title: `签名内容不能为空`,
            icon: 'none',
            duration: 2000
        })
        return false
    }

    Taro.canvasToTempFilePath({
        x: 0,
        y: 0,
        canvasId: 'canvas',
        success(res) {
            Taro.uploadFile({
                url: `xxxx`,
                filePath: res.tempFilePath,
                name: 'file',
                fileName: '签名',
                success: function (ress) {
                    if (JSON.parse(ress.data).code == 0) {
                        imgUrl = JSON.parse(ress.data).data.res.url
                        // 图片已经上传至oss,此处是把图片地址传给服务端
                      } else {
                          Taro.showToast({
                              title: `上传失败,请稍后再试`,
                              icon: 'none',
                              duration: 2000
                          })
                      }
                },
                fail: console.error
            })
        }
    })
}

css:

css 复制代码
.container {
    width: 95%;
    position: absolute;
    height: 95%;
    top: 50%;
    left: 50%;
    transform: translate(-50%, -50%);
    box-sizing: border-box;
    background: #fff;
    border-radius: 5px;
}

.sign-img {
    width: 100%;
    height: 80%;
}
.canvas {
    width: 100%;
    height: 70%;
    border: 1px dashed #aaa;
    box-sizing: border-box;
}

.tips {
    height: 10%;
    display: flex;
    align-items: center;
    justify-content: center;
    text-align: center;
    color: #aaa;
}

.addBtn {
    display: flex;
    align-items: center;
    justify-content: center;
    height: 18%;
    position: fixed;
    bottom: 0;
    width: 100%;
    background: #fff;
    z-index: 100;
}

.addBtn .AtButton {
    height: 80%;
    font-size: 16px;
    width: 40%;
}
相关推荐
丁总学Java9 小时前
微信小程序,点击bindtap事件后,没有跳转到详情页,有可能是app.json中没有正确配置页面路径
微信小程序·小程序·json
说私域10 小时前
基于开源 AI 智能名片、S2B2C 商城小程序的用户获取成本优化分析
人工智能·小程序
mosen86810 小时前
Uniapp去除顶部导航栏-小程序、H5、APP适用
vue.js·微信小程序·小程序·uni-app·uniapp
qq229511650211 小时前
微信小程序的汽车维修预约管理系统
微信小程序·小程序·汽车
尚梦18 小时前
uni-app 封装刘海状态栏(适用小程序, h5, 头条小程序)
前端·小程序·uni-app
小飞哥liac20 小时前
微信小程序的组件
微信小程序
stormjun1 天前
Java基于微信小程序的私家车位共享系统(附源码,文档)
java·微信小程序·共享停车位·私家车共享停车位小程序·停车位共享
paopaokaka_luck1 天前
基于Spring Boot+Vue的助农销售平台(协同过滤算法、限流算法、支付宝沙盒支付、实时聊天、图形化分析)
java·spring boot·小程序·毕业设计·mybatis·1024程序员节
Bessie2341 天前
微信小程序eval无法使用的替代方案
微信小程序·小程序·uni-app
shenweihong1 天前
javascript实现md5算法(支持微信小程序),可分多次计算
javascript·算法·微信小程序