2

我遇到了一个 Web API 的问题,它接收 PDF 文档(PDF 表单)作为 base64 编码字符串,将其放入文件中,填写一些字段(使用iTextSharpPdfStamper类)并将结果发送回为一个base64编码的字符串,它再次被制作成一个文件。

当我之后尝试使用Foxit Reader打开此文件/文档时,我仍然可以编辑这些字段,但是当我尝试使用Adob​​e Acrobat Reader进行相同操作时,这些字段不再可编辑。

在Foxit Reader中打开属性页并查看文档的安全属性时,它说一切都是允许的。但是,在Adob​​e Acrobat Reader中执行相同操作时,它表示不允许评论、填写字段和签名(这是在将其发送到 API 之前)。

不幸的是,我无法指定最终用户使用的 PDF 阅读器,所以我需要它是通用的。

我可以看到许多其他人发布了关于类似问题的帖子,但我没有在其中看到我的特定问题。

有谁知道这里发生了什么以及如何解决它?

编辑:添加代码

我正在这样做(略微简化,但相同):

PdfReader reader = new PdfReader("path to PDF form file");

stamp = new PdfStamper(reader, new FileStream("path to filled PDF form file", FileMode.Create));

AcroFields form = stamp.AcroFields;

IDictionary<string, AcroFields.Item> fs = form.Fields;

foreach (var f in fs)
{
    form.SetField(f.Key, "some value");
}

stamp.Close();

编辑:添加了示例 PDF 文件

示例文件只是在互联网上找到的东西,我记得它在这里的另一篇文章中有链接。

我尝试在此 PDF 表单中填写 2 个字段(名字和姓氏)。

一个文件是从 base64 字符串创建的表单。

d94b6076-983d-47db-b496-a0ba383deda4.pdf

第二文件是填写2个字段后的输出。

d94b6076-983d-47db-b496-a0ba383deda4_filled.pdf

4

1 回答 1

1

您的 PDF 包含混合 AcroForm/XFA 表单定义和使用权限签名。这两个事实都使填表复杂化。

使用权签名

您的 PDF 包含特殊类型的数字签名,即使用权签名。使用此类签名(使用 Adob​​e 为该任务生成的私钥)签名的 PDF 向 Adob​​e Reader 发出信号,表明在处理该 PDF 时应激活某些额外的 Adob​​e Reader 功能。

但是,如果签名无效,Adobe Reader 不仅不会激活那些额外的功能,它甚至会停用某些其他对未签名文档有效的功能。

PdfStamper像这样创造你的

stamp = new PdfStamper(reader, new FileStream("path to filled PDF form file", FileMode.Create));

该构造函数不会在附加模式下初始化标记。因此,加盖戳的 PDF 最终被编写为新的 PDF。这会使使用权限签名无效,并且 Adob​​e Reader 会在再次打开 PDF 时限制功能。

顺便说一句,在使用 Adob​​e Reader 打开 PDF 时,程序会告诉您:

Adobe 阅读器弹出窗口

但是,如果您在附加模式下进行标记,则使用权签名不会失效。您可以使用不同的构造函数来执行此操作:

stamp = new PdfStamper(reader, new FileStream("path to filled PDF form file", FileMode.Create), '\0', true);

最后true激活附加模式。

或者,您可以删除使用权限签名

PdfReader reader = new PdfReader("path to PDF form file");
reader.RemoveUsageRights();

结果不再有使用权签名,特别是没有无效的签名。因此,Adobe Reader 为该文件提供了标准功能。

不幸的是,保存 XFA 表单不在其中,因此 PDF 可以编辑但不能保存在 Adob​​e Reader 中。这可能不是你想要的。

在这些选项上,还可以看到这个旧答案;该答案谈到了“Reader-Enabling”,这是 Adob​​e 术语“应用使用权签名”。

混合 XFA/AcroForm 表单定义

该文件包含一个双重表单定义,一个是原生 PDF 表单类型的 AcroForm 表单定义,以及一个 XFA 表单定义,它是一种 XML 格式,仅将 PDF 用作传输容器。

XFA 表单定义提供更多功能,但只有 Adob​​e 软件完全支持。在混合表单的情况下,iText 对将数据填充到两个表单定义中的支持有限。但是,根据 XFA 定义中使用的确切功能,这可能会失败。

在这种情况下,您可能希望删除 XFA 表单定义并仅使用 AcroForm 之一。

reader.Catalog.GetAsDict(PdfName.ACROFORM).Remove(PdfName.XFA);

在这种方法上,请阅读这个答案和这个iText 文章

于 2018-11-02T10:33:07.123 回答