15

我尝试创建具有多个页面的 PDF,并且需要提前计算每个单独元素(MultiCell)的高度准备分页符。根据文档,有几个函数,比如 GetCharWidth/GetStringWidth 来支持我自己做,但除了潜在的性能损失之外,我可能无论如何都不会做正确的事情。以更优雅的方式实现我的目标的建议?

参考:TCPDF

4

7 回答 7

28

我明白了:D!!!!

创建另一个 pdf2 对象

// pdf2 set x margin to pdf1's xmargin, but y margin to zero
// to make sure that pdf2 has identical settings, you can clone the object (after initializing the main pdf object)
$pdf2 = clone $pdf;
pdf2->addpage
pdf2->writeCell
$height = pdf2->getY()
pdf2->deletePage(pdf2->getPage())
pdf1->checkPageBreak($height);
pdf1->writeCell()

健康度:D

于 2009-12-21T22:38:15.520 回答
22

这是一个老问题,但当前版本的 TCPDF(截至 2011 年 12 月 7 日)有一个名为的函数getStringHeight,允许您在实际调用 MultiCell 之前计算传递给 MultiCell 的字符串的结果高度。然后这个高度可以用于各种事情,原始问题中的计算,以及在制作表格时设置行高等。效果很好。

只是一些信息,以防其他人像我一样偶然发现这个问题,寻找解决这个问题的方法。

于 2011-12-07T14:38:39.643 回答
10

虽然 Carvell 的回答很好,但 TCPDF 提到getStringHeight返回估计的高度。有用的是,那里的文档提供了一种非常全面的技术来获得准确的高度,结果为$height. 至于为什么他们自己不使用这个是一个谜......

// store current object
$pdf->startTransaction();
// store starting values
$start_y = $pdf->GetY();
$start_page = $pdf->getPage();
// call your printing functions with your parameters
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
$pdf->MultiCell($w=0, $h=0, $txt, $border=1, $align='L', $fill=false, $ln=1, $x='', $y='',     $reseth=true, $stretch=0, $ishtml=false, $autopadding=true, $maxh=0);
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// get the new Y
$end_y = $pdf->GetY();
$end_page = $pdf->getPage();
// calculate height
$height = 0;
if ($end_page == $start_page) {
    $height = $end_y - $start_y;
} else {
    for ($page=$start_page; $page <= $end_page; ++$page) {
        $pdf->setPage($page);
        if ($page == $start_page) {
            // first page
            $height = $pdf->h - $start_y - $pdf->bMargin;
        } elseif ($page == $end_page) {
            // last page
            $height = $end_y - $pdf->tMargin;
        } else {
            $height = $pdf->h - $pdf->tMargin - $pdf->bMargin;
        }
    }
}
// restore previous object
$pdf = $pdf->rollbackTransaction();
于 2014-08-14T08:44:51.770 回答
4

根据我的经验,几乎不可能提前计算出单元高度。使用 TCPDF 的分页符处理功能要容易得多,它可以提前告诉您是否要进入分页符。这是示例代码:

$yy = $this->pdf->GetY();

$check_pagebreak = $this->pdf->checkPageBreak($height+$padding,$yy,false);

将 false 更改为 true 以允许自动分页符,否则,您可以自己处理分页符的逻辑,这就是我最终要做的。

此外,如果您可能需要这个,这里还有一个小提示:考虑使用事务功能分两次创建您的文档。第一遍用于计算所有高度和单元格、分页符等。您还可以将所有行高和每页行数存储在数组中。在第二遍中,使用所有正确的信息创建您的文档,并且不需要分页逻辑(第二遍可以从单独的方法运行,以使代码更易于阅读,并且为了您的理智)。

于 2009-09-10T17:30:35.427 回答
3

使用 TCPDF 示例 20

  1. 如果单元格/列在不同页面上结束,那么计算多单元格高度可能是一场噩梦。

  2. 使用事务或额外的 pdf 对象会使事情变得非常缓慢。

  3. 在打印单元格之前使用诸如 getNumLines() 和 getStringHeight() 之类的函数来计算“估计”(参见文档)高度并不总是正确的。特别是如果文本在单元格右边框之前或之后结束 - 导致行被打印在彼此的顶部。

我更喜欢示例 20中使用的技术,其中使用不同页面的最大 Y 值来计算新行的位置。

