预定义模板生成多页PDF文档

6
我有一个需求,需要使用一些预定义的公司模板来生成PDF发票报告。我能够使用iTextSharp创建/生成单页PDF报告。

问题:当发票声明跨越多页时会出现问题。我无法将报告(发票声明)扩展到下一页(第2页)。如果所有数据无法放在一页上,应该写在第二页上,同时仍使用公司模板。
该模板位于以下路径:
HostingEnvironment.MapPath("~/Content/InvoiceTemplate/invoiceTemplate.pdf")

我正在使用iTextSharp库创建文档。以下是用于生成PDF的代码:
public class pdfStatementController : Controller {

        Models.DYNAMICS_EXTEntities _db = new Models.DYNAMICS_EXTEntities();

        //
        // GET: /pdfStatement/


        public ActionResult SendPdfStatement(string InvoiceNumber) {
            try {
                InvoiceNumber = InvoiceNumber.Trim();

                ObjectParameter[] parameters = new ObjectParameter[1];
                parameters[0] = new ObjectParameter("InvoiceNumber", InvoiceNumber);

                List<Models.Statement> statementList = new List<Models.Statement>();
                statementList = _db.ExecuteFunction<Models.Statement>("uspInvoiceStatement", parameters).ToList<Models.Statement>();
                pdfStatementController.WriteInTemplate(statementList);

                return RedirectToAction("Invoice", "Invoice", new { id = statementList.FirstOrDefault().Customer_ID.ToString().Trim() });
            } catch (Exception e) {
                return View("Error");
            }
        } 

    public static void WriteInTemplate(List<Models.Statement> statementList) {
        string invoiceNumber = statementList.FirstOrDefault().Invoice.ToString().Trim();
        string month = null;
        string day = null;
        string year = null;

        PdfReader pdfReader = new PdfReader(
                                  HostingEnvironment.MapPath(
                                       "~/Content/InvoiceTemplate/invoiceTemplate.pdf"));
        FileStream fileStream = new FileStream(
                                   HostingEnvironment.MapPath(
                                      "~/Content/reports/" + invoiceNumber + ".pdf"),
                                      FileMode.Create);
        PdfStamper pdfStamper = new PdfStamper(pdfReader, fileStream);
        AcroFields pdfFields = pdfStamper.AcroFields;

        pdfFields.SetField("BillToCompany", statementList.FirstOrDefault().BillToCompany.ToString().Trim().ToUpper());
        pdfFields.SetField("BillToContact", statementList.FirstOrDefault().BillToContact.ToString().Trim().ToUpper());
        pdfFields.SetField("CustomerId", statementList.FirstOrDefault().Customer_ID);
        pdfFields.SetField("InvoiceNumber", statementList.FirstOrDefault().Invoice.ToString().Trim());
        pdfFields.SetField("JobNumber", statementList.FirstOrDefault().JobNumber.ToString().Trim());
        pdfFields.SetField("Caller", statementList.FirstOrDefault().Caller.ToString().Trim());

        pdfStamper.FormFlattening = true; // generate a flat PDF 
        pdfStamper.Close();
        pdfReader.Close();
    }
}

你是在每个页面都重复使用相同的模板,还是在第二个页面使用不同的模板? - kuujinbo
我在每个页面上使用相同的模板。 - 14578446
2个回答

7

你的代码看起来很不错,只是缺少一些中间步骤。

由于每个页面都使用相同的PDF模板(需要生成两个或更多页面时),所以不要直接使用PdfStamper将内容添加到Document中,而是使用PdfSmartCopyPdfCopy对象。

仍然需要PdfStamper。但是,在这种情况下,它被用于创建一个内存中的(单个)页面,该页面随着您遍历Models.Statement集合而填充数据。

换句话说,PdfSmartCopy/PdfCopy将您的报表作为整体(总页数)进行维护,PdfStamper作为缓冲区,逐页将您的各个报表添加到PDF中。以下是一个简单的工作示例HTTP hander (.ashx):

<%@ WebHandler Language="C#" Class="copyFillTemplate" %>
using System;
using System.Collections.Generic;
using System.IO;
using System.Web;
using iTextSharp.text;
using iTextSharp.text.pdf;

