Java iText PdfCopy创建可编辑的pdf文档
我有一个用于spring启动应用程序的模板pdf文件。我需要根据每个请求的用户输入更新此模板中的值。此外,在请求中,我将获得多个pdf文件,我需要合并这些文件以及更新的文件,这是最终pdf的第一页 我正在使用带弹簧靴的iText。我能够更新模板中的值并合并文件内容,但最终的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
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();
}