2013-06-07 11:44:37 +02:00
|
|
|
<?php
|
|
|
|
|
2013-07-12 11:58:58 +02:00
|
|
|
namespace Icinga\File;
|
2013-06-07 11:44:37 +02:00
|
|
|
|
2014-02-21 11:30:18 +01:00
|
|
|
use DOMPDF;
|
|
|
|
use DOMDocument;
|
|
|
|
use DOMXPath;
|
2014-03-06 12:21:11 +01:00
|
|
|
use Font_Metrics;
|
|
|
|
use Icinga\Application\Icinga;
|
|
|
|
use Icinga\Web\StyleSheet;
|
|
|
|
use Icinga\Web\Url;
|
|
|
|
use Icinga\Exception\ProgrammingError;
|
2013-06-07 11:44:37 +02:00
|
|
|
|
2014-02-05 18:18:22 +01:00
|
|
|
require_once 'vendor/dompdf/dompdf_config.inc.php';
|
2013-06-07 11:44:37 +02:00
|
|
|
|
2014-02-21 11:30:18 +01:00
|
|
|
spl_autoload_register('DOMPDF_autoload');
|
2013-06-07 11:44:37 +02:00
|
|
|
|
2014-02-05 18:18:22 +01:00
|
|
|
class Pdf extends DOMPDF
|
2013-06-07 11:44:37 +02:00
|
|
|
{
|
2014-02-11 17:48:31 +01:00
|
|
|
/**
|
|
|
|
* The amount of table rows that fit on one page before a page-break is inserted.
|
|
|
|
*
|
|
|
|
* @var int
|
|
|
|
*/
|
|
|
|
public $rowsPerPage = 10;
|
|
|
|
|
|
|
|
/**
|
2014-02-21 11:30:18 +01:00
|
|
|
* Wether tables should only start at new pages.
|
2014-02-12 12:05:19 +01:00
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $tableInitialPageBreak = false;
|
|
|
|
|
|
|
|
/**
|
2014-02-21 11:30:18 +01:00
|
|
|
* Whether occurring tables should be split up into smaller tables to avoid
|
|
|
|
* errors in the document layout.
|
2014-02-11 17:48:31 +01:00
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
2014-02-21 11:30:18 +01:00
|
|
|
public $paginateTable = false;
|
2014-02-11 17:48:31 +01:00
|
|
|
|
2014-02-21 14:07:32 +01:00
|
|
|
public function __construct()
|
|
|
|
{
|
2014-03-06 12:21:11 +01:00
|
|
|
$this->set_paper('A4', 'portrait');
|
2014-02-05 18:18:22 +01:00
|
|
|
parent::__construct();
|
|
|
|
}
|
|
|
|
|
2014-03-06 12:21:11 +01:00
|
|
|
protected function assertNoHeadersSent()
|
|
|
|
{
|
|
|
|
if (headers_sent()) {
|
|
|
|
throw new ProgrammingError(
|
|
|
|
'Could not send pdf-response, content already written to output.'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function renderControllerAction($controller)
|
|
|
|
{
|
|
|
|
$this->assertNoHeadersSent();
|
|
|
|
|
|
|
|
ini_set('memory_limit', '384M');
|
|
|
|
ini_set('max_execution_time', 300);
|
|
|
|
|
|
|
|
$request = $controller->getRequest();
|
|
|
|
$layout = $controller->getHelper('layout')->setLayout('pdf');
|
|
|
|
$controller->render();
|
|
|
|
$layout->content = $controller->getResponse();
|
|
|
|
$html = $layout->render();
|
|
|
|
|
|
|
|
$imgDir = Url::fromPath('img');
|
2014-03-07 16:14:48 +01:00
|
|
|
$html = preg_replace('~src="' . $imgDir . '/~', 'src="' . Icinga::app()->getBootstrapDirecory() . '/img/', $html);
|
2014-03-06 12:21:11 +01:00
|
|
|
//echo $html; exit;
|
|
|
|
$this->load_html($html);
|
|
|
|
|
|
|
|
/*
|
|
|
|
// TODO: We need to find a solution for page footers
|
|
|
|
$font = Font_Metrics::get_font("helvetica", "bold");
|
|
|
|
$canvas = $this->get_canvas();
|
|
|
|
$canvas->page_text(555, 750, "{PAGE_NUM}/{PAGE_COUNT}", $font, 10, array(0,0,0));
|
|
|
|
$dompdf->page_script('
|
|
|
|
// $pdf is the variable containing a reference to the canvas object provided by dompdf
|
|
|
|
$pdf->line(10,730,800,730,array(0,0,0),1);
|
|
|
|
');
|
|
|
|
*/
|
|
|
|
$this->render();
|
|
|
|
$this->stream(
|
|
|
|
sprintf(
|
|
|
|
'%s-%s-%d.pdf',
|
|
|
|
$request->getControllerName(),
|
|
|
|
$request->getActionName(),
|
|
|
|
time()
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2014-02-05 18:18:22 +01:00
|
|
|
/**
|
|
|
|
* @param $body
|
|
|
|
* @param $css
|
|
|
|
*/
|
2014-03-06 12:21:11 +01:00
|
|
|
/* public function renderPage($body, $css)
|
2014-02-05 18:18:22 +01:00
|
|
|
{
|
|
|
|
$html =
|
|
|
|
'<html><head></head>'
|
|
|
|
. '<body>'
|
2014-03-06 12:21:11 +01:00
|
|
|
. '<style>' . $css
|
|
|
|
// . Pdf::prepareCss($css)
|
|
|
|
. '</style>'
|
2014-02-05 18:18:22 +01:00
|
|
|
. $body
|
|
|
|
. '</body>'
|
|
|
|
. '</html>';
|
2014-02-11 17:48:31 +01:00
|
|
|
if ($this->paginateTable === true) {
|
2014-02-12 12:05:19 +01:00
|
|
|
$doc = new DOMDocument();
|
|
|
|
@$doc->loadHTML($html);
|
|
|
|
$this->paginateHtmlTables($doc);
|
|
|
|
$html = $doc->saveHtml();
|
2014-02-11 17:48:31 +01:00
|
|
|
}
|
2014-02-05 18:18:22 +01:00
|
|
|
$this->load_html($html);
|
|
|
|
$this->render();
|
2014-03-06 12:21:11 +01:00
|
|
|
}*/
|
2014-02-05 18:18:22 +01:00
|
|
|
|
2014-02-11 17:48:31 +01:00
|
|
|
/**
|
|
|
|
* Split up tables into multiple elements that each contain $rowsPerPage of all original rows
|
|
|
|
*
|
|
|
|
* NOTE: This is a workaround to fix the buggy page-break on table-rows in dompdf.
|
|
|
|
*
|
2014-02-12 12:05:19 +01:00
|
|
|
* @param DOMDocument $doc The html document containing the tables.
|
2014-02-11 17:48:31 +01:00
|
|
|
*
|
2014-02-12 12:05:19 +01:00
|
|
|
* @return array All paginated tables from the document.
|
2014-02-11 17:48:31 +01:00
|
|
|
*/
|
2014-03-06 12:21:11 +01:00
|
|
|
/* private function paginateHtmlTables(DOMDocument $doc)
|
2014-02-11 17:48:31 +01:00
|
|
|
{
|
2014-02-12 12:05:19 +01:00
|
|
|
$xpath = new DOMXPath($doc);
|
|
|
|
$tables = $xpath->query('.//table');
|
|
|
|
$paginated = array();
|
|
|
|
$j = 0;
|
2014-02-11 17:48:31 +01:00
|
|
|
|
|
|
|
foreach ($tables as $table) {
|
|
|
|
$containerType = null;
|
|
|
|
$rows = $xpath->query('.//tr', $table);
|
|
|
|
$rowCnt = $rows->length;
|
|
|
|
$tableCnt = (Integer)ceil($rowCnt / $this->rowsPerPage);
|
2014-02-12 12:05:19 +01:00
|
|
|
$paginated[$j] = array();
|
2014-02-11 17:48:31 +01:00
|
|
|
if ($rowCnt <= $this->rowsPerPage) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// remove all rows from the original parent
|
|
|
|
foreach ($rows as $row) {
|
|
|
|
if (!isset($containerType)) {
|
|
|
|
$containerType = $row->parentNode->nodeName;
|
|
|
|
}
|
|
|
|
$row->parentNode->removeChild($row);
|
|
|
|
}
|
|
|
|
|
|
|
|
// clone table for each additional page and fetch the row containers
|
|
|
|
$containers = array();
|
|
|
|
$pages = array();
|
|
|
|
|
2014-02-12 12:05:19 +01:00
|
|
|
if ($this->tableInitialPageBreak) {
|
|
|
|
$this->pageBreak($doc, $table);
|
|
|
|
}
|
2014-02-11 17:48:31 +01:00
|
|
|
for ($i = 0; $i < $tableCnt; $i++) {
|
|
|
|
// clone table
|
|
|
|
$currentPage = $table->cloneNode(true);
|
|
|
|
$pages[$i] = $currentPage;
|
|
|
|
$table->parentNode->insertBefore($currentPage, $table);
|
|
|
|
|
2014-02-12 12:05:19 +01:00
|
|
|
// put it in current paginated table
|
|
|
|
$paginated[$j] = $currentPage;
|
|
|
|
|
2014-02-11 17:48:31 +01:00
|
|
|
// insert page-break
|
|
|
|
if ($i < $tableCnt - 1) {
|
2014-02-12 12:05:19 +01:00
|
|
|
$this->pageBreak($doc, $table);
|
2014-02-11 17:48:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// fetch row container
|
|
|
|
$container = $xpath->query('.//' . $containerType, $currentPage)->item(0);
|
|
|
|
$containers[$i] = $container;
|
|
|
|
}
|
|
|
|
|
|
|
|
$i = 0;
|
|
|
|
foreach ($rows as $row) {
|
|
|
|
$p = (Integer)floor($i / $this->rowsPerPage);
|
|
|
|
$containers[$p]->appendChild($row);
|
|
|
|
$i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// remove original table
|
|
|
|
$table->parentNode->removeChild($table);
|
2014-02-12 12:05:19 +01:00
|
|
|
$j++;
|
2014-02-11 17:48:31 +01:00
|
|
|
}
|
2014-02-12 12:05:19 +01:00
|
|
|
return $paginated;
|
|
|
|
}
|
|
|
|
|
|
|
|
private function pageBreak($doc, $before)
|
|
|
|
{
|
|
|
|
$div = $doc->createElement('div');
|
|
|
|
$div->setAttribute('style', 'page-break-before: always;');
|
|
|
|
$before->parentNode->insertBefore($div, $before);
|
2014-02-11 17:48:31 +01:00
|
|
|
}
|
2014-03-06 12:21:11 +01:00
|
|
|
*/
|
2014-02-05 18:18:22 +01:00
|
|
|
/**
|
|
|
|
* Prepare the given css for rendering with DOMPDF, by removing or hiding all incompatible
|
|
|
|
* styles
|
|
|
|
*
|
|
|
|
* @param $css The css-string
|
|
|
|
*
|
|
|
|
* @return string A css-string that is ready to use for DOMPDF
|
|
|
|
*/
|
2014-03-06 12:21:11 +01:00
|
|
|
|
|
|
|
// public static function prepareCss($css)
|
|
|
|
// {
|
|
|
|
// $css = preg_replace('/\*:\s*before\s*,\s*/', '', $css);
|
|
|
|
// $css = preg_replace('/\*\s*:\s*after\s*\{[^\}]*\}/', '', $css);
|
|
|
|
// return $css;
|
|
|
|
// }
|
|
|
|
|
2013-06-07 11:44:37 +02:00
|
|
|
}
|