Java 如何仅重命名pdf中第一个找到的重复字段?
在我的PDF上有一个名为Text1的重复字段。现在我想将第一个找到的名为Text1的acrofield重命名为foobar。另一个名为Text1的字段应该保持不变,以便我的新表单包含Text1和foobar字段 我使用的是itext库,其中包含rename函数,但此方法将所有名为Text1的字段重命名为foobar 如果有人想用我的pdf测试它Java 如何仅重命名pdf中第一个找到的重复字段?,java,pdf,itext,Java,Pdf,Itext,在我的PDF上有一个名为Text1的重复字段。现在我想将第一个找到的名为Text1的acrofield重命名为foobar。另一个名为Text1的字段应该保持不变,以便我的新表单包含Text1和foobar字段 我使用的是itext库,其中包含rename函数,但此方法将所有名为Text1的字段重命名为foobar 如果有人想用我的pdf测试它 公共类重命名字段 { 公共静态最终字符串SRC=“c:\\test\u duplicate\u field2.pdf”; 公共静态最终字符串DEST=“
公共类重命名字段
{
公共静态最终字符串SRC=“c:\\test\u duplicate\u field2.pdf”;
公共静态最终字符串DEST=“c:\\test\u duplicate\u field\u mod.pdf”;
公共静态void main(字符串[]args)
抛出DocumentException,IOException
{
文件文件=新文件(DEST);
文件.getParentFile().mkdirs();
新建RenameField().ManufacturePDF(SRC,DEST);
}
公共空操作EPDF(字符串src、字符串dest)
抛出DocumentException,IOException
{
PdfReader读取器=新PdfReader(src);
PdfStamper stamper=新PdfStamper(读取器,新文件输出流(dest));
AcroFields form=stamp.getAcroFields();
表格。重命名字段(“文本1”、“foobar”);
压模关闭();
reader.close();
读卡器=新的PDF读卡器(目的地);
form=reader.getAcroFields();
Map fields=form.getFields();
for(字符串名称:fields.keySet()){
System.out.println(名称);
}
}
}
另一种方法是遍历AcroField.Items。如果有一个项具有多个值字典(在本例中,该字段存在多次),则将进行更改
for (Map.Entry<String, AcroFields.Item> entry: fieldMap.entrySet())
{
// extract Values for Field
String fieldKey = entry.getKey();
AcroFields.Item item = entry.getValue();
PdfDictionary dict;
int numberOfDuplicates = item.values.size();
if (numberOfDuplicates > 1) {
for (int i = 0; i < numberOfDuplicates; i++) {
if (i == 0) {
log.info("first field wont be changed");
} else {
log.info("renaming field " + fieldKey + " round " + i );
item.getMerged(i).put(PdfName.T, new PdfString(fieldKey + "_" + i ));
item.getValue(i).put(PdfName.T, new PdfString(fieldKey + "_" + i));
form.regenerateField(fieldKey);
}
}
}
}
for(Map.Entry:fieldMap.entrySet())
{
//提取字段的值
字符串fieldKey=entry.getKey();
AcroFields.Item=entry.getValue();
PDF字典;
int numberOfDuplicates=item.values.size();
如果(重复数>1){
for(int i=0;i
但这会导致与上面使用itext中的renameField函数得到相同的结果,两个字段名都将更改。在调试期间,我可以看到该项的两个值字典具有相同的对象id,因此当我更改字典[0]的值时,字典[1]中的值也会更改
在我的PDF上有一个名为Text1的重复字段
正如布鲁诺已经在评论中解释的那样,这是错误的。您的PDF只有一个名为“Text1”的字段:
(方法testWidgetToField
)
显然,对于一个通用的解决方案,还有很多工作要做。您混淆了字段和小部件注释的概念。您有一个名为Text1的字段。此字段有两个小部件注释。重命名Text1时,重命名一个字段,该字段仍有两个小部件注释。基本上,我是说你的问题是错误的。你说你有两个字段,但那不是真的。您只有一个字段(带有两个小部件注释)。感谢您的回复,显然您是对的,我混淆了这些概念。你有如何重命名两个小部件注释之一而不是字段的示例代码吗?谢谢你回答@mkl,我目前没有太多时间回答问题。谢谢示例代码。与此同时,我创建了一个解决方案,首先检测重复字段,删除它们,最后用新名称重新创建它们,但这有一个坏的副作用,即并非所有外观都像以前一样。通过您的方法,这种副作用将被消除,因此我将对此进行检查。感谢您的解决方案有一个小问题,分组的重复字段将不会更改,因为PdfName.T只返回点之前的值。例如,如果表单上的字段test1.foobar1是重复的,那么它不会被更改,因为PdfName.T只返回test1。。我如何在这里检索完整的名称?这就是为什么我写了一个通用的解决方案,显然还有很多事情要做。代码只查看字段树的第一级。你得在树上走。再次,您将test1.foobar1字段作为表单上的副本进行讨论。。。该字段不是重复的,它只有多个小部件,但它是一个字段。
for (Map.Entry<String, AcroFields.Item> entry: fieldMap.entrySet())
{
// extract Values for Field
String fieldKey = entry.getKey();
AcroFields.Item item = entry.getValue();
PdfDictionary dict;
int numberOfDuplicates = item.values.size();
if (numberOfDuplicates > 1) {
for (int i = 0; i < numberOfDuplicates; i++) {
if (i == 0) {
log.info("first field wont be changed");
} else {
log.info("renaming field " + fieldKey + " round " + i );
item.getMerged(i).put(PdfName.T, new PdfString(fieldKey + "_" + i ));
item.getValue(i).put(PdfName.T, new PdfString(fieldKey + "_" + i));
form.regenerateField(fieldKey);
}
}
}
}
34 0 obj
<</DA(/Helv 12 Tf 0 g)/FT/Tx/Kids[28 0 R 29 0 R]/T(Text1)>>
endobj
28 0 obj
<</F 4/MK<<>>/P 3 0 R/Parent 34 0 R/Rect[70.305 698.209 220.305 720.209]/Subtype/Widget/Type/Annot>>
endobj
29 0 obj
<</DA(/Helv 12 Tf 0 g)/F 4/MK<<>>/P 3 0 R/Parent 34 0 R/Rect[240.02 697.453 390.02 719.453]/Subtype/Widget/Type/Annot>>
endobj
PdfReader reader = new PdfReader(resource);
PdfDictionary form = reader.getCatalog().getAsDict(PdfName.ACROFORM);
PdfArray fields = form.getAsArray(PdfName.FIELDS);
for (PdfObject object: fields)
{
PdfDictionary field = (PdfDictionary) PdfReader.getPdfObject(object);
if ("Text1".equals(field.getAsString(PdfName.T).toString()))
{
PdfDictionary newField = new PdfDictionary();
PRIndirectReference newFieldRef = reader.addPdfObject(newField);
fields.add(newFieldRef);
newField.putAll(field);
newField.put(PdfName.T, new PdfString("foobar"));
PdfArray newKids = new PdfArray();
newField.put(PdfName.KIDS, newKids);
PdfArray kids = field.getAsArray(PdfName.KIDS);
PdfObject widget = kids.remove(0);
newKids.add(widget);
PdfDictionary widgetDict = (PdfDictionary) PdfReader.getPdfObject(widget);
widgetDict.put(PdfName.PARENT, newFieldRef);
break;
}
}
PdfStamper stamper = new PdfStamper(reader, result);
stamper.close();