app canvas渲染后图片黑色_H5 基于 canvas 实现电子签名并生成PDF文档

(给前端大全加星标,提升前端技能)

转自:coyota666

https://juejin.cn/post/6901273585428463624

前言

电子签名通俗来说就是通过技术手段实现在电子文档上加载电子形式的签名,其作用类似于纸质合同上的手写签名或加盖的公章。虽然电子签名多年来合法性一直遭到质疑,但其在企业工作流审批、请柬、单据保全等场景应用广泛,最近的项目中就有这样一个手写签名并生成PDF文件的需求。

实现思路

  1. 使用canvas来实现手写签名的功能,然后将canvas转化为图片,贴在签名的位置;
  2. 将整个需要生成文档的dom区域使用html2canvas插件转成一张大图;
  3. 使用JsPDF插件将上述图片生成PDF文档;
  4. 对于文件内容较多的情况,需要合理选择分页位置;

生成签名

1. 在tsx中定义canvas画布

 "350" height="150" />

注意:Canvas的宽高必须要使用内联样式定义,这是因为Canvas标签有自己的默认宽高300px×150px。它内联样式定义的width和height是绘画区域(画布)实际宽度和高度,绘制的图形都是在这个上面。如果在style外链文件中定义其width和height,那么这个width和height是Canvas在浏览器中被渲染的高度和宽度。如果Canvas中没有直接定义width和height没或值不正确,就会被设置成默认值{width:300px,height:150px}。所以,如果你在style中外链文件中设置了canvas {width: 200px; height: 200px;},却没有直接在canvas上定义画布宽高,那么此时你输出canvas.height 值依旧为150,canvas.width值依旧为300。也就是一块150×300的画布在200×200的区域渲染,因而图片会出现拉伸、变形等现象。

2. 定义签名函数

 const writing = (
    beginX: number,
    beginY: number,
    stopX: number,
    stopY: number,
    ctx: any,
  ) => {
    ctx.beginPath();  // 开启一条新路径
    ctx.globalAlpha = 1;  // 设置图片的透明度
    ctx.lineWidth = 3;  // 设置线宽
    ctx.strokeStyle = 'red';  // 设置路径颜色
    ctx.moveTo(beginX, beginY);  // 从(beginX, beginY)这个坐标点开始画图
    ctx.lineTo(stopX, stopY);  // 定义从(beginX, beginY)到(stopX, stopY)的线条(该方法不会创建线条)
    ctx.closePath();  // 创建该条路径
    ctx.stroke();  // 实际地绘制出通过 moveTo() 和 lineTo() 方法定义的路径。默认颜色是黑色。
  };

3. 注册监听事件

    let beginX: number, beginY: number;
    const canvas: HTMLCanvasElement = canvasDom.current;
    const ctx = canvas.getContext('2d');
    ctx.fillStyle = '#fff';
    ctx.fillRect(0, 0, canvas.width, canvas.height);
    canvas.addEventListener('touchstart', function(event: any) {
      event.preventDefault(); // 阻止在canvas画布上签名的时候页面跟着滚动
      beginX = event.touches[0].clientX - this.offsetLeft; 
      beginY = event.touches[0].pageY - this.offsetTop;
    });
    canvas.addEventListener('touchmove', (event: any) => {
      event.preventDefault(); // 阻止在canvas画布上签名的时候页面跟着滚动
      event = event.touches[0];let stopX = event.clientX - canvas.offsetLeft;let stopY = event.pageY - canvas.offsetTop;
      writing(beginX, beginY, stopX, stopY, ctx);
      beginX = stopX; // 这一步很关键,需要不断更新起点,否则画出来的是射线簇
      beginY = stopY;
    });

注意

  1. 在注册“touchstart”和“touchmove”事件时,需要阻止默认事件,否则页面会跟着手势上下滑动。
  2. 移动端的每个触摸事件对象中都包括了touches这个属性,它用于描述位于屏幕上的所有手指的一个列表,获取当前事件对象我们习惯性的使用event = event.touches[0],而在PC端则不需要这么操作。
  3. offsetLeft值跟offsetTop值跟父级元素没关系,而是跟其上一级的定位元素(除position:static外的所有定位如fixed,relative,absolute元素)有关系。若上一级定位元素都没有除position:staice外的定位,则这个偏移量是相对于body而言的。
  4. 需要理清移动端事件对象的几个属性,⏬dbce9e3f36dffc1cf9d1fc2985a07b4f.png

clientX/clientY: 触摸位置距离当前body可视区域的x,y坐标;
pageX/pageY: 对于整个页面来说,触摸位置距离body左上角的x,y坐标,包括被scrollTop和scrollLeft的值;
screenX/screenY: 触摸位置距离显示器左边和顶部的x,y距离。
所以,在获取结束点坐标的时候,如果当前页面没有出现滚动条,使用clientY和pageY计算差别不大,如果页面比较长,出现了滚动条,那么就必须要使用pageY来计算。clientX同理,但是移动端通常横向滚动的场景不多,所以用clientX来计算即可。

  1. 在签名(touchmove)这个动作过程中,我们需要不断的更新起点位置,否则画出来是这样?

