推广

前端生成pdf,jspdf+html2Canvas的使用(vue)

iseeyu2年前 (2024-02-21)推广116

image

1、安装jspdf: npm install jspdf –save

2、安装html2Canvas: npm install –save html2canvas

二、代码

1. html

这里使用iframe嵌套需要转成pdf的html文件;

如果pdf内容是动态的,可以将html丢给后端,让后端使用freemarker ftl模板语言或是啥其他的模板语言更改即可,然后将整个html作为字符串传给前端,前端再使用iframe 的 srcdoc属性将内容渲染出来。

image

2. 生成单页

不存在分页内容裂开的情况,但对样式呈现不太友好,有时会出现内容偏移情况,时好时坏的,经过不断的尝试,发生偏移的时候,可以将html2Canvas的width写死,但是值具体是多少,需要你自己猜

// 单页pdf:css高度自适应即可(此处用的一个css,为了实现多页pdf同时不让分页分割图片,css中写死了每页的高度.a4page)
    getOnePdf() {
      var title = "单页报告";
      var dom = document.getElementById("pdf-container");  // 生成pdf的html内容
      html2Canvas(dom, {
        allowTaint: true,
        scrollY: 0,
        scrollX: 0,
        useCORS: true, // 开启跨院
        width: 1000, // 这个宽度,太扯淡了,固定1000,防止偏移!
        height: dom.offsetHeight
      }).then(function(canvas) {
        var contentWidth = canvas.width;
        var contentHeight = canvas.height;
        var pdfWidth = (contentWidth + 10) / 2 * 0.75;
        var pdfHeight = (contentHeight + 200) / 2 * 0.75; // 500为底部留白
        var imgWidth = pdfWidth;
        var imgHeight = contentHeight / 2 * 0.75; //内容图片这里不需要留白的距离
        var pageData = canvas.toDataURL("image/jpeg", 1.0);
        var pdf = new JsPDF("", "pt", [pdfWidth, pdfHeight]);
        pdf.addImage(pageData, "jpeg", 0, 0, imgWidth, imgHeight);
        pdf.save(title + ".pdf");
      });
    },

image

3. 生成多页

分页会出现题:比如图片裂开,表格的某一行从中间断开了等等….我用的笨方法手动去调整iframe的宽、高+html2Canvas的width、height+contentWidth、contentHeight,一点点试出来合适的值(应该有更好的方法,但是技术有限)

// 多页pdf-转换后的样式需要手动调整iframe的宽、高+html2Canvas的width、height+contentWidth、contentHeight
    getManyPdf() {
      let _this = this;
      var dom = document.getElementById("pdf-container"); // 生成pdf的html内容
      html2Canvas(dom, {
        allowTaint: true,
        scrollY: 0,
        scrollX: 0,
        useCORS: true, // 开启跨院
        width: 1000, // 这个宽度,太扯淡了,固定1000,防止偏移!
        height: dom.offsetHeight
      }).then(function(canvas) {
        var contentWidth = 900; // 宽高写死,强制分页
        var contentHeight = 5060;

        //一页pdf显示html页面生成的canvas高度;
        var pageHeight = contentWidth / 592.28 * 841.89;
        //未生成pdf的html页面高度
        var leftHeight = contentHeight;
        //页面偏移
        var position = 0;
        //a4纸的尺寸[595.28,841.89],html页面生成的canvas在pdf中图片的宽高
        var imgWidth = 595.28;
        var imgHeight = 3400; // 为了分页,所以写死。592.28 / contentWidth * contentHeight会导致图片被分割

        var pageData = canvas.toDataURL("image/jpeg", 1.0);
        var pdf = new JsPDF("", "pt", "a4");
        //有两个高度需要区分,一个是html页面的实际高度,和生成pdf的页面高度(841.89)
        //当内容未超过pdf一页显示的范围,无需分页
        if (leftHeight < pageHeight) {
          pdf.addImage(pageData, "JPEG", 0, 0, imgWidth, imgHeight);
        } else {
          while (leftHeight > 0) {
            //arg3-->距离左边距;arg4-->距离上边距;arg5-->宽度;arg6-->高度
            pdf.addImage(pageData, "JPEG", 0, position, imgWidth, imgHeight);
            leftHeight -= pageHeight;
            position -= 841.89;
            //避免添加空白页
            if (leftHeight > 0) {
              pdf.addPage();
            }
          }
        }
        pdf.save(`多页报告}.pdf`);
      });
    }

三、所有代码

  <div class="home-index-container" style="width: 100%; height: 100%; overflow: auto">
    <el-button @click="getOnePdf" type="primary">下载单页</el-button>
    <el-button @click="getManyPdf" type="primary">下载多页</el-button>

    <!-- 转Pdf的容器 -->
    <div id="pdf-container">
      <iframe src="/exportPDF/exportPDF.html" width="1000" height="5200"></iframe>
    </div>
  </div>
</template>