该示例仅打印两列,但我更改了它的 main 函数以能够打印列数组。显然,您可以向数组添加更多数据,例如每列的字体、边框等。

public function MultiRow($columnsArray) {
    $page_start = $this->getPage();
    $y_start    = $this->GetY();

    $pageArray  = array();
    $yArray     = array();

    // traverse through array and print one column at a time.
    $columnCount = count($columnsArray);
    for($i=0; $i<$columnCount; $i++)
    {
        if($i+1 < $columnCount)
        {
            // Current column is not the last column in the row.
            // After printing, the pointer will be moved down to
            // the right-bottom of the column - from where the
            // next multiCell in the following loop will use it
            // via $this->GetX().
            $ln = 2;
        }
        else
        {
            // Current column is the last column in the row.
            // After printing, the pointer will be moved to new line.
            $ln = 1;
        }
        $this->MultiCell(30, 0, $columnsArray[$i], 1, 'L', 1, $ln,
            $this->GetX() ,$y_start, true, 0);

        $pageArray[$i]  = $this->getPage();
        $yArray[$i]     = $this->GetY();

        // Go to page where the row started - to print the
        // next column (if any).
        $this->setPage($page_start);
    }

    // Test if all columns ended on the same page
    $samePage = true;
    foreach ($pageArray as $val) {
       if($val != $pageArray['0']) 
       {
          $samePage = false;
          break;
       }
    }

    // Set the new page and row position by case
    if($samePage == true) 
    {
        // All columns ended on the same page.
        // Get the longest column.
        $newY = max($yArray);
    }
    else
    {
        // Some columns ended on different pages.
        // Get the array-keys (not the values) of all columns that
        // ended on the last page.
        $endPageKeys = array_keys($pageArray, max($pageArray));

        // Get the Y values of all columns that ended on the last page,
        // i.e. get the Y values of all columns with keys in $endPageKeys.
        $yValues = array();
        foreach($endPageKeys as $key)
        {
            $yValues[] = $yArray[$key];
        }

        // Get the largest Y value of all columns that ended on
        // the last page.
        $newY = max($yValues);
    }

    // Go to the last page and start at its largets Y value
    $this->setPage(max($pageArray));
    $this->SetXY($this->GetX(),$newY);
}
于 2015-10-22T10:03:53.680 回答
1

重新访问的帖子:Tcpdf – 使用 MultiCell 的可变高度表行有很多有用的信息。这是一个简短的摘录:

getNumLines()...实际上允许我们确定给定特定宽度的文本字符串将占用多少行。实际上,它允许我们执行之前使用 MultiCell 返回的操作,而无需实际绘制任何内容。这让我们可以用一行代码确定最大单元格高度:

$linecount = max($pdf->getNumLines($row['cell1data'], 80),$pdf->getNumLines($row['cell2data'], 80
于 2009-08-18T14:28:03.090 回答
0

我尝试使用杰伊的答案,它达到了预期的目的,但由于某种原因导致我的徽标没有出现在第一页之后。我不想做深入分析,但与克隆有关。然后我尝试了相同的方法,但使用了事务。这产生了数百个错误。

然后我想出了这个使用相同对象的相当简单的解决方案。

/**
 * Gets an accurate measurement of a cell's rendered height.
 *
 * @param   float   $width     the width of the column to be rendered
 * @param   string  $contents  the contents to be rendered
 *
 * @return float
 */
private function getCellHeight(float $width, string $contents): float
{
    $view        = $this->view;
    $currentPage = $view->getPage();
    $currentX    = $view->GetX();
    $currentY    = $view->GetY();
    $view->AddPage();
    $x     = $view->GetX();
    $start = $view->GetY();
    $view->writeHTMLCell($width, 15, $x, $start, $contents, self::INSTANCE_BORDER, 1);
    $height = $view->GetY() - $start;
    $view->deletePage($view->getPage());
    $view->setPage($currentPage);
    $view->changePosition($currentX, $currentY);

    return $height;
}

由于该writeHTMLCell功能需要 a $h,我使用15,但它可以是您想要的任何东西,$border值也可以。
$ln值需要设置为1,否则在获取y之前会重置该值GetY()
changePosition是我自己的包装器SetXY()

于 2021-03-31T17:29:04.387 回答