vue中使用html2canvas配合jspdf导出pdf(以及在导出时遇到的导出样式问题)

news2025/1/13 14:04:29

指定页面中导出为pdf格式并打包,使用html2canvas先转为图片格式,在利用jspdf转为pdf,最后下载打包为本地压缩包

yarn add html2canvas
yarn add jspdf
1. 注册一个插件并挂载
import html2Canvas from 'html2canvas'
import JsPDF from 'jspdf'
export default {
    install(Vue, options) {
        Vue.prototype.getPdf = function (dom) {
            return new Promise((resolve, reject) => {
                html2Canvas(dom.$el, {
                    allowTaint: true,
                    scale: 2,
                    dpi: 300,
                }).then(function (canvas) {
                    let contentWidth = canvas.width;
                    let contentHeight = canvas.height;
                    let pdfWidth = 595.28; // A4 width in pixels
                    let pdfHeight = (pdfWidth / contentWidth) * contentHeight;
                    let pageData = canvas.toDataURL('image/jpeg', 1.0);

                    let PDF = new JsPDF('p', 'pt', 'a4');
                    let scale = pdfWidth / contentWidth;
                    PDF.addImage(pageData, 'JPEG', 0, 0, pdfWidth, pdfHeight);
                    let remainingHeight = pdfHeight;

                    while (remainingHeight < contentHeight) {
                        PDF.addPage();
                        let offsetY = -remainingHeight * scale;
                        PDF.addImage(pageData, 'JPEG', 0, offsetY, pdfWidth, pdfHeight);
                        remainingHeight += pdfHeight;
                    }
                    PDF.save('export.pdf'); //直接下载
                    // resolve(PDF.output('blob')); //转为blob格式 返回合并下载
                }).catch(reject);
            });
        };
    }
}
2. 页面使用
DownPDFAndFile(){
  this.getPdf('传入ref或者是id')
}

以上正常导出步骤 封装方法 使用 但是如果你的项目中用的是rem布局或者是适配方法,样式或者字体没有显示出来被盖掉请往下看

3.由于在项目中用rem做了适配导致样式变形

在这里插入图片描述

4. 我们可以在html2canvas生成图片时把对应的样式或者添加类名,在生成图片前修改样式,转换为pdf后恢复初始样式
import html2Canvas from 'html2canvas'
import JsPDF from 'jspdf'
/**
 * @param {*} 添加样式类名
 */
function removeChildrenWithClass(parent, className) {
    var children = parent.childNodes;
    if (parent.classList && parent.classList.contains('form_view_item')) {
        parent.childNodes.forEach((item) => {
            item.className += ' is_to_print'
        })
    }
    for (var i = 0; i < children.length; i++) {
        if (children[i].classList && children[i].classList.contains('isprint')) {
            children[i].style.display = 'block'
            if (children[i].classList.contains('is_toprint')) children[i].style.display = 'block'
        }
        if (children[i].classList && children[i].classList.contains('form_view_title')) {
            children[i].style.marginTop = '25px'
        }
        if (children[i].classList && children[i].classList.contains(className)) {
            children[i].style.display = 'none'
        } else if (children[i].childNodes) {
            removeChildrenWithClass(children[i], className);
        }
    }
}
/**
 * @param {*} 恢复初始样式
 */
