| 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;  }}
 |