在我的PDF上,我有一个名为Text1的重复字段 . 现在我想将名为Text1的第一个找到的acrofield重命名为foobar . 名称为Text1的另一个字段应保持不变,以便我的新表单包含字段Text1和foobar .
我正在使用itext库并在那里使用重命名函数,但是此方法会将名称为Text1的所有字段重命名为foobar .
如果有人想用我的pdf测试它,here is a link .
public class RenameField
{
public static final String SRC = "c:\\test_duplicate_field2.pdf";
public static final String DEST = "c:\\test_duplicate_field_mod.pdf";
public static void main(String[] args)
throws DocumentException, IOException
{
File file = new File(DEST);
file.getParentFile().mkdirs();
new RenameField().manipulatePdf(SRC, DEST);
}
public void manipulatePdf(String src, String dest)
throws DocumentException, IOException
{
PdfReader reader = new PdfReader(src);
PdfStamper stamper = new PdfStamper(reader, new FileOutputStream(dest));
AcroFields form = stamper.getAcroFields();
form.renameField("Text1", "foobar");
stamper.close();
reader.close();
reader = new PdfReader(dest);
form = reader.getAcroFields();
Map fields = form.getFields();
for (String name : fields.keySet()) {
System.out.println(name);
}
}
}
另一种方法是遍历AcroField.Items . 如果有一个项目具有多个值字典(在这种情况下该字段存在多次),则将进行更改 .
for (Map.Entry 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);
}
}
}
}
但这导致与上述方法相同的结果与来自itext的renameField函数,两个字段名称都将被更改 . 在调试过程中,我可以看到项目的两个值字典具有相同的object-id,所以当我更改字典[0]的值时,值也将在字典[1]中更改