为PDF添加数字签名字段

Sam*_*mmy 5 c# pdfsharp digital-signature migradoc

我使用PDFsharp和MigraDoc创建一个新的PDF文件。现在,我想添加一个字段,用户可以在其中单击并选择用于数字签名文件的证书。

我在网上发现,使用AcroForms应该可以做到这一点。但是我无法使用,AcroForm因为它始终为null。

我当前的示例代码:

Document document = new Document();

Section section = document.AddSection();
section.AddParagraph("Signature Test");


PdfDocumentRenderer pdfRenderer = new PdfDocumentRenderer(false, PdfFontEmbedding.Always);
pdfRenderer.Document = document;
pdfRenderer.RenderDocument();

// NullPointerException at the following line. AcroForm is null 
pdfRenderer.PdfDocument.AcroForm.Elements.Add(PdfAcroForm.Keys.SigFlags, new PdfInteger(3));

const string filename = "HelloWorld.pdf";
pdfRenderer.PdfDocument.Save(filename);
Process.Start(filename);
Run Code Online (Sandbox Code Playgroud)

为什么此属性为null?我该怎么做才能将其设置为正确的值?

或者更好的是,如何添加一个字段来选择数字证书?

Sam*_*mmy 4

我找到了一个向文档添加签名字段的解决方案。不幸的是我不得不使用反射,因为 auf 缺少对所需对象的访问。

using PdfSharp.Pdf;
using PdfSharp.Pdf.Annotations;
using System;
using System.Linq;
using System.Reflection;

internal sealed class MyPdfSignatureField : PdfAnnotation
{
    public MyPdfSignatureField(PdfDocument document, PdfRectangle rect) : base(document)
    {
        Elements.Add("/FT", new PdfName("/Sig"));
        Elements.Add(Keys.T, new PdfString("Signature1"));
        Elements.Add("/Ff", new PdfInteger(132));
        Elements.Add("/DR", new PdfDictionary());
        Elements.Add(Keys.Subtype, new PdfName("/Widget"));
        Elements.Add("/P", document.Pages[0]);


        PdfDictionary sign = new PdfDictionary(document);
        sign.Elements.Add(Keys.Type, new PdfName("/Sig"));
        sign.Elements.Add("/Filter", new PdfName("/Adobe.PPKLite"));
        sign.Elements.Add("/SubFilter", new PdfName("/adbe.pkcs7.detached"));
        sign.Elements.Add(Keys.M, new PdfDate(DateTime.Now));

        var irefTable = document
            .GetType()
            .GetField("_irefTable", BindingFlags.Instance | BindingFlags.NonPublic).GetValue(document);
        var irefTableAdd = irefTable
            .GetType()
            .GetMethods()
            .Where(m => m.Name == "Add").Skip(1).FirstOrDefault();

        irefTableAdd.Invoke(irefTable, new object[] { sign });

        Elements.Add("/V", sign);

        Elements.Add("/Rect", rect);
        Flags = PdfAnnotationFlags.Print;
        Opacity = 1;
    }
}
Run Code Online (Sandbox Code Playgroud)

并使用:

var sig = new MyPdfSignatureField(document, new PdfRectangle(new XPoint(480, 33.75), new XPoint(612.2, 62.1)));
document.Pages[0].Annotations.Add(sig);
Run Code Online (Sandbox Code Playgroud)

这不是一个很好的解决方案,但它满足了我的基本要求。如果您有更好的想法,请告诉我。

这个类的想法来自这里:https://github.com/empira/PDFsharp/pull/11/files