123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578 |
- <?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
- */
- /**
- * Reflows tables
- *
- * @access private
- * @package dompdf
- */
- class Table_Frame_Reflower extends Frame_Reflower {
- /**
- * Frame for this reflower
- *
- * @var Table_Frame_Decorator
- */
- protected $_frame;
- /**
- * Cache of results between call to get_min_max_width and assign_widths
- *
- * @var array
- */
- protected $_state;
- function __construct(Table_Frame_Decorator $frame) {
- $this->_state = null;
- parent::__construct($frame);
- }
- /**
- * State is held here so it needs to be reset along with the decorator
- */
- function reset() {
- $this->_state = null;
- $this->_min_max_cache = null;
- }
- //........................................................................
- protected function _assign_widths() {
- $style = $this->_frame->get_style();
- // Find the min/max width of the table and sort the columns into
- // absolute/percent/auto arrays
- $min_width = $this->_state["min_width"];
- $max_width = $this->_state["max_width"];
- $percent_used = $this->_state["percent_used"];
- $absolute_used = $this->_state["absolute_used"];
- $auto_min = $this->_state["auto_min"];
- $absolute =& $this->_state["absolute"];
- $percent =& $this->_state["percent"];
- $auto =& $this->_state["auto"];
- // Determine the actual width of the table
- $cb = $this->_frame->get_containing_block();
- $columns =& $this->_frame->get_cellmap()->get_columns();
- $width = $style->width;
- // Calculate padding & border fudge factor
- $left = $style->margin_left;
- $right = $style->margin_right;
-
- $centered = ( $left === "auto" && $right === "auto" );
- $left = $left === "auto" ? 0 : $style->length_in_pt($left, $cb["w"]);
- $right = $right === "auto" ? 0 : $style->length_in_pt($right, $cb["w"]);
- $delta = $left + $right;
-
- if ( !$centered ) {
- $delta += $style->length_in_pt(array(
- $style->padding_left,
- $style->border_left_width,
- $style->border_right_width,
- $style->padding_right),
- $cb["w"]);
- }
-
- $min_table_width = $style->length_in_pt( $style->min_width, $cb["w"] - $delta );
- // min & max widths already include borders & padding
- $min_width -= $delta;
- $max_width -= $delta;
-
- if ( $width !== "auto" ) {
- $preferred_width = $style->length_in_pt($width, $cb["w"]) - $delta;
- if ( $preferred_width < $min_table_width )
- $preferred_width = $min_table_width;
- if ( $preferred_width > $min_width )
- $width = $preferred_width;
- else
- $width = $min_width;
- } else {
- if ( $max_width + $delta < $cb["w"] )
- $width = $max_width;
- else if ( $cb["w"] - $delta > $min_width )
- $width = $cb["w"] - $delta;
- else
- $width = $min_width;
- if ( $width < $min_table_width )
- $width = $min_table_width;
- }
- // Store our resolved width
- $style->width = $width;
- $cellmap = $this->_frame->get_cellmap();
-
- if ( $cellmap->is_columns_locked() ) {
- return;
- }
- // If the whole table fits on the page, then assign each column it's max width
- if ( $width == $max_width ) {
- foreach (array_keys($columns) as $i)
- $cellmap->set_column_width($i, $columns[$i]["max-width"]);
- return;
- }
- // Determine leftover and assign it evenly to all columns
- if ( $width > $min_width ) {
- // We have four cases to deal with:
- //
- // 1. All columns are auto--no widths have been specified. In this
- // case we distribute extra space across all columns weighted by max-width.
- //
- // 2. Only absolute widths have been specified. In this case we
- // distribute any extra space equally among 'width: auto' columns, or all
- // columns if no auto columns have been specified.
- //
- // 3. Only percentage widths have been specified. In this case we
- // normalize the percentage values and distribute any remaining % to
- // width: auto columns. We then proceed to assign widths as fractions
- // of the table width.
- //
- // 4. Both absolute and percentage widths have been specified.
- $increment = 0;
- // Case 1:
- if ( $absolute_used == 0 && $percent_used == 0 ) {
- $increment = $width - $min_width;
- foreach (array_keys($columns) as $i) {
- $cellmap->set_column_width($i, $columns[$i]["min-width"] + $increment * ($columns[$i]["max-width"] / $max_width));
- }
- return;
- }
- // Case 2
- if ( $absolute_used > 0 && $percent_used == 0 ) {
- if ( count($auto) > 0 )
- $increment = ($width - $auto_min - $absolute_used) / count($auto);
- // Use the absolutely specified width or the increment
- foreach (array_keys($columns) as $i) {
- if ( $columns[$i]["absolute"] > 0 && count($auto) )
- $cellmap->set_column_width($i, $columns[$i]["min-width"]);
- else if ( count($auto) )
- $cellmap->set_column_width($i, $columns[$i]["min-width"] + $increment);
- else {
- // All absolute columns
- $increment = ($width - $absolute_used) * $columns[$i]["absolute"] / $absolute_used;
- $cellmap->set_column_width($i, $columns[$i]["min-width"] + $increment);
- }
- }
- return;
- }
- // Case 3:
- if ( $absolute_used == 0 && $percent_used > 0 ) {
- $scale = null;
- $remaining = null;
- // Scale percent values if the total percentage is > 100, or if all
- // values are specified as percentages.
- if ( $percent_used > 100 || count($auto) == 0)
- $scale = 100 / $percent_used;
- else
- $scale = 1;
- // Account for the minimum space used by the unassigned auto columns
- $used_width = $auto_min;
- foreach ($percent as $i) {
- $columns[$i]["percent"] *= $scale;
- $slack = $width - $used_width;
- $w = min($columns[$i]["percent"] * $width/100, $slack);
- if ( $w < $columns[$i]["min-width"] )
- $w = $columns[$i]["min-width"];
- $cellmap->set_column_width($i, $w);
- $used_width += $w;
- }
- // This works because $used_width includes the min-width of each
- // unassigned column
- if ( count($auto) > 0 ) {
- $increment = ($width - $used_width) / count($auto);
- foreach ($auto as $i)
- $cellmap->set_column_width($i, $columns[$i]["min-width"] + $increment);
- }
- return;
- }
- // Case 4:
- // First-come, first served
- if ( $absolute_used > 0 && $percent_used > 0 ) {
- $used_width = $auto_min;
- foreach ($absolute as $i) {
- $cellmap->set_column_width($i, $columns[$i]["min-width"]);
- $used_width += $columns[$i]["min-width"];
- }
- // Scale percent values if the total percentage is > 100 or there
- // are no auto values to take up slack
- if ( $percent_used > 100 || count($auto) == 0 )
- $scale = 100 / $percent_used;
- else
- $scale = 1;
- $remaining_width = $width - $used_width;
- foreach ($percent as $i) {
- $slack = $remaining_width - $used_width;
- $columns[$i]["percent"] *= $scale;
- $w = min($columns[$i]["percent"] * $remaining_width / 100, $slack);
- if ( $w < $columns[$i]["min-width"] )
- $w = $columns[$i]["min-width"];
- $columns[$i]["used-width"] = $w;
- $used_width += $w;
- }
- if ( count($auto) > 0 ) {
- $increment = ($width - $used_width) / count($auto);
- foreach ($auto as $i)
- $cellmap->set_column_width($i, $columns[$i]["min-width"] + $increment);
- }
- return;
- }
- } else { // we are over constrained
- // Each column gets its minimum width
- foreach (array_keys($columns) as $i)
- $cellmap->set_column_width($i, $columns[$i]["min-width"]);
- }
- }
- //........................................................................
- // Determine the frame's height based on min/max height
- protected function _calculate_height() {
- $style = $this->_frame->get_style();
- $height = $style->height;
- $cellmap = $this->_frame->get_cellmap();
- $cellmap->assign_frame_heights();
- $rows = $cellmap->get_rows();
- // Determine our content height
- $content_height = 0;
- foreach ( $rows as $r )
- $content_height += $r["height"];
- $cb = $this->_frame->get_containing_block();
- if ( !($style->overflow === "visible" ||
- ($style->overflow === "hidden" && $height === "auto")) ) {
- // Only handle min/max height if the height is independent of the frame's content
- $min_height = $style->min_height;
- $max_height = $style->max_height;
- if ( isset($cb["h"]) ) {
- $min_height = $style->length_in_pt($min_height, $cb["h"]);
- $max_height = $style->length_in_pt($max_height, $cb["h"]);
- } else if ( isset($cb["w"]) ) {
- if ( mb_strpos($min_height, "%") !== false )
- $min_height = 0;
- else
- $min_height = $style->length_in_pt($min_height, $cb["w"]);
- if ( mb_strpos($max_height, "%") !== false )
- $max_height = "none";
- else
- $max_height = $style->length_in_pt($max_height, $cb["w"]);
- }
- if ( $max_height !== "none" && $min_height > $max_height )
- // Swap 'em
- list($max_height, $min_height) = array($min_height, $max_height);
- if ( $max_height !== "none" && $height > $max_height )
- $height = $max_height;
- if ( $height < $min_height )
- $height = $min_height;
- } else {
- // Use the content height or the height value, whichever is greater
- if ( $height !== "auto" ) {
- $height = $style->length_in_pt($height, $cb["h"]);
- if ( $height <= $content_height )
- $height = $content_height;
- else
- $cellmap->set_frame_heights($height,$content_height);
- } else
- $height = $content_height;
- }
- return $height;
- }
- //........................................................................
- /**
- * @param Block_Frame_Decorator $block
- */
- function reflow(Block_Frame_Decorator $block = null) {
- /**
- * @var Table_Frame_Decorator
- */
- $frame = $this->_frame;
-
- // Check if a page break is forced
- $page = $frame->get_root();
- $page->check_forced_page_break($frame);
- // Bail if the page is full
- if ( $page->is_full() )
- return;
-
- // Let the page know that we're reflowing a table so that splits
- // are suppressed (simply setting page-break-inside: avoid won't
- // work because we may have an arbitrary number of block elements
- // inside tds.)
- $page->table_reflow_start();
-
- // Collapse vertical margins, if required
- $this->_collapse_margins();
- $frame->position();
- // Table layout algorithm:
- // http://www.w3.org/TR/CSS21/tables.html#auto-table-layout
- if ( is_null($this->_state) )
- $this->get_min_max_width();
- $cb = $frame->get_containing_block();
- $style = $frame->get_style();
- // This is slightly inexact, but should be okay. Add half the
- // border-spacing to the table as padding. The other half is added to
- // the cells themselves.
- if ( $style->border_collapse === "separate" ) {
- list($h, $v) = $style->border_spacing;
- $v = $style->length_in_pt($v) / 2;
- $h = $style->length_in_pt($h) / 2;
- $style->padding_left = $style->length_in_pt($style->padding_left, $cb["w"]) + $h;
- $style->padding_right = $style->length_in_pt($style->padding_right, $cb["w"]) + $h;
- $style->padding_top = $style->length_in_pt($style->padding_top, $cb["h"]) + $v;
- $style->padding_bottom = $style->length_in_pt($style->padding_bottom, $cb["h"]) + $v;
- }
- $this->_assign_widths();
- // Adjust left & right margins, if they are auto
- $width = $style->width;
- $left = $style->margin_left;
- $right = $style->margin_right;
- $diff = $cb["w"] - $width;
- if ( $left === "auto" && $right === "auto" ) {
- if ( $diff < 0 ) {
- $left = 0;
- $right = $diff;
- }
- else {
- $left = $right = $diff / 2;
- }
-
- $style->margin_left = "$left pt";
- $style->margin_right = "$right pt";
- } else {
- if ( $left === "auto" ) {
- $left = $style->length_in_pt($cb["w"] - $right - $width, $cb["w"]);
- }
- if ( $right === "auto" ) {
- $left = $style->length_in_pt($left, $cb["w"]);
- }
- }
- list($x, $y) = $frame->get_position();
- // Determine the content edge
- $content_x = $x + $left + $style->length_in_pt(array($style->padding_left,
- $style->border_left_width), $cb["w"]);
- $content_y = $y + $style->length_in_pt(array($style->margin_top,
- $style->border_top_width,
- $style->padding_top), $cb["h"]);
- if ( isset($cb["h"]) )
- $h = $cb["h"];
- else
- $h = null;
- $cellmap = $frame->get_cellmap();
- $col =& $cellmap->get_column(0);
- $col["x"] = $content_x;
- $row =& $cellmap->get_row(0);
- $row["y"] = $content_y;
- $cellmap->assign_x_positions();
- // Set the containing block of each child & reflow
- foreach ( $frame->get_children() as $child ) {
- // Bail if the page is full
- if ( !$page->in_nested_table() && $page->is_full() )
- break;
- $child->set_containing_block($content_x, $content_y, $width, $h);
- $child->reflow();
- if ( !$page->in_nested_table() )
- // Check if a split has occured
- $page->check_page_break($child);
- }
- // Assign heights to our cells:
- $style->height = $this->_calculate_height();
- if ( $style->border_collapse === "collapse" ) {
- // Unset our borders because our cells are now using them
- $style->border_style = "none";
- }
- $page->table_reflow_end();
- // Debugging:
- //echo ($this->_frame->get_cellmap());
-
- if ( $block && $style->float === "none" && $frame->is_in_flow() ) {
- $block->add_frame_to_line($frame);
- $block->add_line();
- }
- }
- //........................................................................
- function get_min_max_width() {
- if ( !is_null($this->_min_max_cache) )
- return $this->_min_max_cache;
- $style = $this->_frame->get_style();
- $this->_frame->normalise();
- // Add the cells to the cellmap (this will calcluate column widths as
- // frames are added)
- $this->_frame->get_cellmap()->add_frame($this->_frame);
- // Find the min/max width of the table and sort the columns into
- // absolute/percent/auto arrays
- $this->_state = array();
- $this->_state["min_width"] = 0;
- $this->_state["max_width"] = 0;
- $this->_state["percent_used"] = 0;
- $this->_state["absolute_used"] = 0;
- $this->_state["auto_min"] = 0;
- $this->_state["absolute"] = array();
- $this->_state["percent"] = array();
- $this->_state["auto"] = array();
- $columns =& $this->_frame->get_cellmap()->get_columns();
- foreach (array_keys($columns) as $i) {
- $this->_state["min_width"] += $columns[$i]["min-width"];
- $this->_state["max_width"] += $columns[$i]["max-width"];
- if ( $columns[$i]["absolute"] > 0 ) {
- $this->_state["absolute"][] = $i;
- $this->_state["absolute_used"] += $columns[$i]["absolute"];
- } else if ( $columns[$i]["percent"] > 0 ) {
- $this->_state["percent"][] = $i;
- $this->_state["percent_used"] += $columns[$i]["percent"];
- } else {
- $this->_state["auto"][] = $i;
- $this->_state["auto_min"] += $columns[$i]["min-width"];
- }
- }
- // Account for margins & padding
- $dims = array($style->border_left_width,
- $style->border_right_width,
- $style->padding_left,
- $style->padding_right,
- $style->margin_left,
- $style->margin_right);
- if ( $style->border_collapse !== "collapse" )
- list($dims[]) = $style->border_spacing;
- $delta = $style->length_in_pt($dims, $this->_frame->get_containing_block("w"));
- $this->_state["min_width"] += $delta;
- $this->_state["max_width"] += $delta;
- return $this->_min_max_cache = array(
- $this->_state["min_width"],
- $this->_state["max_width"],
- "min" => $this->_state["min_width"],
- "max" => $this->_state["max_width"],
- );
- }
- }
|