Server : Apache/2.4.18 (Ubuntu) System : Linux canvaswebdesign 3.13.0-71-generic #114-Ubuntu SMP Tue Dec 1 02:34:22 UTC 2015 x86_64 User : oppastar ( 1041) PHP Version : 7.0.33-0ubuntu0.16.04.15 Disable Function : pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority, Directory : /var/www/ptbos.biz/public_html/application/third_party/dompdf/include/ |
Upload File : |
<?php /** * DOMPDF - PHP5 HTML to PDF renderer * * File: $RCSfile: block_frame_reflower.cls.php,v $ * Created on: 2004-06-17 * * Copyright (c) 2004 - Benj Carson <benjcarson@digitaljunkies.ca> * * 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.dompdf.com/ * * @link http://www.dompdf.com/ * @copyright 2004 Benj Carson * @author Benj Carson <benjcarson@digitaljunkies.ca> * @package dompdf */ /* $Id: block_frame_reflower.cls.php 358 2011-01-30 22:22:47Z fabien.menager $ */ /** * Reflows block frames * * @access private * @package dompdf */ class Block_Frame_Reflower extends Frame_Reflower { // Minimum line width to justify, as fraction of available width const MIN_JUSTIFY_WIDTH = 0.80; /** * @var Block_Frame_Decorator */ protected $_frame; 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 * * @param float $width * @return array */ protected function _calculate_width($width) { $style = $this->_frame->get_style(); $w = $this->_frame->get_containing_block("w"); if( $style->position === "fixed" ) $w = $this->_frame->get_parent()->get_containing_block("w"); $rm = $style->length_in_pt($style->margin_right, $w); $lm = $style->length_in_pt($style->margin_left, $w); $left = $style->length_in_pt($style->left, $w); $right = $style->length_in_pt($style->right, $w); // Handle 'auto' values $dims = array($style->border_left_width, $style->border_right_width, $style->padding_left, $style->padding_right, $width !== "auto" ? $width : 0, $rm !== "auto" ? $rm : 0, $lm !== "auto" ? $lm : 0); // absolutely positioned boxes take the 'left' and 'right' properties into account if ( $style->position === "absolute" || $style->position === "fixed" ) { $absolute = true; $dims[] = $left !== "auto" ? $left : 0; $dims[] = $right !== "auto" ? $right : 0; } else { $absolute = false; } $sum = $style->length_in_pt($dims, $w); // Compare to the containing block $diff = $w - $sum; if ( $diff > 0 ) { if ( $absolute ) { // resolve auto properties: see // http://www.w3.org/TR/CSS21/visudet.html#abs-non-replaced-width if ( $width === "auto" && $left === "auto" && $right === "auto" ) { if ( $lm === "auto" ) $lm = 0; if ( $rm === "auto" ) $rm = 0; // Technically, the width should be "shrink-to-fit" i.e. based on the // preferred width of the content... a little too costly here as a // special case. Just get the width to take up the slack: $left = 0; $right = 0; $width = $diff; } else if ( $width === "auto" ) { if ( $lm === "auto" ) $lm = 0; if ( $rm === "auto" ) $rm = 0; if ( $left === "auto" ) $left = 0; if ( $right === "auto" ) $right = 0; $width = $diff; } else if ( $left === "auto" ) { if ( $lm === "auto" ) $lm = 0; if ( $rm === "auto" ) $rm = 0; if ( $right === "auto" ) $right = 0; $left = $diff; } else if ( $right === "auto" ) { if ( $lm === "auto" ) $lm = 0; if ( $rm === "auto" ) $rm = 0; $right = $diff; } } else { // Find auto properties and get them to take up the slack if ( $width === "auto" ) $width = $diff; else if ( $lm === "auto" && $rm === "auto" ) $lm = $rm = round($diff / 2); else if ( $lm === "auto" ) $lm = $diff; else if ( $rm === "auto" ) $rm = $diff; } } else if ($diff < 0) { // We are over constrained--set margin-right to the difference $rm = $diff; } return array("width"=> $width, "margin_left" => $lm, "margin_right" => $rm, "left" => $left, "right" => $right); } /** * Call the above function, but resolve max/min widths * @return array */ protected function _calculate_restricted_width() { $frame = $this->_frame; $style = $frame->get_style(); $cb = $frame->get_containing_block(); if ( $style->position === "fixed" ) $cb = $frame->get_root()->get_containing_block(); //if ( $style->position === "absolute" ) // $cb = $frame->find_positionned_parent()->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, $left, $right); } /** * Determine the unrestricted height of content within the block * by adding each line's height * @return float */ protected function _calculate_content_height() { $height = 0; foreach ($this->_frame->get_lines() as $line) { $height += $line["h"]; } return $height; } /** * Determine the frame's restricted height * @return array */ protected function _calculate_restricted_height() { $style = $this->_frame->get_style(); $content_height = $this->_calculate_content_height(); $cb = $this->_frame->get_containing_block(); $height = $style->length_in_pt($style->height, $cb["h"]); $top = $style->length_in_pt($style->top, $cb["h"]); $bottom = $style->length_in_pt($style->bottom, $cb["h"]); $margin_top = $style->length_in_pt($style->margin_top, $cb["h"]); $margin_bottom = $style->length_in_pt($style->margin_bottom, $cb["h"]); if ( $style->position === "absolute" || $style->position === "fixed" ) { // see http://www.w3.org/TR/CSS21/visudet.html#abs-non-replaced-height $dims = array($top !== "auto" ? $top : 0, $style->margin_top !== "auto" ? $style->margin_top : 0, $style->padding_top, $style->border_top_width, $height !== "auto" ? $height : 0, $style->border_bottom_width, $style->padding_bottom, $style->margin_bottom !== "auto" ? $style->margin_bottom : 0, $bottom !== "auto" ? $bottom : 0); $sum = $style->length_in_pt($dims, $cb["h"]); $diff = $cb["h"] - $sum; if ( $diff > 0 ) { if ( $height === "auto" && $top === "auto" && $bottom === "auto" ) { if ( $margin_top === "auto" ) $margin_top = 0; if ( $margin_bottom === "auto" ) $margin_bottom = 0; $height = $diff; } else if ( $height === "auto" && $top === "auto" ) { if ( $margin_top === "auto" ) $margin_top = 0; if ( $margin_bottom === "auto" ) $margin_bottom = 0; $height = $content_height; $top = $diff - $content_height; } else if ( $height === "auto" && $bottom === "auto" ) { if ( $margin_top === "auto" ) $margin_top = 0; if ( $margin_bottom === "auto" ) $margin_bottom = 0; $height = $content_height; $bottom = $diff - $content_height; } else if ( $top === "auto" && $bottom === "auto" ) { if ( $margin_top === "auto" ) $margin_top = 0; if ( $margin_bottom === "auto" ) $margin_bottom = 0; $bottom = $diff; } else if ( $top === "auto" ) { if ( $margin_top === "auto" ) $margin_top = 0; if ( $margin_bottom === "auto" ) $margin_bottom = 0; $top = $diff; } else if ( $height === "auto" ) { if ( $margin_top === "auto" ) $margin_top = 0; if ( $margin_bottom === "auto" ) $margin_bottom = 0; $height = $diff; } else if ( $bottom === "auto" ) { if ( $margin_top === "auto" ) $margin_top = 0; if ( $margin_bottom === "auto" ) $margin_bottom = 0; $bottom = $diff; } else { if ( $style->overflow === "visible" ) { // set all autos to zero if ( $margin_top === "auto" ) $margin_top = 0; if ( $margin_bottom === "auto" ) $margin_bottom = 0; if ( $top === "auto" ) $top = 0; if ( $bottom === "auto" ) $bottom = 0; if ( $height === "auto" ) $height = $content_height; } // FIXME: overflow hidden } } } else { // Expand the height if overflow is visible if ( $height === "auto" && $content_height > $height /* && $style->overflow === "visible" */) $height = $content_height; // FIXME: this should probably be moved to a seperate function as per // _calculate_restricted_width // 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 array($height, $margin_top, $margin_bottom, $top, $bottom); } /** * Adjust the justification of each of our lines. * http://www.w3.org/TR/CSS21/text.html#propdef-text-align */ protected function _text_align() { $style = $this->_frame->get_style(); $w = $this->_frame->get_containing_block("w"); $width = $style->length_in_pt($style->width, $w); switch ($style->text_align) { default: case "left": foreach ($this->_frame->get_lines() as $line) { if ( !$line["left"] ) continue; foreach($line["frames"] as $frame) { if ( $frame instanceof Block_Frame_Decorator) continue; $frame->set_position( $frame->get_position("x") + $line["left"] ); } } return; case "right": foreach ($this->_frame->get_lines() as $line) { // Move each child over by $dx $dx = $width - $line["w"] - $line["right"]; foreach($line["frames"] as $frame) { // Block frames are not aligned by text-align if ($frame instanceof Block_Frame_Decorator) continue; $frame->set_position( $frame->get_position("x") + $dx ); } } break; case "justify": // We justify all lines except the last one $lines = $this->_frame->get_lines(); // needs to be a variable (strict standards) $lines = array_splice($lines, 0, -1); foreach($lines as $i => $line) { if ( $line["br"] ) { unset($lines[$i]); } } // One space character's width. Will be used to get a more accurate spacing $space_width = Font_Metrics::get_text_width(" ", $style->font_family, $style->font_size); foreach ($lines as $i => $line) { if ( $line["left"] ) { foreach($line["frames"] as $frame) { if ( !$frame instanceof Text_Frame_Decorator ) continue; $frame->set_position( $frame->get_position("x") + $line["left"] ); } } // Only set the spacing if the line is long enough. This is really // just an aesthetic choice ;) //if ( $line["left"] + $line["w"] + $line["right"] > self::MIN_JUSTIFY_WIDTH * $width ) { // Set the spacing for each child if ( $line["wc"] > 1 ) $spacing = ($width - ($line["left"] + $line["w"] + $line["right"]) + $space_width) / ($line["wc"] - 1); else $spacing = 0; $dx = 0; foreach($line["frames"] as $frame) { if ( !$frame instanceof Text_Frame_Decorator ) continue; $text = $frame->get_text(); $spaces = mb_substr_count($text, " "); $char_spacing = $style->length_in_pt($style->letter_spacing); $_spacing = $spacing + $char_spacing; $frame->set_position( $frame->get_position("x") + $dx ); $frame->set_text_spacing($_spacing); $dx += $spaces * $_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 $line) { // Centre each line by moving each frame in the line by: $dx = ($width + $line["left"] - $line["w"] - $line["right"] ) / 2; foreach ($line["frames"] as $frame) { // Block frames are not aligned by text-align if ($frame instanceof Block_Frame_Decorator) continue; $frame->set_position( $frame->get_position("x") + $dx ); } } break; } } /** * Align inline children vertically. * Aligns each child vertically after each line is reflowed */ function vertical_align() { foreach ( $this->_frame->get_lines() as $i => $line ) { $height = $line["h"]; foreach ( $line["frames"] as $frame ) { $style = $frame->get_style(); if ( $style->display !== "inline" && $style->display !== "text" ) continue; // FIXME? if ( $this instanceof Table_Cell_Frame_Reflower ) $align = $frame->get_frame()->get_style()->vertical_align; else $align = $frame->get_frame()->get_parent()->get_style()->vertical_align; $frame_h = $frame->get_margin_height(); $y = $line["y"]; switch ($align) { case "baseline": $y += $height - $frame_h; break; case "middle": $y += ($height + $frame_h) / 2; break; case "sub": $y += 0.2 * $height; break; case "super": $y += -0.3 * $height; break; case "text-top": case "top": // Not strictly accurate, but good enough for now break; case "text-bottom": case "bottom": $y += $height - $frame_h; break; } $x = $frame->get_position("x"); $frame->set_position($x, $y); } } } function reflow(Frame_Decorator $block = null) { // 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; // Generated content $this->_set_content(); // Collapse margins if required $this->_collapse_margins(); $style = $this->_frame->get_style(); $cb = $this->_frame->get_containing_block(); if ( $style->counter_increment && ($increment = $style->counter_increment) !== "none" ) $this->_frame->increment_counter($increment); if ( $style->position === "fixed" ) $cb = $this->_frame->get_root()->get_containing_block(); // Determine the constraints imposed by this frame: calculate the width // of the content area: list($w, $left_margin, $right_margin, $left, $right) = $this->_calculate_restricted_width(); // Store the calculated properties $style->width = $w . "pt"; $style->margin_left = $left_margin."pt"; $style->margin_right = $right_margin."pt"; $style->left = $left ."pt"; $style->right = $right . "pt"; // Update the position $this->_frame->position(); list($x, $y) = $this->_frame->get_position(); // 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_margin + $style->length_in_pt(array($style->border_left_width, $style->padding_left), $cb["w"]); $cb_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($cb_y); $floating_children = array(); // 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; // Floating siblings if ( DOMPDF_ENABLE_CSS_FLOAT && count($floating_children) ) { $offset_left = 0; $offset_right = 0; // We need to reflow the child to know its initial x position $child->set_containing_block($cb_x, $cb_y, $w, $cb_h); $child->reflow($this->_frame); $current_line = $this->_frame->get_current_line(); foreach ( $floating_children as $child_key => $floating_child ) { $float = $floating_child->get_style()->float; $floating_width = $floating_child->get_margin_width(); $floating_x = $floating_child->get_position("x"); if ( $float === "left" ) { if ($current_line["left"] + $child->get_position("x") > $floating_x + $floating_width) continue; } else { if ($current_line["left"] + $child->get_position("x") + $child->get_margin_width() < $w - $floating_width - $current_line["right"]) continue; } // If the child is still shifted by the floating element if ( $floating_child->get_position("y") + $floating_child->get_margin_height() > $current_line["y"] ) { if ( $float === "left" ) $offset_left += $floating_width; else $offset_right += $floating_width; } // else, the floating element won't shift anymore else { unset($floating_children[$child_key]); } } if ( $offset_left ) $this->_frame->set_current_line(array("left" => $offset_left)); if ( $offset_right ) $this->_frame->set_current_line(array("right" => $offset_right)); } $child->set_containing_block($cb_x, $cb_y, $w, $cb_h); $child->reflow($this->_frame); // Don't add the child to the line if a page break has occurred if ( $page->check_page_break($child) ) break; $child_style = $child->get_style(); if ( DOMPDF_ENABLE_CSS_FLOAT && $child_style->float !== "none") { $floating_children[] = $child; // Remove next frame's beginning whitespace $next = $child->get_next_sibling(); if ( $next && $next instanceof Text_Frame_Decorator) { $next->set_text(ltrim($next->get_text())); } $float_x = $cb_x; $float_y = $this->_frame->get_current_line("y"); $child_style = $child->get_style(); switch( $child_style->float ) { case "left": break; case "right": $width = $w; $float_x += ($width - $child->get_margin_width()); break; } $child->set_position($float_x, $float_y); } } // Determine our height list($height, $margin_top, $margin_bottom, $top, $bottom) = $this->_calculate_restricted_height(); $style->height = $height; $style->margin_top = $margin_top; $style->margin_bottom = $margin_bottom; $style->top = $top; $style->bottom = $bottom; $this->_text_align(); $this->vertical_align(); if ( $block ) { $block->add_frame_to_line($this->_frame); } } }