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('/