public class copyFillTemplate : IHttpHandler {
  public void ProcessRequest (HttpContext context) {
    HttpServerUtility Server = context.Server;
    HttpResponse Response = context.Response;
    Response.ContentType = "application/pdf";
// template used to test __this__ example;
// replace with __your__ PDF template
    string pdfTemplatePath = Server.MapPath(
      "~/app_data/template.pdf"
    );
// this example's test data; replace with __your__ data collection   
    List<Statement> statementList = Statement.GetStatements();

// COPY FROM HERE

    using (Document document = new Document()) {
// PdfSmartCopy reduces PDF file size by reusing parts
// of the PDF template, but uses more memory. you can
// replace PdfSmartCopy with PdfCopy if memory is an issue
      using (PdfSmartCopy copy = new PdfSmartCopy(
        document, Response.OutputStream)
      ) 
      {
        document.Open();
// used to test this example
        int counter = 0;
// generate one page per statement        
        foreach (Statement statment in statementList) {
          ++counter;
// replace this with your PDF form template          
          PdfReader reader = new PdfReader(pdfTemplatePath);
          using (var ms = new MemoryStream()) {
            using (PdfStamper stamper = new PdfStamper(reader, ms)) {
              AcroFields form = stamper.AcroFields;
// replace this with your field data for each page
              form.SetField("title", counter.ToString());
              stamper.FormFlattening = true;
            }
            reader = new PdfReader(ms.ToArray());
// add one page at a time; assumes your template is only one page.
// if your template is more than one page you will need to 
// call GetImportedPage() for each page in your template
            copy.AddPage(copy.GetImportedPage(reader, 1));
          }
        }
      }

// COPY TO HERE

    }
  }
  public bool IsReusable { get { return false; } }

  public class Statement {
    public string FieldName, FieldValue;
    public static List<Statement> GetStatements() {
      List<Statement> s = new List<Statement>();
      for (int i = 0; i < 5; ++i) {s.Add(new Statement());}
      return s;
    }
  }
}

希望内联注释能够有所帮助。你需要移除或替换我用来测试示例代码的某些部分。


只有一个问题:当我调用方法 public void ProcessRequest (HttpContext context) 时,我应该传递什么参数给 context?您能够粘贴调用此方法的方法吗? - 14578446
ProcessRequest 是特定于实现 IHttpHandler 接口的 HTTP 处理程序 (.ashx)。因此,如果您不是通过 HTTP 处理程序在 Web 上下文中提供 PDF,则不能调用此方法。只需使用上面的部分,包括 using (Document document... 块/语句。如果您正在运行 Windows Forms/桌面应用程序,则需要将 Response.OutputStream 替换为 FileStream - kuujinbo
作为.NET的初学者,我无法理解您上面的评论。我只在Web应用程序中使用它。我已经编辑了我的代码,以便更详细地描述我的应用程序。我正在使用MVC3,因此我的类继承控制器类。我知道您的代码可以为我工作,只需要将各个部分组合起来。请帮助我找到这里的HTTP上下文对象。 - 14578446
我可以做到... @kuujinbo 你是个明星,非常感谢你... 给你的回答打5颗星,再次感谢。 :) - 14578446
没问题,每个人都曾经是初学者 :) 问题是我不做MVC3(或根本不做MVC)。 我编辑了上面的代码示例,请复制从这里开始复制复制到此处结束之间的部分,并将其放入您的WriteInTemplate方法中。 另外,请进行内联注释中指定的替换(您可能需要进行一些实验)。 您不需要复制上面的整个ProcessRequest方法。 - kuujinbo

0

虽然最后一个答案非常好,帮助我解决了问题,但我在这里总结一下问题。

问题:我有一个场景需要生成公司提供的模板中的多页PDF文档。需要生成发票报表并通过Microsoft Outlook电子邮件客户端将其附加到电子邮件中。

我使用MVC3、ASP.NET 4.0、Entity Framework

解决方案:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Hosting;
using System.Web.Mvc;
using System.Data;
using System.Data.Objects;
using System.IO;
using iTextSharp;
using iTextSharp.text;
using iTextSharp.text.html;
using iTextSharp.text.pdf;
using iTextSharp.text.xml;

namespace InvoiceSearchTool.Controllers

{


 public class pdfStatementController : Controller
{
    Models.DYNAMICS_EXTEntities _db = new Models.DYNAMICS_EXTEntities();
    //
    // GET: /pdfStatement/