function restoreOriginalState(parent, className) {
    var children = parent.childNodes;
    if (parent.classList && parent.classList.contains('form_view_item')) {
        parent.childNodes.forEach((item) => {
            item.classList.remove('is_to_print');
        });
    }
    for (var i = 0; i < children.length; i++) {
        if (children[i].classList && children[i].classList.contains(className)) {
            children[i].style.display = '';
        }
        if (children[i].classList && children[i].classList.contains('form_view_title')) {
            children[i].style.marginTop = '';
        }
        if (children[i].classList && children[i].classList.contains('isprint')) {
            children[i].style.display = 'none';
        }
        if (children[i].childNodes) {
            restoreOriginalState(children[i], className);
        }
    }
}
export default {
    install(Vue, options) {
        Vue.prototype.getPdf = function (dom) {
            return new Promise((resolve, reject) => {
                removeChildrenWithClass(dom.$el, 'noprint');
                html2Canvas(dom.$el, {
                    allowTaint: true,
                    //如果不要求清晰度可以去掉
                    scale: 2,  //按比例增加分辨率 
                    dpi: 300, //将分辨率提高到特定的 DPI
                }).then(function (canvas) {
                    restoreOriginalState(dom.$el, 'noprint')
                    let contentWidth = canvas.width;
                    let contentHeight = canvas.height;
                    let pdfWidth = 595.28; // A4 width in pixels
                    let pdfHeight = (pdfWidth / contentWidth) * contentHeight;
                    let pageData = canvas.toDataURL('image/jpeg', 1.0);

                    let PDF = new JsPDF('p', 'pt', 'a4');
                    let scale = pdfWidth / contentWidth;
                    PDF.addImage(pageData, 'JPEG', 0, 0, pdfWidth, pdfHeight);
                    let remainingHeight = pdfHeight;

                    while (remainingHeight < contentHeight) {
                        PDF.addPage();
                        let offsetY = -remainingHeight * scale;
                        PDF.addImage(pageData, 'JPEG', 0, offsetY, pdfWidth, pdfHeight);
                        remainingHeight += pdfHeight;
                    }
                    PDF.save('export.pdf'); //直接下载
                    // resolve(PDF.output('blob')); //转为blob格式 返回合并下载
                }).catch(reject);
            });
        };
    }
}
5. 修改完后导出正常

在这里插入图片描述

6. 如果需求是直接下载以下就忽略,如果需求是打包为本地压缩包请继续
// 引入使用attachDownload方法
// 下载事件
async DownPDFAndFile() {
      const pdfinfo = await this.getPdf('传入ref或者是id')
      const pdfList = [
      //参数字段自行修改 为attachDownload 中的字段对应
				  {
				     id: "...",
				     data: pdfinfo,
				  },
      	];
      const ImageList = []
      const VideoList = []
      const ImageAndVideoAndPdfList = [...pdfList,...ImageList,...VideoList]
      const { downloadStatus } = await this.is_downFile(ImageAndVideoAndPdfList);
      // downloadStatus 这个状态为下载状态 true为完成 可以自行添加业务
    },
// 下载方法
async is_downFile(list) {
      if (list.length > 0) {
        const config = {
          downloadList: list,
          suffix: "病历编辑.zip",
        };
        const { downloadStatus } = await attachDownload(config);
        return { downloadStatus };
      }
    },
