vue生成pdf文件步骤及pdf分页隔断处理方法

 更新时间:2024年02月22日 09:49:56   作者:小新²  
最近遇到一个需求,需要把内容下载生成pdf文件,但转换过程中内容总是会被截断,就很难受,从网上找到了解决办法分享给大家,这篇文章主要给大家介绍了关于vue生成pdf文件步骤及pdf分页隔断处理的相关资料,需要的朋友可以参考下

一:PDF生成步骤

1.1 引入所需插件命令

npm install html2canvas   

npm install jspdf

1.2 在utils中创建pdf.js文件

pdf.js完整代码

// 页面导出为pdf格式 //title表示为下载的标题,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的边距,显示区域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剪裁指定区域,并画到前面创建的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;
      }
      // 保存文件
      pdf.save(title + '.pdf');
    });
  }

export default htmlPdf;

1.3 html文件

html完整代码

<template>
  <div>
    <button @click="generate">点击按钮导出pdf</button>
    <div id="pdf-details" class="pdf-details">
      <h1>div里写需要生成的PDF内容的代码</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: '上海市普陀区金沙江路 111 弄'},
        {date: '2016-05-04',name: '王二虎',address: '上海市普陀区金沙江路 112 锤'}, 
        {date: '2016-05-01',name: '王三虎',address: '上海市普陀区金沙江路 113 子'},
        {date: '2016-05-03',name: '王四虎',address: '上海市普陀区金沙江路 114 呢'},
        {date: '2016-05-03',name: '王没虎',address: '上海市普陀区金沙江路 110 弄'}
      ]
    }
  },
  methods: {
    generate () {
      var TypeName = '生成的PDF';
      // 注意这一句
      htmlPdf(TypeName, document.querySelector('#pdf-details'));
    }
  }
}
</script>

1.4 生成演示

二:PDF分页隔断处理

  • 在我们日常开发中生成pdf会遇到内容显示出现隔断问题
  • 接下来我会通过代码来处理这个问题

  • 思路为获取每一行的高度然后根据页高度来计算此行内容是否超出
  • 超出则在上一级兄弟元素添加一个空白块来撑高pad内容

2.1 html代码

  • 需要生成的pdf每一行添加一个相同的class作为标识,此次增加的class为“pdf-details”
  • 调用htmlPdf方法时需要获取class为“pdf-details”的元素传给pdf.js

html完整代码

<template>
  <div>
    <button @click="generate">点击按钮导出pdf</button>
    <div id="pdf-details" >
      <h1 class="pdf-details" style="margin:0;padding:20px">div里写需要生成的PDF内容的代码</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: '上海市普陀区金沙江路 111 弄'},
        {date: '2016-05-04',name: '王二虎',address: '上海市普陀区金沙江路 112 锤'}, 
        {date: '2016-05-01',name: '王三虎',address: '上海市普陀区金沙江路 113 子'},
        {date: '2016-05-03',name: '王四虎',address: '上海市普陀区金沙江路 114 呢'},
        {date: '2016-05-03',name: '测试超长隔断',address: '这是汉字但生成时有隔断,我现在要处理他;这是汉字但生成时有隔断,我现在要处理他;'},
        {date: '2016-05-03',name: '王没虎',address: '上海市普陀区金沙江路 110 弄'},
        {date: '2016-05-03',name: '王没虎',address: '上海市普陀区金沙江路 110 啊'},
        {date: '2016-05-03',name: '王没虎',address: '上海市普陀区金沙江路 110 测'},
        {date: '2016-05-03',name: '王没虎',address: '上海市普陀区金沙江路 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文件

  • 首先获取每一行需要生成的元素来进行遍历
  • 根据当前元素以及遍历过的元素总高度来计算出当前元素添加到pdf中是否超出一页
  • 超出则添加一个空白块 代替当前元素 当前元素移动到第二页

pdf.js完整代码

// 页面导出为pdf格式 //title表示为下载的标题,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++) { //循环获取的元素
      const multiple = Math.ceil((lableList[i].offsetTop + lableList[i].offsetHeight) / pageHeight); //元素的高度
      if (isSplit(lableList, i, multiple * pageHeight)) { //计算是否超出一页
        var _H = '' //向pdf插入空白块的内容高度
        if(lableList[i].localName !== 'tr'){ //判断是不是表格里的内容
          _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插入空白块的内容
        const divParent = lableList[i].parentNode; // 获取该div的父节点
        const next = lableList[i].nextSibling; // 获取div的下一个兄弟节点
        // 判断兄弟节点是否存在
        if (next) {
          // 存在则将新节点插入到div的下一个兄弟节点之前,即div之后
          divParent.insertBefore(newNode, next);
        } else {
          // 否则向节点添加最后一个子节点
          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的边距,显示区域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剪裁指定区域,并画到前面创建的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;
    }
    // 保存文件
    pdf.save(title + '.pdf');
  });
}
  // pdf截断需要一个空白位置来补充
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 如果不是高度存起来
  // 表格里的内容要特殊处理
  // 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 效果

