vue生成pdf文件步驟及pdf分頁隔斷處理方法
更新時間:2024年02月22日 09:49:56 作者:小新2
最近遇到一個需求,需要把內(nèi)容下載生成pdf文件,但轉(zhuǎn)換過程中內(nèi)容總是會被截斷,就很難受,從網(wǎng)上找到了解決辦法分享給大家,這篇文章主要給大家介紹了關(guān)于vue生成pdf文件步驟及pdf分頁隔斷處理的相關(guān)資料,需要的朋友可以參考下
一:PDF生成步驟
1.1 引入所需插件命令
npm install html2canvas npm install jspdf
1.2 在utils中創(chuàng)建pdf.js文件
pdf.js完整代碼
// 頁面導(dǎo)出為pdf格式 //title表示為下載的標(biāo)題,html表示document.querySelector('#myPrintHtml')
import html2Canvas from 'html2canvas';
import JsPDF from 'jspdf';
function htmlPdf(title, html) {
html2Canvas(html, {
allowTaint: false,
taintTest: false,
logging: false,
useCORS: true,
dpi: window.devicePixelRatio * 1,
scale: 1 // 按比例增加分辨率
}).then(canvas => {
var pdf = new JsPDF('p', 'mm', 'a4'); // A4紙,縱向
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);// 可能內(nèi)容不足一頁
// 用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();// 如果后面還有內(nèi)容,添加一個空頁
}
// delete page;
}
// 保存文件
pdf.save(title + '.pdf');
});
}
export default htmlPdf;
1.3 html文件
html完整代碼
<template>
<div>
<button @click="generate">點(diǎn)擊按鈕導(dǎo)出pdf</button>
<div id="pdf-details" class="pdf-details">
<h1>div里寫需要生成的PDF內(nèi)容的代碼</h1>
<table border="1" align="center" cellspacing="0" cellpadding="30">
<tr class="pdf-details">
<th style="width:80px">日期</th>
<th style="width:100px">姓名</th>
<th>地址</th>
</tr>
<tr v-for="(item,index) in tableData" :key="index" class="pdf-details">
<td>{{item.date}}</td>
<td>{{item.name}}</td>
<td>{{item.address}}</td>
</tr>
</table>
</div>
</div>
</template>
<script>
import htmlPdf from '@/utils/pdf.js';
export default {
name: 'pdfGenerate',
data () {
return {
tableData: [
{date: '2016-05-02',name: '王大虎',address: '上海市普陀區(qū)金沙江路 111 弄'},
{date: '2016-05-04',name: '王二虎',address: '上海市普陀區(qū)金沙江路 112 錘'},
{date: '2016-05-01',name: '王三虎',address: '上海市普陀區(qū)金沙江路 113 子'},
{date: '2016-05-03',name: '王四虎',address: '上海市普陀區(qū)金沙江路 114 呢'},
{date: '2016-05-03',name: '王沒虎',address: '上海市普陀區(qū)金沙江路 110 弄'}
]
}
},
methods: {
generate () {
var TypeName = '生成的PDF';
// 注意這一句
htmlPdf(TypeName, document.querySelector('#pdf-details'));
}
}
}
</script>
1.4 生成演示

二:PDF分頁隔斷處理
- 在我們?nèi)粘i_發(fā)中生成pdf會遇到內(nèi)容顯示出現(xiàn)隔斷問題
- 接下來我會通過代碼來處理這個問題

