首页 > 其他分享 >都快2024了,手写签名你还不会吗?

都快2024了,手写签名你还不会吗?

时间:2023-12-04 17:57:24浏览次数:38  
标签:canvas false base64 ctx 2024 签名 touch 手写 event

手写签名(H5\小程序)

手写签名主要是用canvas实现。通过监听手指触碰屏幕事件来完成(touchstart、touchmove、touchcancel)

H5:

这里是用的vue封装一个手写签名的组件
话不多说直接上代码
html:

<template>
  <div class="hand_paint" @touchmove.prevent>
    <div class="hand_content">
      <div class="sign_box" id="signBox">
        <canvas id="signCanvas"></canvas>
      </div>

      <div class="btn_boxs">
        <div @click="clear()">重签</div>
        <div @click="save()">确定</div>
      </div>
    </div>
  </div>
</template>

js:

data() {
  return {
    image: "",
    mousePressed: false, // 是否触碰画板
    c: "",
    ctx: "",

    lastX: 0,
    lastY: 0,
  }
},
mounted() {
  this.image = "";
  this.mousePressed = false;

  this.ctx = document.getElementById("signCanvas").getContext("2d");
  this.c = document.getElementById("signCanvas");
  var signBox = document.getElementById("signBox");
  console.log('clientWidth', signBox.clientWidth)

  this.c.width = signBox.clientWidth; // 设置宽度
  this.c.height = signBox.clientHeight; // 设置高度
  console.log('this.c', this.c.width)

  // 监听touchstart事件,touchmove事件,touchcancel事件等事件
  this.InitThis();
},
methods: {
  InitThis() {
    // 触摸屏
    var _this = this;
    const WIDTH = document.body.clientWidth
    this.c.addEventListener("touchstart", function(event){
      if (event.targetTouches.length == 1) {
        var touch = event.targetTouches[0];
        _this.mousePressed = true;

        _this.Draw(
          touch.pageY - this.offsetLeft,
          WIDTH - this.offsetTop - touch.pageX,
          false
        );
      }
    }, false );
    this.c.addEventListener("touchmove", function(event){
      if (event.targetTouches.length == 1) {
        var touch = event.targetTouches[0];
        if (_this.mousePressed) {
          _this.Draw(
            touch.pageY - this.offsetLeft,
            WIDTH - this.offsetTop - touch.pageX,
            true
          );
        }
      }
    }, false );
    this.c.addEventListener("touchcancel", function(event){
      if (event.targetTouches.length == 1) {
        _this.mousePressed = false;
      }
    }, false );
  },
  // 绘画
  Draw(x, y, isDown) {
    if (isDown) {// 开始移动手指
      this.ctx.beginPath();
      this.ctx.strokeStyle = "#000"; // 颜色
      this.ctx.lineWidth = 3; // 线宽
      this.ctx.lineJoin = "round"; // 拐角类型
      this.ctx.lineMax = 10; // 设置画笔最大线宽
      this.ctx.lineMin = 3; // 设置画笔最小线宽
      this.ctx.linePressure = 1.2; // 设置画笔笔触压力
      this.ctx.smoothness = 30; // 设置画笔笔触大小变化的平滑度
      this.ctx.moveTo(this.lastX, this.lastY);
      this.ctx.lineTo(x, y);
      this.ctx.closePath();
      this.ctx.stroke();
    }
    this.lastX = x;
    this.lastY = y;
  },
  // 清空画板
  clear() {
    this.ctx.setTransform(1, 0, 0, 1, 0, 0);
    this.ctx.clearRect(0, 0, this.ctx.canvas.width, this.ctx.canvas.height);
  },
  // 提交签名
  save() {
    this.checkEmpty(); // 非空验证
  },
  checkEmpty() {
    var c = document.getElementById("signCanvas"); // 获取canvas对象
    if (this.isCanvasBlank(c)) {
      return;
    } else {
      var image = this.c.toDataURL("image/png"); // 得到生成后的签名base64位  url 地址
      // console.log(image); // 打印图片base64 url
      this.dataURLtoFile(image, 'signature.png')
    }
  },
  dataURLtoFile(dataurl, filename) {//将base64转换为文件,dataurl为base64字符串,filename为文件名(必须带后缀名,如.jpg,.png)
    var arr = dataurl.split(','),
        mime = arr[0].match(/:(.*?);/)[1],
        bstr = window.atob(arr[1]),
        n = bstr.length,
        u8arr = new Uint8Array(n);
    while(n--){
      u8arr[n] = bstr.charCodeAt(n);
    }
    let file = new File([u8arr], filename, {type:mime});
    this.aggSignUpload(dataurl, file)
  },
  aggSignUpload(dataurl, file){
    let _this = this

    let formData = new FormData()
    formData.append('file', file);

    // 调用后端接口,通过文件流的方式上传图片
    // 成功回调中将图片base64传回父组件用作展示
        _this.$emit('submit', dataurl)

  },
  // 验证canvas画布内容是否为空
  isCanvasBlank(canvas) {
    var blank = document.createElement("canvas"); // 创建一个空canvas对象
    blank.width = canvas.width;
    blank.height = canvas.height;
    return canvas.toDataURL() == blank.toDataURL(); // 比较值相等则为空
  },
},

