jsPdf+html2Canvas+react实现前端页面导出pdf,并解决jspdf分页图片过长截断问题

第一步 在utils下面创建一个工具文件exportPDF.js

以下每一句代码都有对应的注释,并且针对不同情况,作出了相应的解释,如果有需要的话,请耐心看完。

import html2Canvas from 'html2canvas';
import JsPDF from "jspdf";
/**
 * [获取页面导出的pdf文件]
 * @param   {[Object]}  options  [导出pdf配置项,包括一个title属性设置文件名,以及query属性设置获取元素的条件]
 */
function exportPDF(options) {
    var title = options.title || '标题';// 导出文件名,默认为“标题”
    // 这里获取调用这个方法传过来的参数对象里面的类名,也就是你需要导出为pdf元素的父盒子类名
    const ele = document.getElementsByClassName(options.className || "pdf-cell");
    // 这里面我要解释一下,这里是要为做分页做准备,因为分页是根据每一个dom元素的高度来决定的,也不用细化到每一个dom元素,只是你不希望被截断的dom元素都需要单独的拿出来计算高度,而不能获取整个大页面的高度
    // 这里的children取决于你传过来的父元素,根据实际需求来获取你需要的子元素
    const children = ele[0].children[1].children[0].children;
    
    // 定义一个空数组,来接收需要生成图片计算完高度的dom对象
    let canvas:any = [];
    //  这里的 i 和 j 根据实际需求添加  i 是只你定义的children的每一个dom元素的索引,每计算完一个高度,进行++运算然后递归
    let i = 0; 
    // 这里的 j 是如果你需要对你定义的children的子元素更深层的遍历的话,需要拿到他下面的子元素高度的话,就另定义一个j变量,进行++运算再递归
    let j = 0;    
    // scale是清晰度参数 数值越大,对应的清晰度越清晰,但是相对导出的文件大小也就越大,慎重修改,这里是根据你的窗口大小做判断
    const scale = window.devicePixelRatio > 1 ? window.devicePixelRatio : 2
    // 将dom转化为canvas
    function toCanvas() {
    	// 这里因为我需要深层遍历children的子元素的第三个元素的子元素也就是children的孙子元素,但是前两个我没有要单独计算高度的需求,所以做了这个判断
        if (children.length > 1 && i<=1 ) {
            html2Canvas(children[i], {
                scale:scale,
                dpi: 500, // 导出pdf清晰度
                background: '#fff', // 背景设为白色(默认为黑色)
            }).then(res => { // 计算每个dom的高度,方便后面计算分页
                res.imgWidth = 595.28 / scale / 0.7;
                res.imgHeight = 592.28 / res.width * res.height / scale /0.7;
                canvas.push(res);
                i++;
                // 这里判断我是否已经全部将需要计算高度的节点计算完了,如果计算完高度就会添加到我定义的canvas数组里面,计算完了就执行分页并生成pdf,否则递归继续计算
                if (canvas.length === children.length  + children[2].children.length) {
                    paging();
                    toPdf();
                } else {
                    toCanvas();
                }
            });
        }
       // 前面说过了,这里我需要单独计算第三个子元素的每个子元素的高度,所以如果 i>1了也就意味着遍历到第三个节点了,第三个节点我要再去获取里面的子元素,在这里面在进行转化canvas
        else{
          html2Canvas(children[i].children[j], {
            scale:scale,
            dpi: 500, // 导出pdf清晰度
            background: '#fff', // 背景设为白色(默认为黑色)
        }).then(res => { // 计算每个dom的高度,方便后面计算分页
            res.imgWidth = 595.28 / scale / 0.7;
            res.imgHeight = 592.28 / res.width * res.height / scale/0.7 ;
            canvas.push(res);
            j++;
            
            if (canvas.length === children.length + children[2].children.length-1) {
                paging();
                toPdf();
            } else {
                toCanvas();
            }
        });
          
        }
        
    }
    /**
     * [根据dom的高度初步进行分页,会将canvas组装为一个二维数组]
     */
    // dom 全部转化完成 开始计算分页
    function paging() {
        const imgArr = [[]];
        let pageH = 0;// 页面的高度
        let allH = 0;// 当前组所有dom的高度和
        let j = 0;
        for (let k = 0; k < canvas.length; k++) { // 涉及到k--的操作,使用for循环方便
            pageH += canvas[k].imgHeight;
            if (pageH > 841.89 && canvas[k].imgHeight < 841.89) { // 当某个页面装不下下一个dom时,则分页
                imgArr[j][0].allH = allH - canvas[k].imgHeight;
                allH = pageH = 0;
                k--;
                j++;
                imgArr.push([]);
            } else {
                if (canvas[k].imgHeight > 841.89) { // 特殊情况:某个dom高度大于了页面高度,特殊处理
                    canvas[k].topH = 841.89 - (pageH - canvas[k].imgHeight);// 该dom顶部距离页面上方的距离
                    pageH = (2 * canvas[k].imgHeight - pageH) % 841.89;
                    canvas[k].pageH = pageH;// 该dom底部距离页面上方的距离
                }
                imgArr[j].push(canvas[k]);
                allH += canvas[k].imgHeight;
            }
            if (k === canvas.length - 1) imgArr[j][0].allH = allH;
        }
        canvas = imgArr;
    }
    /**
     * [生成PDF文件]
     */
    function toPdf() {
        const PDF = new JsPDF('p', 'pt', 'a4');
        canvas.forEach((page, index) => {
            let allH = page[0].allH;
            let position = 20;// pdf页面偏移
            if (index !== 0 && allH <= 841.89) PDF.addPage();
            page.forEach(img => {
                if (img.imgHeight < 841.89) { // 当某个dom高度小于页面宽度,直接添加图片
                // 这里的width除以多少取决于你导出的页面在pdf页里面横向偏移的多少,你可以修改这个除数进行对页面的横向调整
                    PDF.addImage(img.toDataURL('image/jpeg', 1.0), 'JPEG', img.imgWidth/5, position, img.imgWidth, img.imgHeight);
                    position += img.imgHeight;
                    allH -= img.imgHeight;
                } else { // 当某个dom高度大于页面宽度,则需另行处理
                    while (allH > 0) {
                        PDF.addImage(img.toDataURL('image/jpeg', 1.0), 'JPEG', img.imgWidth/5, position, img.imgWidth, img.imgHeight);
                        allH -= img.topH || 841.89;
                        position -= img.topH || 841.89;
                        img.topH = 0;
                        if (allH > 0) PDF.addPage();
                    }
                    position = img.pageH;
                }
            });
        });
        PDF.save(title + '.pdf');
    }
    toCanvas();
}
export default exportPDF;

