PDF, utilise fpdf de Olivier PLATHEY * Distribué sous la licence LGPL. * * @author Laurent MINGUET * @version 3.28 - 18/01/2010 */ if (!defined('__CLASS_STYLEHTML__')) { define('__CLASS_STYLEHTML__', true); class styleHTML { var $css = array(); // tableau des CSS var $css_keys = array(); // tableau des clefs CSS, pour l'ordre d'execution var $value = array(); // valeurs actuelles var $table = array(); // tableau d'empilement pour historisation des niveaux var $pdf = null; // référence au PDF parent var $htmlColor = array(); // liste des couleurs HTML var $onlyLeft = false; // indique si on est dans un sous HTML et qu'on bloque à gauche var $defaultFont = null; // fonte par défaut si la fonte demandée n'existe pas /** * Constructeur * * @param &pdf référence à l'objet HTML2PDF parent * @return null */ function styleHTML(&$pdf) { $this->init(); // initialisation $this->pdf = &$pdf; } /** * définit la fonte par défaut si aucun fonte n'est spécifiée, ou si la fonte demandée n'existe pas * * @param string nom de la fonte par defaut. si null : Arial pour fonte non spécifiée, et erreur pour fonte non existante * @return string nom de l'ancienne fonte par defaut */ function setDefaultFont($default = null) { $old = $this->defaultFont; $this->defaultFont = $default; if ($default) $this->value['font-family'] = $default; return $old; } /** * Initialisation du style * * @return null */ function init() { $color = array(); $color['AliceBlue'] = '#F0F8FF'; $color['AntiqueWhite'] = '#FAEBD7'; $color['Aqua'] = '#00FFFF'; $color['Aquamarine'] = '#7FFFD4'; $color['Azure'] = '#F0FFFF'; $color['Beige'] = '#F5F5DC'; $color['Bisque'] = '#FFE4C4'; $color['Black'] = '#000000'; $color['BlanchedAlmond'] = '#FFEBCD'; $color['Blue'] = '#0000FF'; $color['BlueViolet'] = '#8A2BE2'; $color['Brown'] = '#A52A2A'; $color['BurlyWood'] = '#DEB887'; $color['CadetBlue'] = '#5F9EA0'; $color['Chartreuse'] = '#7FFF00'; $color['Chocolate'] = '#D2691E'; $color['Coral'] = '#FF7F50'; $color['CornflowerBlue'] = '#6495ED'; $color['Cornsilk'] = '#FFF8DC'; $color['Crimson'] = '#DC143C'; $color['Cyan'] = '#00FFFF'; $color['DarkBlue'] = '#00008B'; $color['DarkCyan'] = '#008B8B'; $color['DarkGoldenRod'] = '#B8860B'; $color['DarkGray'] = '#A9A9A9'; $color['DarkGrey'] = '#A9A9A9'; $color['DarkGreen'] = '#006400'; $color['DarkKhaki'] = '#BDB76B'; $color['DarkMagenta'] = '#8B008B'; $color['DarkOliveGreen'] = '#556B2F'; $color['Darkorange'] = '#FF8C00'; $color['DarkOrchid'] = '#9932CC'; $color['DarkRed'] = '#8B0000'; $color['DarkSalmon'] = '#E9967A'; $color['DarkSeaGreen'] = '#8FBC8F'; $color['DarkSlateBlue'] = '#483D8B'; $color['DarkSlateGray'] = '#2F4F4F'; $color['DarkSlateGrey'] = '#2F4F4F'; $color['DarkTurquoise'] = '#00CED1'; $color['DarkViolet'] = '#9400D3'; $color['DeepPink'] = '#FF1493'; $color['DeepSkyBlue'] = '#00BFFF'; $color['DimGray'] = '#696969'; $color['DimGrey'] = '#696969'; $color['DodgerBlue'] = '#1E90FF'; $color['FireBrick'] = '#B22222'; $color['FloralWhite'] = '#FFFAF0'; $color['ForestGreen'] = '#228B22'; $color['Fuchsia'] = '#FF00FF'; $color['Gainsboro'] = '#DCDCDC'; $color['GhostWhite'] = '#F8F8FF'; $color['Gold'] = '#FFD700'; $color['GoldenRod'] = '#DAA520'; $color['Gray'] = '#808080'; $color['Grey'] = '#808080'; $color['Green'] = '#008000'; $color['GreenYellow'] = '#ADFF2F'; $color['HoneyDew'] = '#F0FFF0'; $color['HotPink'] = '#FF69B4'; $color['IndianRed'] = '#CD5C5C'; $color['Indigo'] = '#4B0082'; $color['Ivory'] = '#FFFFF0'; $color['Khaki'] = '#F0E68C'; $color['Lavender'] = '#E6E6FA'; $color['LavenderBlush'] = '#FFF0F5'; $color['LawnGreen'] = '#7CFC00'; $color['LemonChiffon'] = '#FFFACD'; $color['LightBlue'] = '#ADD8E6'; $color['LightCoral'] = '#F08080'; $color['LightCyan'] = '#E0FFFF'; $color['LightGoldenRodYellow'] = '#FAFAD2'; $color['LightGray'] = '#D3D3D3'; $color['LightGrey'] = '#D3D3D3'; $color['LightGreen'] = '#90EE90'; $color['LightPink'] = '#FFB6C1'; $color['LightSalmon'] = '#FFA07A'; $color['LightSeaGreen'] = '#20B2AA'; $color['LightSkyBlue'] = '#87CEFA'; $color['LightSlateGray'] = '#778899'; $color['LightSlateGrey'] = '#778899'; $color['LightSteelBlue'] = '#B0C4DE'; $color['LightYellow'] = '#FFFFE0'; $color['Lime'] = '#00FF00'; $color['LimeGreen'] = '#32CD32'; $color['Linen'] = '#FAF0E6'; $color['Magenta'] = '#FF00FF'; $color['Maroon'] = '#800000'; $color['MediumAquaMarine'] = '#66CDAA'; $color['MediumBlue'] = '#0000CD'; $color['MediumOrchid'] = '#BA55D3'; $color['MediumPurple'] = '#9370D8'; $color['MediumSeaGreen'] = '#3CB371'; $color['MediumSlateBlue'] = '#7B68EE'; $color['MediumSpringGreen'] = '#00FA9A'; $color['MediumTurquoise'] = '#48D1CC'; $color['MediumVioletRed'] = '#C71585'; $color['MidnightBlue'] = '#191970'; $color['MintCream'] = '#F5FFFA'; $color['MistyRose'] = '#FFE4E1'; $color['Moccasin'] = '#FFE4B5'; $color['NavajoWhite'] = '#FFDEAD'; $color['Navy'] = '#000080'; $color['OldLace'] = '#FDF5E6'; $color['Olive'] = '#808000'; $color['OliveDrab'] = '#6B8E23'; $color['Orange'] = '#FFA500'; $color['OrangeRed'] = '#FF4500'; $color['Orchid'] = '#DA70D6'; $color['PaleGoldenRod'] = '#EEE8AA'; $color['PaleGreen'] = '#98FB98'; $color['PaleTurquoise'] = '#AFEEEE'; $color['PaleVioletRed'] = '#D87093'; $color['PapayaWhip'] = '#FFEFD5'; $color['PeachPuff'] = '#FFDAB9'; $color['Peru'] = '#CD853F'; $color['Pink'] = '#FFC0CB'; $color['Plum'] = '#DDA0DD'; $color['PowderBlue'] = '#B0E0E6'; $color['Purple'] = '#800080'; $color['Red'] = '#FF0000'; $color['RosyBrown'] = '#BC8F8F'; $color['RoyalBlue'] = '#4169E1'; $color['SaddleBrown'] = '#8B4513'; $color['Salmon'] = '#FA8072'; $color['SandyBrown'] = '#F4A460'; $color['SeaGreen'] = '#2E8B57'; $color['SeaShell'] = '#FFF5EE'; $color['Sienna'] = '#A0522D'; $color['Silver'] = '#C0C0C0'; $color['SkyBlue'] = '#87CEEB'; $color['SlateBlue'] = '#6A5ACD'; $color['SlateGray'] = '#708090'; $color['SlateGrey'] = '#708090'; $color['Snow'] = '#FFFAFA'; $color['SpringGreen'] = '#00FF7F'; $color['SteelBlue'] = '#4682B4'; $color['Tan'] = '#D2B48C'; $color['Teal'] = '#008080'; $color['Thistle'] = '#D8BFD8'; $color['Tomato'] = '#FF6347'; $color['Turquoise'] = '#40E0D0'; $color['Violet'] = '#EE82EE'; $color['Wheat'] = '#F5DEB3'; $color['White'] = '#FFFFFF'; $color['WhiteSmoke'] = '#F5F5F5'; $color['Yellow'] = '#FFFF00'; $color['YellowGreen'] = '#9ACD32'; $this->htmlColor = array(); foreach($color as $key => $val) $this->htmlColor[strtolower($key)] = $val; unset($color); $this->table = array(); $this->value = array(); $this->initStyle(); // initialisation des styles sans héritages $this->resetStyle(); } function initStyle() { $this->value['id_balise'] = 'body'; // balise $this->value['id_name'] = null; // name $this->value['id_id'] = null; // id $this->value['id_class'] = null; // class $this->value['id_lst'] = array('*'); // lst de dependance $this->value['mini-size'] = 1.; // rapport de taille spécifique aux sup, sub $this->value['mini-decal'] = 0; // rapport de position spécifique aux sup, sub $this->value['font-family'] = 'Arial'; $this->value['font-bold'] = false; $this->value['font-italic'] = false; $this->value['font-underline'] = false; $this->value['font-overline'] = false; $this->value['font-linethrough'] = false; $this->value['text-transform'] = 'none'; $this->value['font-size'] = $this->ConvertToMM('10pt'); $this->value['text-indent'] = 0; $this->value['text-align'] = 'left'; $this->value['vertical-align'] = 'middle'; $this->value['line-height'] = 'normal'; $this->value['position'] = null; $this->value['x'] = null; $this->value['y'] = null; $this->value['width'] = 0; $this->value['height'] = 0; $this->value['top'] = null; $this->value['right'] = null; $this->value['bottom'] = null; $this->value['left'] = null; $this->value['float'] = null; $this->value['display'] = null; $this->value['rotate'] = null; $this->value['overflow'] = 'visible'; $this->value['color'] = array(0, 0, 0); $this->value['background'] = array('color' => null, 'image' => null, 'position' => null, 'repeat' => null); $this->value['border'] = array(); $this->value['padding'] = array(); $this->value['margin'] = array(); $this->value['margin-auto'] = false; $this->value['list-style-type'] = ''; $this->value['list-style-image'] = ''; $this->value['xc'] = null; $this->value['yc'] = null; } /** * Initialisation des styles sans héritages * * @param string balise HTML * @return null */ function resetStyle($balise = '') { $collapse = isset($this->value['border']['collapse']) ? $this->value['border']['collapse'] : false; if (!in_array($balise, array('tr', 'td', 'th', 'thead', 'tbody', 'tfoot'))) $collapse = false; $this->value['position'] = null; $this->value['x'] = null; $this->value['y'] = null; $this->value['width'] = 0; $this->value['height'] = 0; $this->value['top'] = null; $this->value['right'] = null; $this->value['bottom'] = null; $this->value['left'] = null; $this->value['float'] = null; $this->value['display'] = null; $this->value['rotate'] = null; $this->value['overflow'] = 'visible'; $this->value['background'] = array('color' => null, 'image' => null, 'position' => null, 'repeat' => null); $this->value['border'] = array( 't' => $this->readBorder('none'), 'r' => $this->readBorder('none'), 'b' => $this->readBorder('none'), 'l' => $this->readBorder('none'), 'radius' => array( 'tl' => array(0, 0), 'tr' => array(0, 0), 'br' => array(0, 0), 'bl' => array(0, 0) ), 'collapse' => $collapse, ); if (!in_array($balise, array('h1', 'h2', 'h3', 'h4', 'h5', 'h6'))) $this->value['margin'] = array('t'=>0,'r'=>0,'b'=>0,'l'=>0); if ($balise=='p') { $this->value['margin']['t'] = null; $this->value['margin']['b'] = null; } $this->value['margin-auto'] = false; if (in_array($balise, array('div'))) $this->value['vertical-align'] = 'top'; if (in_array($balise, array('ul', 'li'))) { $this->value['list-style-type'] = ''; $this->value['list-style-image'] = ''; } if (!in_array($balise, array('tr', 'td'))) { $this->value['padding'] = array( 't' => 0, 'r' => 0, 'b' => 0, 'l' => 0 ); } else { $this->value['padding'] = array( 't' => $this->ConvertToMM('1px'), 'r' => $this->ConvertToMM('1px'), 'b' => $this->ConvertToMM('1px'), 'l' => $this->ConvertToMM('1px') ); } if ($balise=='hr') { $this->value['border'] = array( 't' => $this->readBorder('solid 1px #000000'), 'r' => $this->readBorder('solid 1px #000000'), 'b' => $this->readBorder('solid 1px #000000'), 'l' => $this->readBorder('solid 1px #000000'), 'radius' => array( 'tl' => array(0, 0), 'tr' => array(0, 0), 'br' => array(0, 0), 'bl' => array(0, 0) ), 'collapse' => false, ); $this->ConvertBackground('#FFFFFF', $this->value['background']); } $this->value['xc'] = null; $this->value['yc'] = null; } /** * Initialisation de la font PDF * * @return null */ function FontSet() { $family = strtolower($this->value['font-family']); $b = ($this->value['font-bold'] ? 'B' : ''); $i = ($this->value['font-italic'] ? 'I' : ''); $u = ($this->value['font-underline'] ? 'U' : ''); if ($this->defaultFont) { $style = $b.$i; if($family=='arial') $family='helvetica'; elseif($family=='symbol' || $family=='zapfdingbats') $style=''; $fontkey = $family.$style; if (!isset($this->pdf->fonts[$fontkey])) if (!isset($this->pdf->CoreFonts[$fontkey])) $family = $this->defaultFont; } // taille en mm, à ramener en pt $size = $this->value['font-size']; $size = 72 * $size / 25.4; $this->pdf->setOverline($this->value['font-overline']); $this->pdf->setLinethrough($this->value['font-linethrough']); // application de la fonte $this->pdf->SetFont($family, $b.$i.$u, $this->value['mini-size']*$size); $this->pdf->setMyTextColor($this->value['color']); if ($this->value['background']['color']) $this->pdf->setMyFillColor($this->value['background']['color']); else $this->pdf->setMyFillColor(1.); } /** * Monter d'un niveau dans l'historisation * * @return null */ function save() { $this->table[count($this->table)] = $this->value; } /** * Descendre d'un niveau dans l'historisation * * @return null */ function load() { if (count($this->table)) { $this->value = $this->table[count($this->table)-1]; unset($this->table[count($this->table)-1]); } } function restorePosition(&$current_x, &$current_y) { if ($this->value['y']==$current_y) $current_y = $this->value['yc']; } function setPosition(&$current_x, &$current_y) { $this->value['xc'] = $current_x; $this->value['yc'] = $current_y; if ($this->value['position']=='relative' || $this->value['position']=='absolute') { if ($this->value['right']!==null) { $x = $this->getLastWidth(true) - $this->value['right'] - $this->value['width']; if ($this->value['margin']['r']) $x-= $this->value['margin']['r']; } else { $x = $this->value['left']; if ($this->value['margin']['l']) $x+= $this->value['margin']['l']; } if ($this->value['bottom']!==null) { $y = $this->getLastHeight(true) - $this->value['bottom'] - $this->value['height']; if ($this->value['margin']['b']) $y-= $this->value['margin']['b']; } else { $y = $this->value['top']; if ($this->value['margin']['t']) $y+= $this->value['margin']['t']; } if ($this->value['position']=='relative') { $this->value['x'] = $current_x + $x; $this->value['y'] = $current_y + $y; } else { $this->value['x'] = $this->getLastAbsoluteX()+$x; $this->value['y'] = $this->getLastAbsoluteY()+$y; } } else { $this->value['x'] = $current_x; $this->value['y'] = $current_y; if ($this->value['margin']['l']) $this->value['x']+= $this->value['margin']['l']; if ($this->value['margin']['t']) $this->value['y']+= $this->value['margin']['t']; } $current_x = $this->value['x']; $current_y = $this->value['y']; } /** * Analyse un tableau de style provenant du parseurHTML * * @param string nom de la balise * @param array tableau de style * @return null */ function getSvgStyle($balise, &$param) { // preparation $balise = strtolower($balise); $id = isset($param['id']) ? strtolower(trim($param['id'])) : null; if (!$id) $id = null; $name = isset($param['name']) ? strtolower(trim($param['name'])) : null; if (!$name) $name = null; // lecture de la propriete classe $class = array(); $tmp = isset($param['class']) ? preg_replace('/[\s]+/', ' ', strtolower($param['class'])) : ''; $tmp = explode(' ', $tmp); foreach($tmp as $k => $v) { $v = trim($v); if ($v) $class[] = $v; } // identification de la balise et des styles direct qui pourraient lui être appliqués $this->value['id_balise'] = $balise; $this->value['id_name'] = $name; $this->value['id_id'] = $id; $this->value['id_class'] = $class; $this->value['id_lst'] = array(); $this->value['id_lst'][] = '*'; $this->value['id_lst'][] = $balise; if (count($class)) { foreach($class as $v) { $this->value['id_lst'][] = '*.'.$v; $this->value['id_lst'][] = '.'.$v; $this->value['id_lst'][] = $balise.'.'.$v; } } if ($id) { $this->value['id_lst'][] = '*#'.$id; $this->value['id_lst'][] = '#'.$id; $this->value['id_lst'][] = $balise.'#'.$id; } // style CSS $styles = $this->getFromCSS(); // on ajoute le style propre à la balise $styles = array_merge($styles, $param['style']); $styles['stroke'] = isset($styles['stroke']) ? $this->ConvertToColor($styles['stroke'], $res) : null; $styles['fill'] = isset($styles['fill']) ? $this->ConvertToColor($styles['fill'], $res) : null; $styles['stroke-width'] = isset($styles['stroke-width']) ? $this->ConvertToMM($styles['stroke-width']) : null; $styles['fill-opacity'] = isset($styles['fill-opacity']) ? 1.*$styles['fill-opacity'] : null; if (!$styles['stroke-width']) $styles['stroke'] = null; if (!$styles['stroke']) $styles['stroke-width'] = null; return $styles; } /** * Analyse un tableau de style provenant du parseurHTML * * @param string nom de la balise * @param array tableau de style * @param array tableau initialisant des styles * @return null */ function analyse($balise, &$param, $heritage = null) { // preparation $balise = strtolower($balise); $id = isset($param['id']) ? strtolower(trim($param['id'])) : null; if (!$id) $id = null; $name = isset($param['name']) ? strtolower(trim($param['name'])) : null; if (!$name) $name = null; // lecture de la propriete classe $class = array(); $tmp = isset($param['class']) ? preg_replace('/[\s]+/', ' ', strtolower($param['class'])) : ''; $tmp = explode(' ', $tmp); foreach($tmp as $k => $v) { $v = trim($v); if ($v) $class[] = $v; } // identification de la balise et des styles direct qui pourraient lui être appliqués $this->value['id_balise'] = $balise; $this->value['id_name'] = $name; $this->value['id_id'] = $id; $this->value['id_class'] = $class; $this->value['id_lst'] = array(); $this->value['id_lst'][] = '*'; $this->value['id_lst'][] = $balise; if (count($class)) { foreach($class as $v) { $this->value['id_lst'][] = '*.'.$v; $this->value['id_lst'][] = '.'.$v; $this->value['id_lst'][] = $balise.'.'.$v; } } if ($id) { $this->value['id_lst'][] = '*#'.$id; $this->value['id_lst'][] = '#'.$id; $this->value['id_lst'][] = $balise.'#'.$id; } // style CSS $styles = $this->getFromCSS(); // on ajoute le style propre à la balise $styles = array_merge($styles, $param['style']); if (isset($param['allwidth']) && !isset($styles['width'])) $styles['width'] = '100%'; // mise à zero des styles non hérités $this->resetStyle($balise); if ($heritage) { foreach($heritage as $he_nom => $he_val) { if (is_array($he_val)) { foreach($he_val as $he2_nom => $he2_val) $this->value[$he_nom][$he2_nom] = $he2_val; } else $this->value[$he_nom] = $he_val; } } // interpreration des nouvelles valeurs $correct_width = false; $no_width = true; foreach($styles as $nom => $val) { switch($nom) { case 'font-family': $val = explode(',', $val); $val = trim($val[0]); if ($val) $this->value['font-family'] = $val; break; case 'font-weight': $this->value['font-bold'] = ($val=='bold'); break; case 'font-style': $this->value['font-italic'] = ($val=='italic'); break; case 'text-decoration': $val = explode(' ', $val); $this->value['font-underline'] = (in_array('underline', $val)); $this->value['font-overline'] = (in_array('overline', $val)); $this->value['font-linethrough'] = (in_array('line-through', $val)); break; case 'text-indent': $this->value['text-indent'] = $this->ConvertToMM($val); break; case 'text-transform': if (!in_array($val, array('none', 'capitalize', 'uppercase', 'lowercase'))) $val = 'none'; $this->value['text-transform'] = $val; break; case 'font-size': $val = $this->ConvertToMM($val, $this->value['font-size']); if ($val) $this->value['font-size'] = $val; break; case 'color': $res = null; $this->value['color'] = $this->ConvertToColor($val, $res); if ($balise=='hr') { $this->value['border']['l']['color'] = $this->value['color']; $this->value['border']['t']['color'] = $this->value['color']; $this->value['border']['r']['color'] = $this->value['color']; $this->value['border']['b']['color'] = $this->value['color']; } break; case 'text-align': $val = strtolower($val); if (!in_array($val, array('left', 'right', 'center', 'justify', 'li_right'))) $val = 'left'; $this->value['text-align'] = $val; break; case 'vertical-align': $this->value['vertical-align'] = $val; break; case 'width': $this->value['width'] = $this->ConvertToMM($val, $this->getLastWidth()); if ($this->value['width'] && substr($val, -1)=='%') $correct_width=true; $no_width = false; break; case 'height': $this->value['height'] = $this->ConvertToMM($val, $this->getLastHeight()); break; case 'line-height': if (preg_match('/^[0-9\.]+$/isU', $val)) $val = floor($val*100).'%'; $this->value['line-height'] = $val; break; case 'rotate': if (!in_array($val, array(0, -90, 90, 180, 270, -180, -270))) $val = null; if ($val<0) $val+= 360; $this->value['rotate'] = $val; break; case 'overflow': if (!in_array($val, array('visible', 'hidden'))) $val = 'visible'; $this->value['overflow'] = $val; break; case 'padding': $val = explode(' ', $val); foreach($val as $k => $v) { $v = trim($v); if ($v!='') $val[$k] = $v; else unset($val[$k]); } $val = array_values($val); if (count($val)!=4) { $val = $this->ConvertToMM($val[0], 0); $this->value['padding']['t'] = $val; $this->value['padding']['r'] = $val; $this->value['padding']['b'] = $val; $this->value['padding']['l'] = $val; } else { $this->value['padding']['t'] = $this->ConvertToMM($val[0], 0); $this->value['padding']['r'] = $this->ConvertToMM($val[1], 0); $this->value['padding']['b'] = $this->ConvertToMM($val[2], 0); $this->value['padding']['l'] = $this->ConvertToMM($val[3], 0); } break; case 'padding-top': $this->value['padding']['t'] = $this->ConvertToMM($val, 0); break; case 'padding-right': $this->value['padding']['r'] = $this->ConvertToMM($val, 0); break; case 'padding-bottom': $this->value['padding']['b'] = $this->ConvertToMM($val, 0); break; case 'padding-left': $this->value['padding']['l'] = $this->ConvertToMM($val, 0); break; case 'margin': if ($val=='auto') { $this->value['margin-auto'] = true; break; } $val = explode(' ', $val); foreach($val as $k => $v) { $v = trim($v); if ($v!='') $val[$k] = $v; else unset($val[$k]); } $val = array_values($val); if (count($val)!=4) { $val = $this->ConvertToMM($val[0], 0); $this->value['margin']['t'] = $val; $this->value['margin']['r'] = $val; $this->value['margin']['b'] = $val; $this->value['margin']['l'] = $val; } else { $this->value['margin']['t'] = $this->ConvertToMM($val[0], 0); $this->value['margin']['r'] = $this->ConvertToMM($val[1], 0); $this->value['margin']['b'] = $this->ConvertToMM($val[2], 0); $this->value['margin']['l'] = $this->ConvertToMM($val[3], 0); } break; case 'margin-top': $this->value['margin']['t'] = $this->ConvertToMM($val, 0); break; case 'margin-right': $this->value['margin']['r'] = $this->ConvertToMM($val, 0); break; case 'margin-bottom': $this->value['margin']['b'] = $this->ConvertToMM($val, 0); break; case 'margin-left': $this->value['margin']['l'] = $this->ConvertToMM($val, 0); break; case 'border': $val = $this->readBorder($val); $this->value['border']['t'] = $val; $this->value['border']['r'] = $val; $this->value['border']['b'] = $val; $this->value['border']['l'] = $val; break; case 'border-style': $val = explode(' ', $val); foreach($val as $val_k => $val_v) if (!in_array($val_v, array('solid', 'dotted', 'dashed'))) $val[$val_k] = null; $this->duplicateBorder($val); if ($val[0]) $this->value['border']['t']['type'] = $val[0]; if ($val[1]) $this->value['border']['r']['type'] = $val[1]; if ($val[2]) $this->value['border']['b']['type'] = $val[2]; if ($val[3]) $this->value['border']['l']['type'] = $val[3]; break; case 'border-top-style': if (in_array($val, array('solid', 'dotted', 'dashed'))) $this->value['border']['t']['type'] = $val; break; case 'border-right-style': if (in_array($val, array('solid', 'dotted', 'dashed'))) $this->value['border']['r']['type'] = $val; break; case 'border-bottom-style': if (in_array($val, array('solid', 'dotted', 'dashed'))) $this->value['border']['b']['type'] = $val; break; case 'border-left-style': if (in_array($val, array('solid', 'dotted', 'dashed'))) $this->value['border']['l']['type'] = $val; break; case 'border-color': $res = false; $val = preg_replace('/,[\s]+/', ',', $val); $val = explode(' ', $val); foreach($val as $val_k => $val_v) { $val[$val_k] = $this->ConvertToColor($val_v, $res); if (!$res) $val[$val_k] = null; } $this->duplicateBorder($val); if (is_array($val[0])) $this->value['border']['t']['color'] = $val[0]; if (is_array($val[1])) $this->value['border']['r']['color'] = $val[1]; if (is_array($val[2])) $this->value['border']['b']['color'] = $val[2]; if (is_array($val[3])) $this->value['border']['l']['color'] = $val[3]; break; case 'border-top-color': $res = false; $val = $this->ConvertToColor($val, $res); if ($res) $this->value['border']['t']['color'] = $val; break; case 'border-right-color': $res = false; $val = $this->ConvertToColor($val, $res); if ($res) $this->value['border']['r']['color'] = $val; break; case 'border-bottom-color': $res = false; $val = $this->ConvertToColor($val, $res); if ($res) $this->value['border']['b']['color'] = $val; break; case 'border-left-color': $res = false; $val = $this->ConvertToColor($val, $res); if ($res) $this->value['border']['l']['color'] = $val; break; case 'border-width': $val = explode(' ', $val); foreach($val as $val_k => $val_v) { $val[$val_k] = $this->ConvertToMM($val_v, 0); } $this->duplicateBorder($val); if ($val[0]) $this->value['border']['t']['width'] = $val[0]; if ($val[1]) $this->value['border']['r']['width'] = $val[1]; if ($val[2]) $this->value['border']['b']['width'] = $val[2]; if ($val[3]) $this->value['border']['l']['width'] = $val[3]; break; case 'border-top-width': $val = $this->ConvertToMM($val, 0);; if ($val) $this->value['border']['t']['width'] = $val; break; case 'border-right-width': $val = $this->ConvertToMM($val, 0);; if ($val) $this->value['border']['r']['width'] = $val; break; case 'border-bottom-width': $val = $this->ConvertToMM($val, 0);; if ($val) $this->value['border']['b']['width'] = $val; break; case 'border-left-width': $val = $this->ConvertToMM($val, 0);; if ($val) $this->value['border']['l']['width'] = $val; break; case 'border-collapse': if ($balise=='table') $this->value['border']['collapse'] = ($val=='collapse'); break; case 'border-radius': $val = explode('/', $val); if (count($val)>2) break; $val_h = $this->ConvertToRadius(trim($val[0])); if (count($val_h)<1 || count($val_h)>4) break; if (!isset($val_h[1])) $val_h[1] = $val_h[0]; if (!isset($val_h[2])) $val_h = array($val_h[0], $val_h[0], $val_h[1], $val_h[1]); if (!isset($val_h[3])) $val_h[3] = $val_h[1]; if (isset($val[1])) { $val_v = $this->ConvertToRadius(trim($val[1])); if (count($val_v)<1 || count($val_v)>4) break; if (!isset($val_v[1])) $val_v[1] = $val_v[0]; if (!isset($val_v[2])) $val_v = array($val_v[0], $val_v[0], $val_v[1], $val_v[1]); if (!isset($val_v[3])) $val_v[3] = $val_v[1]; } else $val_v = $val_h; $this->value['border']['radius'] = array( 'tl' => array($val_h[0], $val_v[0]), 'tr' => array($val_h[1], $val_v[1]), 'br' => array($val_h[2], $val_v[2]), 'bl' => array($val_h[3], $val_v[3]) ); break; case 'border-top-left-radius': $val = $this->ConvertToRadius($val); if (count($val)<1 || count($val)>2) break; $this->value['border']['radius']['tl'] = array($val[0], isset($val[1]) ? $val[1] : $val[0]); break; case 'border-top-right-radius': $val = $this->ConvertToRadius($val); if (count($val)<1 || count($val)>2) break; $this->value['border']['radius']['tr'] = array($val[0], isset($val[1]) ? $val[1] : $val[0]); break; case 'border-bottom-right-radius': $val = $this->ConvertToRadius($val); if (count($val)<1 || count($val)>2) break; $this->value['border']['radius']['br'] = array($val[0], isset($val[1]) ? $val[1] : $val[0]); break; case 'border-bottom-left-radius': $val = $this->ConvertToRadius($val); if (count($val)<1 || count($val)>2) break; $this->value['border']['radius']['bl'] = array($val[0], isset($val[1]) ? $val[1] : $val[0]); break; case 'border-top': $this->value['border']['t'] = $this->readBorder($val); break; case 'border-right': $this->value['border']['r'] = $this->readBorder($val); break; case 'border-bottom': $this->value['border']['b'] = $this->readBorder($val); break; case 'border-left': $this->value['border']['l'] = $this->readBorder($val); break; case 'background-color': $this->value['background']['color'] = $this->ConvertBackgroundColor($val); break; case 'background-image': $this->value['background']['image'] = $this->ConvertBackgroundImage($val); break; case 'background-position': $res = null; $this->value['background']['position'] = $this->ConvertBackgroundPosition($val, $res); break; case 'background-repeat': $this->value['background']['repeat'] = $this->ConvertBackgroundRepeat($val); break; case 'background': $this->ConvertBackground($val, $this->value['background']); break; case 'position': if ($val=='absolute') $this->value['position'] = 'absolute'; else if ($val=='relative') $this->value['position'] = 'relative'; else $this->value['position'] = null; break; case 'float': if ($val=='left') $this->value['float'] = 'left'; else if ($val=='right') $this->value['float'] = 'right'; else $this->value['float'] = null; break; case 'display': if ($val=='inline') $this->value['display'] = 'inline'; else if ($val=='block') $this->value['display'] = 'block'; else if ($val=='none') $this->value['display'] = 'none'; else $this->value['display'] = null; break; case 'top': case 'bottom': case 'left': case 'right': $this->value[$nom] = $val; break; case 'list-style': case 'list-style-type': case 'list-style-image': if ($nom=='list-style') $nom = 'list-style-type'; $this->value[$nom] = $val; break; default: break; } } if ($this->value['margin']['t']===null) $this->value['margin']['t'] = $this->value['font-size']; if ($this->value['margin']['b']===null) $this->value['margin']['b'] = $this->value['font-size']; if ($this->onlyLeft) $this->value['text-align'] = 'left'; // correction de la largeur pour correspondre au modèle de boite quick if ($no_width && in_array($balise, array('div')) && $this->value['position']!='absolute') { $this->value['width'] = $this->getLastWidth(); $this->value['width']-= $this->value['margin']['l'] + $this->value['margin']['r']; } else { if ($correct_width) { if (!in_array($balise, array('table', 'div', 'hr'))) { $this->value['width']-= $this->value['padding']['l'] + $this->value['padding']['r']; $this->value['width']-= $this->value['border']['l']['width'] + $this->value['border']['r']['width']; } if (in_array($balise, array('th', 'td'))) { $this->value['width']-= $this->ConvertToMM(isset($param['cellspacing']) ? $param['cellspacing'] : '2px'); } if ($this->value['width']<0) $this->value['width']=0; } else { if ($this->value['width']) { if ($this->value['border']['l']['width']) $this->value['width'] += $this->value['border']['l']['width']; if ($this->value['border']['r']['width']) $this->value['width'] += $this->value['border']['r']['width']; if ($this->value['padding']['l']) $this->value['width'] += $this->value['padding']['l']; if ($this->value['padding']['r']) $this->value['width'] += $this->value['padding']['r']; } } } if ($this->value['height']) { if ($this->value['border']['b']['width']) { $this->value['height'] += $this->value['border']['b']['width']; } if ($this->value['border']['t']['width']) { $this->value['height'] += $this->value['border']['t']['width']; } if ($this->value['padding']['b']) $this->value['height'] += $this->value['padding']['b']; if ($this->value['padding']['t']) $this->value['height'] += $this->value['padding']['t']; } if ($this->value['top']!=null) $this->value['top'] = $this->ConvertToMM($this->value['top'], $this->getLastHeight(true)); if ($this->value['bottom']!=null) $this->value['bottom'] = $this->ConvertToMM($this->value['bottom'], $this->getLastHeight(true)); if ($this->value['left']!=null) $this->value['left'] = $this->ConvertToMM($this->value['left'], $this->getLastWidth(true)); if ($this->value['right']!=null) $this->value['right'] = $this->ConvertToMM($this->value['right'], $this->getLastWidth(true)); if ($this->value['top'] && $this->value['bottom'] && $this->value['height']) $this->value['bottom'] = null; if ($this->value['left'] && $this->value['right'] && $this->value['width']) $this->value['right'] = null; } /** * Récupération de la hauteur de ligne courante * * @return float hauteur en mm */ function getLineHeight() { $val = $this->value['line-height']; if ($val=='normal') $val = '108%'; return $this->ConvertToMM($val, $this->value['font-size']); } /** * Récupération de la largeur de l'objet parent * * @return float largeur */ function getLastWidth($mode = false) { for($k=count($this->table); $k>0; $k--) { if ($this->table[$k-1]['width']) { $w = $this->table[$k-1]['width']; if ($mode) { $w+= $this->table[$k-1]['border']['l']['width'] + $this->table[$k-1]['padding']['l']+0.02; $w+= $this->table[$k-1]['border']['r']['width'] + $this->table[$k-1]['padding']['r']+0.02; } return $w; } } return $this->pdf->w - $this->pdf->lMargin - $this->pdf->rMargin; } /** * Récupération de la hauteur de l'objet parent * * @return float hauteur */ function getLastHeight($mode = false) { for($k=count($this->table); $k>0; $k--) { if ($this->table[$k-1]['height']) { $h = $this->table[$k-1]['height']; if ($mode) { $h+= $this->table[$k-1]['border']['t']['width'] + $this->table[$k-1]['padding']['t']+0.02; $h+= $this->table[$k-1]['border']['b']['width'] + $this->table[$k-1]['padding']['b']+0.02; } return $h; } } return $this->pdf->h - $this->pdf->tMargin - $this->pdf->bMargin; } function getFloat() { if ($this->value['float']=='left') return 'left'; if ($this->value['float']=='right') return 'right'; return null; } function getLastValue($key) { $nb = count($this->table); if ($nb>0) return $this->table[$nb-1][$key]; return null; } function getLastAbsoluteX() { for($k=count($this->table); $k>0; $k--) { if ($this->table[$k-1]['x'] && $this->table[$k-1]['position']) return $this->table[$k-1]['x']; } return $this->pdf->lMargin; } function getLastAbsoluteY() { for($k=count($this->table); $k>0; $k--) { if ($this->table[$k-1]['y'] && $this->table[$k-1]['position']) return $this->table[$k-1]['y']; } return $this->pdf->tMargin; } /** * Récupération des propriétés CSS de la balise en cours * * @return array() tableau des propriétés CSS */ function getFromCSS() { $styles = array(); // style à appliquer $getit = array(); // styles à récuperer // identification des styles direct, et ceux des parents $lst = array(); $lst[] = $this->value['id_lst']; for($i=count($this->table)-1; $i>=0; $i--) $lst[] = $this->table[$i]['id_lst']; // identification des styles à récuperer foreach($this->css_keys as $key => $num) if ($this->getReccursiveStyle($key, $lst)) $getit[$key] = $num; // si des styles sont à recuperer if (count($getit)) { // on les récupère, mais dans l'odre de définition, afin de garder les priorités asort($getit); foreach($getit as $key => $val) $styles = array_merge($styles, $this->css[$key]); } return $styles; } /** * Identification des styles à récuperer, en fonction de la balise et de ses parents * * @param string clef CSS à analyser * @param array() tableau des styles direct, et ceux des parents * @param string prochaine etape * @return boolean clef autorisée ou non */ function getReccursiveStyle($key, $lst, $next = null) { // si propchaine etape, on construit les valeurs if ($next!==null) { if ($next) $key = trim(substr($key, 0, -strlen($next))); // on elève cette etape unset($lst[0]); if (!count($lst)) return false; // pas d'etape possible $lst = array_values($lst); } // pour chaque style direct possible de l'etape en cours foreach($lst[0] as $nom) { if ($key==$nom) return true; // si la clef conrrespond => ok if (substr($key, -strlen(' '.$nom))==' '.$nom && $this->getReccursiveStyle($key, $lst, $nom)) return true; // si la clef est la fin, on analyse ce qui précède } // si on est pas à la premiere etape, on doit analyse toutes les sous etapes if ($next!==null && $this->getReccursiveStyle($key, $lst, '')) return true; // aucun style trouvé return false; } /** * Analyse d'une propriété Border * * @param string propriété border * @return array() propriété décodée */ function readBorder($val) { $none = array('type' => 'none', 'width' => 0, 'color' => array(0, 0, 0)); // valeurs par défault $type = 'solid'; $width = $this->ConvertToMM('1pt'); $color = array(0, 0, 0); // nettoyage des valeurs $val = explode(' ', $val); foreach($val as $k => $v) { $v = trim($v); if ($v) $val[$k] = $v; else unset($val[$k]); } $val = array_values($val); // identification des valeurs $res = null; foreach($val as $key) { if ($key=='none' || $key=='hidden') return $none; if ($this->ConvertToMM($key)!==null) $width = $this->ConvertToMM($key); else if (in_array($key, array('solid', 'dotted', 'dashed'))) $type = $key; else { $tmp = $this->ConvertToColor($key, $res); if ($res) $color = $tmp; } } if (!$width) return $none; return array('type' => $type, 'width' => $width, 'color' => $color); } function duplicateBorder(&$val) { if (count($val)==1) { $val[1] = $val[0]; $val[2] = $val[0]; $val[3] = $val[0]; } else if (count($val)==2) { $val[2] = $val[0]; $val[3] = $val[1]; } else if (count($val)==3) { $val[3] = $val[1]; } } function ConvertBackground($stl, &$res) { // Image $text = '/url\(([^)]*)\)/isU'; if (preg_match($text, $stl, $match)) { $res['image'] = $this->ConvertBackgroundImage($match[0]); $stl = preg_replace($text, '', $stl); $stl = preg_replace('/[\s]+/', ' ', $stl); } // protection des espaces $stl = preg_replace('/,[\s]+/', ',', $stl); $lst = explode(' ', $stl); $pos = ''; foreach($lst as $val) { $ok = false; $color = $this->ConvertToColor($val, $ok); if ($ok) { $res['color'] = $color; } else if ($val=='transparent') { $res['color'] = null; } else { $repeat = $this->ConvertBackgroundRepeat($val); if ($repeat) { $res['repeat'] = $repeat; } else { $pos.= ($pos ? ' ' : '').$val; } } } if ($pos) { $pos = $this->ConvertBackgroundPosition($pos, $ok); if ($ok) $res['position'] = $pos; } } function ConvertBackgroundColor($val) { $res = null; if ($val=='transparent') return null; else return $this->ConvertToColor($val, $res); } function ConvertBackgroundImage($val) { if ($val=='none') return null; else if (preg_match('/^url\(([^)]*)\)$/isU', $val, $match)) return $match[1]; else return null; } function ConvertBackgroundPosition($val, &$res) { $val = explode(' ', $val); if (count($val)<2) { if (!$val[0]) return null; $val[1] = 'center'; } if (count($val)>2) return null; $x = 0; $y = 0; $res = true; if ($val[0]=='left') $x = '0%'; else if ($val[0]=='center') $x = '50%'; else if ($val[0]=='right') $x = '100%'; else if ($val[0]=='top') $y = '0%'; else if ($val[0]=='bottom') $y = '100%'; else if (preg_match('/^[-]?[0-9\.]+%$/isU', $val[0])) $x = $val[0]; else if ($this->ConvertToMM($val[0])) $x = $this->ConvertToMM($val[0]); else $res = false; if ($val[1]=='left') $x = '0%'; else if ($val[1]=='right') $x = '100%'; else if ($val[1]=='top') $y = '0%'; else if ($val[1]=='center') $y = '50%'; else if ($val[1]=='bottom') $y = '100%'; else if (preg_match('/^[-]?[0-9\.]+%$/isU', $val[1])) $y = $val[1]; else if ($this->ConvertToMM($val[1])) $y = $this->ConvertToMM($val[1]); else $res = false; $val[0] = $x; $val[1] = $y; return $val; } function ConvertBackgroundRepeat($val) { switch($val) { case 'repeat': return array(true, true); case 'repeat-x': return array(true, false); case 'repeat-y': return array(false, true); case 'no-repeat': return array(false, false); } return null; } /** * Convertir une longueur en mm * * @param string longueur, avec unité, à convertir * @param float longueur du parent * @return float longueur exprimée en mm */ function ConvertToMM($val, $old=0.) { $val = trim($val); if (preg_match('/^[0-9\.\-]+$/isU', $val)) $val.= 'px'; if (preg_match('/^[0-9\.\-]+px$/isU', $val)) $val = 25.4/96. * str_replace('px', '', $val); else if (preg_match('/^[0-9\.\-]+pt$/isU', $val)) $val = 25.4/72. * str_replace('pt', '', $val); else if (preg_match('/^[0-9\.\-]+in$/isU', $val)) $val = 25.4 * str_replace('in', '', $val); else if (preg_match('/^[0-9\.\-]+mm$/isU', $val)) $val = 1.*str_replace('mm', '', $val); else if (preg_match('/^[0-9\.\-]+%$/isU', $val)) $val = 1.*$old*str_replace('%', '', $val)/100.; else $val = null; return $val; } function ConvertToRadius($val) { $val = explode(' ', $val); foreach($val as $k => $v) { $v = trim($v); if ($v) { $v = $this->ConvertToMM($v, 0); if ($v!==null) $val[$k] = $v; else unset($val[$k]); } else unset($val[$k]); } return array_values($val); } /** * Décomposition d'un code couleur HTML * * @param string couleur au format CSS * @return array(r, v, b) couleur exprimé par ses comporantes R, V, B, de 0 à 255. */ function ConvertToColor($val, &$res) { $val = trim($val); $res = true; if (strtolower($val)=='transparent') return array(null, null, null); if (isset($this->htmlColor[strtolower($val)])) $val = $this->htmlColor[strtolower($val)]; if (preg_match('/^#[0-9A-Fa-f]{6}$/isU', $val)) { $r = floatVal(hexdec(substr($val, 1, 2)))/255.; $v = floatVal(hexdec(substr($val, 3, 2)))/255.; $b = floatVal(hexdec(substr($val, 5, 2)))/255.; $col = array($r, $v, $b); } elseif (preg_match('/^#[0-9A-F]{3}$/isU', $val)) { $r = floatVal(hexdec(substr($val, 1, 1).substr($val, 1, 1)))/255.; $v = floatVal(hexdec(substr($val, 2, 1).substr($val, 2, 1)))/255.; $b = floatVal(hexdec(substr($val, 3, 1).substr($val, 3, 1)))/255.; $col = array($r, $v, $b); } elseif (preg_match('/rgb\([\s]*([0-9%\.]+)[\s]*,[\s]*([0-9%\.]+)[\s]*,[\s]*([0-9%\.]+)[\s]*\)/isU', $val, $match)) { $r = $this->ConvertSubColor($match[1]); $v = $this->ConvertSubColor($match[2]); $b = $this->ConvertSubColor($match[3]); $col = array($r, $v, $b); } elseif (preg_match('/cmyk\([\s]*([0-9%\.]+)[\s]*,[\s]*([0-9%\.]+)[\s]*,[\s]*([0-9%\.]+)[\s]*,[\s]*([0-9%\.]+)[\s]*\)/isU', $val, $match)) { $c = $this->ConvertSubColor($match[1]); $m = $this->ConvertSubColor($match[2]); $y = $this->ConvertSubColor($match[3]); $k = $this->ConvertSubColor($match[4]); $col = array($c, $m, $y, $k); } else { $col = array(0., 0., 0.); $res = false; } return $col; } function ConvertSubColor($c) { if (substr($c, -1)=='%') $c = floatVal(substr($c, 0, -1))/100.; else { $c = floatVal($c); if ($c>1) $c = $c/255.; } return $c; } /** * Analyser une feuille de style * * @param string code CSS * @return null */ function analyseStyle(&$code) { // on remplace tous les espaces, tab, \r, \n, par des espaces uniques $code = preg_replace('/[\s]+/', ' ', $code); // on enlève les commentaires $code = preg_replace('/\/\*.*?\*\//s', '', $code); // on analyse chaque style preg_match_all('/([^{}]+){([^}]*)}/isU', $code, $match); for($k=0; $k on remplie le tableau correspondant $styles = trim($match[2][$k]); $styles = explode(';', $styles); $stl = array(); foreach($styles as $style) { $tmp = explode(':', $style); if (count($tmp)>1) { $cod = $tmp[0]; unset($tmp[0]); $tmp = implode(':', $tmp); $stl[trim(strtolower($cod))] = trim($tmp); } } // décomposition des noms par les , $noms = explode(',', $noms); foreach($noms as $nom) { $nom = trim($nom); // Si il a une fonction spécifique, comme :hover => on zap if (strpos($nom, ':')!==false) continue; if (!isset($this->css[$nom])) $this->css[$nom] = $stl; else $this->css[$nom] = array_merge($this->css[$nom], $stl); } } $this->css_keys = array_flip(array_keys($this->css)); } /** * Extraction des feuille de style du code HTML * * @param string code HTML * @return null */ function readStyle(&$html) { $style = ' '; // extraction des balises link, et suppression de celles-ci dans le code HTML preg_match_all('/]*)>/isU', $html, $match); $html = preg_replace('/]*>/isU', '', $html); $html = preg_replace('/<\/link[^>]*>/isU', '', $html); // analyse de chaque balise foreach($match[1] as $code) { $tmp = array(); // lecture des paramétres du type nom=valeur $prop = '([a-zA-Z0-9_]+)=([^"\'\s>]+)'; preg_match_all('/'.$prop.'/is', $code, $match); for($k=0; $k on garde if (isset($tmp['type']) && strtolower($tmp['type'])=='text/css' && isset($tmp['href'])) { $content = @file_get_contents($tmp['href']); $url = $tmp['href']; if (strpos($url, 'http://')!==false) { $url = str_replace('http://', '', $url); $url = explode('/', $url); $url_main = 'http://'.$url[0].'/'; $url_self = $url; unset($url_self[count($url_self)-1]); $url_self = 'http://'.implode('/', $url_self).'/'; $content = preg_replace('/url\(([^\\\\][^)]*)\)/isU', 'url('.$url_self.'$1)', $content); $content = preg_replace('/url\((\\\\[^)]*)\)/isU', 'url('.$url_main.'$1)', $content); } else { // @todo // $content = preg_replace('/url\(([^)]*)\)/isU', 'url('.dirname($url).'/$1)', $content); } $style.= $content."\n"; } } // extraction des balises style, et suppression de celles-ci dans le code HTML preg_match_all('/]*>(.*)<\/style[^>]*>/isU', $html, $match); $html = preg_replace('/]*>(.*)<\/style[^>]*>/isU', '', $html); // analyse de chaque balise foreach($match[1] as $code) { $code = str_replace('', '', $code); $style.= $code."\n"; } $this->analyseStyle($style); } } }