- 思路為獲取每一行的高度然后根據(jù)頁高度來計算此行內(nèi)容是否超出
- 超出則在上一級兄弟元素添加一個空白塊來撐高pad內(nèi)容
2.1 html代碼
- 需要生成的pdf每一行添加一個相同的class作為標(biāo)識,此次增加的class為“pdf-details”
- 調(diào)用htmlPdf方法時需要獲取class為“pdf-details”的元素傳給pdf.js
html完整代碼
<template>
<div>
<button @click="generate">點(diǎn)擊按鈕導(dǎo)出pdf</button>
<div id="pdf-details" >
<h1 class="pdf-details" style="margin:0;padding:20px">div里寫需要生成的PDF內(nèi)容的代碼</h1>
<div class="pdf-details" style="height:495px">占位</div>
<table border="1" align="center" cellspacing="0" cellpadding="30">
<tr class="pdf-details">
<th style="width:80px">日期</th><th style="width:100px">姓名</th><th>地址</th>
</tr>
<tr v-for="(item,index) in tableData" :key="index" class="pdf-details">
<td>{{item.date}}</td><td>{{item.name}}</td><td>{{item.address}}</td>
</tr>
</table>
</div>
</div>
</template>
<script>
//引用生成pdf方法
import htmlPdf from '@/utils/pdf.js';
export default {
name: 'pdfGenerate',
data () {
return {
tableData: [
{date: '2016-05-02',name: '王大虎',address: '上海市普陀區(qū)金沙江路 111 弄'},
{date: '2016-05-04',name: '王二虎',address: '上海市普陀區(qū)金沙江路 112 錘'},
{date: '2016-05-01',name: '王三虎',address: '上海市普陀區(qū)金沙江路 113 子'},
{date: '2016-05-03',name: '王四虎',address: '上海市普陀區(qū)金沙江路 114 呢'},
{date: '2016-05-03',name: '測試超長隔斷',address: '這是漢字但生成時有隔斷,我現(xiàn)在要處理他;這是漢字但生成時有隔斷,我現(xiàn)在要處理他;'},
{date: '2016-05-03',name: '王沒虎',address: '上海市普陀區(qū)金沙江路 110 弄'},
{date: '2016-05-03',name: '王沒虎',address: '上海市普陀區(qū)金沙江路 110 啊'},
{date: '2016-05-03',name: '王沒虎',address: '上海市普陀區(qū)金沙江路 110 測'},
{date: '2016-05-03',name: '王沒虎',address: '上海市普陀區(qū)金沙江路 110 試'}
]
}
},
methods: {
generate () {
var TypeName = '生成的PDF';
const lableList = document.getElementsByClassName('pdf-details'); // 注意這一句
htmlPdf(TypeName, document.querySelector('#pdf-details'), lableList);
}
}
}
</script>
<style>
td{
padding: 20px;
}
</style>2.2 pdf.js文件
- 首先獲取每一行需要生成的元素來進(jìn)行遍歷
- 根據(jù)當(dāng)前元素以及遍歷過的元素總高度來計算出當(dāng)前元素添加到pdf中是否超出一頁
- 超出則添加一個空白塊 代替當(dāng)前元素 當(dāng)前元素移動到第二頁
pdf.js完整代碼
// 頁面導(dǎo)出為pdf格式 //title表示為下載的標(biāo)題,html表示document.querySelector('#myPrintHtml')
import html2Canvas from 'html2canvas';
import JsPDF from 'jspdf';
var noTableHeight = 0; //table外的元素高度
function htmlPdf(title, html, lableList, type) {// type傳有效值pdf則為橫版
if (lableList) {
const pageHeight = Math.floor(277 * html.scrollWidth / 190) +20; //計算pdf高度
for (let i = 0; i < lableList.length; i++) { //循環(huán)獲取的元素
const multiple = Math.ceil((lableList[i].offsetTop + lableList[i].offsetHeight) / pageHeight); //元素的高度
if (isSplit(lableList, i, multiple * pageHeight)) { //計算是否超出一頁
var _H = '' //向pdf插入空白塊的內(nèi)容高度
if(lableList[i].localName !== 'tr'){ //判斷是不是表格里的內(nèi)容
_H = multiple * pageHeight - (lableList[i].offsetTop + lableList[i].offsetHeight);
}else{
_H = multiple * pageHeight - (lableList[i].offsetTop + lableList[i].offsetHeight + noTableHeight) +20;
}
var newNode = getFooterElement(_H); //向pdf插入空白塊的內(nèi)容
const divParent = lableList[i].parentNode; // 獲取該div的父節(jié)點(diǎn)
const next = lableList[i].nextSibling; // 獲取div的下一個兄弟節(jié)點(diǎn)
// 判斷兄弟節(jié)點(diǎn)是否存在
if (next) {
// 存在則將新節(jié)點(diǎn)插入到div的下一個兄弟節(jié)點(diǎn)之前,即div之后
divParent.insertBefore(newNode, next);
} else {
// 否則向節(jié)點(diǎn)添加最后一個子節(jié)點(diǎn)
divParent.appendChild(newNode);
}
}
}
}
html2Canvas(html, {
allowTaint: false,
taintTest: false,
logging: false,
useCORS: true,
dpi: window.devicePixelRatio * 1,
scale: 1 // 按比例增加分辨率
}).then(canvas => {
var pdf = new JsPDF('p', 'mm', 'a4'); // A4紙,縱向
var ctx = canvas.getContext('2d');
var a4w = type ? 277 : 190; var a4h = type ? 190 : 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);// 可能內(nèi)容不足一頁
// 用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();// 如果后面還有內(nèi)容,添加一個空頁
}
// delete page;
}
// 保存文件
pdf.save(title + '.pdf');
});
}
// pdf截斷需要一個空白位置來補(bǔ)充
function getFooterElement(remainingHeight, fillingHeight = 0) {
const newNode = document.createElement('div');
newNode.style.background = '#ffffff';
newNode.style.width = 'calc(100% + 8px)';
newNode.style.marginLeft = '-4px';
newNode.style.marginBottom = '0px';
newNode.classList.add('divRemove');
newNode.style.height = (remainingHeight + fillingHeight) + 'px';
return newNode;
}
function isSplit(nodes, index, pageHeight) {
// 判斷是不是tr 如果不是高度存起來
// 表格里的內(nèi)容要特殊處理
// tr.offsetTop 是tr到table表格的高度
// 所以計算高速時候要把表格外的高度加起來
// 生成的pdf沒有表格了這里可以不做處理 直接計算就行
if(nodes[index].localName !== 'tr'){ //判斷元素是不是tr
noTableHeight+= nodes[index].clientHeight
}
if(nodes[index].localName !== 'tr'){
return nodes[index].offsetTop + nodes[index].offsetHeight < pageHeight && nodes[index + 1] && nodes[index + 1].offsetTop + nodes[index + 1].offsetHeight > pageHeight;
} else {
return nodes[index].offsetTop + nodes[index].offsetHeight + noTableHeight < pageHeight && nodes[index + 1] && nodes[index + 1].offsetTop + nodes[index + 1].offsetHeight + noTableHeight > pageHeight;
}
}
export default htmlPdf;
2.3 效果

