在iText7中,有没有像iText5的calculateHeights方法?

时间:2017-01-19 10:11:01

标签: itext7

在iText5中,我们可以在需要时获得PdfPTable的高度" public float calculateHeights(boolean firsttime)"。

但在iText7中,我们如何获取当前表高度值(特别是在将表添加到其父元素之前)?

我已经测试了" table.getHeight()"方法,但它返回null。 我还发现在表格渲染对象中我可以得到这个值,但是限制是当表格添加到其父元素时需要触发渲染,所以时间不是我需要的。

因为有时我们需要这个值进行计算来决定" y轴"值。

2 个答案:

答案 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,但结果可能会因元素的配置和属性而异。

我想指出代码的两个重要部分:

  1. 我们将1e4f作为LayoutArea的高度,考虑到这足以放置整个表格。请注意,如果表格无法放入该高度,结果将永远不会超过此给定高度,因此对于您的用例来说它是不正确的(知道表格的总高度)。因此,请务必通过足以放置整个桌子的高度。

  2. .setParent(doc.getRenderer())部分在此非常重要,用于检索继承属性。请注意,我们没有为table元素设置很多属性,甚至是字体,但是这些信息对于了解此元素占用的区域至关重要。因此,此信息将在layout期间从父链继承。您可以通过更改文档的字体document.setFont(newFont);或字体大小:document.setFontSize(24);并观察结果高度变化

  3. 来测试

答案 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);
        }
    }