vue2利用html2canvas+jspdf動態(tài)生成多頁PDF方式
vue2用html2canvas+jspdf動態(tài)生成多頁PDF
業(yè)務需求中,前端把頁面上的內容導出為圖片,pdf,excel是常有的事。
當然,這些工作后端也是能做。秉著前端是萬能的理念,今天就站在前端的角度上,來實現(xiàn)將頁面內容導出為pdf,實現(xiàn)指定div內容導出,并自動拼接成一個多頁的PDF文件
1.PDF相關依賴
vue2腳手架項目中,關于PDF的依賴老生常用的就那么幾個。
pdfmake,vue-print-nb,v-html5-renderer,print-js,html2pdf.js,vue-pdf。
總結起來無非就是達到將內容生成pdf,預覽pdf這兩個業(yè)務場景。
本篇講的是通過html2canvas+jspdf來動態(tài)指定div內容轉化為pdf
html2canvas
可以將 HTML 畫布(canvas)捕捉為圖像,然后 jspdf
可以將這些圖像轉換為 PDF 格式
第一步先安裝依賴
npm install jspdf html2canvas --save # 或者 yarn add jspdf html2canvas
2.效果展示及源碼
導出后一個div為一個pdf頁面,不足的留空白。超出的會自動加頁面補充。
假設,你首頁想指定一個圖片作為封面,中間的一大段內容從第二頁開始,自動往后分頁。那就設置兩個div就行了。
PDF生成后的大致效果:
下面附上代碼,就一個vue文件,方便各位拿來即用,能立馬看到效果:
<template> <div id="pdfinsurancepdf"> <div style="width: 490pt;margin: auto;"> <div class="title" style="padding:20pt"> <el-button type="primary" @click="downpdf">下載PDF</el-button> <el-divider></el-divider> </div> <div id="page1"> <img src="@/assets/1.jpg" style="width:100%;height:auto" alt="封面" /> </div> <div id="page2"> <template> <el-table :data="tableData" style="width: 100%"> <el-table-column prop="date" label="日期" width="180"> </el-table-column> <el-table-column prop="name" label="姓名" width="180"> </el-table-column> <el-table-column prop="address" label="地址"> </el-table-column> </el-table> </template> <div> <pre> 當你在凝視著網(wǎng)頁的時候 網(wǎng)頁也正在凝視著你 流情|導航站 Who is liuqing? 種一棵樹最好的時間是十年前,然后就是現(xiàn)在——致想做還未來得及做的你 How lucky to meet you! 你好,這里是流情 Liuqing ,26歲,技術宅,對有趣的世界和可能有趣的你感到好奇,熱愛IT和國風文化, 從事互聯(lián)網(wǎng)產品/Web開發(fā)相關工作。希望能與你在比特之海的繁星之下相見! </pre> </div> </div> <div id="page3"> <div> <img src="@/assets/2.jpg" style="width:100%;height:auto" alt="結束頁" /> </div> </div> </div> </div> </template>
<script> import html2canvas from 'html2canvas'; import JsPDF from 'jspdf'; export default { name:'insurancepdf', data() { return { compony:"pdf導出多頁測試", tableData: [{ date: '2016-05-02', name: '王小虎', address: '上海市普陀區(qū)金沙江路 1518 弄' }, { date: '2016-05-04', name: '王小虎', address: '上海市普陀區(qū)金沙江路 1517 弄' }, { date: '2016-05-01', name: '王小虎', address: '上海市普陀區(qū)金沙江路 1519 弄' }, { date: '2016-05-03', name: '王小虎', address: '上海市普陀區(qū)金沙江路 1516 弄' }] } }, moutned() { console.log(123) }, methods:{ downpdf(){ const pdf = new JsPDF('p', 'mm', 'a4'); var divIds = ['page1', 'page2','page3']; const promises = divIds.map(divId => { const div = document.getElementById(divId); return html2canvas(div, { allowTaint: false, taintTest: false, logging: false, useCORS: true, dpi: window.devicePixelRatio * 1, scale: 2 //畫布分辨率 }); }); Promise.all(promises).then(canvases => { canvases.forEach((canvas, index) => { if (index > 0) { pdf.addPage(); } this.addCanvasToPdf(pdf, canvas); }); pdf.save('pdf導出多頁測試.pdf'); }).catch(error => { console.error('Error generating PDF:', error); }); }, addCanvasToPdf(pdf,canvas){ // 接下來使用canvas創(chuàng)建PDF var ctx = canvas.getContext('2d'); var a4w = 190; var a4h = 277; // A4大小,210mm x 297mm,四邊各保留10mm的邊距,顯示區(qū)域190x277 var imgHeight = Math.floor(a4h * canvas.width / a4w); // 按A4顯示比例換算一頁圖像的像素高度 var renderedHeight = 0; while (renderedHeight < canvas.height) { var page = document.createElement('canvas'); page.width = canvas.width; page.height = Math.min(imgHeight, canvas.height - renderedHeight);// 可能內容不足一頁 // 用getImageData剪裁指定區(qū)域,并畫到前面創(chuàng)建的canvas對象中 page.getContext('2d').putImageData(ctx.getImageData(0, renderedHeight, canvas.width, Math.min(imgHeight, canvas.height - renderedHeight)), 0, 0); pdf.addImage(page.toDataURL('image/jpeg', 1.0), 'JPEG', 10, 10, a4w, Math.min(a4h, a4w * page.height / page.width)); // 添加圖像到頁面,保留10mm邊距 renderedHeight += imgHeight; if (renderedHeight < canvas.height) { pdf.addPage();// 如果后面還有內容,添加一個空頁 } // delete page; } } } } </script>
<style scoped> #pdfinsurancepdf{ width: 100%; background: #fff; } span{ line-height: 30pt; } </style>
總結
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
Vue-router不允許導航到當前位置(/path)錯誤原因以及修復方式
本文主要介紹了Vue-router不允許導航到當前位置(/path)錯誤原因以及修復方式,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-09-09