在iText5中,我们可以在需要时获得PdfPTable的高度" public float calculateHeights(boolean firsttime)"。
但在iText7中,我们如何获取当前表高度值(特别是在将表添加到其父元素之前)?
我已经测试了" table.getHeight()"方法,但它返回null。 我还发现在表格渲染对象中我可以得到这个值,但是限制是当表格添加到其父元素时需要触发渲染,所以时间不是我需要的。
因为有时我们需要这个值进行计算来决定" y轴"值。
答案 0 :(得分:5)
在iText5中,有关其位置/大小的元素和信息混合在一起,这使您可以在calculateWidths
元素上调用PdfPTable
。
在iText7中,此功能是分开的,这为渲染/布局元素提供了不同的灵活性。
因此,Table
实例是其中一个例子的模型元素,对它们的位置或大小一无所知。调用table.getHeight
会导致null
,因为table
之前未设置HEIGHT
属性。
要计算表格高度,必须使用渲染功能。
对于模型元素,您可以获取表示此模型元素及其所有子元素的渲染器子树,以及layout
在任何给定区域中的子树。要真正了解表格的高度,您需要创建一个有意放置元素全部内容的区域。
PdfDocument pdfDoc = ...
Document doc = ...
Table table = new Table(2)
.addCell(new Cell().add(new Paragraph("cell 1, 1")))
.addCell(new Cell().add(new Paragraph("cell 1, 2")));
LayoutResult result = table.createRendererSubTree().setParent(doc.getRenderer()).layout(
new LayoutContext(new LayoutArea(1, new Rectangle(0, 0, 400, 1e4f))));
System.out.println(result.getOccupiedArea().getBBox().getHeight());
上面的代码为我打印22.982422O
,但结果可能会因元素的配置和属性而异。
我想指出代码的两个重要部分:
我们将1e4f
作为LayoutArea
的高度,考虑到这足以放置整个表格。请注意,如果表格无法放入该高度,结果将永远不会超过此给定高度,因此对于您的用例来说它是不正确的(知道表格的总高度)。因此,请务必通过足以放置整个桌子的高度。
.setParent(doc.getRenderer())
部分在此非常重要,用于检索继承属性。请注意,我们没有为table
元素设置很多属性,甚至是字体,但是这些信息对于了解此元素占用的区域至关重要。因此,此信息将在layout
期间从父链继承。您可以通过更改文档的字体document.setFont(newFont);
或字体大小:document.setFontSize(24);
并观察结果高度变化
答案 1 :(得分:2)
好吧,由于渲染器框架在iText7中的编写方式,在将布局对象添加到父文档之前,还没有办法计算布局对象的高度,因为实际计算了高度布局对象在添加到Document
对象时发生。
然而,您可以重新传输Document
,允许您更改以前添加的元素的内容。使用此功能,您可以模拟表格的渲染,并在添加新元素时保持其高度。 table.getHeight()
仍然无效,因为它检索了height属性,并且该属性当前未在表呈现过程中的任何位置设置。
在下面的示例中,我编写了一个方便的方法,迭代渲染器树并打印出每个表在文档中占据的区域,以向您展示如何获得计算的高度。
示例本身在文档中添加了一些表格,显示占用区域,向每个表格添加一些单元格,显示占用区域(它们是相同的,因为添加到之前添加的元素不会触发布局),最后,手动触发重新布局并显示最终占用区域。
public class DelayedLayout {
public static String DEST = "target/output/StackOverflow/DelayedLayout/delayed.pdf";
public static void main(String[] args)throws IOException, FileNotFoundException{
File file = new File(DEST);
file.getParentFile().mkdirs();
new DelayedLayout().createPdf(DEST);
}
public void createPdf(String dest) throws IOException, FileNotFoundException{
PdfWriter writer = new PdfWriter(dest);
PdfDocument pdfDoc = new PdfDocument(writer);
boolean immediateFlush = false;
boolean relayout = true;
//Set immediate layout to false, so the document doesn't immediatly write render-results to its outputstream
Document doc = new Document(pdfDoc, PageSize.A4,immediateFlush);
Table tOne = createSimpleTable();
for(int i= 0; i< 5; i++) {
//Add a table and some whitespace
doc.add(tOne);
doc.add(new Paragraph(""));
}
System.out.println("\nInitial layout results");
printOccupiedAreasOfTableRenderers(doc.getRenderer());
System.out.println("\nAdding extra cells to the table");
addToTable(tOne);
printOccupiedAreasOfTableRenderers(doc.getRenderer());
System.out.println("\nForcing the document to redo the layout");
if(relayout)doc.relayout();
printOccupiedAreasOfTableRenderers(doc.getRenderer());
doc.close();
}
/**
* Create a very simple table
* @return simple table
*/
private Table createSimpleTable(){
int nrOfCols = 3;
int nrOfRows = 5;
Table res = new Table(nrOfCols);
for(int i= 0; i<nrOfRows;i++){
for(int j = 0; j<nrOfCols;j++){
Cell c = new Cell();
c.add(new Paragraph("["+i+", "+j+"]"));
res.addCell(c);
}
}
return res;
}
/**
* Add some extra cells to an exisiting table
* @param tab table to add cells to
*/
private void addToTable(Table tab){
int nrOfRows = 5;
int nrOfCols = tab.getNumberOfColumns();
for(int i=0; i<nrOfRows*nrOfCols;i++){
Cell c = new Cell();
c.add(new Paragraph("Extra cell"+ i));
tab.addCell(c);
}
}
/**
* Recursively iterate over the renderer tree, writing the occupied area to the console
* @param currentNode current renderer-node to check
*/
private void printOccupiedAreasOfTableRenderers(IRenderer currentNode){
if(currentNode.getClass().equals(TableRenderer.class)){
System.out.println("Table renderer with occupied area: " + currentNode.getOccupiedArea());
}
for (IRenderer child:currentNode.getChildRenderers()) {
printOccupiedAreasOfTableRenderers(child);
}
}