如何在新签名字段中关联先前的签名

问题描述 投票:1回答:1

我有签名的PDF。我想在文档中显示此签名。我可以这样添加一个新的签名字段:

Stamper.addSignature("My Signature", 1, 20f, 10f, 100f, 100f);

但我找不到将其与文档中已有的签名相关联的方法。

我该如何联想呢?

itext sign
1个回答
5
投票

OP希望将文档内签名可视化附加到现有签名。

首先,如果您的文件经过认证且不允许更改,则显然不允许这样做

但是,令人惊讶的是,似乎允许签名但未经证明的文档(我与之合作的示例文件)。

实际上,除非文件经过认证且不允许更改,否则您始终可以填写表格(除了允许表格填写和数字签名认证的文件),甚至修改注释,参见this answer概述。

由于PDF签名是表单字段值,表单字段的可视化是特殊注释,因此允许更改签名可视化作为表单填写或至少作为注释修改。

在iText 5中完成

OP尝试通过添加新的签名字段来实现此目的:

Stamper.addSignature("My Signature", 1, 20f, 10f, 100f, 100f);

但这并没有帮助,因为必须更改现有的签名字段,而不是创建新的签名字段。

使用iText 5.x可以使用通用表单字段操作API完成:

PdfReader pdfReader = new PdfReader(resource);
PdfStamper pdfStamper = new PdfStamper(pdfReader, result, '\0', true);

AcroFields acroFields = pdfStamper.getAcroFields();
for (String signatureName : acroFields.getSignatureNames())
{
    Item field = acroFields.getFieldItem(signatureName);
    field.writeToAll(PdfName.RECT, new PdfArray(new int[]{100,100,200,200}), Item.WRITE_WIDGET);
    field.markUsed(acroFields, Item.WRITE_WIDGET);

    PdfAppearance appearance = PdfAppearance.createAppearance(pdfStamper.getWriter(), 100, 100);
    appearance.setColorStroke(BaseColor.RED);
    appearance.moveTo(0, 0);
    appearance.lineTo(99, 99);
    appearance.moveTo(0, 99);
    appearance.lineTo(99, 0);
    appearance.stroke();

    PdfDictionary appDict = new PdfDictionary();
    appDict.put(PdfName.N, appearance.getIndirectReference());
    field.writeToAll(PdfName.AP, appDict, Item.WRITE_WIDGET);
}

pdfStamper.close();

ChangeSignatureAppearance.java方法testChangeAppearances

此代码为每个集成的PDF签名创建一个新的签名外观,在这种情况下,红色十字标记位于100,100并且大小为100x100,但您可以创建任何您喜欢的外观。

注意:此代码假定隐形签名已与某些文档页面关联。对于尚未与页面关联的不可见签名,必须建立关联。这可能会变成一个不允许的变化,至少它不仅仅是形式填充,因为表单结构也被更改,而不仅仅是其条目。


OP在评论中表示

但我想检索标志的名称并写下它而不是红十字

为此,您只需稍微更改上面的代码:

PdfReader pdfReader = new PdfReader(resource);
PdfStamper pdfStamper = new PdfStamper(pdfReader, result, '\0', true);

AcroFields acroFields = pdfStamper.getAcroFields();
for (String signatureName : acroFields.getSignatureNames())
{
    PdfPKCS7 pkcs7 = acroFields.verifySignature(signatureName);
    X509Certificate signerCert = (X509Certificate) pkcs7.getSigningCertificate();
    String signerName = CertificateInfo.getSubjectFields(signerCert).getField("CN");

    Item field = acroFields.getFieldItem(signatureName);
    field.writeToAll(PdfName.RECT, new PdfArray(new int[]{100,100,200,200}), Item.WRITE_WIDGET);
    field.markUsed(acroFields, Item.WRITE_WIDGET);

    PdfAppearance appearance = PdfAppearance.createAppearance(pdfStamper.getWriter(), 100, 100);
    ColumnText columnText = new ColumnText(appearance);
    Chunk chunk = new Chunk();
    chunk.setSkew(0, 12);
    chunk.append("Signed by:");
    columnText.addElement(new Paragraph(chunk));
    chunk = new Chunk();
    chunk.setTextRenderMode(PdfContentByte.TEXT_RENDER_MODE_FILL_STROKE, 1, BaseColor.BLACK);
    chunk.append(signerName);
    columnText.addElement(new Paragraph(chunk));
    columnText.setSimpleColumn(0, 0, 100, 100);
    columnText.go();

    PdfDictionary appDict = new PdfDictionary();
    appDict.put(PdfName.N, appearance.getIndirectReference());
    field.writeToAll(PdfName.AP, appDict, Item.WRITE_WIDGET);
}

