С# - промежуточный итог itextSharp на каждой странице

Решение для C# вы можете найти в Edit-Part of the Question. Особая благодарность Бруно Лоуаги

Я пытаюсь создать счет через 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);

person Martin S.    schedule 26.09.2016    source источник
comment
Вам нужно реализовать класс PdfPageEventHelper. Ваша реализация будет выполняться каждый раз при создании страницы. См. этот учебник о том, как это реализовать mazsoft.com/blog/post/2008/04/30/   -  person Darren Wainwright    schedule 26.09.2016
comment
Я реализую класс PdfPageEventHelper. Но как я могу понять в public override void OnEndPage(PdfWriter writer, Document doc), где я нахожусь в таблице? На данный момент таблица уже написана. Это событие вызывается в document.Add(table);   -  person Martin S.    schedule 26.09.2016
comment
Некоторые вещи не сразу понятны мне из вашего описания проблемы, поэтому я не уверен, что у меня еще есть ответ. Верны ли следующие предположения: 1. Ваш документ счета состоит из одной большой таблицы, которая может занимать несколько страниц. 2. Ваш счет-фактура не является формой, у вас есть доступ ко всей информации при формировании документа. 3. Вы хотите распечатать промежуточный итог, состоящий из всех продуктов на текущей странице + сумма промежуточных итогов до текущей страницы в верхнем/нижнем колонтитуле этой страницы.   -  person Samuel Huylebroeck    schedule 26.09.2016
comment
1. Все продукты находятся в одной большой таблице, которая может занимать несколько страниц. Продукты имеют описание, поэтому высота строки может варьироваться. 2. Мой счет хранится в базе данных mysql. У меня есть доступ ко всему. 3. Да, я хочу распечатать промежуточные итоги + сумму промежуточных итогов до текущей страницы в нижнем колонтитуле таблицы продуктов.   -  person Martin S.    schedule 26.09.2016


Ответы (1)


Пожалуйста, взгляните на Пример промежуточного итога.

Сначала мы создаем класс для отслеживания промежуточного итога и итога:

class Totals {
    double subtotal = 0;
    double total = 0;
}

Затем мы реализуем интерфейс PdfPCellEvent, который будем использовать для столбца 5:

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();
}

Результат выглядит следующим образом:

введите здесь описание изображения

На первой странице у нас есть 46 основных строк для каждого товара с ценой 10. В нижнем колонтитуле у нас есть промежуточный итог 460. На второй странице у нас есть 4 основных строки для каждого товара с ценой 10. В нижнем колонтитуле , у нас есть промежуточный итог 40. Мы добавили дополнительную таблицу, чтобы имитировать дополнительный нижний колонтитул для общего итога: 500.

person Bruno Lowagie    schedule 26.09.2016
comment
Мы почти на месте! Большое тебе спасибо! Оставалась одна проблема: если, например, строка 46 будет иметь большую высоту и охватывать две страницы, промежуточный итог в вашем примере на первой странице будет правильным. Но на второй странице будет 50, а не 40. Но я исправил это с помощью дополнительной логической переменной. Для всех читателей: приведенный выше код от Бруно Лоуаги — это JAVA-код. Для С# вам нужно перевести. Но работает очень хорошо! Потрясающий. :-) - person Martin S.; 27.09.2016