我使用iText7生成一些带有可编辑表单字段的PDF文档。 我的代码使用以下代码将字段添加到PDF文档中:
@Override
public void draw(DrawContext drawContext) {
super.draw(drawContext);
PdfAcroForm form = PdfAcroForm.getAcroForm(drawContext.getDocument(), true);
PdfFormField field = isMultiline
? PdfFormField.createMultilineText(drawContext.getDocument(), getOccupiedAreaBBox(), fieldName, "")
: PdfFormField.createText(drawContext.getDocument(), getOccupiedAreaBBox(), fieldName, "");
field.setFontSize(defaultFontSize - 1);
form.addField(field);
}
这是iText自定义渲染器类的draw
方法。
生成的PDF文档按预期工作。我可以在Adobe Acrobat Reader中打开它,填写可编辑字段,保存,重新打开等等。
我的一个要求是允许以编程方式展平表单(显然在人类用户填写后),以便之后编辑被禁用。
我使用此代码执行此操作:
PdfDocument pdf = ... // Read a PDF stream and get the document object
PdfAcroForm form = PdfAcroForm.getAcroForm(pdf, false);
form.flattenFields();
pdf.close()
现在的问题是,当我用PDF阅读器打开拼合的PDF时,字段是空白的。
我做了一些调试,看起来这个问题只有在我尝试用 Adobe Reader (版本XI)压缩编辑(和保存)的字段时才会出现。
如果我以编程方式编辑字段,例如使用iText' PdfFormField#setValue(String)
方法,展平的PDF可以很好地呈现这些值。
似乎Adobe Reader正在设置表单字段的某些属性,以防止它们被正确展平......实际情况是这样吗?它有什么办法吗?
谢谢
----------------编辑---------------------- <登记/>
显然问题是由Adobe Reader引起的:由于某种原因,它保存的字段定义没有/XObject
子类型,iText7在form.flattenFields()
方法中检查。
这就是Eclipse调试器中字段字符串的样子:
由iText7生成:
<</BBox [0 0 523 109 ] /Filter /FlateDecode /Length 95 /Matrix [1 0 0 1 0 0 ] /Resources <</Font <</F1 23 0 R >> >> /Subtype /Form /Type /XObject >>
Acrobat Reader保存:
<</BBox [0.0 0.0 523.0 109.0 ] /Filter /FlateDecode /Length 107 /Resources <</Font <</F1 21 0 R >> >> >>
如您所见,/Subtype
部分缺失。每当您尝试禁用表单字段时,例如PdfAcroForm#flattenFields()
或PdfFormField#setReadOnly(boolean)
,iText会尝试使用以下代码在PdfCanvas
上重绘其内容:
if(xObject != null && xObject.getPdfObject().get(PdfName.Subtype) != null) {
[...]
canvas.addXObject(xObject, ...);
}
跳过哪一个,因为Reader字段不再是/XObject
。
我目前的解决方法是用新的字段替换字段:
Map<String,PdfFormField> fields = form.getFormFields();
Set<String> keys = new HashSet<>(fields.keySet()); // avoids concurrent modifications
for(String fieldName : keys) {
PdfFormField field = fields.get(fieldName);
PdfDictionary fieldObject = field.getPdfObject();
PdfFormField newField = field.isMultiline()
? PdfFormField.createMultilineText(pdf, fieldObject.getAsRectangle(PdfName.Rect), fieldName,
field.getValueAsString(), defaultFont, defaultFontSize)
: PdfFormField.createText(pdf, fieldObject.getAsRectangle(PdfName.Rect), fieldName,
field.getValueAsString(), defaultFont, defaultFontSize);
form.replaceField(fieldName, newField);
}
form.flattenFields();
虽然字体信息不是动态设置的,但它是有效的,但这是another story。
答案 0 :(得分:0)
所以最后我用iText RUPS打开了PDF。看起来Acrobat Reader确实设置了子类型,但它是/Widget
。我找到的最后一个解决方法是手动将子类型设置为/XObject
,然后调用form.flatten()
。这也保留了该字段的字体信息。
请注意,手动更改子类型可能会破坏其他内容。我不是一个PDF专家,所以我无法确定。但是现在我只是改变子类型并立即变平,所以它不应该是一个大问题。
此代码的最终版本如下所示:
for(String fieldName : keys) {
PdfFormField field = fields.get(fieldName);
PdfDictionary fieldObject = field.getPdfObject();
if(fieldObject.get(PdfName.Subtype) != PdfName.XObject) {
fieldObject.put(PdfName.Subtype, PdfName.XObject);
}
}
form.flattenFields();
pdf.close();
这样看起来它按预期工作。
答案 1 :(得分:0)
我遇到了同样的问题。看来Adobe Reader是某种原因造成的。我为此找到的另一种解决方案似乎是将字段设置为只读(iText7):
using (iText.Kernel.Pdf.PdfDocument pdfStamper = new iText.Kernel.Pdf.PdfDocument(pdfReader, pdfWriter))
{
iText.Forms.PdfAcroForm pdfForm = iText.Forms.PdfAcroForm.GetAcroForm(pdfStamper, true);
if (pdfForm != null)
{
System.Collections.Generic.IDictionary<String, iText.Forms.Fields.PdfFormField> FormFields = pdfForm.GetFormFields();
foreach (KeyValuePair<String, iText.Forms.Fields.PdfFormField> kv in FormFields)
{
FormFields[kv.Key].SetReadOnly(true);
}
}
WProps.SetFullCompressionMode(false);
pdfStamper.GetWriter().SetCloseStream(false);
}