7. 打包下载使用的是JSZip 和 FileSaver
yarn add jszip
yarn add file-saver
// attachDownload.js
import JSZip from "jszip";
import FileSaver from "file-saver";
export async function attachDownload(config) {
    const { downloadList, suffix } = config
    const zip = new JSZip();
    const cache = {};
    let downloadStatus = false
    const downloadPromises = downloadList.map(async (item) => {
        try {
            if (item.url) {
                let data;
                if(item.type=='.pdf'){
                    data = item.data;
                }else{
                    data = await getImgArrayBuffer(item.url);
                }
                zip.folder(suffix).file(`${item.Title}_${item.FileID}` + item.type, data, { binary: true });
                cache[item.id] = data;
            } else {
                throw new Error(`文件${item.fileName}地址错误,下载失败`);
            }
        } catch (error) {
            console.error("文件获取失败", error);
        }
    });
    try {
        await Promise.all(downloadPromises);
        const content = await zip.generateAsync({ type: "blob" });
        FileSaver.saveAs(content, suffix);
        downloadStatus = true
        return {
            downloadStatus
        }
    } catch (error) {
        console.error("文件压缩失败", error);
    }
}
async function getImgArrayBuffer(url) {
    const response = await fetch(url);
    if (!response.ok) {
        throw new Error(`请求失败: ${response.status}`);
    }
    return await response.blob();
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1422916.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【零基础学习CAPL】——CAN报文的发送(配合Panel面板单次发送)

&#x1f64b;‍♂️【零基础学习CAPL】系列&#x1f481;‍♂️点击跳转 文章目录 1.概述2.面板创建2.1.新建一个Panel2.2. 在Panel中调出控件窗口2.3.控件添加与配置 3.系统变量创建4.系统变量与Panel绑定5.CAPL脚本实现6.效果 1.概述 使用场景&#xff0c;按下面板按钮同时…

人工智能与机器学习——开启智能时代的里程碑

写在前面 前言人工智能与机器学习的概述监督学习、无监督学习和强化学习的基本原理监督学习&#xff1a;无监督学习&#xff1a;强化学习&#xff1a; 机器学习的算法和方法常见的机器学习算法和方法线性回归&#xff1a;决策树&#xff1a;支持向量机&#xff1a;神经网络&…

个人多域名SSL证书推荐

SSL数字证书和通配符SSL证书、多域名通配符SSL证书一样&#xff0c;可以同时保护多个域名站点&#xff0c;但是它们之间还是存在一些区别。其中&#xff0c;最明显的区别就是它们的保护域名网站的类型和适用场景。今天就随SSL盾小编来了解多域名SSL证书。 1.多域名SSL证书可以…

【C/C++】深入理解--函数重载(什么是函数重载?为什么要有函数重载?)

目录 一、前言 二、 函数重载 &#x1f34e;什么是函数重载 &#x1f350;函数重载的条件 &#x1f347;函数重载的注意点 &#x1f349;为什么要有函数重载 &#x1f353;为何C语言不支持函数重载&#xff0c;反倒C可以&#xff1f; &#x1f4a6; Linux环境下演示函数重…

云纱网签约百望云,联手打造数字化产业闭环

近日&#xff0c;百望云签约广东云纱数字科技有限公司&#xff0c;共建数字化发票管理系统&#xff0c;赋能产业链上下游供应商的协同交易与运营&#xff0c;助力企业实现数字化四流合一交易&#xff0c;打造数字化产业闭环。 云纱网是广东云纱数字科技有限公司依托于深厚的产业…

Halcon 几何测量

文章目录 算子Halcon 计算两点之间的距离案例Halcon 计算点到直线的距离Halcon 计算点到区域的距离Halcon 线到区域的距离Halcon 线到线的距离 算子 distance_pp 两点之间的距离算子 distance_pp( : : Row1, Column1, Row2, Column2 : Distance) Row1 点1的行坐标 Column1 点1的…

Django视图函数技巧,从入门到实战

文章目录 Django视图函数1.request对象的方法2.视图函数的常用的返回对象&#xff08;1&#xff09;response对象&#xff08;2&#xff09;JsonResponse对象&#xff08;3&#xff09;redirect() &#xff1a;给浏览器了一个30x的状态码 3.设置响应头和状态码&#xff08;1&am…

【Vue】前端项目引入阿里图标

【Vue&React】前端项目引入阿里图标 方式11、登录自己的iconfont-阿里巴巴矢量图标库&#xff0c;把需要的图标加入到自己的项目中去&#xff1b;2、加入并进入到项目中去选择Font class 并下载到本地3、得到的文件夹如下4. 把红框中的部分粘贴到自己的项目中&#xff08; …

2万块的郎酒,都是我们惯的

文 | 琥珀酒研社 作者 | 五画 当我看到郎酒拿出快2万一瓶纪念酒的时候&#xff0c;我就知道&#xff0c;这场高价酒的喧嚣和吵闹&#xff0c;又到了一个新的高度。 和别的行业有所不同&#xff0c;白酒很少谈智商税&#xff0c;再高的价格&#xff0c;总有个冠冕堂皇的理由。…

28个炫酷的纯CSS特效动画示例(含源代码)

CSS是网页的三驾马车之一&#xff0c;是对页面布局的总管家&#xff0c;2024年了&#xff0c;这里列出28个超级炫酷的纯CSS动画示例&#xff0c;让您的网站更加炫目多彩。 文章目录 1. 涌动的弹簧效果2. 超逼真的3D篮球弹跳&#xff0c;含挤压弹起模态3. 鼠标放div上&#xff0…

力扣hot100 数组中的第K个最大元素 堆 三路划分

Problem: 215. 数组中的第K个最大元素 文章目录 思路复杂度Code 思路 &#x1f468;‍&#x1f3eb; 参考 复杂度 时间复杂度: O ( n ) O(n) O(n) 空间复杂度: O ( log ⁡ n ) O(\log{n}) O(logn) Code class Solution {public int findKthLargest(int[] nums, int k…

【Django自学】Django入门:如何使用django开发一个web项目(非常详细)

测试机器&#xff1a;windows11 x64 python版本&#xff1a;3.11 一、安装Django 安装步骤非常简单&#xff0c;使用pip安装就行 pip install django安装完成之后&#xff0c;python的 Scripts 文件夹下&#xff0c;会多一个 django-admin.exe (管理创建django项目的工具)。…

Linux其实不难,做个四层板设计的终端设备就搞懂了!

前言 想掌握一个技能&#xff0c;还得靠实践。 作者想学习Linux&#xff0c;于是边学边做&#xff0c;成功开源出了一个——还不错的 四层板设计的 终端设备。 项目成本可压缩至100元以内&#xff0c;便于复刻与参考学习&#xff01; 01 开源项目描述 做了一个基于V3S芯片的…

02-opencv简单实例效果和基本介绍-上

机器视觉概述 机器视觉是人工智能正在快速发展的一个分支。简单说来,机器视觉就是用机器代替人眼来做测量和判断。机器视觉系统是通过机器视觉产品(即图像摄取装置,分CMOS和CCD两种)将被摄取目标转换成图像信号,传送给专用的图像处理系统,得到被摄目标的形态信息,根据像素…

聚类(Clustering)理论

一、无监督学习介绍 在这小节中&#xff0c;我将开始介绍聚类算法&#xff0c;这是我们学习的第一个非监督学习算法&#xff0c;我们将要让计算机学习无标签数据而不是此前的标签数据。那么什么是非监督学习呢&#xff1f;在学习机器学习知识的开始我曾简单地介绍过非监督学习&…

【Algorithms 4】算法(第4版)学习笔记 02 - 1.4 算法分析

文章目录 前言参考目录学习笔记1&#xff1a;科学方法2&#xff1a;观察举例&#xff1a;三数之和3&#xff1a;近似4&#xff1a;增长数量级4.1&#xff1a;二分查找 demo4.2&#xff1a;二分查找代码实现4.3&#xff1a;二分查找比较次数的证明&#xff08;比较次数最多为lgN…

贪吃蛇的简易实现

技术要点 贪吃蛇的简易实现会运用到语⾔函数、枚举、结构体、动态内存管理、预处理指令、链表、Win32 API等,接下来首先简单介绍一下会运用到的东西。 Win32 API介绍 Windows 这个多作业系统除了协调应⽤程序的执⾏、分配内存、管理资源之外, 它同时也是⼀个很⼤的服务中⼼…

嵌入式工程师day15(链表)

内存管理 一.内存管理: 1.malloc void *malloc(size_t size); 功能: 申请堆区空间 参数: size:申请堆区空间的大小 返回值: 返回获得的空间的首地址 失败返回NULL 2.free void free(void *ptr); 功能: 释放…

机器学习-3降低损失(Reducing Loss)

机器学习-3降低损失(Reducing Loss) 学习内容来自&#xff1a;谷歌ai学习 https://developers.google.cn/machine-learning/crash-course/framing/check-your-understanding?hlzh-cn 本文作为学习记录1.降低损失&#xff1a;迭代方法 迭代学习 下图展示了机器学习算法用于训…

idea Statistic使用

问题描述&#xff1a;本地idea版本为2018.3.5&#xff0c;安装Statistic插件后没有出现Statistic图标 原因如下&#xff1a;插件版本太新了&#xff0c;需要历史版本 解决办法&#xff1a; IDEA安装代码统计插件Statistic后左下角图标出不来(亲测)_idea statistic不展示-CSD…