iText PdfCopy创建可编辑的pdf文档

时间:2018-11-04 13:53:58

标签: java spring-boot itext

我有一个用于春季启动应用程序的pdf模板文件。我需要根据每个请求的用户输入来更新此模板中的值。同样在请求中,我将获得多个pdf文件,我需要将这些文件与最终pdf的第一页的更新文件合并。

我在Spring Boot中使用iText。我能够更新模板中的值并合并文件内容,但是最终的pdf即将发布,因为可编辑的文件被隐藏了。如果我单击该文件,则可以看到我的值,也可以编辑。

public void mergefiles(Map<String, String> tempData,MultipartFile[] userInfoFiles) 
            throws Exception{



        FileOutputStream mergeOutStream = new FileOutputStream(new File("C:\\UpdateFile\\mergepath\\updatetem.pdf")); //To update user content to Template

        PdfReader reader = new PdfReader(new FileInputStream(new File("C:\\UpdateFile\\template\\template.pdf"))); //Template File Stream
        PdfStamper stamper = new PdfStamper(reader, mergeOutStream);

        stamper.setFormFlattening(false);
        AcroFields form = stamper.getAcroFields();

        Map<String, Item> fieldMap = form.getFields();


        for (String key : fieldMap.keySet()) {
            String fieldValue = dataMap.get(key);
            if (fieldValue != null) {

                form.setField(key, fieldValue);
            }
        }
        //Above part creates updated pdf with read only

        //Below section creates merged file but first page is editable with 
        //filed values are hidden.

        Document mergePdfDoc = new Document();
        PdfCopy pdfCopy;
        boolean smartCopy = false;

        FileOutputStream newmergeOutStream = new FileOutputStream(new File("C:\\UpdateFile\\mergepath\\newmerged.pdf"));

        if(smartCopy)
            pdfCopy = new PdfSmartCopy(mergePdfDoc, newmergeOutStream);
        else
            pdfCopy = new PdfCopy(mergePdfDoc, newmergeOutStream);



        mergePdfDoc.open();

        pdfCopy.addDocument(stamper.getReader());
        pdfCopy.freeReader(stamper.getReader());


        PdfReader[] pdfReader = new PdfReader[userInfoFiles.length];



        for(int i=0; i<=userInfoFiles.length-1;i++) {
                pdfReader[i] = new PdfReader(userInfoFiles[i].getInputStream());
                pdfCopy.addDocument(pdfReader[i]);
                pdfCopy.freeReader(pdfReader[i]);
                pdfReader[i].close();
        }
        stamper.close();
        mergeOutStream.close();
        mergePdfDoc.close();
    }

任何输入为何最终的pdf都是可编辑的形式,而字段值则被隐藏。我必须创建一个合并文档并获取最终文档的ByteArray流作为其输入另一个函数的输入。我正在使用iText5。

1 个答案:

答案 0 :(得分:1)

问题是您将PdfReader的{​​{1}}添加为PdfStamper的输入:

PdfCopy

使用压模的读取器很脏:通过压模应用的某些更改是对读取器所持有的对象进行的,某些更改仅在压模或其输出中使用。

例如在您的情况下,表单字段已在原始pdf中定义。字段值直接添加到该字段。因此,它在阅读器中被更改。但是,外观(包括其当前值的图形的字段可视化)在新的间接对象中生成,该对象被添加到压模输出中。因此,阅读器中仍然有原始的,空的可视化。

因此,在pdf查看器中,pdfCopy.addDocument(stamper.getReader()); 结果最初具有空白字段的外观(因为外观仅在压模中生成),但是在编辑字段时,更改后的值变得可见(因为字段编辑器将使用字段值初始化。)

要解决此问题,请不要使用脏的读取器,而应根据冲压结果创建一个新的干净的读取器。

首先创建图章文件:

PdfCopy

然后合并:

    FileOutputStream mergeOutStream = new FileOutputStream(new File("C:\\UpdateFile\\mergepath\\updatetem.pdf")); //To update user content to Template

    PdfReader reader = new PdfReader(new FileInputStream(new File("C:\\UpdateFile\\template\\template.pdf"))); //Template File Stream
    PdfStamper stamper = new PdfStamper(reader, mergeOutStream);

    stamper.setFormFlattening(false);
    AcroFields form = stamper.getAcroFields();

    Map<String, Item> fieldMap = form.getFields();

    for (String key : fieldMap.keySet()) {
        String fieldValue = dataMap.get(key);
        if (fieldValue != null) {
            form.setField(key, fieldValue);
        }
    }
    stamper.close();
相关问题