* * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this library in the file LICENSE.LGPL; if not, write to the * Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA * 02111-1307 USA * * Alternatively, you may distribute this software under the terms of the * PHP License, version 3.0 or later. A copy of this license should have * been distributed with this file in the file LICENSE.PHP . If this is not * the case, you can obtain a copy at http://www.php.net/license/3_0.txt. * * The latest version of DOMPDF might be available at: * http://www.digitaljunkies.ca/dompdf * * @link http://www.digitaljunkies.ca/dompdf * @copyright 2004 Benj Carson * @author Benj Carson * @package dompdf * @version 0.5.1 */ /* $Id: block_frame_reflower.cls.php,v 1.16 2006/07/07 21:31:02 benjcarson Exp $ */ /** * Reflows block frames * * @access private * @package dompdf */ class Block_Frame_Reflower extends Frame_Reflower { const MIN_JUSTIFY_WIDTH = 0.80; // (Minimum line width to justify, as // fraction of available width) function __construct(Block_Frame_Decorator $frame) { parent::__construct($frame); } //........................................................................ // Calculate the ideal used value for the width property as per: // http://www.w3.org/TR/CSS21/visudet.html#Computing_widths_and_margins protected function _calculate_width($width) { $style = $this->_frame->get_style(); $w = $this->_frame->get_containing_block("w"); $r = $style->length_in_pt($style->margin_right, $w); $l = $style->length_in_pt($style->margin_left, $w); // Handle 'auto' values $dims = array($style->border_left_width, $style->border_right_width, $style->padding_left, $style->padding_right, $width !== "auto" ? $width : 0, $r !== "auto" ? $r : 0, $l !== "auto" ? $l : 0); $sum = $style->length_in_pt($dims, $w); // Compare to the containing block $diff = $w - $sum; if ( $diff > 0 ) { // Find auto properties and get them to take up the slack if ( $width === "auto" ) $width = $diff; else if ( $l === "auto" && $r === "auto" ) $l = $r = round($diff / 2); else if ( $l === "auto" ) $l = $diff; else if ( $r === "auto" ) $r = $diff; } else if ($diff < 0) { // We are over constrained--set margin-right to the difference $r = $diff; } return array("width"=> $width, "margin_left" => $l, "margin_right" => $r); } // Call the above function, but resolve max/min widths protected function _calculate_restricted_width() { $style = $this->_frame->get_style(); $cb = $this->_frame->get_containing_block(); if ( !isset($cb["w"]) ) throw new DOMPDF_Exception("Box property calculation requires containing block width"); // Treat width 100% as auto if ( $style->width === "100%" ) $width = "auto"; else $width = $style->length_in_pt($style->width, $cb["w"]); extract($this->_calculate_width($width)); // Handle min/max width $min_width = $style->length_in_pt($style->min_width, $cb["w"]); $max_width = $style->length_in_pt($style->max_width, $cb["w"]); if ( $max_width !== "none" && $min_width > $max_width) // Swap 'em list($max_width, $min_width) = array($min_width, $max_width); if ( $max_width !== "none" && $width > $max_width ) extract($this->_calculate_width($max_width)); if ( $width < $min_width ) extract($this->_calculate_width($min_width)); return array($width, $margin_left, $margin_right); } //........................................................................ // Determine the unrestricted height of content within the block protected function _calculate_content_height() { // Calculate the actual height $height = 0; // Add the height of all lines foreach ($this->_frame->get_lines() as $line) $height += $line["h"]; return $height; } // Determine the frame's restricted height protected function _calculate_restricted_height() { $style = $this->_frame->get_style(); $content_height = $this->_calculate_content_height(); $cb = $this->_frame->get_containing_block(); $height = $style->height; // Handle percentage heights if ( isset($cb["h"]) ) { $height = $style->length_in_pt($height, $cb["h"]); } else if ( mb_strpos($height, "%") !== false ) $height = "auto"; else $height = $style->length_in_pt($height, $cb["w"]); // Remove margin, padding & borders $height -= $style->length_in_pt( array($style->margin_top, $style->padding_top, $style->border_top_width, $style->border_bottom_width, $style->padding_bottom, $style->margin_bottom), $cb["h"]); // Expand the height if overflow is visible if ( $content_height > $height && $style->overflow === "visible" ) $height = $content_height; // Only handle min/max height if the height is independent of the frame's content if ( !($style->overflow === "visible" || ($style->overflow === "hidden" && $height === "auto")) ) { $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; } return $height; } //........................................................................ protected function _text_align() { $style = $this->_frame->get_style(); $w = $this->_frame->get_containing_block("w"); $width = $style->length_in_pt($style->width, $w); // Adjust the justification of each of our lines. // http://www.w3.org/TR/CSS21/text.html#propdef-text-align switch ($style->text_align) { default: case "left": return; case "right": foreach ($this->_frame->get_lines() as $line) { // Move each child over by $dx $dx = $width - $line["w"]; foreach($line["frames"] as $frame) $frame->set_position( $frame->get_position("x") + $dx ); } break; case "justify": foreach ($this->_frame->get_lines() as $i => $line) { // Only set the spacing if the line is long enough. This is really // just an aesthetic choice ;) if ( $line["w"] > self::MIN_JUSTIFY_WIDTH * $width ) { // Set the spacing for each child if ( $line["wc"] > 1 ) $spacing = ($width - $line["w"]) / ($line["wc"] - 1); else $spacing = 0; $dx = 0; foreach($line["frames"] as $frame) { if ( !$frame instanceof Text_Frame_Decorator ) continue; $frame->set_position( $frame->get_position("x") + $dx ); $frame->set_text_spacing($spacing); $dx += mb_substr_count($frame->get_text(), " ") * $spacing; } // The line (should) now occupy the entire width $this->_frame->set_line($i, null, $width); } } break; case "center": case "centre": foreach ($this->_frame->get_lines() as $i => $line) { // Centre each line by moving each frame in the line by: $dx = ($width - $line["w"]) / 2; foreach ($line["frames"] as $frame) $frame->set_position( $frame->get_position("x") + $dx ); } break; } } /** * Align inline children vertically */ function vertical_align() { // Align each child vertically after each line is reflowed foreach ( $this->_frame->get_lines() as $i => $line ) { foreach ( $line["frames"] as $frame ) { $style = $frame->get_style(); if ( $style->display != "inline" && $style->display != "text" ) continue; $align = $style->vertical_align; $frame_h = $frame->get_margin_height(); switch ($align) { case "baseline": $y = $line["y"] + $line["h"] - $frame_h; break; case "middle": $y = $line["y"] + ($line["h"] + $frame_h) / 2; break; case "sub": $y = $line["y"] + 0.9 * $line["h"]; break; case "super": $y = $line["y"] + 0.1 * $line["h"]; break; case "text-top": case "top": // Not strictly accurate, but good enough for now $y = $line["y"]; break; case "text-bottom": case "bottom": $y = $line["y"] + $line["h"] - $frame_h; break; } $x = $frame->get_position("x"); $frame->set_position($x, $y); } } } //........................................................................ function reflow() { // Check if a page break is forced $page = $this->_frame->get_root(); $page->check_forced_page_break($this->_frame); // Bail if the page is full if ( $page->is_full() ) return; // Collapse margins if required $this->_collapse_margins(); $this->_frame->position(); $style = $this->_frame->get_style(); $cb = $this->_frame->get_containing_block(); list($x, $y) = $this->_frame->get_position(); // Determine the constraints imposed by this frame: calculate the width // of the content area: list($w, $left, $right) = $this->_calculate_restricted_width(); // Store the calculated properties $style->width = $w; $style->margin_left = $left."pt"; $style->margin_right = $right."pt"; // Adjust the first line based on the text-indent property $indent = $style->length_in_pt($style->text_indent, $cb["w"]); $this->_frame->increase_line_width($indent); // Determine the content edge $top = $style->length_in_pt(array($style->margin_top, $style->padding_top, $style->border_top_width), $cb["h"]); $bottom = $style->length_in_pt(array($style->border_bottom_width, $style->margin_bottom, $style->padding_bottom), $cb["h"]); $cb_x = $x + $left + $style->length_in_pt($style->border_left_width, $cb["w"]) + $style->length_in_pt($style->padding_left, $cb["w"]); $cb_y = $line_y = $y + $top; $cb_h = ($cb["h"] + $cb["y"]) - $bottom - $cb_y; // Set the y position of the first line in this block $this->_frame->set_current_line($line_y); // Set the containing blocks and reflow each child foreach ( $this->_frame->get_children() as $child ) { // Bail out if the page is full if ( $page->is_full() ) break; $child->set_containing_block($cb_x, $cb_y, $w, $cb_h); $child->reflow(); // Don't add the child to the line if a page break has occurred if ( $page->check_page_break($child) ) break; // It's okay to add the frame to the line $this->_frame->add_frame_to_line( $child ); } // Determine our height $style->height = $this->_calculate_restricted_height(); $this->_text_align(); $this->vertical_align(); } //........................................................................ } ?>