如何使用 Delphi 将数据从 XML 导入 PDF 表单?

Posted

技术标签:

【中文标题】如何使用 Delphi 将数据从 XML 导入 PDF 表单?【英文标题】:How to import data from XML to PDF Form using Delphi ? 【发布时间】:2017-11-07 16:44:41 【问题描述】:

我有一个 PDF 文件,其中包含文本、数组、图像等。还有文本字段。

我想知道是否以及如何导入格式正确的 XML 文件以填充这些文本字段,例如姓名、姓氏、地址...

我只是想从 Acrobat Reader 的菜单中执行类似的操作:版本-表单选项 - 导入数据但使用 Delphi 编程。

我认为我需要打开 PDF 并使用函数来解析 XML 文件才能填写表格,但目前我在网上没有找到任何好的建议来解释如何做这样的事情。

我希望我的问题是正确的,您可以回答我。

最好的问候。

【问题讨论】:

这在 Acrobat 的完整版中很简单,只需将其 COM 对象导入接口单元,然后在运行时创建它们。也许最新版本的 Reader 也有类似的可能性。 @MartynA 谢谢你的回答,所以我有点理解,但我不知道如何在 Delphi 代码中做到这一点。 好吧。 Acrobat COM 对象库之一是表单填写。您从类型库生成一个 Delphi 导入单元 (something_tlb.pas),然后只需实例化其中的对象以填写表单。 【参考方案1】:

这里是一些示例代码,它使用 Acrobat 的表单填充 COM 对象从使用 Turbopower 的 XML 合作伙伴库解析的 XML 数据中填充 Acrobat 字段。由于一些外部依赖关系,它不能按原样编译,并且需要一些重构,但它应该给你一个大致的想法。 Calibrate 例程添加了一些标记以帮助放置字段。

unit AcrobatXMLu;

interface

uses
  [...]
  Acrobat_tlb, //  main Acrobat COM wrapper
  AFORMAUTLib_TLB;  // Acrobat Forms COM objects

type
  TGenerateFlag = (gfShowAcrobat, gfUseDefaultValues, gfGenerateTestData, gfAddCalibration, gfAlwaysFillCheckboxes);
  TGenerateFlags = set of TGenerateFlag;

function CreateAcrobatFieldsInner(eStartNode: TxpElement;
  const Path: String; Flags : TGenerateFlags; const OutputFile : String): Boolean;

implementation

function CreateAcrobatFieldsInner(eStartNode: TxpElement;
  const Path: String; Flags : TGenerateFlags; const OutputFile : String): Boolean;
