123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790 |
- <?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
- */
- /**
- * Maps table cells to the table grid.
- *
- * This class resolves borders in tables with collapsed borders and helps
- * place row & column spanned table cells.
- *
- * @access private
- * @package dompdf
- */
- class Cellmap {
- /**
- * Border style weight lookup for collapsed border resolution.
- *
- * @var array
- */
- static protected $_BORDER_STYLE_SCORE = array(
- "inset" => 1,
- "groove" => 2,
- "outset" => 3,
- "ridge" => 4,
- "dotted" => 5,
- "dashed" => 6,
- "solid" => 7,
- "double" => 8,
- "hidden" => 9,
- "none" => 0,
- );
- /**
- * The table object this cellmap is attached to.
- *
- * @var Table_Frame_Decorator
- */
- protected $_table;
- /**
- * The total number of rows in the table
- *
- * @var int
- */
- protected $_num_rows;
- /**
- * The total number of columns in the table
- *
- * @var int
- */
- protected $_num_cols;
- /**
- * 2D array mapping <row,column> to frames
- *
- * @var Frame[][]
- */
- protected $_cells;
- /**
- * 1D array of column dimensions
- *
- * @var array
- */
- protected $_columns;
- /**
- * 1D array of row dimensions
- *
- * @var array
- */
- protected $_rows;
- /**
- * 2D array of border specs
- *
- * @var array
- */
- protected $_borders;
- /**
- * 1D Array mapping frames to (multiple) <row, col> pairs, keyed on frame_id.
- *
- * @var Frame[]
- */
- protected $_frames;
- /**
- * Current column when adding cells, 0-based
- *
- * @var int
- */
- private $__col;
- /**
- * Current row when adding cells, 0-based
- *
- * @var int
- */
- private $__row;
-
- /**
- * Tells wether the columns' width can be modified
- *
- * @var bool
- */
- private $_columns_locked = false;
-
- /**
- * Tells wether the table has table-layout:fixed
- *
- * @var bool
- */
- private $_fixed_layout = false;
- //........................................................................
- function __construct(Table_Frame_Decorator $table) {
- $this->_table = $table;
- $this->reset();
- }
-
- function __destruct() {
- clear_object($this);
- }
- //........................................................................
- function reset() {
- $this->_num_rows = 0;
- $this->_num_cols = 0;
- $this->_cells = array();
- $this->_frames = array();
- if ( !$this->_columns_locked ) {
- $this->_columns = array();
- }
-
- $this->_rows = array();
- $this->_borders = array();
- $this->__col = $this->__row = 0;
- }
- //........................................................................
- function lock_columns() {
- $this->_columns_locked = true;
- }
- function is_columns_locked() {
- return $this->_columns_locked;
- }
-
- function set_layout_fixed($fixed) {
- $this->_fixed_layout = $fixed;
- }
- function is_layout_fixed() {
- return $this->_fixed_layout;
- }
-
- function get_num_rows() { return $this->_num_rows; }
- function get_num_cols() { return $this->_num_cols; }
- function &get_columns() {
- return $this->_columns;
- }
- function set_columns($columns) {
- $this->_columns = $columns;
- }
- function &get_column($i) {
- if ( !isset($this->_columns[$i]) ) {
- $this->_columns[$i] = array(
- "x" => 0,
- "min-width" => 0,
- "max-width" => 0,
- "used-width" => null,
- "absolute" => 0,
- "percent" => 0,
- "auto" => true,
- );
- }
-
- return $this->_columns[$i];
- }
- function &get_rows() {
- return $this->_rows;
- }
- function &get_row($j) {
- if ( !isset($this->_rows[$j]) ) {
- $this->_rows[$j] = array(
- "y" => 0,
- "first-column" => 0,
- "height" => null,
- );
- }
-
- return $this->_rows[$j];
- }
- function get_border($i, $j, $h_v, $prop = null) {
- if ( !isset($this->_borders[$i][$j][$h_v]) ) {
- $this->_borders[$i][$j][$h_v] = array(
- "width" => 0,
- "style" => "solid",
- "color" => "black",
- );
- }
-
- if ( isset($prop) ) {
- return $this->_borders[$i][$j][$h_v][$prop];
- }
-
- return $this->_borders[$i][$j][$h_v];
- }
- function get_border_properties($i, $j) {
- return array(
- "top" => $this->get_border($i, $j, "horizontal"),
- "right" => $this->get_border($i, $j+1, "vertical"),
- "bottom" => $this->get_border($i+1, $j, "horizontal"),
- "left" => $this->get_border($i, $j, "vertical"),
- );
- }
- //........................................................................
- function get_spanned_cells(Frame $frame) {
- $key = $frame->get_id();
- if ( !isset($this->_frames[$key]) ) {
- throw new DOMPDF_Exception("Frame not found in cellmap");
- }
- return $this->_frames[$key];
- }
- function frame_exists_in_cellmap(Frame $frame) {
- $key = $frame->get_id();
- return isset($this->_frames[$key]);
- }
-
- function get_frame_position(Frame $frame) {
- global $_dompdf_warnings;
- $key = $frame->get_id();
- if ( !isset($this->_frames[$key]) ) {
- throw new DOMPDF_Exception("Frame not found in cellmap");
- }
- $col = $this->_frames[$key]["columns"][0];
- $row = $this->_frames[$key]["rows"][0];
- if ( !isset($this->_columns[$col])) {
- $_dompdf_warnings[] = "Frame not found in columns array. Check your table layout for missing or extra TDs.";
- $x = 0;
- }
- else {
- $x = $this->_columns[$col]["x"];
- }
- if ( !isset($this->_rows[$row])) {
- $_dompdf_warnings[] = "Frame not found in row array. Check your table layout for missing or extra TDs.";
- $y = 0;
- }
- else {
- $y = $this->_rows[$row]["y"];
- }
- return array($x, $y, "x" => $x, "y" => $y);
- }
- function get_frame_width(Frame $frame) {
- $key = $frame->get_id();
- if ( !isset($this->_frames[$key]) ) {
- throw new DOMPDF_Exception("Frame not found in cellmap");
- }
- $cols = $this->_frames[$key]["columns"];
- $w = 0;
- foreach ($cols as $i) {
- $w += $this->_columns[$i]["used-width"];
- }
-
- return $w;
- }
- function get_frame_height(Frame $frame) {
- $key = $frame->get_id();
- if ( !isset($this->_frames[$key]) ) {
- throw new DOMPDF_Exception("Frame not found in cellmap");
- }
- $rows = $this->_frames[$key]["rows"];
- $h = 0;
- foreach ($rows as $i) {
- if ( !isset($this->_rows[$i]) ) {
- throw new Exception("The row #$i could not be found, please file an issue in the tracker with the HTML code");
- }
-
- $h += $this->_rows[$i]["height"];
- }
-
- return $h;
- }
- //........................................................................
- function set_column_width($j, $width) {
- if ( $this->_columns_locked ) {
- return;
- }
-
- $col =& $this->get_column($j);
- $col["used-width"] = $width;
- $next_col =& $this->get_column($j+1);
- $next_col["x"] = $next_col["x"] + $width;
- }
- function set_row_height($i, $height) {
- $row =& $this->get_row($i);
-
- if ( $row["height"] !== null && $height <= $row["height"] ) {
- return;
- }
- $row["height"] = $height;
- $next_row =& $this->get_row($i+1);
- $next_row["y"] = $row["y"] + $height;
- }
- //........................................................................
- protected function _resolve_border($i, $j, $h_v, $border_spec) {
- $n_width = $border_spec["width"];
- $n_style = $border_spec["style"];
- if ( !isset($this->_borders[$i][$j][$h_v]) ) {
- $this->_borders[$i][$j][$h_v] = $border_spec;
- return $this->_borders[$i][$j][$h_v]["width"];
- }
-
- $border = &$this->_borders[$i][$j][$h_v];
-
- $o_width = $border["width"];
- $o_style = $border["style"];
- if ( ($n_style === "hidden" ||
- $n_width > $o_width ||
- $o_style === "none")
- or
- ($o_width == $n_width &&
- in_array($n_style, self::$_BORDER_STYLE_SCORE) &&
- self::$_BORDER_STYLE_SCORE[ $n_style ] > self::$_BORDER_STYLE_SCORE[ $o_style ]) ) {
- $border = $border_spec;
- }
- return $border["width"];
- }
- //........................................................................
- function add_frame(Frame $frame) {
-
- $style = $frame->get_style();
- $display = $style->display;
- $collapse = $this->_table->get_style()->border_collapse == "collapse";
- // Recursively add the frames within tables, table-row-groups and table-rows
- if ( $display === "table-row" ||
- $display === "table" ||
- $display === "inline-table" ||
- in_array($display, Table_Frame_Decorator::$ROW_GROUPS) ) {
- $start_row = $this->__row;
- foreach ( $frame->get_children() as $child ) {
- $this->add_frame( $child );
- }
- if ( $display === "table-row" ) {
- $this->add_row();
- }
- $num_rows = $this->__row - $start_row - 1;
- $key = $frame->get_id();
- // Row groups always span across the entire table
- $this->_frames[$key]["columns"] = range(0,max(0,$this->_num_cols-1));
- $this->_frames[$key]["rows"] = range($start_row, max(0, $this->__row - 1));
- $this->_frames[$key]["frame"] = $frame;
- if ( $display !== "table-row" && $collapse ) {
- $bp = $style->get_border_properties();
- // Resolve the borders
- for ( $i = 0; $i < $num_rows+1; $i++) {
- $this->_resolve_border($start_row + $i, 0, "vertical", $bp["left"]);
- $this->_resolve_border($start_row + $i, $this->_num_cols, "vertical", $bp["right"]);
- }
- for ( $j = 0; $j < $this->_num_cols; $j++) {
- $this->_resolve_border($start_row, $j, "horizontal", $bp["top"]);
- $this->_resolve_border($this->__row, $j, "horizontal", $bp["bottom"]);
- }
- }
- return;
- }
-
- $node = $frame->get_node();
-
- // Determine where this cell is going
- $colspan = $node->getAttribute("colspan");
- $rowspan = $node->getAttribute("rowspan");
- if ( !$colspan ) {
- $colspan = 1;
- $node->setAttribute("colspan",1);
- }
- if ( !$rowspan ) {
- $rowspan = 1;
- $node->setAttribute("rowspan",1);
- }
- $key = $frame->get_id();
- $bp = $style->get_border_properties();
- // Add the frame to the cellmap
- $max_left = $max_right = 0;
- // Find the next available column (fix by Ciro Mondueri)
- $ac = $this->__col;
- while ( isset($this->_cells[$this->__row][$ac]) ) {
- $ac++;
- }
-
- $this->__col = $ac;
- // Rows:
- for ( $i = 0; $i < $rowspan; $i++ ) {
- $row = $this->__row + $i;
- $this->_frames[$key]["rows"][] = $row;
- for ( $j = 0; $j < $colspan; $j++) {
- $this->_cells[$row][$this->__col + $j] = $frame;
- }
- if ( $collapse ) {
- // Resolve vertical borders
- $max_left = max($max_left, $this->_resolve_border($row, $this->__col, "vertical", $bp["left"]));
- $max_right = max($max_right, $this->_resolve_border($row, $this->__col + $colspan, "vertical", $bp["right"]));
- }
- }
- $max_top = $max_bottom = 0;
- // Columns:
- for ( $j = 0; $j < $colspan; $j++ ) {
- $col = $this->__col + $j;
- $this->_frames[$key]["columns"][] = $col;
- if ( $collapse ) {
- // Resolve horizontal borders
- $max_top = max($max_top, $this->_resolve_border($this->__row, $col, "horizontal", $bp["top"]));
- $max_bottom = max($max_bottom, $this->_resolve_border($this->__row + $rowspan, $col, "horizontal", $bp["bottom"]));
- }
- }
- $this->_frames[$key]["frame"] = $frame;
- // Handle seperated border model
- if ( !$collapse ) {
- list($h, $v) = $this->_table->get_style()->border_spacing;
- // Border spacing is effectively a margin between cells
- $v = $style->length_in_pt($v) / 2;
- $h = $style->length_in_pt($h) / 2;
- $style->margin = "$v $h";
- // The additional 1/2 width gets added to the table proper
- }
- else {
- // Drop the frame's actual border
- $style->border_left_width = $max_left / 2;
- $style->border_right_width = $max_right / 2;
- $style->border_top_width = $max_top / 2;
- $style->border_bottom_width = $max_bottom / 2;
- $style->margin = "none";
- }
- if ( !$this->_columns_locked ) {
- // Resolve the frame's width
- if ( $this->_fixed_layout ) {
- list($frame_min, $frame_max) = array(0, 10e-10);
- }
- else {
- list($frame_min, $frame_max) = $frame->get_min_max_width();
- }
-
- $width = $style->width;
- $val = null;
- if ( is_percent($width) ) {
- $var = "percent";
- $val = (float)rtrim($width, "% ") / $colspan;
- }
- else if ( $width !== "auto" ) {
- $var = "absolute";
- $val = $style->length_in_pt($frame_min) / $colspan;
- }
-
- $min = 0;
- $max = 0;
- for ( $cs = 0; $cs < $colspan; $cs++ ) {
-
- // Resolve the frame's width(s) with other cells
- $col =& $this->get_column( $this->__col + $cs );
-
- // Note: $var is either 'percent' or 'absolute'. We compare the
- // requested percentage or absolute values with the existing widths
- // and adjust accordingly.
- if ( isset($var) && $val > $col[$var] ) {
- $col[$var] = $val;
- $col["auto"] = false;
- }
-
- $min += $col["min-width"];
- $max += $col["max-width"];
- }
-
- if ( $frame_min > $min ) {
- // The frame needs more space. Expand each sub-column
- // FIXME try to avoid putting this dummy value when table-layout:fixed
- $inc = ($this->is_layout_fixed() ? 10e-10 : ($frame_min - $min) / $colspan);
- for ($c = 0; $c < $colspan; $c++) {
- $col =& $this->get_column($this->__col + $c);
- $col["min-width"] += $inc;
- }
- }
-
- if ( $frame_max > $max ) {
- // FIXME try to avoid putting this dummy value when table-layout:fixed
- $inc = ($this->is_layout_fixed() ? 10e-10 : ($frame_max - $max) / $colspan);
- for ($c = 0; $c < $colspan; $c++) {
- $col =& $this->get_column($this->__col + $c);
- $col["max-width"] += $inc;
- }
- }
- }
- $this->__col += $colspan;
- if ( $this->__col > $this->_num_cols )
- $this->_num_cols = $this->__col;
- }
- //........................................................................
- function add_row() {
- $this->__row++;
- $this->_num_rows++;
- // Find the next available column
- $i = 0;
- while ( isset($this->_cells[$this->__row][$i]) ) {
- $i++;
- }
- $this->__col = $i;
- }
- //........................................................................
- /**
- * Remove a row from the cellmap.
- *
- * @param Frame
- */
- function remove_row(Frame $row) {
- $key = $row->get_id();
- if ( !isset($this->_frames[$key]) ) {
- return; // Presumably this row has alredy been removed
- }
- $this->_row = $this->_num_rows--;
- $rows = $this->_frames[$key]["rows"];
- $columns = $this->_frames[$key]["columns"];
- // Remove all frames from this row
- foreach ( $rows as $r ) {
- foreach ( $columns as $c ) {
- if ( isset($this->_cells[$r][$c]) ) {
- $id = $this->_cells[$r][$c]->get_id();
-
- $this->_frames[$id] = null;
- unset($this->_frames[$id]);
-
- $this->_cells[$r][$c] = null;
- unset($this->_cells[$r][$c]);
- }
- }
-
- $this->_rows[$r] = null;
- unset($this->_rows[$r]);
- }
- $this->_frames[$key] = null;
- unset($this->_frames[$key]);
- }
- /**
- * Remove a row group from the cellmap.
- *
- * @param Frame $group The group to remove
- */
- function remove_row_group(Frame $group) {
- $key = $group->get_id();
- if ( !isset($this->_frames[$key]) ) {
- return; // Presumably this row has alredy been removed
- }
-
- $iter = $group->get_first_child();
- while ($iter) {
- $this->remove_row($iter);
- $iter = $iter->get_next_sibling();
- }
- $this->_frames[$key] = null;
- unset($this->_frames[$key]);
- }
- /**
- * Update a row group after rows have been removed
- *
- * @param Frame $group The group to update
- * @param Frame $last_row The last row in the row group
- */
- function update_row_group(Frame $group, Frame $last_row) {
- $g_key = $group->get_id();
- $r_key = $last_row->get_id();
- $r_rows = $this->_frames[$r_key]["rows"];
- $this->_frames[$g_key]["rows"] = range( $this->_frames[$g_key]["rows"][0], end($r_rows) );
- }
- //........................................................................
- function assign_x_positions() {
- // Pre-condition: widths must be resolved and assigned to columns and
- // column[0]["x"] must be set.
- if ( $this->_columns_locked ) {
- return;
- }
-
- $x = $this->_columns[0]["x"];
- foreach ( array_keys($this->_columns) as $j ) {
- $this->_columns[$j]["x"] = $x;
- $x += $this->_columns[$j]["used-width"];
- }
- }
- function assign_frame_heights() {
- // Pre-condition: widths and heights of each column & row must be
- // calcluated
- foreach ( $this->_frames as $arr ) {
- $frame = $arr["frame"];
- $h = 0;
- foreach( $arr["rows"] as $row ) {
- if ( !isset($this->_rows[$row]) ) {
- // The row has been removed because of a page split, so skip it.
- continue;
- }
-
- $h += $this->_rows[$row]["height"];
- }
- if ( $frame instanceof Table_Cell_Frame_Decorator ) {
- $frame->set_cell_height($h);
- }
- else {
- $frame->get_style()->height = $h;
- }
- }
- }
- //........................................................................
- /**
- * Re-adjust frame height if the table height is larger than its content
- */
- function set_frame_heights($table_height, $content_height) {
- // Distribute the increased height proportionally amongst each row
- foreach ( $this->_frames as $arr ) {
- $frame = $arr["frame"];
- $h = 0;
- foreach ($arr["rows"] as $row ) {
- if ( !isset($this->_rows[$row]) ) {
- continue;
- }
- $h += $this->_rows[$row]["height"];
- }
-
- if ( $content_height > 0 ) {
- $new_height = ($h / $content_height) * $table_height;
- }
- else {
- $new_height = 0;
- }
- if ( $frame instanceof Table_Cell_Frame_Decorator ) {
- $frame->set_cell_height($new_height);
- }
- else {
- $frame->get_style()->height = $new_height;
- }
- }
- }
- //........................................................................
- // Used for debugging:
- function __toString() {
- $str = "";
- $str .= "Columns:<br/>";
- $str .= pre_r($this->_columns, true);
- $str .= "Rows:<br/>";
- $str .= pre_r($this->_rows, true);
- $str .= "Frames:<br/>";
- $arr = array();
- foreach ( $this->_frames as $key => $val ) {
- $arr[$key] = array("columns" => $val["columns"], "rows" => $val["rows"]);
- }
-
- $str .= pre_r($arr, true);
- if ( php_sapi_name() == "cli" ) {
- $str = strip_tags(str_replace(array("<br/>","<b>","</b>"),
- array("\n",chr(27)."[01;33m", chr(27)."[0m"),
- $str));
- }
-
- return $str;
- }
- }
|