<script>
import html2Canvas from "html2canvas";
import JsPDF from "jspdf";
export default {
  data() {
    return {};
  },
  methods: {
    // 单页pdf:css高度自适应即可(此处用的一个css,为了实现多页pdf同时不让分页分割图片,css中写死了每页的高度.a4page)
    getOnePdf() {
      var title = "单页报告";
      var dom = document.getElementById("pdf-container");  // 生成pdf的html内容
      html2Canvas(dom, {
        allowTaint: true,
        scrollY: 0,
        scrollX: 0,
        useCORS: true, // 开启跨院
        width: 1000, // 这个宽度,太扯淡了,固定1000,防止偏移!
        height: dom.offsetHeight
      }).then(function(canvas) {
        var contentWidth = canvas.width;
        var contentHeight = canvas.height;
        var pdfWidth = (contentWidth + 10) / 2 * 0.75;
        var pdfHeight = (contentHeight + 200) / 2 * 0.75; // 500为底部留白
        var imgWidth = pdfWidth;
        var imgHeight = contentHeight / 2 * 0.75; //内容图片这里不需要留白的距离
        var pageData = canvas.toDataURL("image/jpeg", 1.0);
        var pdf = new JsPDF("", "pt", [pdfWidth, pdfHeight]);
        pdf.addImage(pageData, "jpeg", 0, 0, imgWidth, imgHeight);
        pdf.save(title + ".pdf");
      });
    },
    // 多页pdf-转换后的样式需要手动调整iframe的宽、高+html2Canvas的width、height+contentWidth、contentHeight
    getManyPdf() {
      let _this = this;
      var dom = document.getElementById("pdf-container"); // 生成pdf的html内容
      html2Canvas(dom, {
        allowTaint: true,
        scrollY: 0,
        scrollX: 0,
        useCORS: true, // 开启跨院
        width: 1000, // 这个宽度,太扯淡了,固定1000,防止偏移!
        height: dom.offsetHeight
      }).then(function(canvas) {
        var contentWidth = 900; // 宽高写死,强制分页
        var contentHeight = 5060;

        //一页pdf显示html页面生成的canvas高度;
        var pageHeight = contentWidth / 592.28 * 841.89;
        //未生成pdf的html页面高度
        var leftHeight = contentHeight;
        //页面偏移
        var position = 0;
        //a4纸的尺寸[595.28,841.89],html页面生成的canvas在pdf中图片的宽高
        var imgWidth = 595.28;
        var imgHeight = 3400; // 为了分页,所以写死。592.28 / contentWidth * contentHeight会导致图片被分割

        var pageData = canvas.toDataURL("image/jpeg", 1.0);
        var pdf = new JsPDF("", "pt", "a4");
        //有两个高度需要区分,一个是html页面的实际高度,和生成pdf的页面高度(841.89)
        //当内容未超过pdf一页显示的范围,无需分页
        if (leftHeight < pageHeight) {
          pdf.addImage(pageData, "JPEG", 0, 0, imgWidth, imgHeight);
        } else {
          while (leftHeight > 0) {
            //arg3-->距离左边距;arg4-->距离上边距;arg5-->宽度;arg6-->高度
            pdf.addImage(pageData, "JPEG", 0, position, imgWidth, imgHeight);
            leftHeight -= pageHeight;
            position -= 841.89;
            //避免添加空白页
            if (leftHeight > 0) {
              pdf.addPage();
            }
          }
        }
        pdf.save(`多页报告}.pdf`);
      });
    }
  },
  created() {
    
  },
  mounted() {}
};
</script>

遗留问题

1. 生成多页pdf,分页内容裂开的问题

扫描二维码推送至手机访问。

版权声明:本文由西安泽虎代运营发布,如需转载请注明出处。

转载请注明出处https://www.0291.com.cn/post/57465.html

相关文章

大一市场营销怎么学得更好?

大一市场营销怎么学得更好?

大一怎么学得更好?...

淘宝新手开店有多久扶持期,70岁了能在淘宝开店不(淘宝扶持期 重要吗)

淘宝新手开店有多久扶持期,70岁了能在淘宝开店不(淘宝扶持期 重要吗)

如果是全新开的店,那么从开店之日算起,扶持期是90天。当然现在淘宝由于卖家过多,扶持期流量也不明显,需要看自己有经营能力。...

如果有一种不储存能量,只储存负熵的电池,会不会十分有用 ...

如果有一种不储存能量,只储存负熵的电池,会不会十分有用 ...

DNA就很接近你說的分子。使用的時候主要是取其信息而非能量。...

百度推广减肥广告效果如何?如何提高软文加粉量?

随着人们水平的提高及的诱惑,再加上生活饮食作息的不规律,导致很多用户的体重超重,特别是对一些女性来说,看着别人的“魔鬼身材”总是非常羡慕,可以说是大多数女性的一生不变的“事业”。另外,肥胖会引发很多病变,所以随着人们对健康的重视,都会尝试很多减肥产品来进行瘦身,换句话说,减...

这就是所谓的富人思维?

就好像有人说的,全国每人给你一块钱,你拥有十几亿,是亿万富翁?这有问题吗?没问题,但这是富人思维吗?完全不是,它是一点都不现实,首先,为什么每个人会给你一块钱,假如你给每个人说几句祝福语,给你一块钱,那没问题,但是你为什么会给十多亿人都说祝福语?就算你没日没夜说一辈子也说不...

拼多多直播带货攻略:如何在拼多多上玩转直播带货?

拼多多直播带货攻略:如何在拼多多上玩转直播带货?

随着电商行业的飞速发展,直播带货已经成为了一种新的销售模式。而在众多电商平台中,拼多多的直播带货功能更是吸引了众多商家和主播的目光。那么,拼多多上面怎么直播带货呢?我将为大家详细解析拼多多直播带货的攻略,让你轻松玩转直播带货。 我们要明确直播带货的目标。在拼多多上进行直播带货,主要是...

现在,非常期待与您的又一次邂逅

我们努力让每一部企业宣传片和抖音短视频成为商业大片