Javascript 将 HTML 页面生成 PDF 并下载

论坛 期权论坛 期权     
前端大全   2019-7-27 14:55   7097   0
(点击上方公众号,可快速关注)


作者:linwalker
https://segmentfault.com/a/1190000009211079
最近碰到个需求,需要把当前页面生成 pdf,并下载。弄了几天,自己整理整理,记录下来,我觉得应该会有人需要 :)
项目源码地址:https://github.com/linwalker/render-html-to-pdf
[h3]html2canvas[/h3]简介我们可以直接在浏览器端使用html2canvas,对整个或局部页面进行“截图”。但这并不是真的截图,而是通过遍历页面DOM结构,收集所有元素信息及相应样式,渲染出canvas image。
由于html2canvas只能将它能处理的生成canvas image,因此渲染出来的结果并不是100%与原来一致。但它不需要服务器参与,整个图片都由客户端浏览器生成,使用很方便。
使用使用的API也很简洁,下面代码可以将某个元素渲染成canvas:
    1. html2canvas(element, {
    复制代码
    1.     onrendered: function(canvas) {
    复制代码
    1.         // canvas is the final rendered  element
    复制代码
    1.     }
    复制代码
    1. });
    复制代码
通过onrendered方法,可以将生成的canvas进行回调,比如插入到页面中:
    1. html2canvas(element, {
    复制代码
    1.     onrendered: function(canvas) {
    复制代码
    1.        document.body.appendChild(canvas);
    复制代码
    1.     }
    复制代码
    1. });
    复制代码
做个小例子(demo1)代码如下:
    1. [/code]
    2. [*][code]  
    复制代码
    1.     html2canvas example
    复制代码
    1.     ...
    复制代码
    1.   
    复制代码
    1.   
    复制代码
    1.    
    复制代码
    1.       
    复制代码
    1.         
    复制代码
    1.           one
    复制代码
    1.           ...
    复制代码
    1.         
    复制代码
    1.       
    复制代码
    1.    
    复制代码
    1.    
    复制代码
    1.       
    复制代码
    1.         it is a title
    复制代码
    1.         Stone Giant
    复制代码
    1.         ...
    复制代码
    1.      
    复制代码
    1.       
    复制代码
    1.         
    复制代码
    1.         Stone Giant
    复制代码
    1.         Coming ...
    复制代码
    1.         以一团石头...
    复制代码
    1.       
    复制代码
    1.    
    复制代码
    1.     write by linwalker @2017
    复制代码
    1.    
    复制代码
    1.    
    复制代码
    1.         html2canvas(document.body, {
    复制代码
    1.           onrendered:function(canvas) {
    复制代码
    1.             document.body.appendChild(canvas)
    复制代码
    1.           }
    复制代码
    1.         })
    复制代码
    1.    
    复制代码
    1.   
    复制代码
    1. [/code]
    2. [/list]这个例子将页面body中的元素渲染成canvas,并插入到body中。
    3. [h3][b]jsPDF[/b][/h3]jsPDF库可以用于浏览器端生成PDF。
    4. 文字生成PDF使用方法如下:
    5. [list=1][*][code]// 默认a4大小,竖直方向,mm单位的PDF
    复制代码
    1. var doc = new jsPDF();
    复制代码
    1. [/code]
    2. [*][code]// 添加文本‘Download PDF’
    复制代码
    1. doc.text('Download PDF!', 10, 10);
    复制代码
    1. doc.save('a4.pdf');
    复制代码
图片生成PDF使用方法如下:
    1. // 三个参数,第一个方向,第二个单位,第三个尺寸格式
    复制代码
    1. var doc = new jsPDF('landscape','pt',[205, 115])
    复制代码
    1. [/code]
    2. [*][code]// 将图片转化为dataUrl
    复制代码
    1. var imageData = ‘...’;
    复制代码
    1. [/code]
    2. [*][code]doc.addImage(imageData, 'PNG', 0, 0, 205, 115);
    复制代码
    1. doc.save('a4.pdf');
    复制代码
文字与图片生成PDF
    1. // 三个参数,第一个方向,第二个尺寸,第三个尺寸格式
    复制代码
    1. var doc = new jsPDF('landscape','pt',[205, 155])
    复制代码
    1. [/code]
    2. [*][code]// 将图片转化为dataUrl
    复制代码
    1. var imageData = ‘...’;
    复制代码
    1. [/code]
    2. [*][code]//设置字体大小
    复制代码
    1. doc.setFontSize(20);
    复制代码
    1. [/code]
    2. [*][code]//10,20这两参数控制文字距离左边,与上边的距离
    复制代码
    1. doc.text('Stone', 10, 20);
    复制代码
    1. [/code]
    2. [*][code]// 0, 40, 控制文字距离左边,与上边的距离
    复制代码
    1. doc.addImage(imageData, 'PNG', 0, 40, 205, 115);
    复制代码
    1. doc.save('a4.pdf')
    复制代码
