使用jspdf生成pdf報表
由于前臺html已經(jīng)動態(tài)生成報表,而且,前臺有一個功能,一個date range組件,當你拖動的時候,報表會在不提交到后臺的情況下動態(tài)變化。
因此需要用到js生成生報表:
用到的組件:
jquery.js jspdf.js canvg.js html2canvas.js jspdf.plugin.autotable.js
前臺動態(tài)生成的chart現(xiàn)在一般是用的html5的canvas或者是svg,很不幸運,我遇到的是svg, 如果是flash沒研究過。
由于報表還需要保持原h(huán)tml頁面的外觀,但是又不是整個html,真正需要轉(zhuǎn)換成pdf報表的是:html+svg
前提:jsPDF 支持html,但支持不是很好,當你用一個html直接生成pdf的時候,其實他只保留了html里面文本,樣式,結構都丟失了。
比如:table就丟失了。
jsPDF不支持svg導入。
思路:將svg轉(zhuǎn)換成canvas,再將html+canvas轉(zhuǎn)換成canvas,再使用html2canvas將canvas轉(zhuǎn)換成圖片,最后將圖片寫入pdf.
table的話就使用:jspdf.plugin.autotable.js
firefox: html2canvas不能直接將svg+html轉(zhuǎn)換成canvas --> 先將svg元素轉(zhuǎn)換成canvas --> html+canvas轉(zhuǎn)換成canvas
chrome: html2canvas可以直接將svg+html轉(zhuǎn)換成canvas
//將指定節(jié)點下面的所有svg轉(zhuǎn)換成canvas
//這里需要:canvg.js
function svg2canvas (targetElem) {
var nodesToRecover = [];
var nodesToRemove = [];
var svgElem = targetElem.find('svg');
svgElem.each(function(index, node) {
var parentNode = node.parentNode;
var svg = node.outerHTML;
var canvas = document.createElement('canvas');
canvg(canvas, svg);
nodesToRecover.push({
parent: parentNode,
child: node
});
parentNode.removeChild(node);
nodesToRemove.push({
parent: parentNode,
child: canvas
});
parentNode.appendChild(canvas);
});
}
//這里是將html(文本)在一個iframe里面打開
//主要是排除其它元素的干擾導致不成功,之前是一直輸出不成功,所示才使用iframe
//這段代碼是官網(wǎng)摳下來的。
//還有個問題就是:如果將頁面的chart轉(zhuǎn)換成canvas了,那web頁面報表動態(tài)變化的功能將丟失。
function openWithIframe(html){
var iframe = document.createElement('iframe');
iframe.setAttribute("id", "myFrmame");
var $iframe = $(iframe);
$iframe.css({
'visibility': 'hidden', 'position':'static', 'z-index':'4'
}).width($(window).width()).height($(window).height());
$('body').append(iframe);
var ifDoc = iframe.contentWindow.document;
//這里做是將報表使用到的css重新寫入到iframe中,根據(jù)自身的需要
var style = "<link href='/javax.faces.resource/css/auth.css.jsf' rel='stylesheet' type='text/css'>";
style+="<link href='/javax.faces.resource/css/common.css.jsf' rel='stylesheet' type='text/css'>";
style+="<link href='/javax.faces.resource/css/dc.css.jsf' rel='stylesheet' type='text/css'>";
html = "<!DOCTYPE html><html><head>"+style+"</head><body>"+html+"</body></html>"
ifDoc.open();
ifDoc.write(html);
ifDoc.close();
/*
//這里做一些微調(diào),根據(jù)自身的需要
var fbody = $iframe.contents().find("body");
fbody.find("#chart-center").removeAttr("width");
fbody.find(".page-container").css("width", "370px");
fbody.find(".center-container").css("width", "600px");
fbody.find("#severity-chart svg").attr("width", "370");
fbody.find("#status-chart svg").attr("width", "300");
*/
return fbody;
}
//導出pdf
function exportAsPDF(){
//得到要導出pdf的html根節(jié)點
var chartCenter = document.getElementById("chart-center").outerHTML;
var fbody = openWithIframe(chartCenter);
svg2canvas(fbody);
//html2canvas官網(wǎng)的標準方法
html2canvas(fbody, {
onrendered: function(canvas) {
//var myImage = canvas.toDataURL("image/png");
//alert(myImage);
//window.open(myImage);
/*
canvas.toBlob(function(blob) {
saveAs(blob, "report.png");
}, "image/png");
*/
//將圖片轉(zhuǎn)換成:base64編碼的jpg圖片。
var imgData = canvas.toDataURL('image/jpeg');
//alert(imgData);
//l:橫向, p:縱向
var doc = new jsPDF('l', 'pt', 'a3');
//var doc = new jsPDF('p', 'mm', [290, 210]);
//var doc = new jsPDF();//默認是A4,由于我的報表比較大,所以專門設置了尺寸。
doc.setFontSize(22);
doc.setFontType("bolditalic");
doc.text(500, 30, "Ticket Report"); //x:500, y:30
doc.addImage(imgData, 'jpeg', 10, 60); //寫入位置:x:10, y:60
doc.addPage(); //新建一頁
//這里就是把將table寫入到pdf里面。
var res = doc.autoTableHtmlToJson(document.getElementById("tickets-summary-table"), true);
doc.autoTable(res.columns, res.data);
doc.save('ticket.report_'+new Date().getTime()+'.pdf');
$('#myFrmame').remove(); //最后將iframe刪除
},
background:"#fff", //這里給生成的圖片默認背景,不然的話,如果你的html根節(jié)點沒有設置背景的話,會用黑色填充。
allowTaint: true //避免一些不識別的圖片干擾,默認為false,遇到不識別的圖片干擾則會停止處理html2canvas
});
};
以上所述就是本文的全部內(nèi)容了,希望大家能夠喜歡。
相關文章
jQuery使用中可能被XSS攻擊的一些危險環(huán)節(jié)提醒
XSS指的是跨站腳本攻擊,比如人們常說的向$傳入字符串或者字符串轉(zhuǎn)換可執(zhí)行函數(shù)等一些安全方面值得注意的細節(jié),下面就為大家整理了jQuery使用中可能被XSS攻擊的一些危險環(huán)節(jié)提醒2016-05-05
輕松學習jQuery插件EasyUI EasyUI實現(xiàn)拖動基本操作
這篇文章主要幫大家輕松學習jQuery插件EasyUI,并利用EasyUI實現(xiàn)拖動基本操作,文章并提供了一個學校課程表簡單實例,感興趣的小伙伴們可以參考一下2015-11-11
jQuery使用after()方法在元素后面添加多項內(nèi)容的方法
這篇文章主要介紹了jQuery使用after()方法在元素后面添加多項內(nèi)容的方法,實例分析了jQuery中after方法的功能及在元素后面添加內(nèi)容的技巧,非常具有實用價值,需要的朋友可以參考下2015-03-03
jQuery插件passwordStrength密碼強度指標詳解
這篇文章主要為大家詳細介紹了jQuery插件passwordStrength密碼強度指標實現(xiàn)代碼,感興趣的小伙伴們可以參考一下2016-06-06
jQuery插件ajaxFileUpload實現(xiàn)異步上傳文件效果
jQuery插件AjaxFileUpload用來實現(xiàn)ajax文件上傳,該插件使用非常簡單,接下來我們來看下用AjaxFileUpload插件實現(xiàn)文件上傳的方法,有需要的小伙伴可以參考下2015-04-04