pdfStamper.close();

ChangeSignatureAppearance.java方法testChangeAppearancesWithName

如果样本文件BouncyCastle必须注册为安全提供者。

而上述警告仍然适用。

在iText 7中完成

由于iText 7最近已经发布,上面的代码可以像这样移植到它:

try (   PdfReader pdfReader = new PdfReader(resource);
        PdfWriter pdfWriter = new PdfWriter(result);
        PdfDocument pdfDocument = new PdfDocument(pdfReader, pdfWriter, new StampingProperties().useAppendMode()))
{
    SignatureUtil signatureUtil = new SignatureUtil(pdfDocument);
    PdfAcroForm acroForm = PdfAcroForm.getAcroForm(pdfDocument, false);

    for (String name : signatureUtil.getSignatureNames())
    {
        PdfFormField field = acroForm.getField(name);
        field.setModified();
        for (PdfWidgetAnnotation pdfWidgetAnnotation : field.getWidgets())
        {
            pdfWidgetAnnotation.setRectangle(new PdfArray(new int[]{100, 100, 200, 200}));

            PdfFormXObject form = new PdfFormXObject(new Rectangle(100, 100));
            PdfCanvas canvas = new PdfCanvas(form, pdfDocument);
            canvas.setStrokeColor(Color.RED);
            canvas.moveTo(0, 0);
            canvas.lineTo(99, 99);
            canvas.moveTo(0, 99);
            canvas.lineTo(99, 0);
            canvas.stroke();

            pdfWidgetAnnotation.setNormalAppearance(form.getPdfObject());
        }
    }
}

ChangeSignatureAppearance.java方法testChangeAppearances

此代码需要iText 7工件kernelformssign

适用于上述iText 5代码的相同警告:

注意:此代码假定隐形签名已与某些文档页面关联。


主题名称的变体如下所示:

try (   PdfReader pdfReader = new PdfReader(resource);
        PdfWriter pdfWriter = new PdfWriter(result);
        PdfDocument pdfDocument = new PdfDocument(pdfReader, pdfWriter, new StampingProperties().useAppendMode()))
{
    SignatureUtil signatureUtil = new SignatureUtil(pdfDocument);
    PdfAcroForm acroForm = PdfAcroForm.getAcroForm(pdfDocument, false);

    for (String name : signatureUtil.getSignatureNames())
    {
        PdfPKCS7 pkcs7 = signatureUtil.verifySignature(name);
        X509Certificate signerCert = (X509Certificate) pkcs7.getSigningCertificate();
        String signerName = CertificateInfo.getSubjectFields(signerCert).getField("CN");
        PdfFormField field = acroForm.getField(name);
        field.setModified();
        for (PdfWidgetAnnotation pdfWidgetAnnotation : field.getWidgets())
        {
            pdfWidgetAnnotation.setRectangle(new PdfArray(new int[]{100, 100, 200, 200}));

            PdfFormXObject form = new PdfFormXObject(new Rectangle(100, 100));
            Canvas canvas = new Canvas(form, pdfDocument);
            canvas.add(new Paragraph().setItalic().add("Signed by:"));
            canvas.add(new Paragraph().setBold().add(signerName));

            pdfWidgetAnnotation.setNormalAppearance(form.getPdfObject());
        }
    }
}

ChangeSignatureAppearance.java方法testChangeAppearancesWithName