總結(jié)
到此這篇關(guān)于vue生成pdf文件步驟及pdf分頁隔斷處理方法的文章就介紹到這了,更多相關(guān)vue生成pdf及分頁隔斷內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Vue Element前端應(yīng)用開發(fā)之常規(guī)Element界面組件
在我們開發(fā)BS頁面的時候,往往需要了解常規(guī)界面組件的使用,小到最普通的單文本輸入框、多文本框、下拉列表,以及按鈕、圖片展示、彈出對話框、表單處理、條碼二維碼等等,本篇隨筆基于普通表格業(yè)務(wù)的展示錄入的場景介紹這些常規(guī)Element組件的使用2021-05-05
Vue3+TypeScript實現(xiàn)PDF預(yù)覽組件
這篇文章主要為大家詳細(xì)介紹了如何基于Vue3+TypeScript實現(xiàn)PDF預(yù)覽組件,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-04-04
vue 實現(xiàn)一個簡單的全局調(diào)用彈窗案例
這篇文章主要介紹了vue 實現(xiàn)一個簡單的全局調(diào)用彈窗案例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-09-09
Vue 菜單欄點(diǎn)擊切換單個class(高亮)的方法
今天小編就為大家分享一篇Vue 菜單欄點(diǎn)擊切換單個class(高亮)的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-08-08

