Вход Регистрация
Файл: _rootinc/dompdf/include/text_frame_reflower.cls.php
Строк: 339
<?
/**
 * DOMPDF - PHP5 HTML to PDF renderer
 *
 * File: $RCSfile: text_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: text_frame_reflower.cls.php 186 2009-10-19 22:42:06Z eclecticgeek@gmail.com $ */

/**
 * Reflows text frames.
 *
 * @access private
 * @package dompdf
 */
class Text_Frame_Reflower extends Frame_Reflower {

  protected 
$_block_parent// Nearest block-level ancestor

  
function __construct(Text_Frame_Decorator $frame) {
    
parent::__construct($frame);
    
$this->_block_parent null;

    
// Handle text transform
    
$transform $this->_frame->get_style()->text_transform;
    switch ( 
strtolower($transform) ) {
    case 
"capitalize":
      
$this->_frame->set_textucwords($this->_frame->get_text()) );
      break;

    case 
"uppercase":
      
$this->_frame->set_textstrtoupper($this->_frame->get_text()) );
      break;

    case 
"lowercase":
      
$this->_frame->set_textstrtolower($this->_frame->get_text()) );
      break;

    default:
      
// Do nothing
      
break;
    }
  }

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

  
protected function _collapse_white_space($text) {
    
//$text = $this->_frame->get_text();
//     if ( $this->_block_parent->get_current_line("w") == 0 )
//       $text = ltrim($text, " nrt");
    
return preg_replace("/[sn]+/u"" "$text);
  }

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

  
protected function _line_break($text) {
    
$style $this->_frame->get_style();
    
$size $style->font_size;
    
$font $style->font_family;

    
// Determine the available width
    
$line_width $this->_frame->get_containing_block("w");
    
$current_line_width $this->_block_parent->get_current_line("w");

    
$available_width $line_width $current_line_width;

    
// split the text into words
    
$words preg_split('/([s-]+)/u'$text, -1PREG_SPLIT_DELIM_CAPTURE);
    
$wc count($words);

    
// Account for word-spacing
    
$word_spacing $style->length_in_pt($style->word_spacing);

    
// Determine the frame width including margin, padding & border
    
$text_width Font_Metrics::get_text_width($text$font$size$word_spacing);
    
$mbp_width =
      
$style->length_in_pt( array( $style->margin_left,
                                   
$style->border_left_width,
                                   
$style->padding_left,
                                   
$style->padding_right,
                                   
$style->border_right_width,
                                   
$style->margin_right), $line_width );
    
$frame_width $text_width $mbp_width;

// Debugging:
//    pre_r("Text: '" . htmlspecialchars($text). "'");
//    pre_r("width: " .$frame_width);
//    pre_r("textwidth + delta: $text_width + $mbp_width");
//    pre_r("font-size: $size");
//    pre_r("cb[w]: " .$line_width);
//    pre_r("available width: " . $available_width);
//    pre_r("current line width: " . $current_line_width);

//     pre_r($words);

    
if ( $frame_width <= $available_width )
      return 
false;

    
// Determine the split point
    
$width 0;
    
$str "";
    
reset($words);

    for (
$i 0$i count($words); $i += 2) {
      
$word $words[$i] . (isset($words[$i+1]) ? $words[$i+1] : "");
      
$word_width Font_Metrics::get_text_width($word$font$size$word_spacing);
      if ( 
$width $word_width $mbp_width $available_width )
        break;

      
$width += $word_width;
      
$str .= $word;

    }

    
// The first word has overflowed.   Force it onto the line
    
if ( $current_line_width == && $width == ) {
      
$width += $word_width;
      
$str .= $word;
    }

    
$offset mb_strlen($str);

// More debugging:
//     pre_var_dump($str);
//     pre_r("Width: ". $width);
//     pre_r("Offset: " . $offset);

    
return $offset;

  }

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

  
protected function _newline_break($text) {

    if ( (
$i mb_strpos($text"n")) === false)
      return 
false;

    return 
$i+1;

  }

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

  
protected function _layout_line() {
    
$style $this->_frame->get_style();
    
$text $this->_frame->get_text();
    
$size $style->font_size;
    
$font $style->font_family;
    
$word_spacing $style->length_in_pt($style->word_spacing);

    
// Determine the text height
    
$style->height Font_Metrics::get_font_height$font$size );

    
$split false;
    
$add_line false;

    
// Handle text transform:
    // http://www.w3.org/TR/CSS21/text.html#propdef-text-transform

    
switch ($style->text_transform) {

    default:
      break;

    case 
"capitalize":
      
$text mb_convert_case($textMB_CASE_TITLE'UTF-8');
      break;

    case 
"uppercase":
      
$text mb_convert_case($textMB_CASE_UPPER'UTF-8');
      break;

    case 
"lowercase":
      
$text mb_convert_case($textMB_CASE_LOWER'UTF-8');
      break;

    }
    
    
// Handle white-space property:
    // http://www.w3.org/TR/CSS21/text.html#propdef-white-space

    
switch ($style->white_space) {

    default:
    case 
"normal":
      
$this->_frame->set_text$text $this->_collapse_white_space($text) );
      if ( 
$text == "" )
        break;

      
$split $this->_line_break($text);
      break;

    case 
"pre":
      
$split $this->_newline_break($text);
      
