项目源码地址:https://github.com/linwalker/render-html-to-pdf
html2canvas
简介
我们可以直接在浏览器端利用html2canvas,对全体或局部页面进行“截图”。但这并不是真的截图,而是通过遍历页面DOM构造,网络所有元素信息及相应样式,渲染出canvas image。
由于html2canvas只能将它能处理的天生canvas image,因此渲染出来的结果并不是100%与原来同等。但它不须要做事器参与,全体图片都由客户端浏览器天生,利用很方便。
利用
利用的API也很简洁,下面代码可以将某个元素渲染成canvas:
通过onrendered方法,可以将天生的canvas进行回调,比如插入到页面中:
做个小例子(demo1)代码如下:
这个例子将页面body中的元素渲染成canvas,并插入到body中。
nvas,并插入到body中。
jsPDF
jsPDF库可以用于浏览器端天生PDF。
笔墨天生PDF
利用方法如下:
图片天生PDF
利用方法如下:
笔墨与图片天生PDF
天生pdf须要把转化的元素添加到jsPDF实例中,也有添加html的功能,但某些元素无法天生在pdf中,因此可以利用html2canvas + jsPDF的办法将页面转成pdf。通过html2canvas将遍历页面元素,并渲染天生canvas,然后将canvas图片格式添加到jsPDF实例,天生pdf。
html2canvas + jsPDF
单页
将demo1的例子修正下:
如果页面内容根据a4比例转化后高度超过a4纸高度呢,天生的pdf会怎么样?会分页吗?
你可以试试,验证一下自己的想法。
jsPDF供应了一个很有用的API, addPage(),我们可以通过 pdf.addPage(),来添加一页pdf,然后通过 pdf.addImage(...),将图片授予这页pdf来显示。
那么我们如何确定哪里分页?
这个问题好回答,我们可以设置一个 pageHeight,超过这个高度的内容放入下一页pdf。
来捋一下思路,将html页面内容天生canvas图片,通过 addImage将第一页图片添加到pdf中,超过一页内容,通过 addPage()添加pdf页数,然后再通过 addImage将下一页图片添加到pdf中。
嗯~,很好!
巴特,难道没有创造问题吗?
这个方法实现的条件是 — — 我们能根据 pageHeight先将整页内容天生的canvas图片分割成对应的小图片,然后一个萝卜一个坑,一页一页 addImage进去。
What? 想一想我们的canvas是肿么来的,不用拉上去,直接看下面:
这里的 body便是要天生canvas的元素工具,一个元素天生一个canvas;那么我们须要一页一页的canvas,也便是说。。。
你以为可能吗? 我以为不太现实,按这思路要获取页面上不同位置的DOM元素,然后通过 htnl2canvas(element,option)来处理,先不说能不能刚好在每个 pageHeight的位置刚好找到一个DOM元素,就算找到了,这样做累不累。
累的话 :可以看看下面这种方法。
多页
我供应的思路是我们只天生一个canvas,对就一个,转化元素便是你要转成pdf内容的母元素,在这篇demo里便是 body了;其他不变,也是超过一页内容就 addPage,然后addImage,只不过这里添加的是同一个canvas。
当然这样做只会涌现多页重复的pdf,那到底怎么实现精确分页显示。实在紧张利用了jsPDF的两点:
超过jsPDF实例格式尺寸的内容不显示( varpdf=newjsPDF('','pt','a4');demo中便是a4纸的尺寸)addImage有两个参数可以掌握图片在pdf中的位置虽然每一页pdf上显示的图片是相同的,但我们通过调度图片的位置,产生了分页的错觉。以第二页为例,将竖直方向上的偏移设置为 -841.89即一张a4纸的高度,又由于超过a4纸高度范围的图片不显示,以是第二页显示了图片竖直方向上[841.89,1682.78]范围内的内容,这就得到了分页的效果,以此类推。
还是看代码吧:
两边留边距
修正imgWidth,并且在addImage时x方向参数设置你要的边距,详细代码如下:
想要学习JS、HTML、CSS的可以私信帐号“学习互换”加群一起谈论,领取学习视频资料。