Вход Регистрация
Файл: _rootinc/dompdf/include/block_frame_reflower.cls.php
Строк: 476
<?
/**
 * 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.digitaljunkies.ca/dompdf
 *
 * @link http://www.digitaljunkies.ca/dompdf
 * @copyright 2004 Benj Carson
 * @author Benj Carson <benjcarson@digitaljunkies.ca>
 * @package dompdf
 * @version 0.5.1
 */

/* $Id: block_frame_reflower.cls.php 186 2009-10-19 22:42:06Z eclecticgeek@gmail.com $ */

/**
 * 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");

    
$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 ) {

      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;

    }

    
$ret = array("width"=> $width"margin_left" => $lm"margin_right" => $rm"left" => $left"right" => $right);

    return 
$ret;
  }

  
// 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$left$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->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 !== "height" $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 ) {

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

  }

  
//........................................................................

  
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"] > )
            
$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($inull$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();

    
$style $this->_frame->get_style();
    
$cb $this->_frame->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;
    
$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($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;

      
// If the frame is not absolutely positioned, It's okay to add the frame
      // to the line
      
if ( $child->get_style()->position != "absolute" &&
           
$child->get_style()->position != "fixed" ) {
        
$this->_frame->add_frame_to_line$child );
      }
    }

    
// 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();
  }

  
//........................................................................

}
?>
Онлайн: 1
Реклама