C# - 每页上的 itextSharp 小计
Posted
技术标签:
【中文标题】C# - 每页上的 itextSharp 小计【英文标题】:C# - itextSharp subtotal on every page 【发布时间】:2016-09-26 12:09:48 【问题描述】:您可以在问题的编辑部分找到 C# 的解决方案。 特别感谢 Bruno Lowagie
我正在尝试通过 C# 中的 iTextSharp 创建发票。它工作得很好,但是当我尝试在每一页上打印小计时,我遇到了问题。
这是我用来为产品创建表格的部分代码:
PdfPTable table = new PdfPTable(5);
table.WidthPercentage = 100;
float[] widths = new float[] 10f, 30f, 120f, 30f, 30f ;
table.SetWidths(widths);
table.SkipLastFooter = true;
table.SpacingAfter = 10;
//Cells to Write Header & Footer
AddCell(table, "Pos.", PdfPCell.ALIGN_RIGHT, BORDER_LTB, 0);
AddCell(table, "Menge", PdfPCell.ALIGN_RIGHT, BORDER_TB);
AddCell(table, "Text", PdfPCell.ALIGN_LEFT, BORDER_TB);
AddCell(table, "Einzelpreis ", PdfPCell.ALIGN_RIGHT, BORDER_TB);
AddCell(table, "Summe", PdfPCell.ALIGN_RIGHT, BORDER_RTB);
AddCell(table, "SUBTOTAL", PdfPCell.ALIGN_LEFT, BORDER_LTB, 7, 4);
AddCell(table, "", PdfPCell.ALIGN_LEFT, BORDER_RTB);
table.HeaderRows = 2;
table.FooterRows = 1;
//Cells with positions of invoice
for (int i = 0; i < beleg.Einzel.Count; i++)
AddCell(table, beleg.Einzel[i].be_lfd_nr.ToString(), PdfPCell.ALIGN_LEFT, BORDER_LTB);
AddCell(table, beleg.Einzel[i].be_art_menge.ToString("N", _gbVars.NFI3D), PdfPCell.ALIGN_RIGHT, BORDER_TB);
AddCell(table, (beleg.Einzel[i].be_art_bez.ToString() + "\n" + beleg.Einzel[i].be_pos_text.ToString()).Trim(), PdfPCell.ALIGN_LEFT, BORDER_TB);
AddCell(table, beleg.Einzel[i].be_summe_preis.ToString("N", _gbVars.NFI2D), PdfPCell.ALIGN_RIGHT, BORDER_TB);
AddCell(table, beleg.Einzel[i].be_summe_netto.ToString("N", _gbVars.NFI2D), PdfPCell.ALIGN_RIGHT, BORDER_RTB);
table.SplitLate = false;
document.Add(table);
Border_RTB 等是我的边框样式的常量。
这段代码生成一个如下所示的表格:
+------------------------------------------+
| Pos | Menge | Text | Einzelpreis | Summe |
+------------------------------------------+
| 1 | 2 | Text | 10.00 | 20.00 |
+------------------------------------------+
| 2 | 1 | Text | 10.00 | 10.00 |
+------------------------------------------+
| 3 | 4 | Text | 10.00 | 40.00 |
+------------------------------------------+
| 4 | 2 | Text | 10.00 | 20.00 |
+------------------------------------------+
|SUBTOTAL | |
+------------------------------------------+
表格可以流到下一页,我想在页脚处写下每一页的小计。事件“OnEndPage”对我没有帮助,因为该事件不“知道”页面在表格中的哪个位置损坏。
谁能告诉我如何获得每一页的小计?有什么解决方案可以让我在页面上总结一些内容并将其打印在 pdf 文件上?
对不起,如果我的描述不是那么好。我的英语有问题。 :-/
编辑:
这里是解决方案。特别感谢布鲁诺。他已经为我指明了道路。 在我的版本中,小计不会在每一页上重置。我想这是我对问题的描述有误。
新的类:
public class Totals
public double subtotal = 0;
public double total = 0;
public class SubTotalEvent : IPdfPCellEvent
Double price;
Totals totals;
bool printTotals = false; //If we just whant to print out the Subtotal, this will be true.
bool CellWrittenOnce = false; //Bool to SUM price just once (if row flows to a second page)
public SubTotalEvent(Totals totals, double price)
printTotals = false;
this.totals = totals;
this.price = price;
public SubTotalEvent(Totals totals)
this.totals = totals;
printTotals = true;
public void CellLayout(PdfPCell cell, Rectangle position, PdfContentByte[] canvases)
if (printTotals)
PdfContentByte canvas = canvases[PdfPTable.TEXTCANVAS];
ColumnText.ShowTextAligned(canvas, Element.ALIGN_LEFT, new Phrase(totals.subtotal.ToString()), position.GetLeft(0) + 2, position.GetBottom(0) + 2, 0);
return;
if (!CellWrittenOnce)
totals.subtotal += price;
totals.total += price;
CellWrittenOnce = true;
创建表格的代码:
PdfPTable table = new PdfPTable(5);
table.WidthPercentage = 100;
float[] widths = new float[] 10f, 30f, 120f, 30f, 30f ;
table.SetWidths(widths);
table.SkipLastFooter = true;
table.SpacingAfter = 10;
AddCell(new PdfPCell(new Phrase("item")));
AddCell(new PdfPCell(new Phrase("amount")));
AddCell(new PdfPCell(new Phrase("text")));
AddCell(new PdfPCell(new Phrase("price")));
AddCell(new PdfPCell(new Phrase("sum")));
Totals totals = new Totals();
PdfPCell cel = new PdfPCell(new Phrase("Subtotal"));
cel.Colspan = 4;
table.AddCell(cel);
cel = new PdfPCell();
cel.CellEvent = new SubTotalEvent(totals);
table.AddCell(cel);
table.HeaderRows = 2;
table.FooterRows = 1;
for (int i = 0; i < beleg.Einzel.Count; i++)
AddCell(new PdfPCell(new Phrase(i.toString())));
AddCell(new PdfPCell(new Phrase("2")));
AddCell(new PdfPCell(new Phrase("ItemText")));
AddCell(new PdfPCell(new Phrase("10.00")));
cell = new PdfPCell(new Phrase("20.00"));
cell.CellEvent = new SubTotalEvent(totals, 20.00);
table.AddCell(cell);
table.SplitLate = false;
document.Add(table);
【问题讨论】:
你需要实现PdfPageEventHelper
类。每次创建页面时都会执行您的实现。请参阅本教程了解如何实现它mazsoft.com/blog/post/2008/04/30/…
我实现了PdfPageEventHelper
类。但是我如何在public override void OnEndPage(PdfWriter writer, Document doc)
中找出我在桌子上的位置?此时表已经写好了。此事件在document.Add(table);
调用
从您的问题描述中我无法立即清楚地了解某些事情,所以我不确定我是否有答案。以下假设是否正确: 1. 您的发票文档由一个可能跨越多个页面的大表组成。 2. 您的发票不是表格,您可以在生成文件时访问所有信息。 3.您要在该页面的页眉/页脚中打印由当前页面上所有产品组成的小计+直到当前页面的小计总和。
1.这些产品都在一个大的单表中,可以跨越多个页面。产品有描述,因此行高可能会有所不同。 2. 我的发票存储在 mysql-Database 中。我可以访问所有内容。 3. 是的,我想在产品表的页脚中打印小计 + 小计的总和。
【参考方案1】:
请查看SubTotal 示例。
首先我们创建一个类来跟踪小计和总计:
class Totals
double subtotal = 0;
double total = 0;
然后我们实现将在第 5 列中使用的PdfPCellEvent
接口:
class SubTotalEvent implements PdfPCellEvent
Double price;
Totals totals;
public SubTotalEvent(Totals totals, double price)
this.totals = totals;
this.price = price;
public SubTotalEvent(Totals totals)
this.totals = totals;
price = null;
@Override
public void cellLayout(PdfPCell cell, Rectangle position, PdfContentByte[] canvases)
if (price == null)
PdfContentByte canvas = canvases[PdfPTable.TEXTCANVAS];
ColumnText.showTextAligned(canvas, Element.ALIGN_LEFT,
new Phrase(String.valueOf(totals.subtotal)),
position.getLeft() + 2, position.getBottom() + 2, 0);
totals.subtotal = 0;
return;
totals.subtotal += price;
totals.total += price;
我们定义了两个构造函数:一个用于包含价格的普通单元格。一个用于包含小计的页脚单元格(在这种情况下,我们不传递价格)。
这就是我们使用这个单元格事件的方式:
public void createPdf(String dest) throws IOException, DocumentException
Totals totals = new Totals();
Document document = new Document();
PdfWriter.getInstance(document, new FileOutputStream(dest));
document.open();
PdfPTable table = new PdfPTable(5);
table.setWidths(new int[]1, 1, 1, 3, 3);
// header
table.addCell("Pos");
table.addCell("Menge");
table.addCell("Text");
table.addCell("Einzerpreis");
table.addCell("Summe");
// footer
PdfPCell cell = new PdfPCell(new Phrase("Subtotal"));
cell.setColspan(4);
table.addCell(cell);
cell = new PdfPCell();
cell.setCellEvent(new SubTotalEvent(totals));
table.addCell(cell);
// definitions
table.setHeaderRows(2);
table.setFooterRows(1);
// table body
for(int r = 0; r < 50; )
table.addCell(String.valueOf(++r));
table.addCell("1");
table.addCell("text");
table.addCell("10.0");
cell = new PdfPCell(new Phrase("10.0"));
cell.setCellEvent(new SubTotalEvent(totals, 10));
table.addCell(cell);
document.add(table);
// extra footer
table = new PdfPTable(5);
table.setWidths(new int[]1, 1, 1, 3, 3);
cell = new PdfPCell(new Phrase("Grand total"));
cell.setColspan(4);
table.addCell(cell);
table.addCell(String.valueOf(totals.total));
document.add(table);
document.close();
结果如下:
在第一页上,价格为 10 的商品每行有 46 行正文。在页脚中,我们有 460 的小计。在第二页上,价格为 10 的商品每行有 4 行正文。在页脚中,我们有一个小计 40。我们添加了一个额外的表格来模拟总计:500 的额外页脚。
【讨论】:
我们快到了!非常感谢!还有一个问题:例如,如果第 46 行的高度较大并且流过两页,那么第一页示例中的小计将是正确的。但在第二页上,它将是 50,而不是 40。但我用一个额外的 bool 变量修复了这个问题。对于所有读者:Bruno Lowagie 上面的代码是 JAVA 代码。对于 C#,您必须翻译。但是效果很好!惊人的。 :-)以上是关于C# - 每页上的 itextSharp 小计的主要内容,如果未能解决你的问题,请参考以下文章
带有 % extends 的每页上的 Django 页脚和页眉