我正在生成一个PdfPTable并将其添加到章节中,然后将章节添加到我的文档中。我的表有几个单元格,其中RowSpan> 1,有时那些跨越页面边界,当发生这种情况时,我希望再次出现跨越的单元格。换句话说,如果我的表看起来像这样:
+------------------+------------------+
| Cell 1 RowSpan=1 | Cell 2 RowSpan=3 |
+------------------+------------------+
| Cell 3 RowSpan=1 | |
+------------------+------------------+
| Cell 4 RowSpan=1 | |
+------------------+------------------+
但是在打印时,第2行和第3行之间有分页符,我希望表格看起来像这样:
+--------+--------+
| Cell 1 | Cell 2 |
+--------+ |
| Cell 3 | |
+--------+--------+
Page Break
+--------+--------+
| Cell 4 | Cell 2 |
+--------+--------+
这要归功于我在这里学到的一个技巧,我有一个实现IPdfPCellEvent的类,它实例化并附加到具有RowSpan>的单元格。 1:
public class CellEvent : IPdfPCellEvent
{
Phrase m_phrase;
bool m_first = true;
public CellEvent(Phrase phrase)
{
m_phrase = phrase;
}
void IPdfPCellEvent.CellLayout(PdfPCell cell, Rectangle r, PdfContentByte[] canvases)
{
if (m_first)
m_first = false;
else
{
ColumnText ct = cell.Column;
ct.Canvases = canvases;
m_phrase.Leading = m_phrase.Font.Size;
ct.AddElement(m_phrase);
ct.Go();
}
}
}
如果多次调用CellLayout(),则会发生后续调用,因为RowSpan已经溢出到新页面(我上面描述的确切情况),因此我手动重新绘制单元格文本。这很有效。
除。
如果我的细胞2高于我的细胞1:
+------------------+-------------------------+
| Cell 1 RowSpan=1 | Cell 2 Line 1 RowSpan=2 |
| | Cell 2 Line 2 |
+------------------+-------------------------+
| Cell 3 RowSpan=1 | |
+------------------+-------------------------+
表格在这两行之间断开,我看到的是:
+--------+---------------+
| Cell 1 | Cell 2 Line 1 |
| | Cell 2 Line 2 |
+--------+---------------+
Page Break
+--------+---------------+
| Cell 3 | Cell 2 Line 1 |
+--------+---------------+
因为,当然,表格不知道我想在第2行中绘制两行文本。
我可以将第2行设置为对于两行总是足够高,但是如果表恰好落在第1行和第2行之间没有分页符的情况下,它会生成不必要的白色第2行中的空格:
+--------+---------------+
| Cell 1 | Cell 2 Line 1 |
| | Cell 2 Line 2 |
+--------| |
| Cell 3 | |
| | | <- un-necessary space
+--------+---------------+
我尝试添加一个实现IPdfPTableEvent并将其附加到我的文档的类,但看起来在执行完所有布局后调用TableLayout(),所以我没有机会潜入并调整行高
我想我可以使用PdfPTable.KeepRowsTogether()来防止行间距中间的分页,但这不是首选格式。
有没有办法“参与”表格的布局,因为它被渲染到PDF文档中,以便我可以调整行的高度?
P.S。我刚刚创建了一个PdfPTable的子类,我重写GetRowHeight(idx)并返回我想要的行高。它对生成的PDF没有影响。
答案 0 :(得分:0)
我提出的解决方案是非迭代的(请参阅评论中的@ChrisHaas的想法),但需要更多的代码。代码有点混乱(反映我用来计算解决方案的所有迭代)所以我不打算全部上传,但基本的要点是:
public class TableRowInfo
{
public int height;
}
public class TableEvent : IPdfPTableEventAfterSplit
{
Dictionary<PdfPRow, TableRowInfo> m_rows = null;
public bool LoadSpans(PdfPTable table)
{
// Fill in m_rows, mapping PdfPRow to information about the height
// of cells in that row, keeping row-spanning in mind
}
void IPdfPTableEventAfterSplit.AfterSplitTable(PdfPTable table, PdfPRow rowToStartNextPage, int startIdx)
{
TableRowInfo rowInfo = m_rows[rowToStartNextPage];
if (rowInfo.height > rowToStartNextPage.MaxHeights)
{
PdfPCell[] cells = rowToStartNextPage.GetCells();
foreach (PdfPCell cell in cells)
{
if (cell != null)
cell.MinimumHeight = rowInfo.height;
}
rowToStartNextPage.MaxHeights = rowInfo.height;
}
}
}
PdfPTable table = new PdfPTable(...);
// Fill in the rows of the table here
table.TableEvent = new TableEvent();
table.TableEvent.LoadSpans(table);