296 lines
8.4 KiB
PHP
296 lines
8.4 KiB
PHP
|
<?php
|
||
|
/**
|
||
|
* @package dompdf
|
||
|
* @link http://dompdf.github.com/
|
||
|
* @author Benj Carson <benjcarson@digitaljunkies.ca>
|
||
|
* @license http://www.gnu.org/copyleft/lesser.html GNU Lesser General Public License
|
||
|
*/
|
||
|
namespace Dompdf;
|
||
|
|
||
|
use Dompdf\Renderer\AbstractRenderer;
|
||
|
use Dompdf\Renderer\Block;
|
||
|
use Dompdf\Renderer\Image;
|
||
|
use Dompdf\Renderer\ListBullet;
|
||
|
use Dompdf\Renderer\TableCell;
|
||
|
use Dompdf\Renderer\TableRowGroup;
|
||
|
use Dompdf\Renderer\Text;
|
||
|
|
||
|
use Dompdf\Frame;
|
||
|
|
||
|
/**
|
||
|
* Concrete renderer
|
||
|
*
|
||
|
* Instantiates several specific renderers in order to render any given frame.
|
||
|
*
|
||
|
* @package dompdf
|
||
|
*/
|
||
|
class Renderer extends AbstractRenderer
|
||
|
{
|
||
|
|
||
|
/**
|
||
|
* Array of renderers for specific frame types
|
||
|
*
|
||
|
* @var AbstractRenderer[]
|
||
|
*/
|
||
|
protected $_renderers;
|
||
|
|
||
|
/**
|
||
|
* Cache of the callbacks array
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
private $_callbacks;
|
||
|
|
||
|
/**
|
||
|
* Advance the canvas to the next page
|
||
|
*/
|
||
|
function new_page()
|
||
|
{
|
||
|
$this->_canvas->new_page();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Render frames recursively
|
||
|
*
|
||
|
* @param Frame $frame the frame to render
|
||
|
*/
|
||
|
public function render(Frame $frame)
|
||
|
{
|
||
|
global $_dompdf_debug;
|
||
|
|
||
|
if ($_dompdf_debug) {
|
||
|
echo $frame;
|
||
|
flush();
|
||
|
}
|
||
|
|
||
|
$style = $frame->get_style();
|
||
|
|
||
|
if (in_array($style->visibility, array("hidden", "collapse"))) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$display = $style->display;
|
||
|
|
||
|
// Starts the CSS transformation
|
||
|
if ($style->transform && is_array($style->transform)) {
|
||
|
$this->_canvas->save();
|
||
|
list($x, $y) = $frame->get_padding_box();
|
||
|
$origin = $style->transform_origin;
|
||
|
|
||
|
foreach ($style->transform as $transform) {
|
||
|
list($function, $values) = $transform;
|
||
|
if ($function === "matrix") {
|
||
|
$function = "transform";
|
||
|
}
|
||
|
|
||
|
$values = array_map("floatval", $values);
|
||
|
$values[] = $x + (float)$style->length_in_pt($origin[0], $style->width);
|
||
|
$values[] = $y + (float)$style->length_in_pt($origin[1], $style->height);
|
||
|
|
||
|
call_user_func_array(array($this->_canvas, $function), $values);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
switch ($display) {
|
||
|
|
||
|
case "block":
|
||
|
case "list-item":
|
||
|
case "inline-block":
|
||
|
case "table":
|
||
|
case "inline-table":
|
||
|
$this->_render_frame("block", $frame);
|
||
|
break;
|
||
|
|
||
|
case "inline":
|
||
|
if ($frame->is_text_node()) {
|
||
|
$this->_render_frame("text", $frame);
|
||
|
} else {
|
||
|
$this->_render_frame("inline", $frame);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case "table-cell":
|
||
|
$this->_render_frame("table-cell", $frame);
|
||
|
break;
|
||
|
|
||
|
case "table-row-group":
|
||
|
case "table-header-group":
|
||
|
case "table-footer-group":
|
||
|
$this->_render_frame("table-row-group", $frame);
|
||
|
break;
|
||
|
|
||
|
case "-dompdf-list-bullet":
|
||
|
$this->_render_frame("list-bullet", $frame);
|
||
|
break;
|
||
|
|
||
|
case "-dompdf-image":
|
||
|
$this->_render_frame("image", $frame);
|
||
|
break;
|
||
|
|
||
|
case "none":
|
||
|
$node = $frame->get_node();
|
||
|
|
||
|
if ($node->nodeName === "script") {
|
||
|
if ($node->getAttribute("type") === "text/php" ||
|
||
|
$node->getAttribute("language") === "php"
|
||
|
) {
|
||
|
// Evaluate embedded php scripts
|
||
|
$this->_render_frame("php", $frame);
|
||
|
} elseif ($node->getAttribute("type") === "text/javascript" ||
|
||
|
$node->getAttribute("language") === "javascript"
|
||
|
) {
|
||
|
// Insert JavaScript
|
||
|
$this->_render_frame("javascript", $frame);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Don't render children, so skip to next iter
|
||
|
return;
|
||
|
|
||
|
default:
|
||
|
break;
|
||
|
|
||
|
}
|
||
|
|
||
|
// Starts the overflow: hidden box
|
||
|
if ($style->overflow === "hidden") {
|
||
|
list($x, $y, $w, $h) = $frame->get_padding_box();
|
||
|
|
||
|
// get border radii
|
||
|
$style = $frame->get_style();
|
||
|
list($tl, $tr, $br, $bl) = $style->get_computed_border_radius($w, $h);
|
||
|
|
||
|
if ($tl + $tr + $br + $bl > 0) {
|
||
|
$this->_canvas->clipping_roundrectangle($x, $y, (float)$w, (float)$h, $tl, $tr, $br, $bl);
|
||
|
} else {
|
||
|
$this->_canvas->clipping_rectangle($x, $y, (float)$w, (float)$h);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$stack = array();
|
||
|
|
||
|
foreach ($frame->get_children() as $child) {
|
||
|
// < 0 : nagative z-index
|
||
|
// = 0 : no z-index, no stacking context
|
||
|
// = 1 : stacking context without z-index
|
||
|
// > 1 : z-index
|
||
|
$child_style = $child->get_style();
|
||
|
$child_z_index = $child_style->z_index;
|
||
|
$z_index = 0;
|
||
|
|
||
|
if ($child_z_index !== "auto") {
|
||
|
$z_index = intval($child_z_index) + 1;
|
||
|
} elseif ($child_style->float !== "none" || $child->is_positionned()) {
|
||
|
$z_index = 1;
|
||
|
}
|
||
|
|
||
|
$stack[$z_index][] = $child;
|
||
|
}
|
||
|
|
||
|
ksort($stack);
|
||
|
|
||
|
foreach ($stack as $by_index) {
|
||
|
foreach ($by_index as $child) {
|
||
|
$this->render($child);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Ends the overflow: hidden box
|
||
|
if ($style->overflow === "hidden") {
|
||
|
$this->_canvas->clipping_end();
|
||
|
}
|
||
|
|
||
|
if ($style->transform && is_array($style->transform)) {
|
||
|
$this->_canvas->restore();
|
||
|
}
|
||
|
|
||
|
// Check for end frame callback
|
||
|
$this->_check_callbacks("end_frame", $frame);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Check for callbacks that need to be performed when a given event
|
||
|
* gets triggered on a frame
|
||
|
*
|
||
|
* @param string $event the type of event
|
||
|
* @param Frame $frame the frame that event is triggered on
|
||
|
*/
|
||
|
protected function _check_callbacks($event, $frame)
|
||
|
{
|
||
|
if (!isset($this->_callbacks)) {
|
||
|
$this->_callbacks = $this->_dompdf->getCallbacks();
|
||
|
}
|
||
|
|
||
|
if (is_array($this->_callbacks) && isset($this->_callbacks[$event])) {
|
||
|
$info = array(0 => $this->_canvas, "canvas" => $this->_canvas,
|
||
|
1 => $frame, "frame" => $frame);
|
||
|
$fs = $this->_callbacks[$event];
|
||
|
foreach ($fs as $f) {
|
||
|
if (is_callable($f)) {
|
||
|
if (is_array($f)) {
|
||
|
$f[0]->{$f[1]}($info);
|
||
|
} else {
|
||
|
$f($info);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Render a single frame
|
||
|
*
|
||
|
* Creates Renderer objects on demand
|
||
|
*
|
||
|
* @param string $type type of renderer to use
|
||
|
* @param Frame $frame the frame to render
|
||
|
*/
|
||
|
protected function _render_frame($type, $frame)
|
||
|
{
|
||
|
|
||
|
if (!isset($this->_renderers[$type])) {
|
||
|
|
||
|
switch ($type) {
|
||
|
case "block":
|
||
|
$this->_renderers[$type] = new Block($this->_dompdf);
|
||
|
break;
|
||
|
|
||
|
case "inline":
|
||
|
$this->_renderers[$type] = new Renderer\Inline($this->_dompdf);
|
||
|
break;
|
||
|
|
||
|
case "text":
|
||
|
$this->_renderers[$type] = new Text($this->_dompdf);
|
||
|
break;
|
||
|
|
||
|
case "image":
|
||
|
$this->_renderers[$type] = new Image($this->_dompdf);
|
||
|
break;
|
||
|
|
||
|
case "table-cell":
|
||
|
$this->_renderers[$type] = new TableCell($this->_dompdf);
|
||
|
break;
|
||
|
|
||
|
case "table-row-group":
|
||
|
$this->_renderers[$type] = new TableRowGroup($this->_dompdf);
|
||
|
break;
|
||
|
|
||
|
case "list-bullet":
|
||
|
$this->_renderers[$type] = new ListBullet($this->_dompdf);
|
||
|
break;
|
||
|
|
||
|
case "php":
|
||
|
$this->_renderers[$type] = new PhpEvaluator($this->_canvas);
|
||
|
break;
|
||
|
|
||
|
case "javascript":
|
||
|
$this->_renderers[$type] = new JavascriptEmbedder($this->_dompdf);
|
||
|
break;
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$this->_renderers[$type]->render($frame);
|
||
|
}
|
||
|
}
|