关于js实现手写签名

原有的业务逻辑是用户线下签字,然后拍照后将照片上传到后台系统。用户将手机端照片转移到电脑也很复杂。特别是对于非自己的电脑的图片上传。所以将业务转到ipad或者surface上,减少操作逻辑。

1. 实现操作逻辑

这个直接参考MDN的触摸事件的文档。里面提供了完整的代码。

将其整理成React组件形式如下:

js 复制代码
import React, { useEffect, forwardRef, useImperativeHandle, useState } from 'react';

const Signature: React.FC = (props, ref) => {
    const color = "#000";
    useImperativeHandle(ref, () => ({
        getImage: () => new Promise((resolve,reject) => {
            // 返回处理后的数据形式
        })
    }));

    useEffect(() => {
        const el = document.querySelector("#signCanvas");
        el?.addEventListener("touchstart", handleStart, false);
        el?.addEventListener("touchend", handleEnd, false);
        el?.addEventListener("touchmove", handleMove, false);
    }, [])
    
    // 所有方法参考MDN内的实现方法
    const ongoingTouches = [];
    function copyTouch(touch) {}
    function ongoingTouchIndexById(idToFind) { }
    function handleStart(evt) {}
    function handleMove(evt) {}
    function handleEnd(evt) {}
  return (
        <canvas id="signCanvas" width="600" height="600" style={{border: "solid #aaa 1px"}}>
          你的浏览器不支持 canvas 元素。
        </canvas>
  );
};

export default forwardRef(Signature);

2. 注意细节

1. 获取canvas元素

ini 复制代码
// MDN中获取元素的类型
const el = document.getElementsByTagName("canvas")[0];

如果代码中已经有了canvas元素,比如antd中使用的QrCode组件。

2. 坐标设置(无滚动条)

MDN中的demo使用的pageX和pageY默认使用的是屏幕左上角,我使用的时候用的是弹窗打开,弹窗位于页面中间。 此时应该基于弹窗的位置进行设置。

为啥是top*2?试出来的,我也不知道

js 复制代码
// 在start方法中挪动坐标位置
    function handleStart(evt) {
        evt.preventDefault();
        const el = document.querySelector("#signCanvas");
        const ctx = el?.getContext("2d");
        const { left, top } = el?.getBoundingClientRect();
        ctx?.translate(0 - left, 0 - top*2);
        .......
     }
// 每次画完再把坐标复原,不然,坐标每画一次挪一次,就越来越远了。
    function handleEnd(evt) {
        evt.preventDefault();
        const el = document.querySelector("#signCanvas");
        const ctx = el?.getContext("2d");       
        const { left, top } = el?.getBoundingClientRect();
        ctx?.translate(0 + left, 0 + top*2);  
        ......
    }

3. 坐标取值(有滚动条)

当弹窗下的页面出现滚动条时,pageX,pageY会把滚动条计算进去。此时要改成screenX,screenY

4. 将签名转图片保存

1. 转base64图片

js 复制代码
            const el = document.querySelector("#signCanvas");
            el?.toBlob((blob) => {
                var reader = new FileReader();
                reader.readAsDataURL(blob); 
                reader.onloadend = function() {
                  var base64data = reader.result;   
                  resolve(base64data)        
                }
            },  "image/png")

2. 转file类型

js 复制代码
// 解析文件
export const base64ToFile = (dataurl) => {
  const arr = dataurl.split(',');
  const mime = arr[0].match(/:(.*?);/)[1];
  const bstr = window.atob(arr[1]);
  let n = bstr.length;
  const u8arr = new Uint8Array(n);
  while (n--) {
    u8arr[n] = bstr.charCodeAt(n);
  }
  let blob = new File([u8arr], 'file.png', { type: mime });
  return blob; 
};

// 将base64转为file
  const file = base64ToFile(base64data);   

5. 兼容性

用到了Touch事件,不支持safari浏览器

相关推荐
面朝大海,春不暖,花不开4 分钟前
Spring Boot MVC自动配置与Web应用开发详解
前端·spring boot·mvc
知否技术4 分钟前
2025微信小程序开发实战教程(一)
前端·微信小程序
玲小珑5 分钟前
Auto.js 入门指南(五)实战项目——自动脚本
android·前端
Sparkxuan5 分钟前
IntersectionObserver的用法
前端
玲小珑6 分钟前
Auto.js 入门指南(四)Auto.js 基础概念
android·前端
全栈技术负责人6 分钟前
Webpack性能优化:构建速度与体积优化策略
前端·webpack·node.js
爱吃肉的小鹿8 分钟前
浏览器渲染的核心流程及详细解析(2025.6月最新)
前端
贩卖纯净水.13 分钟前
webpack打包学习
前端·学习·webpack
敲键盘的小夜猫25 分钟前
RunnablePassthrough介绍和透传参数实战
java·服务器·前端
独立开阀者_FwtCoder33 分钟前
MySQL FULLTEXT索引解析:为什么它能大幅提升文本搜索性能?
前端·javascript·面试