总结 

到此这篇关于vue生成pdf文件步骤及pdf分页隔断处理方法的文章就介绍到这了,更多相关vue生成pdf及分页隔断内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • ElementUI 修改默认样式的几种办法(小结)

    ElementUI 修改默认样式的几种办法(小结)

    这篇文章主要介绍了ElementUI 修改默认样式的几种办法(小结),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-07-07
  • Vue.js组件间通信方式总结【推荐】

    Vue.js组件间通信方式总结【推荐】

    组件之间通信分为三种:父-子;子-父;跨级组件通信。下面,就组件间如何通信做一些总结。需要的朋友可以参考下
    2018-11-11
  • ElementUI下拉组件el-select一次从后端获取选项并设置默认值方式

    ElementUI下拉组件el-select一次从后端获取选项并设置默认值方式

    这篇文章主要介绍了ElementUI下拉组件el-select一次从后端获取选项并设置默认值方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-01-01
  • 解决vue this.$forceUpdate() 处理页面刷新问题(v-for循环值刷新等)

    解决vue this.$forceUpdate() 处理页面刷新问题(v-for循环值刷新等)

    这篇文章主要介绍了解决vue this.$forceUpdate() 处理页面刷新问题(v-for循环值刷新等),解决方法是使用this.$forceUpdate()强制刷新,文章给大家分享了代码案例,需要的朋友参考下吧
    2018-07-07
  • Vue关于自定义事件的$event传参问题

    Vue关于自定义事件的$event传参问题

    这篇文章主要介绍了Vue关于自定义事件的$event传参问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-10-10
  • Element中Slider滑块的具体使用

    Element中Slider滑块的具体使用

    这篇文章主要介绍了Element中Slider滑块的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-07-07
  • VueJs打包之后遇到的坑及解决

    VueJs打包之后遇到的坑及解决

    这篇文章主要介绍了VueJs打包之后遇到的坑及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-05-05
  • 详解vue组件化开发-vuex状态管理库

    详解vue组件化开发-vuex状态管理库

    本篇文章主要介绍了详解vue组件化开发-vuex状态管理库,具有一定的参考价值,有兴趣的可以了解一下。
    2017-04-04
  • vue项目中如何配置env环境的实现

    vue项目中如何配置env环境的实现

    本文主要介绍了vue项目中如何配置env环境的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-07-07
  • vue bus全局事件中心简单Demo详解

    vue bus全局事件中心简单Demo详解

    ue-bus 提供了一个全局事件中心,并将其注入每一个组件,你可以像使用内置事件流一样方便的使用全局事件。这篇文章给大家介绍了vue bus全局事件中心简单Demo,需要的朋友参考下吧
    2018-02-02

最新评论