此代码另外使用iText 7工件layout。此外,在样本文档的情况下,BouncyCastle必须像上面的iText 5代码一样注册为安全提供者。

上述警告仍然适用。

在Adobe Acrobat Reader DC中查看它

我在Adobe Acrobat Reader DC中使用无形签名的空白文档BLANK-signed.pdf对此进行了测试:

Screen shot of original blank signed PDF

使用上面的代码操作文件后,我得到:

Screen shot of blank signed PDF with added appearance

有关未签名更改的警告是正确的,但在再次签名后,即使该警告也会消失:

Screen shot of blank signed PDF with added appearance and signed again


具有签名者名称的变体如下所示:

Screen shot of blank signed PDF with added appearance with signer name

附录:签名的多次出现

OP在评论中提问

此方法仅在文档的第一页中标记它。如何在文档的所有页面中标记?

上述方法通常实际上不会在第一页上标记,而是在与签名相关联的页面上标记。但是,由于隐形签名通常与第一页相关联,因此可以理解为什么它会出现。

此外,单个签名字段的多次出现并非普遍支持(虽然ISO 32000-1实际上并未禁止),并且即将推出的ISO 32000-2将禁止这些出现。因此,走这条路并不是最好的主意。

但是,如果没有办法,你可以在iText 7中尝试这样的事情:

try (   PdfReader pdfReader = new PdfReader(resource);
        PdfWriter pdfWriter = new PdfWriter(result);
        PdfDocument pdfDocument = new PdfDocument(pdfReader, pdfWriter, new StampingProperties().useAppendMode()))
{
    SignatureUtil signatureUtil = new SignatureUtil(pdfDocument);
    PdfAcroForm acroForm = PdfAcroForm.getAcroForm(pdfDocument, false);

    for (String name : signatureUtil.getSignatureNames())
    {
        PdfPKCS7 pkcs7 = signatureUtil.verifySignature(name);
        X509Certificate signerCert = (X509Certificate) pkcs7.getSigningCertificate();
        String signerName = CertificateInfo.getSubjectFields(signerCert).getField("CN");
        PdfFormField field = acroForm.getField(name);
        field.setModified();

        Rectangle rectangle = new Rectangle(100, 100);
        PdfFormXObject form = new PdfFormXObject(rectangle);
        Canvas canvas = new Canvas(form, pdfDocument);
        canvas.add(new Paragraph().setItalic().add("Signed by:"));
        canvas.add(new Paragraph().setBold().add(signerName));

        for (PdfWidgetAnnotation pdfWidgetAnnotation : field.getWidgets())
        {
            PdfDictionary pageObject = pdfWidgetAnnotation.getPageObject();
            PdfPage page = pdfDocument.getPage(pageObject);
            page.removeAnnotation(pdfWidgetAnnotation);

            pdfWidgetAnnotation.releaseFormFieldFromWidgetAnnotation();
        }

        for (int pageNumber = 1; pageNumber <= pdfDocument.getNumberOfPages(); pageNumber++)
        {
            PdfPage pdfPage = pdfDocument.getPage(pageNumber);
            PdfWidgetAnnotation pdfWidgetAnnotation = new PdfWidgetAnnotation(rectangle);
            pdfWidgetAnnotation.setNormalAppearance(form.getPdfObject());
            pdfWidgetAnnotation.setPage(pdfPage);
            field.addKid(pdfWidgetAnnotation);
            pdfPage.addAnnotation(pdfWidgetAnnotation);
        }
    }
}

ChangeSignatureAppearance.java方法testChangeAppearancesWithNameAllPages

这首先删除签名字段的任何现有注释,然后将新的注释添加到所有字段。

就像上面一样,这将发出关于无符号更改的警告,这毕竟是真的。

顺便说一句,如果在一个页面上使用带有可视化的签名文档并更改上面的代码而不删除原始注释,则可以轻松地将该注释的副本添加到所有页面,并且当前的Adobe Acrobat Reader甚至不会显示一个警告!用于检查已签名文档中的更改的Reader代码确实很奇怪......

© www.soinside.com 2019 - 2024. All rights reserved.