PDF * Distribué sous la licence LGPL. * * @author Laurent MINGUET * @version 3.31 */ class styleHTML { 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 var $value = array(); // valeurs actuelles var $css = array(); // tableau des CSS var $css_keys = array(); // tableau des clefs CSS, pour l'ordre d'execution var $table = array(); // tableau d'empilement pour historisation des niveaux /** * Constructeur * * @param &pdf référence à l'objet HTML2PDF parent * @return null */ function styleHTML(&$pdf) { $this->init(); // initialisation $this->setPdfParent($pdf); } function setPdfParent(&$pdf) { $this->pdf = &$pdf; } function setOnlyLeft() { $this->value['text-align'] = 'left'; $this->onlyLeft = true; } function getOldValues() { return isset($this->table[count($this->table)-1]) ? $this->table[count($this->table)-1] : $this->value; } /** * 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() { global $webcolor; $this->htmlColor = &$webcolor; $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', 'fieldset'))) $this->value['vertical-align'] = 'top'; if (in_array($balise, array('fieldset', 'legend'))) { $radius = $this->ConvertToMM('5px'); $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(), 'collapse' => false, ); $this->value['border']['radius'] = array( 'tl' => array($radius, $radius), 'tr' => array($radius, $radius), 'br' => array($radius, $radius), 'bl' => array($radius, $radius)); } 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' : ''); $d = ($this->value['font-linethrough'] ? 'D' : ''); $o = ($this->value['font-overline'] ? 'O' : ''); if ($this->defaultFont) { $style = $b.$i; if($family=='arial') $family='helvetica'; elseif($family=='symbol' || $family=='zapfdingbats') $style=''; $fontkey = $family.$style; if (!$this->pdf->isLoadedFont($fontkey)) $family = $this->defaultFont; } if($family=='arial') $family='helvetica'; elseif($family=='symbol' || $family=='zapfdingbats') $style=''; // taille en mm, à ramener en pt $size = $this->value['font-size']; $size = 72 * $size / 25.4; $this->pdf->setOverline($o ? true : false); $this->pdf->setLinethrough($d ? true : false); // 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() { if ($this->value['y']==$this->pdf->getY()) $this->pdf->setY($this->value['yc'], false); } function setPosition() { $current_x = $this->pdf->getX(); $current_y = $this->pdf->getY(); $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']; } $this->pdf->setXY($this->value['x'], $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']) ? strtolower(trim($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 (!isset($this->value['svg'])) { $this->value['svg'] = array( 'stroke' => null, 'stroke-width' => $this->ConvertToMM('1pt'), 'fill' => null, 'fill-opacity' => null, ); } 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($styles['stroke'])) $this->value['svg']['stroke'] = $this->ConvertToColor($styles['stroke'], $res); if (isset($styles['stroke-width'])) $this->value['svg']['stroke-width'] = $this->ConvertToMM($styles['stroke-width']); if (isset($styles['fill'])) $this->value['svg']['fill'] = $this->ConvertToColor($styles['fill'], $res); if (isset($styles['fill-opacity'])) $this->value['svg']['fill-opacity'] = 1.*$styles['fill-opacity']; return $this->value['svg']; } /** * 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']) ? strtolower(trim($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); $this->duplicateBorder($val); $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); $this->duplicateBorder($val); $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; } } $return = true; 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', 'fieldset')) && $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', 'fieldset', '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'); $return = false; } 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; return $return; } /** * 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)-1; $k>=0; $k--) { if ($this->table[$k]['width']) { $w = $this->table[$k]['width']; if ($mode) { $w+= $this->table[$k]['border']['l']['width'] + $this->table[$k]['padding']['l'] + 0.02; $w+= $this->table[$k]['border']['r']['width'] + $this->table[$k]['padding']['r'] + 0.02; } return $w; } } return $this->pdf->getW() - $this->pdf->getlMargin() - $this->pdf->getrMargin(); } /** * Récupération de la hauteur de l'objet parent * * @return float hauteur */ function getLastHeight($mode = false) { for($k=count($this->table)-1; $k>=0; $k--) { if ($this->table[$k]['height']) { $h = $this->table[$k]['height']; if ($mode) { $h+= $this->table[$k]['border']['t']['width'] + $this->table[$k]['padding']['t'] + 0.02; $h+= $this->table[$k]['border']['b']['width'] + $this->table[$k]['padding']['b'] + 0.02; } return $h; } } return $this->pdf->getH() - $this->pdf->gettMargin() - $this->pdf->getbMargin(); } 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)-1; $k>=0; $k--) { if ($this->table[$k]['x'] && $this->table[$k]['position']) return $this->table[$k]['x']; } return $this->pdf->getlMargin(); } function getLastAbsoluteY() { for($k=count($this->table)-1; $k>=0; $k--) { if ($this->table[$k]['y'] && $this->table[$k]['position']) return $this->table[$k]['y']; } return $this->pdf->gettMargin(); } /** * 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', 'double'))) $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)]; $r = floatVal(hexdec(substr($val, 0, 2))); $v = floatVal(hexdec(substr($val, 2, 2))); $b = floatVal(hexdec(substr($val, 4, 2))); $col = array($r/255., $v/255., $b/255.); } elseif (preg_match('/^#[0-9A-Fa-f]{6}$/isU', $val)) { $r = floatVal(hexdec(substr($val, 1, 2))); $v = floatVal(hexdec(substr($val, 3, 2))); $b = floatVal(hexdec(substr($val, 5, 2))); $col = array($r/255., $v/255., $b/255.); } elseif (preg_match('/^#[0-9A-F]{3}$/isU', $val)) { $r = floatVal(hexdec(substr($val, 1, 1).substr($val, 1, 1))); $v = floatVal(hexdec(substr($val, 2, 1).substr($val, 2, 1))); $b = floatVal(hexdec(substr($val, 3, 1).substr($val, 3, 1))); $col = array($r/255., $v/255., $b/255.); } 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); } }