PDF模板填充新姿势,开箱即用

写在前面

由于之前使用Itext5工具填充PDF模板后,会导致填充后的PDF文件体积变得很庞大。

怀疑了嵌入字体、PDF模板编辑转换和编辑等等的原因,但最后都无功而返,查阅了官方文档,也没得出解决方案。

因此,退而求其次,换上了Itext7,官方说过,性能相较Itext5更出色。

依赖

<dependency>
    <groupId>com.itextpdf</groupId>
    <artifactId>itext7-core</artifactId>
    <version>8.0.1</version>
    <type>pom</type>
</dependency>
<dependency>
    <groupId>com.itextpdf</groupId>
    <artifactId>bouncy-castle-adapter</artifactId>
    <version>8.0.1</version>
</dependency>

注:引入bouncy-castle-adapter是因为Itext7解析PDF时用到了加密算法,因此可能还需引入

<dependency>
    <groupId>org.bouncycastle</groupId>
    <artifactId>bcprov-jdk15on</artifactId>
    <version>1.70</version>
</dependency>
<dependency>
    <groupId>org.bouncycastle</groupId>
    <artifactId>bcpkix-jdk15on</artifactId>
    <version>1.70</version>
</dependency>

工具

import com.gynsh.utils.image.ImageCompressUtils;
import com.gynsh.utils.pdf.font.FontDict;
import com.itextpdf.forms.PdfAcroForm;
import com.itextpdf.forms.fields.PdfFormCreator;
import com.itextpdf.forms.fields.PdfFormField;
import com.itextpdf.io.font.PdfEncodings;
import com.itextpdf.io.image.ImageData;
import com.itextpdf.io.image.ImageDataFactory;
import com.itextpdf.kernel.font.PdfFont;
import com.itextpdf.kernel.font.PdfFontFactory;
import com.itextpdf.kernel.geom.Rectangle;
import com.itextpdf.kernel.pdf.*;
import com.itextpdf.kernel.pdf.annot.PdfAnnotation;
import com.itextpdf.kernel.pdf.annot.PdfWidgetAnnotation;
import com.itextpdf.kernel.pdf.canvas.PdfCanvas;
import lombok.extern.slf4j.Slf4j;

import java.io.*;
import java.util.HashMap;
import java.util.Map;
import java.util.Optional;

/**
 * PDF 模板填充工具类(基于itext7)
 */
@Slf4j
public final class Pdf7TemplateFillUtil {
    /**
     * 待填充的PDF模板文档
     */
    private final transient PdfDocument templateDocument;

    /**
     * 填充后PDF文档数据
     */
    private final transient ByteArrayOutputStream destByteOutStream;

    /**
     * 填充及返回填充后的数据
     *
     * @param fillData - 待填充数据
     * @return - 填充后bytes数据
     */
    public byte[] fill(Map<String, Object> fillData) throws IOException {
        try {
            PdfFont font = PdfFontFactory.createFont("fons/AlibabaPuHuiTi-3-35-Thin.ttf",
                    PdfEncodings.IDENTITY_H, PdfFontFactory.EmbeddingStrategy.PREFER_EMBEDDED);
            PdfAcroForm form = PdfAcroForm.getAcroForm(templateDocument, true);
            fillData.forEach((keyword, value) -> {
                Optional.ofNullable(form.getField(keyword)).ifPresent(templateFormField -> {
                    if (value instanceof byte[]) {
                        PdfArray pos = templateFormField.getWidgets().get(0).getRectangle();
                        float x = pos.getAsNumber(0).floatValue();
                        float y = pos.getAsNumber(1).floatValue();
                        float width = pos.getAsNumber(2).floatValue() - x;
                        float height = pos.getAsNumber(3).floatValue() - y;
                        Rectangle rectangle = new Rectangle(x, y, width, height);
                        PdfWidgetAnnotation widget = new PdfWidgetAnnotation(rectangle);
                        PdfFormField formField = PdfFormCreator.createFormField(widget, templateDocument);
                        PdfPage annotationPage = findAnnotationPage(keyword);
                        if (annotationPage != null) {
                            doFillFieldImage(annotationPage, formField, (byte[]) value);
                        }
                    } else {
                        if (value instanceof Boolean) {
                            templateFormField.setValue(String.valueOf(value), true);
                        } else {
                            templateFormField.setValue(String.valueOf(value));
                        }
                        templateFormField.setFontAndSize(font, templateFormField.getFontSize());
                        templateFormField.setReadOnly(true);
                    }
                    form.partialFormFlattening(keyword);
                });
            });
            form.flattenFields();
        } finally {
            if (templateDocument != null) {
                templateDocument.close();
            }
        }
        return destByteOutStream.toByteArray();
    }