在页面调用

import exportPDF from '@/pages/common/utils/exportPDF'
...


  const toPdf = () =>{
    exportPDF({className:"toPdf",title:caseAuthor})
  }
return (
//  这里如果按钮不需要被导出的话 一定要拿到你传的类名的盒子外面
	<div className='toPdf'>
      <Button onClick={toPdf}>导出pdf</Button>
      <Layout style={{ backgroundColor: '#fff' }}>
        <Content className={css.casecontent}>
        	......
        </Content>
      </Layout>
    </div>
)

代码到这就结束了,用的html2canvas是1.4.1版本的,所以不涉及svg不能解析的问题,如果你的svg不能解析,或者格式混乱的话,检查一下对应的样式是否生效,如果没有生效,你可以把对应的样式放在行内来写。

但是还有一定的问题没有解决,如果页面过于复杂,导出的话会有浏览器性能问题,因为要递归去获取dom节点,嵌套比较深的话,可能导出会很慢,如果有解决方案,还请大佬指点

还要实现一个需求,就是在不渲染这个页面,通过其他页面路由的id去获取到这个页面,但是不跳转,前端是完全没有知觉的去通过这个页面对应的id去生成pdf,目前我还没有想到解决办法,如果大家有好的方案和建议还请多多指点

  • 6
    点赞
  • 32
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值