var
  List : TxpNodeList;
  X : TxpNode;
  N,
  Max : Integer;
  E : TxpElement;
  Align,
  S : String;
  AddedFields : TStringlist;

  procedure CreateField(E : TxpElement);
  var
    FieldName,
    FieldType,
    MappedFieldType : String;
    PageNo : Integer;
    ALeft,
    ARight,
    ATop,
    ABottom,
    AWidth,
    AHeight : Single;
    S : String;
    Field : IField;  //  Acrobat form field
    IsMultiLine : Boolean;
  begin
     FieldName := E.GetAttribute('Name');
     if CompareText(FieldName, 'Name1') = 0 then
       NoOp;
     FieldType := LowerCase(E.GetAttribute('PdfFieldType'));
     MappedFieldType := FieldType;
     if CompareText(MappedFieldType, 'checkbox') = 0 then
       MappedFieldType := 'text';
     IsMultiLine := CompareText(MappedFieldType, 'memo') = 0;
     if (CompareText(MappedFieldType, 'Text') = 0) or (CompareText(MappedFieldType, 'Memo') = 0) then
       MappedFieldType := 'text';

     Align := LowerCase(E.GetAttribute('Align'));
     S := E.GetAttribute('Page');
     if S = '' then
       S := TxpElement(E.ParentNode).GetAttribute('Page');
     if S <> '' then
       PageNo := StrToInt(S) - 1
     else
       PageNo := 0;
     S := GetHeritableAttribute(E, 'XPos', 'FieldXPos');
     ALeft := StrToInt(S);

     ATop := E.GetAttributeInt('YPos');

     S := GetHeritableAttribute(E, 'Width', 'FieldWidth');
     if S <> '' then
       AWidth := StrToInt(S)
     else
       AWidth := 60;
     S := GetHeritableAttribute(E, 'Height', 'FieldHeight');
     if S <> '' then
       AHeight := StrToInt(S)
     else
       AHeight := 20;
     ARight := ALeft + AWidth;
     ABottom := ATop - AHeight;

     try
       Field := Acrobat.Fields.Add(FieldName, MappedFieldType, PageNo, ALeft, ATop, ARight, ABottom) as IField;
       if True or (AddedFields.IndexOf(FieldName) < 0) then begin
         if CompareText(MappedFieldType, 'text') = 0 then begin
           S := GetHeritableAttribute(E, 'TextFont', 'FieldTextFont');
           if S <> '' then
             Field.Set_TextFont(S);
           S := GetHeritableAttribute(E, 'TextSize', 'FieldTextSize');
           if S <> '' then
             if not (CompareText(S, 'Auto') = 0) then
               Field.Set_TextSize(StrToInt(S));
           if IsMultiLine then
             Field.Set_IsMultiline(True);

           S := E.GetAttribute('DefaultValue');
           S := StringReplace(S, #10, #10#13, [rfReplaceAll]);
           if S <> '' then begin
             Field.Set_Value(S);
           end
           else begin
             if CompareText(FieldType, 'checkbox') = 0 then begin
               if gfAlwaysFillCheckboxes in Flags then
                 Field.Set_Value('X')
             end
             else begin
               if gfGenerateTestData in Flags then
                 Field.Set_Value(Format('(%s)', [FieldName]));
             end;
           end;
           if Align <> '' then
             Field.Set_Alignment(Align);
         end
         else begin
           if CompareText(MappedFieldType, 'checkbox') = 0 then begin
           end;
         end;
       end;
       if AddedFields.IndexOf(FieldName) < 0 then
         AddedFields.Add(FieldName)
     except
       ShowMessage('Error adding field ' + FieldName);
     end;
  end;

  procedure Calibrate;
  var
    X,
    Y,
    N,
    M,
    Page : Integer;
    Field : IField;  // Acrobat form field

    procedure AddField(X, Y : Integer);
    var
      S : String;
      FieldName : String;
    begin
      if X < 40  then
        S := Format('Y:%d', [Y])
      else
        S := Format('X:%d', [X]);
      FieldName := Format('X%dY%d', [X, Y]);
      Field := Acrobat.Fields.Add(FieldName, 'text', Page, X, Y, X + 40, Y - 15) as IField;
      Field.Set_TextFont('Courier');
      Field.Set_TextSize(10);
      Field.Set_Value(S);

    end;
  begin
    for Page := 0 to ((Acrobat.AcroApp.GetActiveDoc as CAcroAVDoc).GetPDDoc as CAcroPDDoc).GetNumPages - 1 do begin
      for N := 1 to 15 do
        AddField(40 * N, 0);
      N := 15;
      for M := 0 to 60 do
        AddField(0, N * M);
    end;
  end;

begin
  AddedFields := TStringlist.Create;
  AddedFields.Sorted := True;
  Result := False;
  try
    List := eStartNode.SelectNodes(Path);
    try
      Max := List.Length - 1;
      for N := 0 to Max do begin
        X := List.Item(N);
        if not (X is TxpElement) then Continue;
        E := TxpElement(X);
        S := E.GetAttribute('YPos');
        if S <> '' then
          CreateField(E);
      end;
      Result := True;
    finally
      List.Free;
    end;
    if gfAddCalibration in Flags then
      Calibrate;
    Acrobat.DocV.GetPDDoc.Save(PDSaveFull, OutputFile);
  finally
    AddedFields.Free;
  end;
end;

end.

正如编写的那样,这段代码添加了 XML 中定义的字段,并在执行过程中将它们填满。很明显;填充现有字段是一件小事。

一些用于 Delphi 的第 3 方 PDF 库也可能能够填写 Acrobat 字段。此外,命令行库 PDFtk 还可以填充 Acrobat 字段并执行 Acrobat COM 对象无法执行的其他操作,例如“展平” PDF 表单,它可以有效地将字段中的文本合并到主文档中,因此不再可编辑为形式。请参阅https://www.pdflabs.com/docs/pdftk-man-page/ 了解更多信息。

【讨论】:

非常感谢。我会进入它并尝试一切。它似乎给了我答案,或者至少是一条很好的路线。再次感谢

以上是关于如何使用 Delphi 将数据从 XML 导入 PDF 表单?的主要内容,如果未能解决你的问题,请参考以下文章

SQLite - 如何从 XML 文件插入 JPG 图像(使用 Delphi 2009)

如何使用delphi将Excel文件导入Access数据库

如何在不写入磁盘的情况下将 XML 从 Delphi 传递到 C#?

delphi 如何在Delphi中执行将Excel表格里的内容导入数据库中相应表

XSD 包含导入的 xsd 时的 Delphi XE2 XML 数据绑定向导错误

Delphi 中如何将EXCEL表导入到已知的数据库中,再进行操作