    /**
     * 图片填充
     *
     * @param newPage   - 当前页
     * @param formField - 表单文本域
     * @param imgBytes  - 图片文件字节数组
     */
    private void doFillFieldImage(PdfPage newPage, PdfFormField formField, byte[] imgBytes) {
        Rectangle rtl = formField.getWidgets().get(0).getRectangle().toRectangle(); // 获取表单域的xy坐标
        PdfCanvas canvas = new PdfCanvas(newPage);
        ImageData img = ImageDataFactory.create(imgBytes);
        if (Float.compare(img.getWidth(), rtl.getWidth()) <= 0 && Float.compare(img.getHeight(), rtl.getHeight()) <= 0) {// 不处理
            canvas.addImageAt(img, rtl.getX(), rtl.getY(), true);
        } else {
            // 压缩图片。计算得到图片放缩的最大比例
            float scale = Math.max(img.getWidth() / rtl.getWidth(), img.getHeight() / rtl.getHeight());
            int imgWidth = Math.round(img.getWidth() / scale);
            int imgHeight = Math.round(img.getHeight() / scale);
            // 压缩图片
            byte[] compressImgBytes;
            try {
                compressImgBytes = ImageCompressUtils.resizeByThumbnails(imgBytes, imgWidth, imgHeight);
            } catch (IOException e) {
                throw new RuntimeException(e);
            }
            img = ImageDataFactory.create(compressImgBytes);
            canvas.addImageAt(img, rtl.getX(), rtl.getY(), true);
        }
    }

    /**
     * 根据表单域关键字查找当前关键字所在页对象(PdfPage)
     *
     * @param keyword - 关键字
     * @return - page object
     */
    private PdfPage findAnnotationPage(String keyword) {
        int pages = templateDocument.getNumberOfPages();
        for (int index = 1; index <= pages; index++) {
            PdfPage page = templateDocument.getPage(index);
            for (PdfAnnotation annotation : page.getAnnotations()) {
                PdfString title = annotation.getPdfObject().getAsString(PdfName.T);
                if (title != null && keyword.equals(String.valueOf(title))) {
                    return page;
                }
            }
        }
        return null;
    }

    /**
     * 获取模板文件表单域关键字位置信息
     */
    private Map<Integer, Map<String, float[]>> getFormKeywordsPos() {
        int pages = templateDocument.getNumberOfPages();
        Map<Integer, Map<String, float[]>> maps = new HashMap<>(pages);
        for (int index = 1; index <= pages; index++) {
            maps.putIfAbsent(index, new HashMap<>());
            PdfPage page = templateDocument.getPage(index);
            // 获取当前页的表单域
            int finalIndex = index;
            page.getAnnotations().forEach(anno -> {
                PdfString title = anno.getTitle();
                PdfArray rectangle = anno.getRectangle();
                float x = rectangle.getAsNumber(0).floatValue();
                float y = rectangle.getAsNumber(1).floatValue();
                float width = rectangle.getAsNumber(2).floatValue() - x;
                float height = rectangle.getAsNumber(3).floatValue() - y;
                maps.get(finalIndex).put(title.getValue(), new float[]{x, y, width, height});
            });
        }
        return maps;
    }

    private Pdf7TemplateFillUtil(PdfDocument templateDocument, ByteArrayOutputStream destByteOutStream) {
        this.templateDocument = templateDocument;
        this.destByteOutStream = destByteOutStream;
    }

    public static Pdf7TemplateFillUtil load(String template) throws IOException {
        return init(new PdfReader(template));
    }

    public static Pdf7TemplateFillUtil load(File template) throws IOException {
        return init(new PdfReader(template));
    }

    public static Pdf7TemplateFillUtil load(byte[] template) throws IOException {
        return init(new PdfReader(new ByteArrayInputStream(template)));
    }

    public static Pdf7TemplateFillUtil load(InputStream template) throws IOException {
        return init(new PdfReader(template));
    }

    private static Pdf7TemplateFillUtil init(PdfReader template) {
        ByteArrayOutputStream dest = new ByteArrayOutputStream();
        PdfWriter pdfWriter = new PdfWriter(dest);
        pdfWriter.setCompressionLevel(9);// 压缩级别
        PdfDocument pdfDocument = new PdfDocument(template, pdfWriter);
        return new Pdf7TemplateFillUtil(pdfDocument, dest);
    }
}

特别注意

字体加载有一个巨坑,一定要放在具体的方法中去加载,保证用完即销毁。

否则填充时会出错:Pdf indirect object belongs to other PDF document....

  • 3
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

流沙QS

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值