js完成html轉(zhuǎn)換pdf
發(fā)表時間:2024-05-12 來源:明輝站整理相關(guān)軟件相關(guān)文章人氣:
[摘要]在項目開發(fā)的時候,我們偶爾或者是經(jīng)常會遇到一些難題,關(guān)于用js吧html頁面轉(zhuǎn)換成pdf也是一個難題,意思是說相當(dāng)于把整個頁面截下來,然后保存成pdf。其實,能夠?qū)崿F(xiàn)html轉(zhuǎn)pdf的方法還是挺多的,大概有以下幾種:1、大部分瀏覽器就有這個功能。然而我們客戶要的可不是這個,人家要的是能夠在系統(tǒng)中主...
在項目開發(fā)的時候,我們偶爾或者是經(jīng)常會遇到一些難題,關(guān)于用js吧html頁面轉(zhuǎn)換成pdf也是一個難題,意思是說相當(dāng)于把整個頁面截下來,然后保存成pdf。
其實,能夠?qū)崿F(xiàn)html轉(zhuǎn)pdf的方法還是挺多的,大概有以下幾種:
1、大部分瀏覽器就有這個功能。然而我們客戶要的可不是這個,人家要的是能夠在系統(tǒng)中主動觸發(fā)的導(dǎo)出為pdf功能,所以這種方案pass。
2、利用第三方工具。我找到了一種利用wkhtmltopdf這種工具來導(dǎo)出的方案,自己在我們的項目中試了一下,效果不好,而且對svg圖片的支持也不行。pass。
3、還有一種是利用iText類后臺生成java文件。但因為需要導(dǎo)出的這個頁面是動態(tài)頁面,而且直接把頁面?zhèn)鹘o后臺會丟失大量樣式,所以還是pass。
最后沒什么好的辦法,只能退而求其次,想著要不先把html頁面轉(zhuǎn)成圖片,再把圖片導(dǎo)出為pdf。因為要支持用戶導(dǎo)出下載,而且要保留樣式,所以最好是純js前端實現(xiàn)。
html轉(zhuǎn)canvas的話,就用html2canvas這個js,這個網(wǎng)上介紹比較多了,這里就不廢話了。
比較麻煩的是svg圖片,直接用html2canvas無法把svg標(biāo)簽的內(nèi)容轉(zhuǎn)成canvas,最后查了一圈資料后,鎖定了canvg這個js。canvg是谷歌的一個插件,可以將svg標(biāo)簽內(nèi)容轉(zhuǎn)成canvas。具體到我們的項目,還有一個難點,就是如何把glyphicons這種字體圖標(biāo)也轉(zhuǎn)成canvas,因為在不同瀏覽器下對這種字體圖標(biāo)的支持是完全不一樣的。最后找到的方法是用char code來替換這些字體圖標(biāo),重新繪制成canvas。由canvas生成圖片不用廢話。由圖片生成pdf用jsPDF實現(xiàn)。 折騰了大半天,總算把整個流程打通了,接下來一步一步貼上代碼。
第一步:把對應(yīng)dom節(jié)點里所有的svg元素替換成canvas
svg2canvas: function(targetElem) {
var svgElem = targetElem.find('svg');
svgElem.each(function(index, node) {
var parentNode = node.parentNode;
//由于現(xiàn)在的IE不支持直接對svg標(biāo)簽node取內(nèi)容,所以需要在當(dāng)前標(biāo)簽外面套一層div,通過外層div的innerHTML屬性來獲取
var tempNode = document.createElement('div');
tempNode.appendChild(node);
var svg = tempNode.innerHTML;
var canvas = document.createElement('canvas');
//轉(zhuǎn)換
canvg(canvas, svg);
parentNode.appendChild(canvas);
});
}
第二步:把glyphicons字體轉(zhuǎn)成canvas。如果項目中沒有用到glyphicons字體圖標(biāo),可忽略這一步
glyphicons2canvas: function(targetElem, fontClassName, fontFamilyName) {
var iconElems = targetElem.find('.' + fontClassName);
iconElems.each(function(index, inconNode) {
var fontSize = $(inconNode).css("font-size");
var iconColor = $(inconNode).css("color");
var styleContent = $(inconNode).attr('style');
//去掉"px"
fontSize = fontSize.replace("px", "");
var charCode = getCharCodeByGlyphiconsName(iconName);
var myCanvas = document.createElement('canvas');
//把canva寬高各增加2是為了顯示圖標(biāo)完整
myCanvas.width = parseInt(fontSize) + 2;
myCanvas.height = parseInt(fontSize) + 2;
myCanvas.style = styleContent;
var ctx = myCanvas.getContext('2d');
//設(shè)置繪圖內(nèi)容的顏色
ctx.fillStyle = iconColor;
//設(shè)置繪圖的字體大小以及font-family的名字
ctx.font = fontSize + 'px ' + fontFamilyName;
ctx.fillText(String.fromCharCode(charCode), 1, parseInt(fontSize) + 1);
$(inconNode).replaceWith(myCanvas);
});
}
//根據(jù)glyphicons/glyphicon圖標(biāo)的類名獲取到對應(yīng)的char code
getCharCodeByGlyphiconsName: function(iconName) {
switch (iconName) {
case("glyphicons-resize-full"):
return "0xE216";
case ("glyphicons-chevron-left"):
return "0xE225";
default:
return "";
}
}
第三步:html轉(zhuǎn)canvas轉(zhuǎn)圖片再轉(zhuǎn)pdf。
html2canvas($("#myExportArea"), {
onrendered: function(canvas) {
var imgData = canvas.toDataURL('image/jpeg');
var img = new Image();
img.src = imgData;
//根據(jù)圖片的尺寸設(shè)置pdf的規(guī)格,要在圖片加載成功時執(zhí)行,之所以要*0.225是因為比例問題
img.onload = function() {
//此處需要注意,pdf橫置和豎置兩個屬性,需要根據(jù)寬高的比例來調(diào)整,不然會出現(xiàn)顯示不完全的問題
if (this.width > this.height) {
var doc = new jsPDF('l', 'mm', [this.width * 0.225, this.height * 0.225]);
} else {
var doc = new jsPDF('p', 'mm', [this.width * 0.225, this.height * 0.225]);
}
doc.addImage(imgData, 'jpeg', 0, 0, this.width * 0.225, this.height * 0.225);
//根據(jù)下載保存成不同的文件名
doc.save('report_pdf_' + new Date().getTime() + '.pdf');
}
},
background: "#fff",
//這里給生成的圖片默認背景,不然的話,如果你的html根節(jié)點沒設(shè)置背景的話,會用黑色填充。
allowTaint: true //避免一些不識別的圖片干擾,默認為false,遇到不識別的圖片干擾則會停止處理html2canvas
});
雖然這種方法可以將html頁面轉(zhuǎn)換成pdf樣式,但是生成的pdf效果明顯不如正常截圖來的清晰,如果大家有更好的辦法,歡迎指點。
相關(guān)推薦:
C#怎么將 HTML轉(zhuǎn)換為圖片或 PDF?
關(guān)于C#如何將Word轉(zhuǎn)換成PDF的方法匯總
使用HTML生成一個PDF實例代碼
以上就是js實現(xiàn)html轉(zhuǎn)成pdf的詳細內(nèi)容,更多請關(guān)注php中文網(wǎng)其它相關(guān)文章!
網(wǎng)站建設(shè)是一個廣義的術(shù)語,涵蓋了許多不同的技能和學(xué)科中所使用的生產(chǎn)和維護的網(wǎng)站。