Java iText PdfCopy创建可编辑的pdf文档

Java iText PdfCopy创建可编辑的pdf文档,java,spring-boot,itext,Java,Spring Boot,Itext,我有一个用于spring启动应用程序的模板pdf文件。我需要根据每个请求的用户输入更新此模板中的值。此外,在请求中,我将获得多个pdf文件,我需要合并这些文件以及更新的文件,这是最终pdf的第一页 我正在使用带弹簧靴的iText。我能够更新模板中的值并合并文件内容,但最终的pdf文件是可编辑的,文件是隐藏的。如果我点击该字段,我可以看到我的值,也可以编辑 public void mergefiles(Map<String, String> tempData,MultipartFile

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

我正在使用带弹簧靴的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。

问题是,您将PdfStamper所基于的PdfReader作为输入添加到您的PdfCopy中:

母版所使用的读卡器是脏的:通过母版应用的一些更改是对读卡器所持有的对象进行的,一些仅在母版或其输出中

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

因此,在pdf查看器中,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();
然后合并:

    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();

    PdfReader reader = new PdfReader(new FileInputStream(new File("C:\\UpdateFile\\mergepath\\updatetem.pdf")));
    pdfCopy.addDocument(reader);
    pdfCopy.freeReader(reader);

    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();
    }

    mergeOutStream.close();
    mergePdfDoc.close();
}

感谢您的详细解释。我已经根据评论更改了逻辑,现在我可以看到带有字段的最终pdf,但仍处于编辑模式。我可以编辑pdf。这是我们期望的,还是我们可以让pdfcopy创建只读最终pdf的任何方法。我通过设置stamper.setFormFlatingTrue来修复它。感谢您详细的书写以理解原始版本。@springbootlearner确实,这些字段是可编辑的,因为没有理由不编辑它们;。展平是一种选择,或者您可以将其设置为只读。
    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();

    PdfReader reader = new PdfReader(new FileInputStream(new File("C:\\UpdateFile\\mergepath\\updatetem.pdf")));
    pdfCopy.addDocument(reader);
    pdfCopy.freeReader(reader);

    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();
    }

    mergeOutStream.close();
    mergePdfDoc.close();
}