一、vue2导出PDF使用步骤

1、安装html2canvas,将页面html转换成图片

npm install --save html2canvas

卸载:

npm uninstall html2canvas

指定版本安装:

npm install --save html2canvas@1.0.0-rc.4

2、安装jspdf,将图片生成pdf

npm install jspdf --save

3、定义全局函数

在指定位置创建一个htmlToPdf.js文件,我个人习惯放在('src/components/utils/htmlToPdf')

// 导出页面为PDF格式
import html2Canvas from 'html2canvas'
import JsPDF from 'jspdf'
export default{
  install(Vue, options) {
    Vue.prototype.exportToPdf = function(domId, fileName) {
      html2Canvas(document.getElementById(domId), {
        allowTaint: true
      }).then(function(canvas) {
        const contentWidth = canvas.width
        const contentHeight = canvas.height
        const pageHeight = contentWidth / 592.28 * 841.89
        let leftHeight = contentHeight
        let position = 0
        const imgWidth = 595.28
        const imgHeight = 592.28 / contentWidth * contentHeight
        const pageData = canvas.toDataURL('image/jpeg', 1.0)
        const PDF = new JsPDF('', 'pt', 'a4')
        if (leftHeight < pageHeight) {
          PDF.addImage(pageData, 'JPEG', 0, 0, imgWidth, imgHeight)
        } else {
          while (leftHeight > 0) {
            PDF.addImage(pageData, 'JPEG', 0, position, imgWidth, imgHeight)
            leftHeight -= pageHeight
            position -= 841.89
            if (leftHeight > 0) {
              PDF.addPage()
            }
          }
        }
        PDF.save(fileName + '.pdf')
      }
      )
    }
  }
}

4、在main.js中全局注册

import htmlToPdf from '@/components/utils/htmlToPdf'
Vue.use(htmlToPdf)

5、在需要的导出的页面调用我们的exportToPdf方法即可

<el-button style="float: right; background-color: lavender" @click="exportToPdf('pdfDom01', '企业信息')" >导出PDF</el-button>

二、html2canvas导出PDFtextarea多行问题

1、将textarea元素标签换成div

<el-form-item label="药材批号及批量定制原则:" >
        <div class="textarea">{{sVp.mmMcp}}</div>
<!--        <el-input v-model="sVp.mmMcp" :autosize="{ minRows: 4, maxRows: 4}" type="textarea" class="inputClass" size="small" style="width: 700px"/>-->
      </el-form-item>

2、设置样式,重点word-wrap: break-word;

.textarea {
    width: 700px;
    height: auto;
    min-height: 100px;
    resize: none;
    border: solid;
    border-radius: 5px;
    overflow: auto;
    border-color: #E5E7E8;
    /*color: rgb(181, 17, 4);*/
    display: inline-block;
    /* 允许长单词换行到下一行: */
    word-wrap: break-word;
  }

三、vue3导出PDF使用步骤

1、安装html2canvas,将页面html转换成图片

npm install --save html2canvas

2、安装jspdf,将图片生成pdf

npm install jspdf --save

3、定义全局函数

在指定位置创建一个htmlToPdf.js文件,我个人习惯放在('src/components/utils/htmlToPdf')

// 页面导出为pdf格式
import html2Canvas from 'html2canvas';
import jsPDF from 'jspdf';

const htmlToPdf = {
    getPdf(title, id) {
        html2Canvas(
            document.querySelector(id),
            {
                allowTaint: false,
                taintTest: false,
                logging: false,
                useCORS: true,
                dpi: window.devicePixelRatio * 4, //将分辨率提高到特定的DPI 提高四倍
                scale: 4, //按比例增加分辨率
            }
        ).then((canvas) => {
            var pdf = new jsPDF('p', 'mm', 'a4'); //A4纸,纵向
            var ctx = canvas.getContext('2d'),
                a4w = 190,
                a4h = 272, //A4大小,210mm x 297mm,四边各保留10mm的边距,显示区域190x277
                imgHeight = Math.floor((a4h * canvas.width) / a4w), //按A4显示比例换算一页图像的像素高度
                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 htmlToPdf;

 

5、使用

<template>
  <div>
    <div id="pdfDom">
      <p>这是一段文本</p>
      <img src="../assets/logo.png">
    </div>
    <el-button size="mini" type="primary" @click="pdfFunc" :loading="loadingag">
      转成pdf
    </el-button>
  </div>
</template>

<script>
import htmlToPdf from "@/components/utils/htmlToPdf";

export default {
  data() {
    return {
      loadingFlag: false
    }
  },
  methods: {
    // 容器id="pdfCompany"
    pdfFunc() {
      this.loadingFlag = true;    // 动画加载事件
      // 调用htmlToPdf工具函数
      htmlToPdf.getPdf('文件标题',"#pdfDom");
      // 定时器模拟按钮loading动画的时间
      setTimeout(() => {
        this.loadingFlag = false;
      }, 1000);
    }
  }
}
</script>

效果如下:

点击"转成pdf",

但是会出现分页隔断问题,如下所示:

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

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

调用

<template>
  <div>
    <button @click="generate">点击按钮导出pdf</button>
    <div id="pdf-details" >
      ......
        <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 '@/components/utils/htmlToPdf.js';
export default {
  name: 'pdfGenerate',
  data () {
    return {
      tableData: [
        
      ]
    }
  },
  methods: {
    generate () {
      var TypeName = '生成的PDF';
      const lableList = document.getElementsByClassName('pdf-details');   // 注意这一句
      htmlPdf(TypeName, document.querySelector('#pdf-details'), lableList);
    }
  }
}
</script>

但是还是存在分页隔断的问题

posted on 2023-10-19 10:46  周文豪  阅读(6506)  评论(0编辑  收藏  举报