2b7372e477c5f6d7706dde229fb9d8bb.png其实这个原理和微积分很相似,线段本质上就是由无穷多个小线段组成,宏观一点来看可以把线段当成一个个长度很小的小线段首尾相连构成。所以我一直觉得编程编到最后就是考验一个人的数学能力,交并集、逻辑思维、算法等都能看到数学的身影。最后生成签名如下:cde6368d52c0f634db641eea737c9ed5.png

生成PDF文档

html2canvas是一款将HTML代码转换成Canvas的插件,因此需要用一个div包裹住需要打印的内容区域,获得这个dom节点。

html2Canvas(dom, {
    allowTaint: true,
    width: dom.offsetWidth, //设置获取到的canvas宽度
    height: dom.offsetHeight, //设置获取到的canvas高度
    x: 0, //页面在水平方向滚动的距离
    y: 0, //页面在垂直方向滚动的距离
   })

注意:此处需要设置width和height及x,y,否则当页面内容只有一页的时候没有问题,但是若页面内容有很多页的时候,就会出现生成的图片只有一小部分有内容的现象。问题就出现在这个配置参数上,若没有设置宽高,则默认只取当前视口的内容,丢弃掉其他超出当前视口的内容。
设置打印参数:

const print = () => {let dom: HTMLElement = pdfDom.current;
    html2Canvas(dom, {
      allowTaint: true,
      width: dom.offsetWidth, //设置获取到的canvas宽度
      height: dom.offsetHeight, //设置获取到的canvas高度
      x: 0, //页面在水平方向滚动的距离
      y: 0, //页面在垂直方向滚动的距离
    }).then((canvas: HTMLCanvasElement) => {let canvasWidth = canvas.width;let canvasHeight = canvas.height;let pageHeight = (canvasWidth / 592.28) * 841.89; // 一页A4 pdf能显示的canvas高度let imgWidth = 595.28; // 设置图片宽度和A4纸宽度相等let imgHeight = (592.28 / canvasWidth) * canvasHeight;//等比例换算成A4纸的高度let totalHeight = imgHeight; // 需要打印的图片总高度,初始状态和图片高度相等let pageData = canvas.toDataURL('image/png', 1.0);let PDF = new JsPDF('p', 'pt', 'a4', true);if (totalHeight         PDF.addImage(pageData, 'JPEG', 0, 0, imgWidth, imgHeight); // 从顶部开始打印
      } else {let top = 0;   // 打印初始区域while (totalHeight > 0) {
          PDF.addImage(pageData, 'JPEG', 0, top, imgWidth, imgHeight);  // 从图片顶部往下top位置开始打印
          totalHeight -= pageHeight;
          top -= 841.89;if (totalHeight > 0) {
            PDF.addPage();
          }
        }
      }
      PDF.save('test.pdf');
    });
  };

选择分页位置

按照上述步骤生成了一份PDF文档,但是当PDF页数有很多的时候,会有这样的问题⏬da01ef62e530f5b8c190c3062aa757a1.png可以看到,分页的时候从这段文字这里懒腰截断了。这显然不是我们想要看到的效果,如何解决这个问题呢??

  • PDF文档页数较少的情况

可以在开发测试的时候预先在将要分页的地方插入一个padding,就是提前预留分页位置

  • PDF文档页数较多

对于这种情况,笔者尝试遍历要打印的dom节点的子节点,将每一页所能打印的dom节点高度累加,若超过了页面所能承载的最大高度,则将最后一个节点增加padding,打印完毕将样式还原。这种方法因为要计算每个dom节点的高度,非常耗性能,也要求页面dom元素的颗粒度较细,否则会出现一个页面有大块空白,完全无法模拟出word生成pdf的那种效果,所以就不展开讨论了。如若有读者有比较好的解放方案,欢迎不吝赐教,感谢~❤️

推荐阅读  点击标题可跳转

1、烧脑!JS+Canvas 带你体验「偶消奇不消」的智商挑战

2、canvas 中普通动效与粒子动效的实现

3、基于 HTML5 Canvas 的交互式地铁线路图

觉得本文对你有帮助?请分享给更多人

推荐关注「前端大全」,提升前端技能

9ab387e10935d32789c480fc14cee3d9.png

点赞和在看就是最大的支持❤️

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/550718.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

浏览器播放视频时蓝牙耳机自动关机

原因是谷歌浏览器新增的音量调节界面插件的问题,解决方法:进网址设置 chrome://flags/#hardware-media-key-handling 里面Hardware Media Key Handling 改成disabled,然后点击右下角浏览器重启就OK了。不仅是chrome有效,新版的Mic…

前端为什么说github很重要_私域流量|为什么说私域流量很重要?

前面几篇文章我们分析了私域流量的概念,现在我们来分析一下私域流量的好处。为什么说私域流量很重要呢?首先,私域流量可以把营销成本降低,较快地获取用户反馈。之前在公域流量上进行的交易,都是你买我卖,你…

java中几种基本数据类型之间的转换

int类型转字符串 int a 10 ; String str a"";int a 10; String str Integer.toString(a);int a 10; String str String.valueOf(a); 字符串转int类型 String str "10"; int a Integer.parseInt(str);String str "10"; int a Intege…

