PDF convertor * distributed under the LGPL License * * @author Laurent MINGUET * @version 4.03 */ class HTML2PDF_parsingCss { /** * reference to the pdf object * @var TCPDF */ protected $_pdf = null; protected $_htmlColor = array(); // list of the HTML colors protected $_onlyLeft = false; // flag if we are in a sub html => only "text-align:left" is used protected $_defaultFont = null; // default font to use if the asked font does not exist public $value = array(); // current values public $css = array(); // css values public $cssKeys = array(); // css key, for the execution order public $table = array(); // level history /** * Constructor * * @param &HTML2PDF_myPdf reference to the PDF $object * @access public */ public function __construct(&$pdf) { $this->_init(); $this->setPdfParent($pdf); } /** * Set the HTML2PDF parent object * * @param &HTML2PDF reference to the HTML2PDF parent $object * @access public */ public function setPdfParent(&$pdf) { $this->_pdf = &$pdf; } /** * Inform that we want only "test-align:left" because we are in a sub HTML * * @access public */ public function setOnlyLeft() { $this->value['text-align'] = 'left'; $this->_onlyLeft = true; } /** * Get the vales of the parent, if exist * * @return array CSS values * @access public */ public function getOldValues() { return isset($this->table[count($this->table)-1]) ? $this->table[count($this->table)-1] : $this->value; } /** * define the Default Font to use, if the font does not exist, or if no font asked * * @param string default font-family. If null : Arial for no font asked, and error fot ont does not exist * @return string old default font-family * @access public */ public function setDefaultFont($default = null) { $old = $this->_defaultFont; $this->_defaultFont = $default; if ($default) $this->value['font-family'] = $default; return $old; } /** * Init the object * * @access protected */ protected function _init() { // get the Web Colors from TCPDF require(K_PATH_MAIN.'htmlcolors.php'); $this->_htmlColor = $webcolor; // init the Style $this->table = array(); $this->value = array(); $this->initStyle(); // Init the styles without legacy $this->resetStyle(); } /** * Init the CSS Style * * @access public */ public function initStyle() { $this->value['id_tag'] = 'body'; // tag name $this->value['id_name'] = null; // tag - attribute name $this->value['id_id'] = null; // tag - attribute id $this->value['id_class'] = null; // tag - attribute class $this->value['id_lst'] = array('*'); // tag - list of legacy $this->value['mini-size'] = 1.; // specific size report for sup, sub $this->value['mini-decal'] = 0; // specific position report for 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; } /** * Init the CSS Style without legacy * * @param string tag name * @access public */ public function resetStyle($tagName = '') { // prepare somme values $border = $this->readBorder('solid 1px #000000'); $units = array( '1px' => $this->convertToMM('1px'), '5px' => $this->convertToMM('5px'), ); // prepare the Collapse attribute $collapse = isset($this->value['border']['collapse']) ? $this->value['border']['collapse'] : false; if (!in_array($tagName, array('tr', 'td', 'th', 'thead', 'tbody', 'tfoot'))) $collapse = false; // set the global css values $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, ); // specific values for some tags if (!in_array($tagName, array('h1', 'h2', 'h3', 'h4', 'h5', 'h6'))) { $this->value['margin'] = array('t'=>0,'r'=>0,'b'=>0,'l'=>0); } if (in_array($tagName, array('input', 'select', 'textarea'))) { $this->value['border']['t'] = null; $this->value['border']['r'] = null; $this->value['border']['b'] = null; $this->value['border']['l'] = null; } if ($tagName=='p') { $this->value['margin']['t'] = null; $this->value['margin']['b'] = null; } if ($tagName=='blockquote') { $this->value['margin']['t'] = 3; $this->value['margin']['r'] = 3; $this->value['margin']['b'] = 3; $this->value['margin']['l'] = 6; } $this->value['margin-auto'] = false; if (in_array($tagName, array('blockquote', 'div', 'fieldset'))) { $this->value['vertical-align'] = 'top'; } if (in_array($tagName, array('fieldset', 'legend'))) { $this->value['border'] = array( 't' => $border, 'r' => $border, 'b' => $border, 'l' => $border, 'radius' => array( 'tl' => array($units['5px'], $units['5px']), 'tr' => array($units['5px'], $units['5px']), 'br' => array($units['5px'], $units['5px']), 'bl' => array($units['5px'], $units['5px']) ), 'collapse' => false, ); } if (in_array($tagName, array('ul', 'li'))) { $this->value['list-style-type'] = ''; $this->value['list-style-image'] = ''; } if (!in_array($tagName, array('tr', 'td'))) { $this->value['padding'] = array( 't' => 0, 'r' => 0, 'b' => 0, 'l' => 0 ); } else { $this->value['padding'] = array( 't' => $units['1px'], 'r' => $units['1px'], 'b' => $units['1px'], 'l' => $units['1px'] ); } if ($tagName=='hr') { $this->value['border'] = array( 't' => $border, 'r' => $border, 'b' => $border, 'l' => $border, '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; } /** * Init the PDF Font * * @access public */ public 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' : ''); // font style $style = $b.$i; if ($this->_defaultFont) { 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=''; // complete style $style.= $u.$d.$o; // size : mm => pt $size = $this->value['font-size']; $size = 72 * $size / 25.4; // apply the font $this->_pdf->SetFont($family, $style, $this->value['mini-size']*$size); $this->_pdf->setTextColorArray($this->value['color']); if ($this->value['background']['color']) $this->_pdf->setFillColorArray($this->value['background']['color']); else $this->_pdf->setFillColor(255); } /** * add a level in the CSS history * * @access public */ public function save() { array_push($this->table, $this->value); } /** * remove a level in the CSS history * * @access public */ public function load() { if (count($this->table)) { $this->value = array_pop($this->table); } } /** * restore the Y positiony (used after a span) * * @access public */ public function restorePosition() { if ($this->value['y']==$this->_pdf->getY()) $this->_pdf->setY($this->value['yc'], false); } /** * set the New position for the current Tag * * @access public */ public function setPosition() { // get the current position $currentX = $this->_pdf->getX(); $currentY = $this->_pdf->getY(); // save it $this->value['xc'] = $currentX; $this->value['yc'] = $currentY; 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'] = $currentX + $x; $this->value['y'] = $currentY + $y; } else { $this->value['x'] = $this->_getLastAbsoluteX()+$x; $this->value['y'] = $this->_getLastAbsoluteY()+$y; } } else { $this->value['x'] = $currentX; $this->value['y'] = $currentY; if ($this->value['margin']['l']) $this->value['x']+= $this->value['margin']['l']; if ($this->value['margin']['t']) $this->value['y']+= $this->value['margin']['t']; } // save the new position $this->_pdf->setXY($this->value['x'], $this->value['y']); } /** * Analise the CSS style to convert it into Form style * * @access public * @param array styles */ public function getFormStyle() { $prop = array(); $prop['alignment'] = $this->value['text-align']; if (isset($this->value['background']['color']) && is_array($this->value['background']['color'])) { $prop['fillColor'] = $this->value['background']['color']; } if (isset($this->value['border']['t']['color'])) { $prop['strokeColor'] = $this->value['border']['t']['color']; } if (isset($this->value['border']['t']['width'])) { $prop['lineWidth'] = $this->value['border']['t']['width']; } if (isset($this->value['border']['t']['type'])) { $prop['borderStyle'] = $this->value['border']['t']['type']; } if (!empty($this->value['color'])) { $prop['textColor'] = $this->value['color']; } if (!empty($this->value['font-size'])) { $prop['textSize'] = $this->value['font-size']; } return $prop; } /** * Analise the CSS style to convert it into SVG style * * @access public * @param string tag name * @param array styles */ public function getSvgStyle($tagName, &$param) { // prepare $tagName = strtolower($tagName); $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; // read the class attribute $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; } // identify the tag, and the direct styles $this->value['id_tag'] = $tagName; $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'][] = $tagName; 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'][] = $tagName.'.'.$v; } } if ($id) { $this->value['id_lst'][] = '*#'.$id; $this->value['id_lst'][] = '#'.$id; $this->value['id_lst'][] = $tagName.'#'.$id; } // CSS style $styles = $this->_getFromCSS(); // adding the style from the tag $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 the css properties from the HTML parsing * * @access public * @param string $tagName * @param array $param * @param array $legacy */ public function analyse($tagName, &$param, $legacy = null) { // prepare the informations $tagName = strtolower($tagName); $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; // get the class names to use $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; } // prepare the values, and the list of css tags to identify $this->value['id_tag'] = $tagName; $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'][] = $tagName; if (count($class)) { foreach ($class as $v) { $this->value['id_lst'][] = '*.'.$v; $this->value['id_lst'][] = '.'.$v; $this->value['id_lst'][] = $tagName.'.'.$v; } } if ($id) { $this->value['id_lst'][] = '*#'.$id; $this->value['id_lst'][] = '#'.$id; $this->value['id_lst'][] = $tagName.'#'.$id; } // get the css styles from class $styles = $this->_getFromCSS(); // merge with the css styles from tag $styles = array_merge($styles, $param['style']); if (isset($param['allwidth']) && !isset($styles['width'])) $styles['width'] = '100%'; // reset some styles, depending on the tag name $this->resetStyle($tagName); // add the legacy values if ($legacy) { foreach ($legacy as $legacyName => $legacyValue) { if (is_array($legacyValue)) { foreach($legacyValue as $legacy2Name => $legacy2Value) $this->value[$legacyName][$legacy2Name] = $legacy2Value; } else { $this->value[$legacyName] = $legacyValue; } } } // some flags $correctWidth = false; $noWidth = true; // read all the css styles 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 ($tagName=='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)=='%') $correctWidth=true; $noWidth = 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 $valK => $valV) { if (!in_array($valV, array('solid', 'dotted', 'dashed'))) { $val[$valK] = 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 $valK => $valV) { $val[$valK] = $this->convertToColor($valV, $res); if (!$res) { $val[$valK] = 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 $valK => $valV) { $val[$valK] = $this->convertToMM($valV, 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 ($tagName=='table') $this->value['border']['collapse'] = ($val=='collapse'); break; case 'border-radius': $val = explode('/', $val); if (count($val)>2) { break; } $valH = $this->convertToRadius(trim($val[0])); if (count($valH)<1 || count($valH)>4) { break; } if (!isset($valH[1])) $valH[1] = $valH[0]; if (!isset($valH[2])) $valH = array($valH[0], $valH[0], $valH[1], $valH[1]); if (!isset($valH[3])) $valH[3] = $valH[1]; if (isset($val[1])) { $valV = $this->convertToRadius(trim($val[1])); if (count($valV)<1 || count($valV)>4) { break; } if (!isset($valV[1])) $valV[1] = $valV[0]; if (!isset($valV[2])) $valV = array($valV[0], $valV[0], $valV[1], $valV[1]); if (!isset($valV[3])) $valV[3] = $valV[1]; } else { $valV = $valH; } $this->value['border']['radius'] = array( 'tl' => array($valH[0], $valV[0]), 'tr' => array($valH[1], $valV[1]), 'br' => array($valH[2], $valV[2]), 'bl' => array($valH[3], $valV[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; // only for P tag 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']; // force the text align to left, if asked by html2pdf if ($this->_onlyLeft) $this->value['text-align'] = 'left'; // correction on the width (quick box) if ($noWidth && in_array($tagName, array('div', 'blockquote', 'fieldset')) && $this->value['position']!='absolute') { $this->value['width'] = $this->getLastWidth(); $this->value['width']-= $this->value['margin']['l'] + $this->value['margin']['r']; } else { if ($correctWidth) { if (!in_array($tagName, array('table', 'div', 'blockquote', '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($tagName, 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; } /** * get the height of the current line * * @access public * @return float $height in mm */ public function getLineHeight() { $val = $this->value['line-height']; if ($val=='normal') $val = '108%'; return $this->convertToMM($val, $this->value['font-size']); } /** * get the width of the parent * * @access public * @param boolean $mode true => adding padding and border * @return float $width in mm */ public 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(); } /** * get the height of the parent * * @access public * @param boolean $mode true => adding padding and border * @return float $height in mm */ public 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(); } /** * get the value of the float property * * @access public * @return $float left/right */ public function getFloat() { if ($this->value['float']=='left') return 'left'; if ($this->value['float']=='right') return 'right'; return null; } /** * get the last value for a specific key * * @access public * @param string $key * @return mixed */ public function getLastValue($key) { $nb = count($this->table); if ($nb>0) { return $this->table[$nb-1][$key]; } else { return null; } } /** * get the last absolute X * * @access protected * @return float $x */ protected 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(); } /** * get the last absolute Y * * @access protected * @return float $y */ protected 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(); } /** * get the CSS properties of the current tag * * @access protected * @return array $styles */ protected function _getFromCSS() { // styles to apply $styles = array(); // list of the selectors to get in the CSS files $getit = array(); // get the list of the selectors of each tags $lst = array(); $lst[] = $this->value['id_lst']; for ($i=count($this->table)-1; $i>=0; $i--) { $lst[] = $this->table[$i]['id_lst']; } // foreach selectors in the CSS files, verify if it match with the list of selectors foreach ($this->cssKeys as $key => $num) { if ($this->_getReccursiveStyle($key, $lst)) { $getit[$key] = $num; } } // if we have selectors if (count($getit)) { // get them, but in the definition order, because of priority asort($getit); foreach ($getit as $key => $val) $styles = array_merge($styles, $this->css[$key]); } return $styles; } /** * identify if the selector $key match with the list of tag selectors * * @access protected * @param string $key CSS selector to analyse * @param array $lst list of the selectors of each tags * @param string $next next step of parsing the selector * @return boolean */ protected function _getReccursiveStyle($key, $lst, $next = null) { // if next step if ($next!==null) { // we remove this step if ($next) $key = trim(substr($key, 0, -strlen($next))); array_shift($lst); // if no more step to identify => return false if (!count($lst)) { return false; } } // for each selector of the current step foreach ($lst[0] as $name) { // if selector = key => ok if ($key==$name) { return true; } // if the end of the key = the selector and the next step is ok => ok if (substr($key, -strlen(' '.$name))==' '.$name && $this->_getReccursiveStyle($key, $lst, $name)) { return true; } } // if we are not in the first step, we analyse the sub steps (the pareng tag of the current tag) if ($next!==null && $this->_getReccursiveStyle($key, $lst, '')) { return true; } // no corresponding found return false; } /** * Analyse a border * * @access public * @param string $css css border properties * @return array border properties */ public function readBorder($css) { // border none $none = array('type' => 'none', 'width' => 0, 'color' => array(0, 0, 0)); // default value $type = 'solid'; $width = $this->convertToMM('1pt'); $color = array(0, 0, 0); // clean up the values $css = explode(' ', $css); foreach ($css as $k => $v) { $v = trim($v); if ($v) $css[$k] = $v; else unset($css[$k]); } $css = array_values($css); // read the values $res = null; foreach ($css as $value) { // if no border => return none if ($value=='none' || $value=='hidden') { return $none; } // try to convert the value as a distance $tmp = $this->convertToMM($value); // if the convert is ok => it is a width if ($tmp!==null) { $width = $tmp; // else, it could be the type } else if (in_array($value, array('solid', 'dotted', 'dashed', 'double'))) { $type = $value; // else, it could be the color } else { $tmp = $this->convertToColor($value, $res); if ($res) $color = $tmp; } } // if no witdh => return none if (!$width) return $none; // return the border properties return array('type' => $type, 'width' => $width, 'color' => $color); } /** * duplicate the borders if needed * * @access protected * @param &array $val */ protected function _duplicateBorder(&$val) { // 1 value => L => RTB if (count($val)==1) { $val[1] = $val[0]; $val[2] = $val[0]; $val[3] = $val[0]; // 2 values => L => R & T => B } else if (count($val)==2) { $val[2] = $val[0]; $val[3] = $val[1]; // 3 values => T => B } else if (count($val)==3) { $val[3] = $val[1]; } } /** * Analyse a background * * @access public * @param string $css css background properties * @param &array $value parsed values (by reference, because, ther is a legacy of the parent CSS properties) */ public function convertBackground($css, &$value) { // is there a image ? $text = '/url\(([^)]*)\)/isU'; if (preg_match($text, $css, $match)) { // get the image $value['image'] = $this->convertBackgroundImage($match[0]); // remove if from the css properties $css = preg_replace($text, '', $css); $css = preg_replace('/[\s]+/', ' ', $css); } // protect some spaces $css = preg_replace('/,[\s]+/', ',', $css); // explode the values $css = explode(' ', $css); // background position to parse $pos = ''; // foreach value foreach ($css as $val) { // try to parse the value as a color $ok = false; $color = $this->convertToColor($val, $ok); // if ok => it is a color if ($ok) { $value['color'] = $color; // else if transparent => no coloàr } else if ($val=='transparent') { $value['color'] = null; // else } else { // try to parse the value as a repeat $repeat = $this->convertBackgroundRepeat($val); // if ok => it is repeat if ($repeat) { $value['repeat'] = $repeat; // else => it could only be a position } else { $pos.= ($pos ? ' ' : '').$val; } } } // if we have a position to parse if ($pos) { // try to read it $pos = $this->convertBackgroundPosition($pos, $ok); if ($ok) $value['position'] = $pos; } } /** * parse a background color * * @access public * @param string $css * @return string $value */ public function convertBackgroundColor($css) { $res = null; if ($css=='transparent') return null; else return $this->convertToColor($css, $res); } /** * parse a background image * * @access public * @param string $css * @return string $value */ public function convertBackgroundImage($css) { if ($css=='none') return null; else if (preg_match('/^url\(([^)]*)\)$/isU', $css, $match)) return $match[1]; else return null; } /** * parse a background position * * @access public * @param string $css * @param &boolean $res flag if conver is ok or not * @return array ($x, $y) */ public function convertBackgroundPosition($css, &$res) { // init the res $res = false; // explode the value $css = explode(' ', $css); // we must have 2 values. if 0 or >2 : error. if 1 => put center for 2 if (count($css)<2) { if (!$css[0]) return null; $css[1] = 'center'; } if (count($css)>2) return null; // prepare the values $x = 0; $y = 0; $res = true; // convert the first value if ($css[0]=='left') $x = '0%'; else if ($css[0]=='center') $x = '50%'; else if ($css[0]=='right') $x = '100%'; else if ($css[0]=='top') $y = '0%'; else if ($css[0]=='bottom') $y = '100%'; else if (preg_match('/^[-]?[0-9\.]+%$/isU', $css[0])) $x = $css[0]; else if ($this->convertToMM($css[0])) $x = $this->convertToMM($css[0]); else $res = false; // convert the second value if ($css[1]=='left') $x = '0%'; else if ($css[1]=='right') $x = '100%'; else if ($css[1]=='top') $y = '0%'; else if ($css[1]=='center') $y = '50%'; else if ($css[1]=='bottom') $y = '100%'; else if (preg_match('/^[-]?[0-9\.]+%$/isU', $css[1])) $y = $css[1]; else if ($this->convertToMM($css[1])) $y = $this->convertToMM($css[1]); else $res = false; // return the values return array($x, $y); } /** * parse a background repeat * * @access public * @param string $css * @return string $value */ public function convertBackgroundRepeat($css) { switch($css) { 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; } /** * convert a distance to mm * * @access public * @param string $css distance to convert * @param float $old parent distance * @return float $value */ public function convertToMM($css, $old=0.) { $css = trim($css); if (preg_match('/^[0-9\.\-]+$/isU', $css)) $css.= 'px'; if (preg_match('/^[0-9\.\-]+px$/isU', $css)) $css = 25.4/96. * str_replace('px', '', $css); else if (preg_match('/^[0-9\.\-]+pt$/isU', $css)) $css = 25.4/72. * str_replace('pt', '', $css); else if (preg_match('/^[0-9\.\-]+in$/isU', $css)) $css = 25.4 * str_replace('in', '', $css); else if (preg_match('/^[0-9\.\-]+mm$/isU', $css)) $css = 1.*str_replace('mm', '', $css); else if (preg_match('/^[0-9\.\-]+%$/isU', $css)) $css = 1.*$old*str_replace('%', '', $css)/100.; else $css = null; return $css; } /** * convert a css radius * * @access public * @param string $css * @return float $value */ public function convertToRadius($css) { // explode the value $css = explode(' ', $css); foreach ($css as $k => $v) { $v = trim($v); if ($v) { $v = $this->convertToMM($v, 0); if ($v!==null) { $css[$k] = $v; } else { unset($css[$k]); } } else { unset($css[$k]); } } return array_values($css); } /** * convert a css color * * @access public * @param string $css * @param &boolean $res * @return array (r,g, b) */ public function convertToColor($css, &$res) { // prepare the value $css = trim($css); $res = true; // if transparent => return null if (strtolower($css)=='transparent') return array(null, null, null); // HTML color if (isset($this->_htmlColor[strtolower($css)])) { $css = $this->_htmlColor[strtolower($css)]; $r = floatVal(hexdec(substr($css, 0, 2))); $v = floatVal(hexdec(substr($css, 2, 2))); $b = floatVal(hexdec(substr($css, 4, 2))); return array($r, $v, $b); } // like #FFFFFF if (preg_match('/^#[0-9A-Fa-f]{6}$/isU', $css)) { $r = floatVal(hexdec(substr($css, 1, 2))); $v = floatVal(hexdec(substr($css, 3, 2))); $b = floatVal(hexdec(substr($css, 5, 2))); return array($r, $v, $b); } // like #FFF if (preg_match('/^#[0-9A-F]{3}$/isU', $css)) { $r = floatVal(hexdec(substr($css, 1, 1).substr($css, 1, 1))); $v = floatVal(hexdec(substr($css, 2, 1).substr($css, 2, 1))); $b = floatVal(hexdec(substr($css, 3, 1).substr($css, 3, 1))); return array($r, $v, $b); } // like rgb(100, 100, 100) if (preg_match('/rgb\([\s]*([0-9%\.]+)[\s]*,[\s]*([0-9%\.]+)[\s]*,[\s]*([0-9%\.]+)[\s]*\)/isU', $css, $match)) { $r = $this->_convertSubColor($match[1]); $v = $this->_convertSubColor($match[2]); $b = $this->_convertSubColor($match[3]); return array($r*255., $v*255., $b*255.); } // like cmyk(100, 100, 100, 100) if (preg_match('/cmyk\([\s]*([0-9%\.]+)[\s]*,[\s]*([0-9%\.]+)[\s]*,[\s]*([0-9%\.]+)[\s]*,[\s]*([0-9%\.]+)[\s]*\)/isU', $css, $match)) { $c = $this->_convertSubColor($match[1]); $m = $this->_convertSubColor($match[2]); $y = $this->_convertSubColor($match[3]); $k = $this->_convertSubColor($match[4]); return array($c*100., $m*100., $y*100., $k*100.); } $res = false; return array(0., 0., 0.); } /** * color value to convert * * @access protected * @param string $c * @return float $c 0.->1. */ protected 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; } /** * read a css content * * @access protected * @param &string $code */ protected function _analyseStyle(&$code) { // clean the spaces $code = preg_replace('/[\s]+/', ' ', $code); // remove the comments $code = preg_replace('/\/\*.*?\*\//s', '', $code); // split each CSS code "selector { value }" preg_match_all('/([^{}]+){([^}]*)}/isU', $code, $match); // for each CSS code for ($k=0; $k1) { $cod = $tmp[0]; unset($tmp[0]); $tmp = implode(':', $tmp); $css[trim(strtolower($cod))] = trim($tmp); } } // explode the names $names = explode(',', $names); // save the values for each names foreach ($names as $name) { // clean the name $name = trim($name); // if a selector with somethink lige :hover => continue if (strpos($name, ':')!==false) continue; // save the value if (!isset($this->css[$name])) $this->css[$name] = $css; else $this->css[$name] = array_merge($this->css[$name], $css); } } // get he list of the keys $this->cssKeys = array_flip(array_keys($this->css)); } /** * Extract the css files from a html code * * @access public * @param string &$html */ public function readStyle(&$html) { // the CSS content $style = ' '; // extract the link tags, and remove them in the html code preg_match_all('/]*)>/isU', $html, $match); $html = preg_replace('/]*>/isU', '', $html); $html = preg_replace('/<\/link[^>]*>/isU', '', $html); // analyse each link tag foreach ($match[1] as $code) { $tmp = array(); // read the attributes name=value $prop = '([a-zA-Z0-9_]+)=([^"\'\s>]+)'; preg_match_all('/'.$prop.'/is', $code, $match); for ($k=0; $k we keep it if (isset($tmp['type']) && strtolower($tmp['type'])=='text/css' && isset($tmp['href'])) { // get the href $url = $tmp['href']; // get the content of the css file $content = @file_get_contents($url); // if "http://" in the url if (strpos($url, 'http://')!==false) { // get the domain "http://xxx/" $url = str_replace('http://', '', $url); $url = explode('/', $url); $urlMain = 'http://'.$url[0].'/'; // get the absolute url of the path $urlSelf = $url; unset($urlSelf[count($urlSelf)-1]); $urlSelf = 'http://'.implode('/', $urlSelf).'/'; // adapt the url in the css content $content = preg_replace('/url\(([^\\\\][^)]*)\)/isU', 'url('.$urlSelf.'$1)', $content); $content = preg_replace('/url\((\\\\[^)]*)\)/isU', 'url('.$urlMain.'$1)', $content); } else { // @TODO correction on url in absolute on a local css content // $content = preg_replace('/url\(([^)]*)\)/isU', 'url('.dirname($url).'/$1)', $content); } // add to the CSS content $style.= $content."\n"; } } // extract the style tags des tags style, and remove them in the html code preg_match_all('/]*>(.*)<\/style[^>]*>/isU', $html, $match); $html = preg_replace('/]*>(.*)<\/style[^>]*>/isU', '', $html); // analyse each style tags foreach ($match[1] as $code) { // add to the CSS content $code = str_replace('', '', $code); $style.= $code."\n"; } //analyse the css content $this->_analyseStyle($style); } }