FormSubmitXml.vb
 '' '' This code is part of Document Solutions for PDF demos. '' Copyright (c) MESCIUS inc. All rights reserved. '' Imports System.Drawing Imports System.IO Imports System.Linq Imports System.Xml Imports System.Text Imports System.Collections.Generic Imports GrapeCity.Documents.Pdf Imports GrapeCity.Documents.Pdf.AcroForms Imports GrapeCity.Documents.Pdf.Actions Imports GrapeCity.Documents.Pdf.Annotations Imports GrapeCity.Documents.Text '' NOTE: This sample is obsolete as of DsPdf v3. Please see the new FormDataSubmit '' sample for a better solution. '' '' This sample creates an AcroForm PDF that can be submitted to the server. '' It relies on the server to put the submitted data into an XML, '' import that XML into a PDF containing a similar form, '' And send the form with loaded data back to the client. '' Note that the produced PDF with filled form fields '' Is shown in the client browser's default PDF viewer. '' The code Is similar to FormSubmit. Public Class FormSubmitXml Function CreatePDF(ByVal stream As Stream) As Integer Dim doc = New GcPdfDocument() Dim page = doc.NewPage() Dim rc = Util.AddNote("Fill the fields in the form and click 'Submit' to send it back to the server. " + "The sample server will put the submitted data into an XML, feed that XML" + "to a PDF with a different but compatible form, and send the resulting form" + "filled with the submitted data back to your browser." + "Note that the form with the submitted data is opened in the browser's default PDF viewer," + "and does not have the 'Submit' and 'Reset' buttons.", page) Dim g = page.Graphics Dim tf = New TextFormat() With {.Font = StandardFonts.Times, .FontSize = 14} Dim ip = New PointF(72, rc.Bottom + 36) Dim fldOffset = 72 * 2 + 46 Dim fldHeight = tf.FontSize * 1.2F Dim dY = 32 '' Text field g.DrawString("First name:", tf, ip) Dim fldFirstName = New TextField() With {.Name = "FirstName", .Value = "John"} fldFirstName.Widget.Page = page fldFirstName.Widget.Rect = New RectangleF(ip.X + fldOffset, ip.Y, 72 * 3, fldHeight) fldFirstName.Widget.DefaultAppearance.Font = tf.Font fldFirstName.Widget.DefaultAppearance.FontSize = tf.FontSize doc.AcroForm.Fields.Add(fldFirstName) ip.Y += dY '' Text field g.DrawString("Last name:", tf, ip) Dim fldLastName = New TextField() With {.Name = "LastName", .Value = "Smith"} fldLastName.Widget.Page = page fldLastName.Widget.Rect = New RectangleF(ip.X + fldOffset, ip.Y, 72 * 3, fldHeight) fldLastName.Widget.DefaultAppearance.Font = tf.Font fldLastName.Widget.DefaultAppearance.FontSize = tf.FontSize doc.AcroForm.Fields.Add(fldLastName) ip.Y += dY '' Checkbox g.DrawString("Subscribe to Mailing List:", tf, ip) Dim fldCheckbox = New CheckBoxField() With {.Name = "Subscribe", .Checked = True} fldCheckbox.Widget.Page = page fldCheckbox.Widget.Rect = New RectangleF(ip.X + fldOffset, ip.Y, fldHeight, fldHeight) doc.AcroForm.Fields.Add(fldCheckbox) ip.Y += dY '' Multiline TextBox g.DrawString("Additional information:", tf, ip) Dim fldAdditionalInfo = New TextField() With {.Name = "AdditionalInfo", .Multiline = True} fldAdditionalInfo.Widget.Page = page fldAdditionalInfo.Widget.Rect = New RectangleF(ip.X + fldOffset, ip.Y, 72 * 3, fldHeight * 2) fldAdditionalInfo.Widget.DefaultAppearance.Font = tf.Font fldAdditionalInfo.Widget.DefaultAppearance.FontSize = tf.FontSize doc.AcroForm.Fields.Add(fldAdditionalInfo) ip.Y += dY * 2 '' Submit form button: Dim btnSubmit = New PushButtonField() btnSubmit.Widget.Rect = New RectangleF(ip.X + fldOffset, ip.Y, 72, fldHeight) btnSubmit.Widget.ButtonAppearance.Caption = "Submit" btnSubmit.Widget.Highlighting = HighlightingMode.Invert btnSubmit.Widget.Page = page '' The URL for the submission btnSubmit.Widget.Activate = New ActionSubmitForm("/Samples/HandleFormSubmitXml") doc.AcroForm.Fields.Add(btnSubmit) '' Reset form button: Dim btnReset = New PushButtonField() btnReset.Widget.Rect = New RectangleF(ip.X + fldOffset + 72 * 1.5F, ip.Y, 72, fldHeight) btnReset.Widget.ButtonAppearance.Caption = "Reset" btnReset.Widget.Highlighting = HighlightingMode.Invert btnReset.Widget.Page = page btnReset.Widget.Activate = New ActionResetForm() doc.AcroForm.Fields.Add(btnReset) ip.Y += dY '' Done doc.Save(stream) Return doc.Pages.Count End Function '' '' NOTE: the code below Is used by the web sample browser controller When the form '' prepared by this sample Is submitted, it Is Not directly called by the CreatePDF() method. '' '' Creates a GcPdfDocument, loads an AcroForm PDF into it, And fills it with data '' using the GcPdfDocument.ImportFormDataFromXML() method. '' '' This method Is called by the samples controller when the form prepared by this sample '' Is submitted by the user. The samples controller parses the client response And builds '' the 'values' collection filling it with the submitted field values, then calls '' this method to prepare the XML, imports it into a newly created PDF, And returns '' the resulting PDF to the controller, which sends it back to the client. Public Shared Function ImportFormData(ByVal values As List(Of FieldExportEntry)) As Stream Dim pdf = New GcPdfDocument() Using fs = New FileStream(Path.Combine("Resources", "PDFs", "ImportFormXML.pdf"), FileMode.Open, FileAccess.Read) pdf.Load(fs) Using ms = New MemoryStream() SaveFieldsToXML(values, ms) ms.Seek(0, SeekOrigin.Begin) pdf.ImportFormDataFromXML(ms) End Using Dim outMs = New MemoryStream() pdf.Save(outMs) outMs.Seek(0, SeekOrigin.Begin) Return outMs End Using End Function '' Represents a form field And its value(s) for export to XML. Public Class FieldExportEntry Public Property Name As String Public Property Values As List(Of String) '' Note: this sample does Not support child fields: '' public List<FieldTreeNode> Children { get set } End Class '' Saves the fields And their values to a stream. '' '' This method Is similar to GcPdfDocument.ExportFormDataToXML(), with the following '' important limitations: '' - it does Not support child fields (field.Children collection) '' - it does Not handle fields with names that are Not valid XML names (xfdf:original). Public Shared Sub SaveFieldsToXML(ByVal values As List(Of FieldExportEntry), ByVal stream As Stream) Dim xws = New XmlWriterSettings() With { .Indent = True, .CloseOutput = False, .Encoding = Encoding.UTF8 } Using xw = XmlWriter.Create(stream, xws) xw.WriteStartElement("fields") xw.WriteAttributeString("xmlns", "xfdf", Nothing, "http:''ns.adobe.com/xfdf-transition/") For Each ftn In values xw.WriteStartElement(ftn.Name) For Each v In ftn.Values xw.WriteStartElement("value") '' NOTE: the values In the array are formed by the client PDF viewer, '' And it represents 'on' checkbox values as 'true', while ImportFormDataFromXML '' expects 'on' values to be represented as "Yes" (that's how ExportFormDataToXML '' works, similar to Acrobat). This here Is a quick And dirty hack just for the '' sake of this sample: If v = "true" Then xw.WriteString("Yes") Else xw.WriteString(v) End If xw.WriteEndElement() Next xw.WriteEndElement() Next xw.WriteEndElement() End Using End Sub End Class