Warning: file_get_contents(/data/phpspider/zhask/data//catemap/9/java/399.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Java 如何仅重命名pdf中第一个找到的重复字段?_Java_Pdf_Itext - Fatal编程技术网

Java 如何仅重命名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=“

在我的PDF上有一个名为Text1的重复字段。现在我想将第一个找到的名为Text1的acrofield重命名为foobar。另一个名为Text1的字段应该保持不变,以便我的新表单包含Text1和foobar字段

我使用的是itext库,其中包含rename函数,但此方法将所有名为Text1的字段重命名为foobar

如果有人想用我的pdf测试它

公共类重命名字段
{
公共静态最终字符串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();