Vue頁(yè)面生成PDF的最佳方法推薦
前言
最近項(xiàng)目有個(gè)需求,將系統(tǒng)統(tǒng)計(jì)的數(shù)據(jù)生成分析報(bào)告,然后可以導(dǎo)出成PDF。
這種方法可以有兩種,一種是直接調(diào)用打印,用戶通過瀏覽器提供的打印頁(yè)面手動(dòng)選擇導(dǎo)出PDF。當(dāng)然這種方式兼容性差,且體驗(yàn)不好,顯然不是我們想要的效果。
那么第二種方法的實(shí)現(xiàn)思路是什么呢?
首先生成報(bào)告頁(yè)面,也就是常規(guī)頁(yè)面;
然后將頁(yè)面轉(zhuǎn)換成圖片( 用到的組件 html2canvas );
最后將圖片導(dǎo)出成PDF( 用到的組件 jspdf )。
安裝依賴
npm install --save html2canvas // 頁(yè)面轉(zhuǎn)圖片 npm install jspdf --save // 圖片轉(zhuǎn)pdf
頁(yè)面轉(zhuǎn)圖片
新建一個(gè) index.vue 頁(yè)面:
<template> <div ref="pdf"> 這是待轉(zhuǎn)換的頁(yè)面,點(diǎn)擊 <button @click="handleExport">導(dǎo)出</button> 按鈕,完成導(dǎo)出操作。 </div> </template> <script> import {downloadPDF} from "@/util/pdf.js" //工具方法,導(dǎo)出操作 export default { name: "pdf", data() { return {}; }, methods: { handleExport(){ downloadPDF(this.$refs.pdf) } } }; </script>
頁(yè)面很簡(jiǎn)單,一段話加上一個(gè)導(dǎo)出按鈕。整個(gè)頁(yè)面被設(shè)置別名 (ref=“pdf”),導(dǎo)出的時(shí)候通過別名導(dǎo)出整個(gè)頁(yè)面。
按鈕點(diǎn)擊事件中調(diào)用了一個(gè)工具方法 downloadPDF(),來自于工具類 pdf.js:
import html2canvas from "html2canvas"; export const downloadPDF = page => { html2canvas(page).then(function(canvas) { page.appendChild(canvas); }); };
也很簡(jiǎn)單,首先引入了 html2canvas,定義downloadPDF方法,接受一個(gè)參數(shù)就是要導(dǎo)出的內(nèi)容,可以使用id獲取,也可以使用 ref 。
運(yùn)行結(jié)果:
當(dāng)我們點(diǎn)擊導(dǎo)出按鈕后,在現(xiàn)有的頁(yè)面增加了一行相同的內(nèi)容。通過元素檢查器可以看到增加的這行內(nèi)容實(shí)際是一個(gè) canvas 圖像,里面的按鈕已經(jīng)不可以操作了。
這樣,頁(yè)面轉(zhuǎn)換圖片的過程就完成了。
圖片轉(zhuǎn)PDF
首先給index.vue改造一下,為了讓我們導(dǎo)出的pdf好看一點(diǎn),使用el-table做個(gè)表格,其它的導(dǎo)出邏輯不變:
<template> <div ref="pdf"> <el-table :data="tableData" style="width: 700px"> <el-table-column prop="date" label="時(shí)間" width="180"> </el-table-column> <el-table-column prop="name" label="姓名" width="180"> </el-table-column> <el-table-column prop="address" label="對(duì)我說"> </el-table-column> </el-table> <button @click="handleExport">導(dǎo)出</button> </div> </template> <script> import {downloadPDF} from "@/util/pdf.js" //工具方法,導(dǎo)出操作 export default { name: "pdf", data() { return { tableData: [{ date: '昨天', name: '劉德華', address: '你好帥' }, { date: '今天', name: '郭富城', address: '你最帥' }, { date: '明天', name: '張學(xué)友', address: '你賊帥' }, { date: '每天', name: '黎明', address: '我沒你帥' }] }; }, methods: { handleExport(){ downloadPDF(this.$refs.pdf) } } }; </script>
效果圖:
然后改造一下pdf.js,增加上pdf轉(zhuǎn)換邏輯
import html2canvas from "html2canvas"; import jsPDF from "jspdf"; export const downloadPDF = page => { html2canvas(page).then(function(canvas) { canvas2PDF(canvas); }); }; const canvas2PDF = canvas => { let contentWidth = canvas.width; let contentHeight = canvas.height; let imgHeight = contentHeight; let imgWidth = contentWidth; // 第一個(gè)參數(shù): l:橫向 p:縱向 // 第二個(gè)參數(shù):測(cè)量單位("pt","mm", "cm", "m", "in" or "px") let pdf = new jsPDF("l", "pt"); pdf.addImage( canvas.toDataURL("image/jpeg", 1.0), "JPEG", 0, 0, imgWidth, imgHeight ); pdf.save("導(dǎo)出.pdf"); };
canvas2PDF 方法中創(chuàng)建了jsPDF的實(shí)例,然后添加上一步生成的圖片,最后保存導(dǎo)出pdf文件。
導(dǎo)出的文件效果:
A4打印適配
因?yàn)椴糠中枨笥行枰傻捻?yè)面使用A4紙打印,那么pdf生成時(shí)的寬高尺寸就不能像上面一樣設(shè)定。需要按照A4紙的尺寸比例調(diào)整(其它打印需求同理,使用對(duì)應(yīng)紙張比例即可)
那么改造一下 pdf.js
import html2canvas from "html2canvas"; import jsPDF from "jspdf"; export const downloadPDF = page => { html2canvas(page).then(function(canvas) { canvas2PDF(canvas); }); }; const canvas2PDF = canvas => { let contentWidth = canvas.width; let contentHeight = canvas.height; //a4紙的尺寸[595.28,841.89],html頁(yè)面生成的canvas在pdf中圖片的寬高 let imgWidth = 595.28; let imgHeight = 592.28/contentWidth * contentHeight; // 第一個(gè)參數(shù): l:橫向 p:縱向 // 第二個(gè)參數(shù):測(cè)量單位("pt","mm", "cm", "m", "in" or "px") let pdf = new jsPDF("p", "pt"); pdf.addImage( canvas.toDataURL("image/jpeg", 1.0), "JPEG", 0, 0, imgWidth, imgHeight ); pdf.save("導(dǎo)出.pdf"); };
這里面將 imgWidth 和 imgHeight 按照A4紙比例調(diào)整后,導(dǎo)出的pdf就可以正常比例打印了。
總結(jié)
到此這篇關(guān)于Vue頁(yè)面生成PDF的文章就介紹到這了,更多相關(guān)Vue頁(yè)面生成PDF內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
利用vue + koa2 + mockjs模擬數(shù)據(jù)的方法教程
這篇文章主要給大家介紹了關(guān)于利用vue + koa2 + mockjs模擬數(shù)據(jù)的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2017-11-11Vue?中如何使用?el-date-picker?限制只能選擇當(dāng)天、當(dāng)天之前或當(dāng)天之后日期的方法詳解
在Vue前端開發(fā)中,使用 el-date-picker 組件進(jìn)行日期選擇是常見的需求,有時(shí)候我們需要限制用戶只能選擇當(dāng)天、當(dāng)天之前或當(dāng)天之后的日期,本文將詳細(xì)介紹如何使用 el-date-picker 組件實(shí)現(xiàn)這些限制,讓你能夠輕松應(yīng)對(duì)各種日期選擇場(chǎng)景,需要的朋友可以參考下2023-09-09Vue圖片瀏覽組件v-viewer用法分析【支持旋轉(zhuǎn)、縮放、翻轉(zhuǎn)等操作】
這篇文章主要介紹了Vue圖片瀏覽組件v-viewer用法,結(jié)合實(shí)例形式分析了v-viewer的基本功能與使用方法,包括旋轉(zhuǎn)、縮放、翻轉(zhuǎn)等操作技巧,需要的朋友可以參考下2019-11-11VUE路由動(dòng)態(tài)加載實(shí)例代碼講解
在本文里小編給大家整理了關(guān)于VUE路由動(dòng)態(tài)加載實(shí)例代碼以及相關(guān)知識(shí)點(diǎn),需要的朋友們學(xué)習(xí)下。2019-08-08vuex項(xiàng)目中登錄狀態(tài)管理的實(shí)踐過程
由于狀態(tài)零散地分布在許多組件和組件之間的交互中,大型應(yīng)用復(fù)雜度也經(jīng)常逐漸增長(zhǎng),為了解決這個(gè)問題,Vue 提供 vuex,這篇文章主要給大家介紹了關(guān)于vuex項(xiàng)目中登錄狀態(tài)管理的相關(guān)資料,需要的朋友可以參考下2021-09-09一篇文章告訴你如何實(shí)現(xiàn)Vue前端分頁(yè)和后端分頁(yè)
這篇文章主要為大家介紹了如何實(shí)現(xiàn)Vue前端分頁(yè)和后端分頁(yè),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2021-12-12element-ui中的select下拉列表設(shè)置默認(rèn)值方法
今天小編就為大家分享一篇element-ui中的select下拉列表設(shè)置默認(rèn)值方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-08-08vue3封裝一個(gè)帶動(dòng)畫的關(guān)閉按鈕示例詳解
這篇文章主要為大家介紹了vue3封裝一個(gè)帶動(dòng)畫的關(guān)閉按鈕示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09