css:
由于H5无法完美的强制让浏览器横屏,所以这里使用媒体查询来达到横屏的效果。

@media screen and (orientation: portrait) { // 当屏幕处于纵向时
  .hand_paint {
    position: absolute;
    width: 100vh;
    height: 100vw;
    top: 0;
    left: 100vw;
    -webkit-transform: rotate(90deg);
    -moz-transform: rotate(90deg);
    -ms-transform: rotate(90deg);
    transform: rotate(90deg);
    transform-origin: 0% 0%;
  }
}

小程序:

小程序与H5不同,没有封装为组件,做成了一个公共页面。因为可以在page的json文件中设置:

横屏 "pageOrientation": "landscape"

竖屏 "pageOrientation": "portrait" (默认)

自动切换横竖屏 "pageOrientation": "auto"

如果是用的uni-app,则:

{
  "path" : "路径",
  "style" : {
    "navigationBarTitleText": "手写签名",
    "pageOrientation": "landscape"  // 属性值与上面一致
  }
}

注意!!!小程序在横屏的时候,页面可视区域发生了变化,所以在写css样式的时候需要注意。

这里可以使用vmin。vmin是一种视窗单位,也是相对单位,取的是视图窗口宽度百分比vw和高度百分比vh的较小值。使用vmin的好处是可以让小程序页面不论是横屏还是竖屏,字体大小都能保持一致。

我开发小程序用的uni-app,首先声明一个scss函数:

@function tovmin($rpx){//$rpx为需要转换的字号
    @return #{$rpx * 100 / 750}vmin;
}

使用时就把原本单位为rpx的数值带入函数即可,例如:40rpx ==> tovmin(40);

和H5的canvas不同,小程序只需要改成这样:

<canvas id="signCanvas" canvas-id="signCanvas" class="sign_canvas" disable-scroll @touchstart='handlerTouchstart' @touchmove='handlerTouchmove' @touchend='handlerTouchend' @touchcancel='handlerTouchend'></canvas>

disable-scroll这个属性是为了防止在手指滑动时页面发生移动 。
js:

data() {
    return {
      mousePressed: false, // 是否触碰画板
      c: "",
      ctx: "",
      
	  arrX: [],
	  arrY: [],
	  arrZ: [],
	  isPaint: false, // 是否绘画
			
	  type: ''
    }
},
onReady() {
	this.mousePressed = false;
	
		
	this.ctx = uni.createCanvasContext('signCanvas');
	this.ctx.lineWidth = 3; // 线宽
	this.ctx.lineCap = 'round'; // 线条端点类型
	this.ctx.lineJoin = 'round'; // 线条拐角类型
	this.ctx.globalAlpha = 1; // 透明度
	console.log('ctx',this.ctx)
},
methods: {
	handlerTouchstart(event){
		this.arrX = []
		this.arrY = []
		this.arrZ = []
		if (event.changedTouches.length == 1) {
			 var touch = event.changedTouches[0];
			 // console.log('touchstart',event)
			 this.mousePressed = true;
			
			 this.arrZ.push(0);
			 this.arrX.push(Math.floor(touch.x));
			 this.arrY.push(Math.floor(touch.y));
		}
	},
	handlerTouchmove(event){
		if (event.changedTouches.length == 1) {
			 var touch = event.changedTouches[0];
			 if (this.mousePressed) {
			   this.arrZ.push(1);
			   this.arrX.push(Math.floor(touch.x));
			   this.arrY.push(Math.floor(touch.y));
					
				// 绘画
				for (var i = 0; i < this.arrZ.length; i++) {
					if (this.arrZ[i] == 0) {
						this.ctx.moveTo(this.arrX[i], this.arrY[i])
					} else {
						this.ctx.lineTo(this.arrX[i], this.arrY[i])
					};
				};
				this.ctx.stroke();
				this.ctx.draw(true);
				this.isPaint = true
			 }
		}
			
			
	},
	handlerTouchend(event){
		if (event.changedTouches.length == 1) {
			 this.mousePressed = false;
		}
	},
    // 清空画板
    clearArea() {
			let canvasw, canvash;
			const res = uni.getSystemInfoSync();

			canvasw = res.windowWidth * 1.2; //设备宽度
			canvash = res.windowHeight * 1.2; //设备高度
			console.log('canvasw',canvasw)
			console.log('canvash',canvash)

			this.arrX = []
			this.arrY = []
			this.arrZ = []
			this.isPaint = false
			this.ctx.clearRect(0, 0, canvasw, canvash);
			this.ctx.draw(true);
    },
    // 提交签名
    saveSign() {
            let _this = this
			if (!this.isPaint) { // 画布为空
				return false;
			};
			//生成图片
			const fileManager = wx.getFileSystemManager();
			uni.showLoading({title: '加载中', mask: true})
			uni.canvasToTempFilePath({
				x: 0,
				y: 0,
				canvasId: 'signCanvas',
				success: function (res) {
					//将图片转换为base64 的格式
					uni.hideLoading()
					let base64 = 'data:image/jpg;base64,' + fileManager.readFileSync(res.tempFilePath, 'base64');
					// console.log(base64)
					//图片路径
					// console.log('tempFilePath', res.tempFilePath)
					_this.uploadImg(res.tempFilePath, base64)
				}
			})
    },
    // 上传图片文件
	uploadImg(tempFilePath, base64){
		uni.showLoading({
			 title: '上传中'
		});
		let _this = this
		return new Promise((resolve, reject) => {
			uni.uploadFile({
				method: "post",
				header: {
					"token": userinfo.token,
					"Content-Type": "multipart/form-data"
				},
				url: '接口地址',
				filePath: tempFilePath, // 生成的文件路径
				name: 'file',
				formData: '请求参数',
				success: (res) => {
					uni.hideLoading()
					console.log('res', res)
				},
				fail(err) {
					uni.hideLoading()
					console.log('err', err)
				}
			})
		})
	},
},