生成pdf需要把转化的元素添加到jsPDF实例中,也有添加html的功能,但某些元素无法生成在pdf中,因此可以使用html2canvas + jsPDF的方式将页面转成pdf。通过html2canvas将遍历页面元素,并渲染生成canvas,然后将canvas图片格式添加到jsPDF实例,生成pdf。
[h1]html2canvas + jsPDF[/h1]单页将demo1的例子修改下:
    1. [/code]
    2. [*][code]
    复制代码
    1.       var downPdf = document.getElementById("renderPdf");
    复制代码
    1.       downPdf.onclick = function() {
    复制代码
    1.           html2canvas(document.body, {
    复制代码
    1.               onrendered:function(canvas) {
    复制代码
    1. [/code]
    2. [*][code]                  //返回图片dataURL,参数:图片格式和清晰度(0-1)
    复制代码
    1.                   var pageData = canvas.toDataURL('image/jpeg', 1.0);
    复制代码
    1. [/code]
    2. [*][code]                  //方向默认竖直,尺寸ponits,格式a4[595.28,841.89]
    复制代码
    1.                   var pdf = new jsPDF('', 'pt', 'a4');
    复制代码
    1. [/code]
    2. [*][code]                  //addImage后两个参数控制添加图片的尺寸,此处将页面高度按照a4纸宽高比列进行压缩
    复制代码
    1.                   pdf.addImage(pageData, 'JPEG', 0, 0, 595.28, 592.28/canvas.width * canvas.height );
    复制代码
    1. [/code]
    2. [*][code]                  pdf.save('stone.pdf');
    复制代码
    1. [/code]
    2. [*][code]              }
    复制代码
    1.           })
    复制代码
    1.       }
    复制代码
    1. [/code]
    2. [/list]如果页面内容根据a4比例转化后高度超过a4纸高度呢,生成的pdf会怎么样?会分页吗?
    3. 你可以试试,验证一下自己的想法。
    4. jsPDF提供了一个很有用的API, [code]addPage()
    复制代码
    ,我们可以通过
    1. pdf.addPage()
    复制代码
    ,来添加一页pdf,然后通过
    1. pdf.addImage(...)
    复制代码
    ,将图片赋予这页pdf来显示。
    那么我们如何确定哪里分页?
    这个问题好回答,我们可以设置一个
    1. pageHeight
    复制代码
    ,超过这个高度的内容放入下一页pdf。
    来捋一下思路,将html页面内容生成canvas图片,通过
    1. addImage
    复制代码
    将第一页图片添加到pdf中,超过一页内容,通过
    1. addPage()
    复制代码
    添加pdf页数,然后再通过
    1. addImage
    复制代码
    将下一页图片添加到pdf中。
    嗯~,很好!巴特,难道没有发现问题吗?
    这个方法实现的前提是 — — 我们能根据
    1. pageHeight
    复制代码
    先将整页内容生成的canvas图片分割成对应的小图片,然后一个萝卜一个坑,一页一页
    1. addImage
    复制代码
    进去。
    What? 想一想我们的canvas是肿么来的,不用拉上去,直接看下面:
      1. html2canvas(document.body, {
      复制代码
      1.     onrendered:function(canvas) {
      复制代码
      1.      //it is here we handle the canvas
      复制代码
      1.     }
      复制代码
      1. })
      复制代码
    这里的
    1. body
    复制代码
    就是要生成canvas的元素对象,一个元素生成一个canvas;那么我们需要一页一页的canvas,也就是说。。。
    你觉得可能吗? 我觉得不太现实,按这思路要获取页面上不同位置的DOM元素,然后通过
    1. htnl2canvas(element,option)
    复制代码
    来处理,先不说能不能刚好在每个
    1. pageHeight
    复制代码
    的位置刚好找到一个DOM元素,就算找到了,这样做累不累。
    累的话 :)可以看看下面这种方法。
    多页我提供的思路是我们只生成一个canvas,对就一个,转化元素就是你要转成pdf内容的母元素,在这篇demo里就是
    1. body
    复制代码
    了;其他不变,也是超过一页内容就
    1. addPage
    复制代码
    ,然后
    1. addImage
    复制代码
    ,只不过这里添加的是同一个canvas。
    当然这样做只会出现多页重复的pdf,那到底怎么实现正确分页显示。其实主要利用了jsPDF的两点:
    • 超过jsPDF实例格式尺寸的内容不显示(
      1. varpdf=newjsPDF('','pt','a4');
      复制代码
      demo中就是a4纸的尺寸)
    • addImage有两个参数可以控制图片在pdf中的位置
    虽然每一页pdf上显示的图片是相同的,但我们通过调整图片的位置,产生了分页的错觉。以第二页为例,将竖直方向上的偏移设置为
    1. -841.89
    复制代码
    即一张a4纸的高度,又因为超过a4纸高度范围的图片不显示,所以第二页显示了图片竖直方向上[841.89,1682.78]范围内的内容,这就得到了分页的效果,以此类推。
    还是看代码吧:
      1. html2canvas(document.body, {
      复制代码
      1.   onrendered:function(canvas) {
      复制代码
      1. [/code]
      2. [*][code]      var contentWidth = canvas.width;
      复制代码
      1.       var contentHeight = canvas.height;
      复制代码
      1. [/code]
      2. [*][code]      //一页pdf显示html页面生成的canvas高度;
      复制代码
      1.       var pageHeight = contentWidth / 592.28 * 841.89;
      复制代码
      1.       //未生成pdf的html页面高度
      复制代码
      1.       var leftHeight = contentHeight;
      复制代码
      1.       //页面偏移
      复制代码
      1.       var position = 0;
      复制代码
      1.       //a4纸的尺寸[595.28,841.89],html页面生成的canvas在pdf中图片的宽高
      复制代码
      1.       var imgWidth = 595.28;
      复制代码
      1.       var imgHeight = 592.28/contentWidth * contentHeight;
      复制代码
      1. [/code]
      2. [*][code]      var pageData = canvas.toDataURL('image/jpeg', 1.0);
      复制代码
      1. [/code]
      2. [*][code]      var pdf = new jsPDF('', 'pt', 'a4');
      复制代码
      1. [/code]
      2. [*][code]      //有两个高度需要区分,一个是html页面的实际高度,和生成pdf的页面高度(841.89)
      复制代码
      1.       //当内容未超过pdf一页显示的范围,无需分页
      复制代码
      1.       if (leftHeight < pageHeight) {
      复制代码
      1.       pdf.addImage(pageData, 'JPEG', 0, 0, imgWidth, imgHeight );
      复制代码
      1.       } else {
      复制代码
      1.           while(leftHeight > 0) {
      复制代码
      1.               pdf.addImage(pageData, 'JPEG', 0, position, imgWidth, imgHeight)
      复制代码
      1.               leftHeight -= pageHeight;
      复制代码
      1.               position -= 841.89;
      复制代码
      1.               //避免添加空白页
      复制代码
      1.               if(leftHeight > 0) {
      复制代码
      1.                 pdf.addPage();
      复制代码
      1.               }
      复制代码
      1.           }
      复制代码
      1.       }
      复制代码
      1. [/code]
      2. [*][code]      pdf.save('content.pdf');
      复制代码
      1.   }
      复制代码
      1. })
      复制代码
    两边留边距修改imgWidth,并且在addImage时x方向参数设置你要的边距,具体代码如下:
      1. var imgWidth = 555.28;
      复制代码
      1. var imgHeight = 555.28/contentWidth * contentHeight;
      复制代码
      1. ...
      复制代码
      1. pdf.addImage(pageData, 'JPEG', 20, 0, imgWidth, imgHeight );
      复制代码
      1. ...
      复制代码
      1. pdf.addImage(pageData, 'JPEG', 20, position, imgWidth, imgHeight);
      复制代码


    【关于投稿】


    如果大家有原创好文投稿,请直接给公号发送留言。


    ① 留言格式:
    【投稿】+《 文章标题》+ 文章链接

    ② 示例:
    【投稿】《不要自称是程序员,我十多年的 IT 职场总结》:http://blog.jobbole.com/94148/

    ③ 最后请附上您的个人简介哈~






    觉得本文对你有帮助?请分享给更多人
    关注「前端大全」,提升前端技能


    [url=http://mp.weixin.qq.com/s?__biz=MzAxODE2MjM1MA==&mid=2651554450&idx=1&sn=8602a6ce4f2f2c8b12a8ad04e0092c2c&chksm=80255553b752dc45b5c26642d62e39c285f4851473fdc294e7b642c94ea0051874fb484faafa&scene=21#wechat_redirect][/url]
分享到 :
0 人收藏
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

积分:195
帖子:39
精华:0
期权论坛 期权论坛
发布
内容

下载期权论坛手机APP