$add_line $split !== false;
      break;

    case 
"nowrap":
      
$this->_frame->set_text$text $this->_collapse_white_space($text) );
      break;

    case 
"pre-wrap":
      
$split $this->_newline_break($text);

      if ( (
$tmp $this->_line_break($text)) !== false ) {
        
$add_line $split $tmp;
        
$split min($tmp$split);
      } else
        
$add_line true;

      break;

    case 
"pre-line":
      
// Collapse white-space except for n
      
$this->_frame->set_text$text preg_replace"/[ t]+/u"" "$text ) );

      if ( 
$text == "" )
        break;

      
$split $this->_newline_break($text);

      if ( (
$tmp $this->_line_break($text)) !== false ) {
        
$add_line $split $tmp;
        
$split min($tmp$split);
      } else
        
$add_line true;

      break;

    }

    
// Handle degenerate case
    
if ( $text === "" )
      return;

    if ( 
$split !== false) {

      
// Handle edge cases
      
if ( $split == && $text == " " ) {
        
$this->_frame->set_text("");
        return;
      }

      if ( 
$split == ) {

        
// Trim newlines from the beginning of the line
        //$this->_frame->set_text(ltrim($text, "nr"));

        
$this->_block_parent->add_line();
        
$this->_frame->position();

        
// Layout the new line
        
$this->_layout_line();

      } else if ( 
$split mb_strlen($this->_frame->get_text()) ) {

        
// split the line if required
        
$this->_frame->split_text($split);

        
// Remove any trailing newlines
        
$t $this->_frame->get_text();

        if ( 
$split && $t{$split-1} == "n" )
          
$this->_frame->set_textmb_substr($t0, -1) );

      }

      if ( 
$add_line ) {
        
$this->_block_parent->add_line();
        
$this->_frame->position();
      }

      
// Set our new width
      
$this->_frame->recalculate_width();

    } else {

      
$this->_frame->recalculate_width();

    }
  }

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

  
function reflow() {

    
$this->_block_parent $this->_frame->find_block_parent();

    
// Left trim the text if this is the first text on the line and we're
    // collapsing white space
//     if ( $this->_block_parent->get_current_line("w") == 0 &&
//          ($this->_frame->get_style()->white_space != "pre" ||
//           $this->_frame->get_style()->white_space != "pre-wrap") ) {
//       $this->_frame->set_text( ltrim( $this->_frame->get_text() ) );
//     }
    
    
$this->_frame->position();

    
$this->_layout_line();

  }

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

  // Returns an array(0 => min, 1 => max, "min" => min, "max" => max) of the
  // minimum and maximum widths of this frame
  
function get_min_max_width() {

    
$style $this->_frame->get_style();
    
$this->_block_parent $this->_frame->find_block_parent();
    
$line_width $this->_frame->get_containing_block("w");

    
$str $text $this->_frame->get_text();
    
$size $style->font_size;
    
$font $style->font_family;

    
$spacing $style->length_in_pt($style->word_spacing);

    switch(
$style->white_space) {

    default:
    case 
"normal":
      
$str preg_replace("/[sn]+/u"," "$str);
    case 
"pre-wrap":
    case 
"pre-line":

      
// Find the longest word (i.e. minimum length)

      // This technique (using arrays & an anonymous function) is actually
      // faster than doing a single-pass character by character scan.  Heh,
      // yes I took the time to bench it ;)
      
$words array_flip(preg_split("/[s-]+/u",$str, -1PREG_SPLIT_DELIM_CAPTURE));
      
array_walk($wordscreate_function('&$val,$str',
                                         
'$val = Font_Metrics::get_text_width($str, "'.$font.'", '.$size.', '.$spacing.');'));
      
arsort($words);
      
$min reset($words);
      break;

    case 
"pre":
      
$lines array_flip(preg_split("/n/u"$str));
      
array_walk($linescreate_function('&$val,$str',
                                         
'$val = Font_Metrics::get_text_width($str, "'.$font.'", '.$size.', '.$spacing.');'));

      
arsort($lines);
      
$min reset($lines);
      break;

    case 
"nowrap":
      
$min Font_Metrics::get_text_width($this->_collapse_white_space($str), $font$size$spacing);
      break;

    }

    switch (
$style->white_space) {

    default:
    case 
"normal":
    case 
"nowrap":
      
$str preg_replace("/[sn]+/u"," "$text);
      break;

    case 
"pre-line":
      
$str preg_replace"/[ t]+/u"" "$text);

    case 
"pre-wrap":
      
// Find the longest word (i.e. minimum length)
      
$lines array_flip(preg_split("/n/"$text));
      
array_walk($linescreate_function('&$val,$str',
                                         
'$val = Font_Metrics::get_text_width($str, "'.$font.'", '.$size.', '.$spacing.');'));
      
arsort($lines);
      
reset($lines);
      
$str key($lines);
      break;

    }

    
$max Font_Metrics::get_text_width($str$font$size$spacing);
    
    
$delta $style->length_in_pt(array($style->margin_left,
                                        
$style->border_left_width,
                                        
$style->padding_left,
                                        
$style->padding_right,
                                        
$style->border_right_width,
                                        
$style->margin_right), $line_width);
    
$min += $delta;
    
$max += $delta;

    return array(
$min$max"min" => $min"max" => $max);

  }

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