idea的jsp如何显示语法高亮_如何啃下Python学习中的三块硬骨头?

Python 根式字【51CTO.com快译】众所周知,作为一门开源的高级编程语言,Python的用途十分广泛。它可以被用于不同的应用场景中,包括:开发基于桌面和Web的应用程序,分析访问数据,开发后端网站和人工智能等。虽…

java中数组的定义

int[] a new int[10];int[] a new int[]{元素1,元素2};int[] a {元素1,元素2};

几种经典排序算法

文章转载自:指月小筑 原文链接:链接

centos7盘符 linux_linux_centos7_扩展磁盘空间

那些经过PV创建啊 ,vg 扩展啊 还有lv扩展的 都略过了。就说说我遇到的错误吧。先说说系统环境侬 这就是系统环境咯Linux oracle.localdomain 3.10.0-514.el7.x86_64 #1 SMP Tue Nov 22 16:42:41 UTC 2016 x86_64 x86_64 x86_64 GNU/Linux然后是自己报错咯[rootoracl…

关于BigInteger的加减乘除使用

public static void main(String[] args) {BigInteger bg1 new BigInteger("12345678912");// int类型转BigIntegerint value 12345678912;BigInteger bg2 BigInteger.valueOf(value);//1.加法System.out.println(bg1.add(bg2));//2.减法System.out.println(bg1.s…

unity隔一段时间再显示_Unity3D内置倒计时!从此再不拖延!

短时间内再次突然诈尸,不过这次依然不是新视频。没错又是来发开源项目。“嘿,我一定要在接下来的一个小时的时间里实现这个功能。”“诶?XXX出了个新视频,先来看看。” .......于是一天过去了。无论你是极度拖延,还是在…

关于eclipse的一些简单配置

自动导包 在eclispe中,打开 Window > Preferences > Java > Editor > Save Actions 然后选中 Organize impots

python elif可以单独使用_Celery在python中的单独使用

简单使用:1.目录结构-app_task.py-worker.py-result.py2.在需要进行异步执行的文件app_task.py中导入celery,并实例化出一个对象,传入消息中间和数据存储配置参数broker redis://127.0.0.1:6379/1 #使用redis第一个库backend redis://127.0.0.1:6379/2…

Eclipse配置自动补齐键为alt+/

在顶栏找到window,选择下拉栏中的Preferences 在Preferences的搜索框输入keys,在右边的候选区域找到Content Assist,在Binding中输入Alt/ Apply and Close即可

python高级属性 用法 编程_python高级编程之面向对象高级编程

1 面向对象编程面向对象这节比较简单,就稍微总结几个特殊的点。特殊方法__init__前后分别有两个下划线,__init__方法的第一个参数永远是self,表示创建的实例本身,因此,在__init__方法内部,就可以把各种属性…

1命名规则 sentinel_Alibaba Sentinel 规则参数总结

本文总结了Alibaba Sentinel各种规则的参数。基于Sentinel 1.6.2编写,未来如果本文不再适用,可自行点击每一节 参考 一览的链接前往查看如何配置。一、流控规则1.1 配置1.2 参数Field说明默认值resource资源名,资源名是限流规则的作用对象cou…

java文件读写类及其用法介绍

见链接:转载于zhangbinu的博客

公共界面_公共建筑东营市档案馆能耗计量分析系统

摘要:本文介绍东营市档案馆能耗管理系统,采用智能电力仪表采集配电现场的各种电参量,仪表就地组网后通过现场总线通讯并远传至后台,通过Acrel-5000型建筑能耗监测系统实现配电回路用电的监测分析。关键词:大型公共建筑…

IDEA、pycharm白嫖攻略

本方法仅适用于高校大学生 JetBrains系列下的idea、pycharm优秀的代码提示让人爱不释手,但是相比较eclipse,idea的巨大缺点就是:要收费。 我看到身边的许多同学为了使用idea、pycharm而到处搜索激活码或者破解软件 ,但是激活码还是…

com.alibaba.easyexcel导出指定的列_使用Python导入导出Excel表格

这篇文章的目的是讲解使用python导入导出Excel表格,目前还不涉及数据处理,主要实现为数据的展示。第一步,准备材料python3.8的安装包一个Excel表格第二步,安装软件python可以采用默认安装的方式,安装完之后把python和p…

2021-08-10 HDFS Web报错Couldn‘t preview the file.

在查看文件内容时&#xff0c;在红框位置出现报错“Couldn’t preview the file” 解决方法 第一步&#xff1a;修改hdfs-site.xml&#xff0c;添加配置信息 <property><name>dfs.webhdfs.enabled</name><value>true</value> </property&…

2021-08-10 maven配置阿里云仓库

方式一&#xff1a;全局配置 可以添加阿里云的镜像到maven的setting.xml配置中&#xff0c;这样就不需要每次在pom中&#xff0c;添加镜像仓库的配置&#xff0c;在mirrors节点下面添加子节点&#xff1a; <mirror><id>nexus-aliyun</id><mirrorOf>ce…