标签:canvas,false,base64,ctx,2024,签名,touch,手写,event
From: https://www.cnblogs.com/monkeySoft/p/17875560.html

相关文章

  • 手写类似于BetterScroll样式的左右联动菜单 uni-app+vue3+ts (使用了script setup语法
     注意:在模拟器用鼠标滚动是不会切换光标的,因为使用的是触摸滑动。【自定义类型贴在最后了】script部分如下:import{onMounted}from'vue'importtype{orderDetail}from'@/types/category'importtype{mainArr}from'@/types/main-arr'import{nextTick,ref}......
  • 2024年PMP考试会有变化吗?你关心的来了!
    2024年PMP®考试大纲依然使用项目管理协会(PMI协会)于2021年1月发布的考试内容大纲。  PMP®考试题目将分布在以下三个领域: 1【人员】考试内容所占比例42% 2【过程】考试内容所占比例50% 3【业务环境】考试内容所占比例8% 在考试中约一半的内容将体现项目管理中的预测法,另一......
  • 2024年软考报名费用是多少?全国大汇总!
    软考(计算机技术与软件专业技术资格(水平)考试)是纳入全国专业技术人员职业资格证书制度统一规划,实行大纲、试题、标准、证书均统一的考试办法。其目的是科学、公正地对全国计算机与软件专业技术人员进行职业资格、专业技术资格认定和专业技术水平测试。关于2024年软考报名的费用是多少......
  • 2023-2024 CTU Open Contest
    A.Beth'sCookiesn=int(input())s=input()res=[]foriins:ifres==[]:res.append(i)elifi=='(':ifres[-1]==')':res.append("*")res.append(i)else:......
  • Windows驱动中使用数字签名验证控制设备访问权限
    1.背景  在一般的驱动开发时,创建了符号链接后在应用层就可以访问打开我们的设备并进行通讯。  但我们有时候不希望非自己的进程访问我们的设备并进行交互,虽然可以使用IoCreateDeviceSecure来创建有安全描述符的设备,但大数的用户账户为了方便都是管理员,因此该方法不太完整......
  • 2023-2024-1 20231413 《计算机基础与程序设计》第十周学习总结
    2023-2024-120231413《计算机基础与程序设计》第十周学习总结1.作业信息班级:2023-2024-1-计算机基础与程序设计作业要求:2023-2024-1《计算机基础与程序设计》教学进程目标:自学教材:计算机科学概论第12、13、14章并完成云班课测试《C语言程序设计》第9章并完成云班课测试......
  • 2023-2024-4 20232419《网络空间安全导论》第4章预习总结
    系统安全基础系统安全概述总结:整体论和还原论是研究系统安全的重要方法。系统安全原理总结:安全控制与数学关系最为密切。系统安全结构问题与解决问题:何为分时系统?解决:是使一台计算机采用时间片轮转的方式同时为几个、几十个甚至几百个用户服务的一种操作系统。基于AI......
  • 2023-2024-1 20231309 《计算机基础与程序设计》第十周学习总结
    2023-2024-120231309《计算机基础与程序设计》第十周学习总结作业信息这个作业属于哪个课程2023-2024-1-计算机基础与程序设计这个作业要求在哪里2023-2024-1计算机基础与程序设计第十周作业这个作业的目标作业正文2023-2024-120231309《计算机基础与程......
  • 2023-2024-1 20231415 《计算机基础与程序设计》第十周学习总结
    这个作业属于那个课程https://edu.cnblogs.com/campus/besti/2023-2024-1-CFAP/这个作业要求在哪里https://www.cnblogs.com/rocedu/p/9577842.html#WEEK10这个作业的目标信息系统、数据库与SQL、人工智能与专家系统、人工神经网络、模拟与离散事件、排队系统、天气与......
  • 2023-2024-120232325《网络空间安全导论》第四周学习
    教材内容总结问题1.整体论与还原论2.sql的应用数据库管理:SQL被用于创建、修改和管理数据库中的表格、索引、视图和存储过程等数据库对象。数据查询:SQL用于从数据库中检索数据,可以通过SELECT语句来执行简单或复杂的查询操作,以满足特定的数据需求。数据更新:SQL可以用于更新......