    public ActionResult SendPdfStatement(string InvoiceNumber)
    {
        try
        {

            InvoiceNumber = InvoiceNumber.Trim();


            List<Models.Statement> statementList = new List<Models.Statement>();
            //this is if you use entity framework
            {  
               ObjectParameter[] parameters = new ObjectParameter[1];
            parameters[0] = new ObjectParameter("InvoiceNumber", InvoiceNumber);

               statementList = _db.ExecuteFunction<Models.Statement>("uspInvoiceStatement", parameters).ToList<Models.Statement>();
             }

            //others can simply use line like
            //statementList = GetStatementList(inviceNumber);

            pdfStatementController.WriteInTemplate(statementList);

            return RedirectToAction("Invoice", "Invoice", new { id = statementList.FirstOrDefault().Customer_ID.ToString().Trim() });
        }
        catch (Exception e)
        {
            return View("Error");
        }


    }



public static void WriteInTemplate(List<Models.Statement> statementList)
    {

        try

        {

            string invoiceNumber = statementList.FirstOrDefault().Invoice.ToString().Trim();

            using (Document document = new Document())
            {
                FileStream fileStream = new FileStream(HostingEnvironment.MapPath("~/Content/reports/" + invoiceNumber + ".pdf"), FileMode.Create);
                using (PdfSmartCopy smartCopy = new PdfSmartCopy(document, fileStream))
                {
                    document.Open();

                    PdfReader pdfReader = new PdfReader(HostingEnvironment.MapPath("~/Content/InvoiceTemplate/invoiceTemplate.pdf"));
                            using (var memoryStream = new MemoryStream())
                            {
                                using (PdfStamper pdfStamper = new PdfStamper(pdfReader, memoryStream))
                                {
                                    string month = null;
                                    string day = null;
                                    string year = null;

                                    AcroFields pdfFields = pdfStamper.AcroFields;
                                    {//billing address
                                        pdfFields.SetField("BillToCompany", statementList.FirstOrDefault().BillToCompany.ToString().Trim().ToUpper());
                                        pdfFields.SetField("BillToContact", statementList.FirstOrDefault().BillToContact.ToString().Trim().ToUpper());

                                        pdfFields.SetField("ShipToCompany", statementList.FirstOrDefault().ShipToCompany.ToString().Trim().ToUpper());
                                        pdfFields.SetField("ShipToContact", statementList.FirstOrDefault().ShipToContact.ToString().Trim().ToUpper());
                                        pdfFields.SetField("PONumber", statementList.FirstOrDefault().PurchaseOrderNo.ToString().Trim());
                                        pdfFields.SetField("OrderNumber", statementList.FirstOrDefault().Order_Number.ToString().Trim());
                                        pdfFields.SetField("ShippingMethod", statementList.FirstOrDefault().Shipping_Method.ToString().Trim());
                                        pdfFields.SetField("PaymentTerms", statementList.FirstOrDefault().Payment_Terms.ToString().Trim());
                                   }
                                   pdfStamper.FormFlattening = true; // generate a flat PDF 

                                }
                                pdfReader = new PdfReader(memoryStream.ToArray());
                                smartCopy.AddPage(smartCopy.GetImportedPage(pdfReader, 1));
                            }
                }
            }

            emailController.CreateMessageWithAttachment(invoiceNumber);
        }
        catch (Exception e)
        {
        }

    }

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Mvc;
using Outlook = Microsoft.Office.Interop.Outlook;
using System.Net;
using System.Net.Mail;
using System.Web.Hosting;
using System.Net.NetworkInformation;
using System.Data.Objects;

namespace InvoiceSearchTool.Controllers
{


 public class emailController : Controller

 {
    //
    // GET: /email/
    public static void CreateMessageWithAttachment(string invoiceNumber)
    {
        try
        {
            Outlook.Application oApp = new Outlook.Application();
            Outlook.MailItem email = (Outlook.MailItem)(oApp.CreateItem(Outlook.OlItemType.olMailItem));

            Models.DYNAMICS_EXTEntities _db = new Models.DYNAMICS_EXTEntities();

            string recipient = null;
            string messageBody = null;
            #region set email recipients
            {
                ObjectParameter[] parameters = new ObjectParameter[1];
                parameters[0] = new ObjectParameter("InvoiceNumber", invoiceNumber);

                List<Models.EmailAddress> emailList = _db.ExecuteFunction<Models.EmailAddress>("uspGetEmailAddress", parameters).ToList<Models.EmailAddress>();
                if(!string.IsNullOrEmpty(emailList[0].Email.ToString()))
                    recipient = emailList[0].Email.ToString().Trim();
                else
                    recipient = " ";
                email.Recipients.Add(recipient);
            }
            #endregion

            //email subject                 
            email.Subject = "Invoice # " + invoiceNumber;

            #region set email Text
            {
                Models.EmailText emailText = _db.ExecuteFunction<Models.EmailText>("uspEmailText").SingleOrDefault();

                messageBody = emailText.EmailTextLine1.ToString().Trim() + "\n\n\n\n\n\n\n\n\n";
                messageBody += emailText.EmailTextLine2.ToString().Trim() + "\n";
                messageBody += emailText.EmailTextLine3.ToString().Trim();

                email.Body = messageBody;
            }
            #endregion

            #region email attachment
            {
                string fileName = invoiceNumber.Trim();
                string filePath = HostingEnvironment.MapPath("~/Content/reports/");
                filePath = filePath + fileName + ".pdf";
                fileName += ".pdf";
                int iPosition = (int)email.Body.Length + 1;
                int iAttachType = (int)Outlook.OlAttachmentType.olByValue;
                Outlook.Attachment oAttach = email.Attachments.Add(filePath, iAttachType, iPosition, fileName);
            }
            #endregion

            email.Display();
            //uncomment below line to SendAutomatedEmail emails atomaticallly
            //((Outlook.MailItem)email).Send(); 
        }
        catch (Exception e)
        {

        }

    }

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接