spip_nursit/lib/mpdf/classes/cssmgr.php
2023-06-01 17:30:12 +02:00

2044 lines
74 KiB
PHP

<?php
class cssmgr
{
var $mpdf = null;
var $tablecascadeCSS;
var $cascadeCSS;
var $CSS;
var $tbCSSlvl;
public function __construct(mPDF $mpdf)
{
$this->mpdf = $mpdf;
$this->tablecascadeCSS = array();
$this->CSS = array();
$this->cascadeCSS = array();
$this->tbCSSlvl = 0;
}
function ReadCSS($html)
{
preg_match_all('/<style[^>]*media=["\']([^"\'>]*)["\'].*?<\/style>/is', $html, $m);
for ($i = 0; $i < count($m[0]); $i++) {
if ($this->mpdf->CSSselectMedia && !preg_match('/(' . trim($this->mpdf->CSSselectMedia) . '|all)/i', $m[1][$i])) {
$html = preg_replace('/' . preg_quote($m[0][$i], '/') . '/', '', $html);
}
}
preg_match_all('/<link[^>]*media=["\']([^"\'>]*)["\'].*?>/is', $html, $m);
for ($i = 0; $i < count($m[0]); $i++) {
if ($this->mpdf->CSSselectMedia && !preg_match('/(' . trim($this->mpdf->CSSselectMedia) . '|all)/i', $m[1][$i])) {
$html = preg_replace('/' . preg_quote($m[0][$i], '/') . '/', '', $html);
}
}
// mPDF 5.5.02
// Remove Comment tags <!-- ... --> inside CSS as <style> in HTML document
// Remove Comment tags /* ... */ inside CSS as <style> in HTML document
// But first, we replace upper and mixed case closing style tag with lower
// case so we can use str_replace later.
preg_match_all('/<style.*?>(.*?)<\/style>/si',$html,$m);
if (count($m[1])) {
for ($i = 0; $i < count($m[1]); $i++) {
// Remove comment tags
$sub = preg_replace('/(<\!\-\-|\-\->)/s',' ',$m[1][$i]);
$sub = '>'.preg_replace('|/\*.*?\*/|s',' ',$sub).'</style>';
$html = str_replace('>'.$m[1][$i].'</style>', $sub, $html);
}
}
$html = preg_replace('/<!--mpdf/i', '', $html);
$html = preg_replace('/mpdf-->/i', '', $html);
$html = preg_replace('/<\!\-\-.*?\-\->/s', ' ', $html);
$match = 0; // no match for instance
$regexp = ''; // This helps debugging: showing what is the REAL string being processed
$CSSext = array();
//CSS inside external files
$regexp = '/<link[^>]*rel=["\']stylesheet["\'][^>]*href=["\']([^>"\']*)["\'].*?>/si';
$x = preg_match_all($regexp, $html, $cxt);
if ($x) {
$match += $x;
$CSSext = $cxt[1];
}
$regexp = '/<link[^>]*href=["\']([^>"\']*)["\'][^>]*?rel=["\']stylesheet["\'].*?>/si';
$x = preg_match_all($regexp, $html, $cxt);
if ($x) {
$match += $x;
$CSSext = array_merge($CSSext, $cxt[1]);
}
// look for @import stylesheets
//$regexp = '/@import url\([\'\"]{0,1}([^\)]*?\.css)[\'\"]{0,1}\)/si';
$regexp = '/@import url\([\'\"]{0,1}([^\)]*?\.css(\?\S+)?)[\'\"]{0,1}\)/si';
$x = preg_match_all($regexp, $html, $cxt);
if ($x) {
$match += $x;
$CSSext = array_merge($CSSext, $cxt[1]);
}
// look for @import without the url()
//$regexp = '/@import [\'\"]{0,1}([^;]*?\.css)[\'\"]{0,1}/si';
$regexp = '/@import [\'\"]{0,1}([^;]*?\.css(\?\S+)?)[\'\"]{0,1}/si';
$x = preg_match_all($regexp, $html, $cxt);
if ($x) {
$match += $x;
$CSSext = array_merge($CSSext, $cxt[1]);
}
$ind = 0;
$CSSstr = '';
if (!is_array($this->cascadeCSS))
$this->cascadeCSS = array();
while ($match) {
$path = $CSSext[$ind];
$path = htmlspecialchars_decode($path); // mPDF 6
$this->mpdf->GetFullPath($path);
$CSSextblock = $this->mpdf->_get_file($path);
if ($CSSextblock) {
// look for embedded @import stylesheets in other stylesheets
// and fix url paths (including background-images) relative to stylesheet
//$regexpem = '/@import url\([\'\"]{0,1}(.*?\.css)[\'\"]{0,1}\)/si';
$regexpem = '/@import url\([\'\"]{0,1}(.*?\.css(\?\S+)?)[\'\"]{0,1}\)/si';
$xem = preg_match_all($regexpem, $CSSextblock, $cxtem);
$cssBasePath = preg_replace('/\/[^\/]*$/', '', $path) . '/';
if ($xem) {
foreach ($cxtem[1] AS $cxtembedded) {
// path is relative to original stlyesheet!!
$this->mpdf->GetFullPath($cxtembedded, $cssBasePath);
$match++;
$CSSext[] = $cxtembedded;
}
}
$regexpem = '/(background[^;]*url\s*\(\s*[\'\"]{0,1})([^\)\'\"]*)([\'\"]{0,1}\s*\))/si';
$xem = preg_match_all($regexpem, $CSSextblock, $cxtem);
if ($xem) {
for ($i = 0; $i < count($cxtem[0]); $i++) {
// path is relative to original stlyesheet!!
$embedded = $cxtem[2][$i];
if (!preg_match('/^data:image/i', $embedded)) { // mPDF 5.5.13
$this->mpdf->GetFullPath($embedded, $cssBasePath);
$CSSextblock = preg_replace('/' . preg_quote($cxtem[0][$i], '/') . '/', ($cxtem[1][$i] . $embedded . $cxtem[3][$i]), $CSSextblock);
}
}
}
$CSSstr .= ' ' . $CSSextblock;
}
$match--;
$ind++;
} //end of match
$match = 0; // reset value, if needed
// CSS as <style> in HTML document
$regexp = '/<style.*?>(.*?)<\/style>/si';
$match = preg_match_all($regexp, $html, $CSSblock);
if ($match) {
$tmpCSSstr = implode(' ', $CSSblock[1]);
$regexpem = '/(background[^;]*url\s*\(\s*[\'\"]{0,1})([^\)\'\"]*)([\'\"]{0,1}\s*\))/si';
$xem = preg_match_all($regexpem, $tmpCSSstr, $cxtem);
if ($xem) {
for ($i = 0; $i < count($cxtem[0]); $i++) {
$embedded = $cxtem[2][$i];
if (!preg_match('/^data:image/i', $embedded)) { // mPDF 5.5.13
$this->mpdf->GetFullPath($embedded);
$tmpCSSstr = preg_replace('/' . preg_quote($cxtem[0][$i], '/') . '/', ($cxtem[1][$i] . $embedded . $cxtem[3][$i]), $tmpCSSstr);
}
}
}
$CSSstr .= ' ' . $tmpCSSstr;
}
// Remove comments
$CSSstr = preg_replace('|/\*.*?\*/|s', ' ', $CSSstr);
$CSSstr = preg_replace('/[\s\n\r\t\f]/s', ' ', $CSSstr);
if (preg_match('/@media/', $CSSstr)) {
preg_match_all('/@media(.*?)\{(([^\{\}]*\{[^\{\}]*\})+)\s*\}/is', $CSSstr, $m);
for ($i = 0; $i < count($m[0]); $i++) {
if ($this->mpdf->CSSselectMedia && !preg_match('/(' . trim($this->mpdf->CSSselectMedia) . '|all)/i', $m[1][$i])) {
$CSSstr = preg_replace('/' . preg_quote($m[0][$i], '/') . '/', '', $CSSstr);
} else {
$CSSstr = preg_replace('/' . preg_quote($m[0][$i], '/') . '/', ' ' . $m[2][$i] . ' ', $CSSstr);
}
}
}
// Replace any background: url(data:image... with temporary image file reference
preg_match_all("/(url\(data:image\/(jpeg|gif|png);base64,(.*?)\))/si", $CSSstr, $idata); // mPDF 5.7.2
if (count($idata[0])) {
for ($i = 0; $i < count($idata[0]); $i++) {
$file = _MPDF_TEMP_PATH . '_tempCSSidata' . RAND(1, 10000) . '_' . $i . '.' . $idata[2][$i];
//Save to local file
file_put_contents($file, base64_decode($idata[3][$i]));
// $this->mpdf->GetFullPath($file); // ? is this needed - NO mPDF 5.6.03
$CSSstr = str_replace($idata[0][$i], 'url("' . $file . '")', $CSSstr); // mPDF 5.5.17
}
}
$CSSstr = preg_replace('/(<\!\-\-|\-\->)/s', ' ', $CSSstr);
// mPDF 5.7.4 URLs
// Characters "(" ")" and ";" in url() e.g. background-image, cause problems parsing the CSS string
// URLencode ( and ), but change ";" to a code which can be converted back after parsing (so as not to confuse ;
// with a segment delimiter in the URI)
$tempmarker = '%ZZ';
if (strpos($CSSstr, 'url(') !== false) {
preg_match_all('/url\(\"(.*?)\"\)/', $CSSstr, $m);
for ($i = 0; $i < count($m[1]); $i++) {
$tmp = str_replace(array('(', ')', ';'), array('%28', '%29', $tempmarker), $m[1][$i]);
$CSSstr = preg_replace('/' . preg_quote($m[0][$i], '/') . '/', 'url(\'' . $tmp . '\')', $CSSstr);
}
preg_match_all('/url\(\'(.*?)\'\)/', $CSSstr, $m);
for ($i = 0; $i < count($m[1]); $i++) {
$tmp = str_replace(array('(', ')', ';'), array('%28', '%29', $tempmarker), $m[1][$i]);
$CSSstr = preg_replace('/' . preg_quote($m[0][$i], '/') . '/', 'url(\'' . $tmp . '\')', $CSSstr);
}
preg_match_all('/url\(([^\'\"].*?[^\'\"])\)/', $CSSstr, $m);
for ($i = 0; $i < count($m[1]); $i++) {
$tmp = str_replace(array('(', ')', ';'), array('%28', '%29', $tempmarker), $m[1][$i]);
$CSSstr = preg_replace('/' . preg_quote($m[0][$i], '/') . '/', 'url(\'' . $tmp . '\')', $CSSstr);
}
}
if ($CSSstr) {
$classproperties = array(); // mPDF 6
preg_match_all('/(.*?)\{(.*?)\}/', $CSSstr, $styles);
for ($i = 0; $i < count($styles[1]); $i++) {
// SET array e.g. $classproperties['COLOR'] = '#ffffff';
$stylestr = trim($styles[2][$i]);
$stylearr = explode(';', $stylestr);
foreach ($stylearr AS $sta) {
if (trim($sta)) {
// Changed to allow style="background: url('http://www.bpm1.com/bg.jpg')"
$tmp = explode(':', $sta, 2);
$property = $tmp[0];
if (isset($tmp[1])) {
$value = $tmp[1];
} else {
$value = '';
}
$value = str_replace($tempmarker, ';', $value); // mPDF 5.7.4 URLs
$property = trim($property);
$value = preg_replace('/\s*!important/i', '', $value);
$value = trim($value);
if ($property && ($value || $value === '0')) {
// Ignores -webkit-gradient so doesn't override -moz-
if ((strtoupper($property) == 'BACKGROUND-IMAGE' || strtoupper($property) == 'BACKGROUND') && preg_match('/-webkit-gradient/i', $value)) {
continue;
}
$classproperties[strtoupper($property)] = $value;
}
}
}
$classproperties = $this->fixCSS($classproperties);
$tagstr = strtoupper(trim($styles[1][$i]));
$tagarr = explode(',', $tagstr);
$pageselectors = false; // used to turn on $this->mpdf->mirrorMargins
foreach ($tagarr AS $tg) {
// mPDF 5.7.4
if (preg_match('/NTH-CHILD\((\s*(([\-+]?\d*)N(\s*[\-+]\s*\d+)?|[\-+]?\d+|ODD|EVEN)\s*)\)/', $tg, $m)) {
$tg = preg_replace('/NTH-CHILD\(.*\)/', 'NTH-CHILD(' . str_replace(' ', '', $m[1]) . ')', $tg);
}
$tags = preg_split('/\s+/', trim($tg));
$level = count($tags);
$t = '';
$t2 = '';
$t3 = '';
if (trim($tags[0]) == '@PAGE') {
if (isset($tags[0])) {
$t = trim($tags[0]);
}
if (isset($tags[1])) {
$t2 = trim($tags[1]);
}
if (isset($tags[2])) {
$t3 = trim($tags[2]);
}
$tag = '';
if ($level == 1) {
$tag = $t;
} else if ($level == 2 && preg_match('/^[:](.*)$/', $t2, $m)) {
$tag = $t . '>>PSEUDO>>' . $m[1];
if ($m[1] == 'LEFT' || $m[1] == 'RIGHT') {
$pageselectors = true;
} // used to turn on $this->mpdf->mirrorMargins
} else if ($level == 2) {
$tag = $t . '>>NAMED>>' . $t2;
} else if ($level == 3 && preg_match('/^[:](.*)$/', $t3, $m)) {
$tag = $t . '>>NAMED>>' . $t2 . '>>PSEUDO>>' . $m[1];
if ($m[1] == 'LEFT' || $m[1] == 'RIGHT') {
$pageselectors = true;
} // used to turn on $this->mpdf->mirrorMargins
}
if (isset($this->CSS[$tag]) && $tag) {
$this->CSS[$tag] = $this->array_merge_recursive_unique($this->CSS[$tag], $classproperties);
} else if ($tag) {
$this->CSS[$tag] = $classproperties;
}
} else if ($level == 1) { // e.g. p or .class or #id or p.class or p#id
if (isset($tags[0])) {
$t = trim($tags[0]);
}
if ($t) {
$tag = '';
if (preg_match('/^[.](.*)$/', $t, $m)) {
$tag = 'CLASS>>' . $m[1];
} else if (preg_match('/^[#](.*)$/', $t, $m)) {
$tag = 'ID>>' . $m[1];
} else if (preg_match('/^\[LANG=[\'\"]{0,1}([A-Z\-]{2,11})[\'\"]{0,1}\]$/', $t, $m)) {
$tag = 'LANG>>' . strtolower($m[1]);
} // mPDF 6 Special case for lang as attribute selector
else if (preg_match('/^:LANG\([\'\"]{0,1}([A-Z\-]{2,11})[\'\"]{0,1}\)$/', $t, $m)) {
$tag = 'LANG>>' . strtolower($m[1]);
} // mPDF 6 Special case for lang as attribute selector
else if (preg_match('/^(' . $this->mpdf->allowedCSStags . ')[.](.*)$/', $t, $m)) {
$tag = $m[1] . '>>CLASS>>' . $m[2];
} else if (preg_match('/^(' . $this->mpdf->allowedCSStags . ')\s*:NTH-CHILD\((.*)\)$/', $t, $m)) {
$tag = $m[1] . '>>SELECTORNTHCHILD>>' . $m[2];
} else if (preg_match('/^(' . $this->mpdf->allowedCSStags . ')[#](.*)$/', $t, $m)) {
$tag = $m[1] . '>>ID>>' . $m[2];
} else if (preg_match('/^(' . $this->mpdf->allowedCSStags . ')\[LANG=[\'\"]{0,1}([A-Z\-]{2,11})[\'\"]{0,1}\]$/', $t, $m)) {
$tag = $m[1] . '>>LANG>>' . strtolower($m[2]);
} // mPDF 6 Special case for lang as attribute selector
else if (preg_match('/^(' . $this->mpdf->allowedCSStags . '):LANG\([\'\"]{0,1}([A-Z\-]{2,11})[\'\"]{0,1}\)$/', $t, $m)) {
$tag = $m[1] . '>>LANG>>' . strtolower($m[2]);
} // mPDF 6 Special case for lang as attribute selector
else if (preg_match('/^(' . $this->mpdf->allowedCSStags . ')$/', $t)) {
$tag = $t;
}
if (isset($this->CSS[$tag]) && $tag) {
$this->CSS[$tag] = $this->array_merge_recursive_unique($this->CSS[$tag], $classproperties);
} else if ($tag) {
$this->CSS[$tag] = $classproperties;
}
}
} else {
$tmp = array();
for ($n = 0; $n < $level; $n++) {
if (isset($tags[$n])) {
$t = trim($tags[$n]);
} else {
$t = '';
}
if ($t) {
$tag = '';
if (preg_match('/^[.](.*)$/', $t, $m)) {
$tag = 'CLASS>>' . $m[1];
} else if (preg_match('/^[#](.*)$/', $t, $m)) {
$tag = 'ID>>' . $m[1];
} else if (preg_match('/^\[LANG=[\'\"]{0,1}([A-Z\-]{2,11})[\'\"]{0,1}\]$/', $t, $m)) {
$tag = 'LANG>>' . strtolower($m[1]);
} // mPDF 6 Special case for lang as attribute selector
else if (preg_match('/^:LANG\([\'\"]{0,1}([A-Z\-]{2,11})[\'\"]{0,1}\)$/', $t, $m)) {
$tag = 'LANG>>' . strtolower($m[1]);
} // mPDF 6 Special case for lang as attribute selector
else if (preg_match('/^(' . $this->mpdf->allowedCSStags . ')[.](.*)$/', $t, $m)) {
$tag = $m[1] . '>>CLASS>>' . $m[2];
} else if (preg_match('/^(' . $this->mpdf->allowedCSStags . ')\s*:NTH-CHILD\((.*)\)$/', $t, $m)) {
$tag = $m[1] . '>>SELECTORNTHCHILD>>' . $m[2];
} else if (preg_match('/^(' . $this->mpdf->allowedCSStags . ')[#](.*)$/', $t, $m)) {
$tag = $m[1] . '>>ID>>' . $m[2];
} else if (preg_match('/^(' . $this->mpdf->allowedCSStags . ')\[LANG=[\'\"]{0,1}([A-Z\-]{2,11})[\'\"]{0,1}\]$/', $t, $m)) {
$tag = $m[1] . '>>LANG>>' . strtolower($m[2]);
} // mPDF 6 Special case for lang as attribute selector
else if (preg_match('/^(' . $this->mpdf->allowedCSStags . '):LANG\([\'\"]{0,1}([A-Z\-]{2,11})[\'\"]{0,1}\)$/', $t, $m)) {
$tag = $m[1] . '>>LANG>>' . strtolower($m[2]);
} // mPDF 6 Special case for lang as attribute selector
else if (preg_match('/^(' . $this->mpdf->allowedCSStags . ')$/', $t)) {
$tag = $t;
}
if ($tag)
$tmp[] = $tag;
else {
break;
}
}
}
if ($tag) {
$x = &$this->cascadeCSS;
foreach ($tmp AS $tp) {
$x = &$x[$tp];
}
$x = $this->array_merge_recursive_unique($x, $classproperties);
$x['depth'] = $level;
}
}
}
if ($pageselectors) {
$this->mpdf->mirrorMargins = true;
}
$properties = array();
$values = array();
$classproperties = array();
}
} // end of if
//Remove CSS (tags and content), if any
$regexp = '/<style.*?>(.*?)<\/style>/si'; // it can be <style> or <style type="txt/css">
$html = preg_replace($regexp, '', $html);
//print_r($this->CSS); exit;
//print_r($this->cascadeCSS); exit;
return $html;
}
function readInlineCSS($html)
{
$html = htmlspecialchars_decode($html); // mPDF 5.7.4 URLs
// mPDF 5.7.4 URLs
// Characters "(" ")" and ";" in url() e.g. background-image, cause probems parsing the CSS string
// URLencode ( and ), but change ";" to a code which can be converted back after parsing (so as not to confuse ;
// with a segment delimiter in the URI)
$tempmarker = '%ZZ';
if (strpos($html, 'url(') !== false) {
preg_match_all('/url\(\"(.*?)\"\)/', $html, $m);
for ($i = 0; $i < count($m[1]); $i++) {
$tmp = str_replace(array('(', ')', ';'), array('%28', '%29', $tempmarker), $m[1][$i]);
$html = preg_replace('/' . preg_quote($m[0][$i], '/') . '/', 'url(\'' . $tmp . '\')', $html);
}
preg_match_all('/url\(\'(.*?)\'\)/', $html, $m);
for ($i = 0; $i < count($m[1]); $i++) {
$tmp = str_replace(array('(', ')', ';'), array('%28', '%29', $tempmarker), $m[1][$i]);
$html = preg_replace('/' . preg_quote($m[0][$i], '/') . '/', 'url(\'' . $tmp . '\')', $html);
}
preg_match_all('/url\(([^\'\"].*?[^\'\"])\)/', $html, $m);
for ($i = 0; $i < count($m[1]); $i++) {
$tmp = str_replace(array('(', ')', ';'), array('%28', '%29', $tempmarker), $m[1][$i]);
$html = preg_replace('/' . preg_quote($m[0][$i], '/') . '/', 'url(\'' . $tmp . '\')', $html);
}
}
//Fix incomplete CSS code
$size = strlen($html) - 1;
if (substr($html, $size, 1) != ';')
$html .= ';';
//Make CSS[Name-of-the-class] = array(key => value)
$regexp = '|\\s*?(\\S+?):(.+?);|i';
preg_match_all($regexp, $html, $styleinfo);
$properties = $styleinfo[1];
$values = $styleinfo[2];
//Array-properties and Array-values must have the SAME SIZE!
$classproperties = array();
for ($i = 0; $i < count($properties); $i++) {
// Ignores -webkit-gradient so doesn't override -moz-
if ((strtoupper($properties[$i]) == 'BACKGROUND-IMAGE' || strtoupper($properties[$i]) == 'BACKGROUND') && preg_match('/-webkit-gradient/i', $values[$i])) {
continue;
}
$values[$i] = str_replace($tempmarker, ';', $values[$i]); // mPDF 5.7.4 URLs
$classproperties[strtoupper($properties[$i])] = trim($values[$i]);
}
return $this->fixCSS($classproperties);
}
function _fix_borderStr($bd)
{
preg_match_all("/\((.*?)\)/", $bd, $m);
if (count($m[1])) {
for ($i = 0; $i < count($m[1]); $i++) {
$sub = preg_replace("/ /", "", $m[1][$i]);
$bd = preg_replace('/' . preg_quote($m[1][$i], '/') . '/si', $sub, $bd);
}
}
$prop = preg_split('/\s+/', trim($bd));
$w = 'medium';
$c = '#000000';
$s = 'none';
if (count($prop) == 1) {
// solid
if (in_array($prop[0], $this->mpdf->borderstyles) || $prop[0] == 'none' || $prop[0] == 'hidden') {
$s = $prop[0];
}
// #000000
else if (is_array($this->mpdf->ConvertColor($prop[0]))) {
$c = $prop[0];
}
// 1px
else {
$w = $prop[0];
}
} else if (count($prop) == 2) {
// 1px solid
if (in_array($prop[1], $this->mpdf->borderstyles) || $prop[1] == 'none' || $prop[1] == 'hidden') {
$w = $prop[0];
$s = $prop[1];
}
// solid #000000
else if (in_array($prop[0], $this->mpdf->borderstyles) || $prop[0] == 'none' || $prop[0] == 'hidden') {
$s = $prop[0];
$c = $prop[1];
}
// 1px #000000
else {
$w = $prop[0];
$c = $prop[1];
}
} else if (count($prop) == 3) {
// Change #000000 1px solid to 1px solid #000000 (proper)
if (substr($prop[0], 0, 1) == '#') {
$c = $prop[0];
$w = $prop[1];
$s = $prop[2];
}
// Change solid #000000 1px to 1px solid #000000 (proper)
else if (substr($prop[0], 1, 1) == '#') {
$s = $prop[0];
$c = $prop[1];
$w = $prop[2];
}
// Change solid 1px #000000 to 1px solid #000000 (proper)
else if (in_array($prop[0], $this->mpdf->borderstyles) || $prop[0] == 'none' || $prop[0] == 'hidden') {
$s = $prop[0];
$w = $prop[1];
$c = $prop[2];
} else {
$w = $prop[0];
$s = $prop[1];
$c = $prop[2];
}
} else {
return '';
}
$s = strtolower($s);
return $w . ' ' . $s . ' ' . $c;
}
function fixCSS($prop)
{
if (!is_array($prop) || (count($prop) == 0))
return array();
$newprop = array();
foreach ($prop AS $k => $v) {
if ($k != 'BACKGROUND-IMAGE' && $k != 'BACKGROUND' && $k != 'ODD-HEADER-NAME' && $k != 'EVEN-HEADER-NAME' && $k != 'ODD-FOOTER-NAME' && $k != 'EVEN-FOOTER-NAME' && $k != 'HEADER' && $k != 'FOOTER') {
$v = strtolower($v);
}
if ($k == 'FONT') {
$s = trim($v);
preg_match_all('/\"(.*?)\"/', $s, $ff);
if (count($ff[1])) {
foreach ($ff[1] AS $ffp) {
$w = preg_split('/\s+/', $ffp);
$s = preg_replace('/\"' . $ffp . '\"/', $w[0], $s);
}
}
preg_match_all('/\'(.*?)\'/', $s, $ff);
if (count($ff[1])) {
foreach ($ff[1] AS $ffp) {
$w = preg_split('/\s+/', $ffp);
$s = preg_replace('/\'' . $ffp . '\'/', $w[0], $s);
}
}
$s = preg_replace('/\s*,\s*/', ',', $s);
$bits = preg_split('/\s+/', $s);
if (count($bits) > 1) {
$k = 'FONT-FAMILY';
$v = $bits[(count($bits) - 1)];
$fs = $bits[(count($bits) - 2)];
if (preg_match('/(.*?)\/(.*)/', $fs, $fsp)) {
$newprop['FONT-SIZE'] = $fsp[1];
$newprop['LINE-HEIGHT'] = $fsp[2];
} else {
$newprop['FONT-SIZE'] = $fs;
}
if (preg_match('/(italic|oblique)/i', $s)) {
$newprop['FONT-STYLE'] = 'italic';
} else {
$newprop['FONT-STYLE'] = 'normal';
}
if (preg_match('/bold/i', $s)) {
$newprop['FONT-WEIGHT'] = 'bold';
} else {
$newprop['FONT-WEIGHT'] = 'normal';
}
if (preg_match('/small-caps/i', $s)) {
$newprop['TEXT-TRANSFORM'] = 'uppercase';
}
}
} else if ($k == 'FONT-FAMILY') {
$aux_fontlist = explode(",", $v);
$found = 0;
foreach ($aux_fontlist AS $f) {
$fonttype = trim($f);
$fonttype = preg_replace('/["\']*(.*?)["\']*/', '\\1', $fonttype);
$fonttype = preg_replace('/ /', '', $fonttype);
$v = strtolower(trim($fonttype));
if (isset($this->mpdf->fonttrans[$v]) && $this->mpdf->fonttrans[$v]) {
$v = $this->mpdf->fonttrans[$v];
}
if ((!$this->mpdf->onlyCoreFonts && in_array($v, $this->mpdf->available_unifonts)) ||
in_array($v, array('ccourier', 'ctimes', 'chelvetica')) ||
($this->mpdf->onlyCoreFonts && in_array($v, array('courier', 'times', 'helvetica', 'arial'))) ||
in_array($v, array('sjis', 'uhc', 'big5', 'gb'))) {
$newprop[$k] = $v;
$found = 1;
break;
}
}
if (!$found) {
foreach ($aux_fontlist AS $f) {
$fonttype = trim($f);
$fonttype = preg_replace('/["\']*(.*?)["\']*/', '\\1', $fonttype);
$fonttype = preg_replace('/ /', '', $fonttype);
$v = strtolower(trim($fonttype));
if (isset($this->mpdf->fonttrans[$v]) && $this->mpdf->fonttrans[$v]) {
$v = $this->mpdf->fonttrans[$v];
}
if (in_array($v, $this->mpdf->sans_fonts) || in_array($v, $this->mpdf->serif_fonts) || in_array($v, $this->mpdf->mono_fonts)) {
$newprop[$k] = $v;
break;
}
}
}
}
// mPDF 5.7.1
else if ($k == 'FONT-VARIANT') {
if (preg_match('/(normal|none)/', $v, $m)) { // mPDF 6
$newprop['FONT-VARIANT-LIGATURES'] = $m[1];
$newprop['FONT-VARIANT-CAPS'] = $m[1];
$newprop['FONT-VARIANT-NUMERIC'] = $m[1];
$newprop['FONT-VARIANT-ALTERNATES'] = $m[1];
} else {
if (preg_match_all('/(no-common-ligatures|\bcommon-ligatures|no-discretionary-ligatures|\bdiscretionary-ligatures|no-historical-ligatures|\bhistorical-ligatures|no-contextual|\bcontextual)/i', $v, $m)) {
$newprop['FONT-VARIANT-LIGATURES'] = implode(' ', $m[1]);
}
if (preg_match('/(all-small-caps|\bsmall-caps|all-petite-caps|\bpetite-caps|unicase|titling-caps)/i', $v, $m)) {
$newprop['FONT-VARIANT-CAPS'] = $m[1];
}
if (preg_match_all('/(lining-nums|oldstyle-nums|proportional-nums|tabular-nums|diagonal-fractions|stacked-fractions)/i', $v, $m)) {
$newprop['FONT-VARIANT-NUMERIC'] = implode(' ', $m[1]);
}
if (preg_match('/(historical-forms)/i', $v, $m)) {
$newprop['FONT-VARIANT-ALTERNATES'] = $m[1];
}
}
} else if ($k == 'MARGIN') {
$tmp = $this->expand24($v);
$newprop['MARGIN-TOP'] = $tmp['T'];
$newprop['MARGIN-RIGHT'] = $tmp['R'];
$newprop['MARGIN-BOTTOM'] = $tmp['B'];
$newprop['MARGIN-LEFT'] = $tmp['L'];
}
/* -- BORDER-RADIUS -- */ else if ($k == 'BORDER-RADIUS' || $k == 'BORDER-TOP-LEFT-RADIUS' || $k == 'BORDER-TOP-RIGHT-RADIUS' || $k == 'BORDER-BOTTOM-LEFT-RADIUS' || $k == 'BORDER-BOTTOM-RIGHT-RADIUS') {
$tmp = $this->border_radius_expand($v, $k);
if (isset($tmp['TL-H']))
$newprop['BORDER-TOP-LEFT-RADIUS-H'] = $tmp['TL-H'];
if (isset($tmp['TL-V']))
$newprop['BORDER-TOP-LEFT-RADIUS-V'] = $tmp['TL-V'];
if (isset($tmp['TR-H']))
$newprop['BORDER-TOP-RIGHT-RADIUS-H'] = $tmp['TR-H'];
if (isset($tmp['TR-V']))
$newprop['BORDER-TOP-RIGHT-RADIUS-V'] = $tmp['TR-V'];
if (isset($tmp['BL-H']))
$newprop['BORDER-BOTTOM-LEFT-RADIUS-H'] = $tmp['BL-H'];
if (isset($tmp['BL-V']))
$newprop['BORDER-BOTTOM-LEFT-RADIUS-V'] = $tmp['BL-V'];
if (isset($tmp['BR-H']))
$newprop['BORDER-BOTTOM-RIGHT-RADIUS-H'] = $tmp['BR-H'];
if (isset($tmp['BR-V']))
$newprop['BORDER-BOTTOM-RIGHT-RADIUS-V'] = $tmp['BR-V'];
}
/* -- END BORDER-RADIUS -- */
else if ($k == 'PADDING') {
$tmp = $this->expand24($v);
$newprop['PADDING-TOP'] = $tmp['T'];
$newprop['PADDING-RIGHT'] = $tmp['R'];
$newprop['PADDING-BOTTOM'] = $tmp['B'];
$newprop['PADDING-LEFT'] = $tmp['L'];
} else if ($k == 'BORDER') {
if ($v == '1') {
$v = '1px solid #000000';
} else {
$v = $this->_fix_borderStr($v);
}
$newprop['BORDER-TOP'] = $v;
$newprop['BORDER-RIGHT'] = $v;
$newprop['BORDER-BOTTOM'] = $v;
$newprop['BORDER-LEFT'] = $v;
} else if ($k == 'BORDER-TOP') {
$newprop['BORDER-TOP'] = $this->_fix_borderStr($v);
} else if ($k == 'BORDER-RIGHT') {
$newprop['BORDER-RIGHT'] = $this->_fix_borderStr($v);
} else if ($k == 'BORDER-BOTTOM') {
$newprop['BORDER-BOTTOM'] = $this->_fix_borderStr($v);
} else if ($k == 'BORDER-LEFT') {
$newprop['BORDER-LEFT'] = $this->_fix_borderStr($v);
} else if ($k == 'BORDER-STYLE') {
$e = $this->expand24($v);
if (!empty($e)) {
$newprop['BORDER-TOP-STYLE'] = $e['T'];
$newprop['BORDER-RIGHT-STYLE'] = $e['R'];
$newprop['BORDER-BOTTOM-STYLE'] = $e['B'];
$newprop['BORDER-LEFT-STYLE'] = $e['L'];
}
} else if ($k == 'BORDER-WIDTH') {
$e = $this->expand24($v);
if (!empty($e)) {
$newprop['BORDER-TOP-WIDTH'] = $e['T'];
$newprop['BORDER-RIGHT-WIDTH'] = $e['R'];
$newprop['BORDER-BOTTOM-WIDTH'] = $e['B'];
$newprop['BORDER-LEFT-WIDTH'] = $e['L'];
}
} else if ($k == 'BORDER-COLOR') {
$e = $this->expand24($v);
if (!empty($e)) {
$newprop['BORDER-TOP-COLOR'] = $e['T'];
$newprop['BORDER-RIGHT-COLOR'] = $e['R'];
$newprop['BORDER-BOTTOM-COLOR'] = $e['B'];
$newprop['BORDER-LEFT-COLOR'] = $e['L'];
}
} else if ($k == 'BORDER-SPACING') {
$prop = preg_split('/\s+/', trim($v));
if (count($prop) == 1) {
$newprop['BORDER-SPACING-H'] = $prop[0];
$newprop['BORDER-SPACING-V'] = $prop[0];
} else if (count($prop) == 2) {
$newprop['BORDER-SPACING-H'] = $prop[0];
$newprop['BORDER-SPACING-V'] = $prop[1];
}
} else if ($k == 'TEXT-OUTLINE') { // mPDF 5.6.07
$prop = preg_split('/\s+/', trim($v));
if (trim(strtolower($v)) == 'none') {
$newprop['TEXT-OUTLINE'] = 'none';
} else if (count($prop) == 2) {
$newprop['TEXT-OUTLINE-WIDTH'] = $prop[0];
$newprop['TEXT-OUTLINE-COLOR'] = $prop[1];
} else if (count($prop) == 3) {
$newprop['TEXT-OUTLINE-WIDTH'] = $prop[0];
$newprop['TEXT-OUTLINE-COLOR'] = $prop[2];
}
} else if ($k == 'SIZE') {
$prop = preg_split('/\s+/', trim($v));
if (preg_match('/(auto|portrait|landscape)/', $prop[0])) {
$newprop['SIZE'] = strtoupper($prop[0]);
} else if (count($prop) == 1) {
$newprop['SIZE']['W'] = $this->mpdf->ConvertSize($prop[0]);
$newprop['SIZE']['H'] = $this->mpdf->ConvertSize($prop[0]);
} else if (count($prop) == 2) {
$newprop['SIZE']['W'] = $this->mpdf->ConvertSize($prop[0]);
$newprop['SIZE']['H'] = $this->mpdf->ConvertSize($prop[1]);
}
} else if ($k == 'SHEET-SIZE') {
$prop = preg_split('/\s+/', trim($v));
if (count($prop) == 2) {
$newprop['SHEET-SIZE'] = array($this->mpdf->ConvertSize($prop[0]), $this->mpdf->ConvertSize($prop[1]));
} else {
if (preg_match('/([0-9a-zA-Z]*)-L/i', $v, $m)) { // e.g. A4-L = A$ landscape
$ft = $this->mpdf->_getPageFormat($m[1]);
$format = array($ft[1], $ft[0]);
} else {
$format = $this->mpdf->_getPageFormat($v);
}
if ($format) {
$newprop['SHEET-SIZE'] = array($format[0] / _MPDFK, $format[1] / _MPDFK);
}
}
} else if ($k == 'BACKGROUND') {
$bg = $this->parseCSSbackground($v);
if ($bg['c']) {
$newprop['BACKGROUND-COLOR'] = $bg['c'];
} else {
$newprop['BACKGROUND-COLOR'] = 'transparent';
}
/* -- BACKGROUNDS -- */
if ($bg['i']) {
$newprop['BACKGROUND-IMAGE'] = $bg['i'];
if ($bg['r']) {
$newprop['BACKGROUND-REPEAT'] = $bg['r'];
}
if ($bg['p']) {
$newprop['BACKGROUND-POSITION'] = $bg['p'];
}
} else {
$newprop['BACKGROUND-IMAGE'] = '';
}
/* -- END BACKGROUNDS -- */
}
/* -- BACKGROUNDS -- */ else if ($k == 'BACKGROUND-IMAGE') {
if (preg_match('/(-moz-)*(repeating-)*(linear|radial)-gradient\(.*\)/i', $v, $m)) {
$newprop['BACKGROUND-IMAGE'] = $m[0];
continue;
}
if (preg_match('/url\([\'\"]{0,1}(.*?)[\'\"]{0,1}\)/i', $v, $m)) {
$newprop['BACKGROUND-IMAGE'] = $m[1];
} else if (strtolower($v) == 'none') {
$newprop['BACKGROUND-IMAGE'] = '';
}
} else if ($k == 'BACKGROUND-REPEAT') {
if (preg_match('/(repeat-x|repeat-y|no-repeat|repeat)/i', $v, $m)) {
$newprop['BACKGROUND-REPEAT'] = strtolower($m[1]);
}
} else if ($k == 'BACKGROUND-POSITION') {
$s = $v;
$bits = preg_split('/\s+/', trim($s));
// These should be Position x1 or x2
if (count($bits) == 1) {
if (preg_match('/bottom/', $bits[0])) {
$bg['p'] = '50% 100%';
} else if (preg_match('/top/', $bits[0])) {
$bg['p'] = '50% 0%';
} else {
$bg['p'] = $bits[0] . ' 50%';
}
} else if (count($bits) == 2) {
// Can be either right center or center right
if (preg_match('/(top|bottom)/', $bits[0]) || preg_match('/(left|right)/', $bits[1])) {
$bg['p'] = $bits[1] . ' ' . $bits[0];
} else {
$bg['p'] = $bits[0] . ' ' . $bits[1];
}
}
if ($bg['p']) {
$bg['p'] = preg_replace('/(left|top)/', '0%', $bg['p']);
$bg['p'] = preg_replace('/(right|bottom)/', '100%', $bg['p']);
$bg['p'] = preg_replace('/(center)/', '50%', $bg['p']);
if (!preg_match('/[\-]{0,1}\d+(in|cm|mm|pt|pc|em|ex|px|%)* [\-]{0,1}\d+(in|cm|mm|pt|pc|em|ex|px|%)*/', $bg['p'])) {
$bg['p'] = false;
}
}
if ($bg['p']) {
$newprop['BACKGROUND-POSITION'] = $bg['p'];
}
}
/* -- END BACKGROUNDS -- */ else if ($k == 'IMAGE-ORIENTATION') {
if (preg_match('/([\-]*[0-9\.]+)(deg|grad|rad)/i', $v, $m)) {
$angle = $m[1] + 0;
if (strtolower($m[2]) == 'deg') {
$angle = $angle;
} else if (strtolower($m[2]) == 'grad') {
$angle *= (360 / 400);
} else if (strtolower($m[2]) == 'rad') {
$angle = rad2deg($angle);
}
while ($angle < 0) {
$angle += 360;
}
$angle = ($angle % 360);
$angle /= 90;
$angle = round($angle) * 90;
$newprop['IMAGE-ORIENTATION'] = $angle;
}
} else if ($k == 'TEXT-ALIGN') {
if (preg_match('/["\'](.){1}["\']/i', $v, $m)) {
$d = array_search($m[1], $this->mpdf->decimal_align);
if ($d !== false) {
$newprop['TEXT-ALIGN'] = $d;
}
if (preg_match('/(center|left|right)/i', $v, $m)) {
$newprop['TEXT-ALIGN'] .= strtoupper(substr($m[1], 0, 1));
} else {
$newprop['TEXT-ALIGN'] .= 'R';
} // default = R
} else if (preg_match('/["\'](\\\[a-fA-F0-9]{1,6})["\']/i', $v, $m)) {
$utf8 = codeHex2utf(substr($m[1], 1, 6));
$d = array_search($utf8, $this->mpdf->decimal_align);
if ($d !== false) {
$newprop['TEXT-ALIGN'] = $d;
}
if (preg_match('/(center|left|right)/i', $v, $m)) {
$newprop['TEXT-ALIGN'] .= strtoupper(substr($m[1], 0, 1));
} else {
$newprop['TEXT-ALIGN'] .= 'R';
} // default = R
} else {
$newprop[$k] = $v;
}
}
// mpDF 6 Lists
else if ($k == 'LIST-STYLE') {
if (preg_match('/none/i', $v, $m)) {
$newprop['LIST-STYLE-TYPE'] = 'none';
$newprop['LIST-STYLE-IMAGE'] = 'none';
}
if (preg_match('/(lower-roman|upper-roman|lower-latin|lower-alpha|upper-latin|upper-alpha|decimal|disc|circle|square|arabic-indic|bengali|devanagari|gujarati|gurmukhi|kannada|malayalam|oriya|persian|tamil|telugu|thai|urdu|cambodian|khmer|lao|cjk-decimal|hebrew)/i', $v, $m)) {
$newprop['LIST-STYLE-TYPE'] = strtolower(trim($m[1]));
} else if (preg_match('/U\+([a-fA-F0-9]+)/i', $v, $m)) {
$newprop['LIST-STYLE-TYPE'] = strtolower(trim($m[1]));
}
if (preg_match('/url\([\'\"]{0,1}(.*?)[\'\"]{0,1}\)/i', $v, $m)) {
$newprop['LIST-STYLE-IMAGE'] = strtolower(trim($m[1]));
}
if (preg_match('/(inside|outside)/i', $v, $m)) {
$newprop['LIST-STYLE-POSITION'] = strtolower(trim($m[1]));
}
} else {
$newprop[$k] = $v;
}
}
return $newprop;
}
function setCSSboxshadow($v)
{
$sh = array();
$c = preg_match_all('/(rgba|rgb|device-cmyka|cmyka|device-cmyk|cmyk|hsla|hsl)\(.*?\)/', $v, $x); // mPDF 5.6.05
for ($i = 0; $i < $c; $i++) {
$col = preg_replace('/,/', '*', $x[0][$i]);
$v = preg_replace('/' . preg_quote($x[0][$i], '/') . '/', $col, $v);
}
$ss = explode(',', $v);
foreach ($ss AS $s) {
$new = array('inset' => false, 'blur' => 0, 'spread' => 0);
if (preg_match('/inset/i', $s)) {
$new['inset'] = true;
$s = preg_replace('/\s*inset\s*/', '', $s);
}
$p = explode(' ', trim($s));
if (isset($p[0])) {
$new['x'] = $this->mpdf->ConvertSize(trim($p[0]), $this->mpdf->blk[$this->mpdf->blklvl - 1]['inner_width'], $this->mpdf->FontSize, false);
}
if (isset($p[1])) {
$new['y'] = $this->mpdf->ConvertSize(trim($p[1]), $this->mpdf->blk[$this->mpdf->blklvl - 1]['inner_width'], $this->mpdf->FontSize, false);
}
if (isset($p[2])) {
if (preg_match('/^\s*[\.\-0-9]/', $p[2])) {
$new['blur'] = $this->mpdf->ConvertSize(trim($p[2]), $this->mpdf->blk[$this->mpdf->blklvl - 1]['inner_width'], $this->mpdf->FontSize, false);
} else {
$new['col'] = $this->mpdf->ConvertColor(preg_replace('/\*/', ',', $p[2]));
}
if (isset($p[3])) {
if (preg_match('/^\s*[\.\-0-9]/', $p[3])) {
$new['spread'] = $this->mpdf->ConvertSize(trim($p[3]), $this->mpdf->blk[$this->mpdf->blklvl - 1]['inner_width'], $this->mpdf->FontSize, false);
} else {
$new['col'] = $this->mpdf->ConvertColor(preg_replace('/\*/', ',', $p[3]));
}
if (isset($p[4])) {
$new['col'] = $this->mpdf->ConvertColor(preg_replace('/\*/', ',', $p[4]));
}
}
}
if (!$new['col']) {
$new['col'] = $this->mpdf->ConvertColor('#888888');
}
if (isset($new['y'])) {
array_unshift($sh, $new);
}
}
return $sh;
}
function setCSStextshadow($v)
{
$sh = array();
$c = preg_match_all('/(rgba|rgb|device-cmyka|cmyka|device-cmyk|cmyk|hsla|hsl)\(.*?\)/', $v, $x); // mPDF 5.6.05
for ($i = 0; $i < $c; $i++) {
$col = preg_replace('/,/', '*', $x[0][$i]);
$v = preg_replace('/' . preg_quote($x[0][$i], '/') . '/', $col, $v);
}
$ss = explode(',', $v);
foreach ($ss AS $s) {
$new = array('blur' => 0);
$p = explode(' ', trim($s));
if (isset($p[0])) {
$new['x'] = $this->mpdf->ConvertSize(trim($p[0]), $this->mpdf->FontSize, $this->mpdf->FontSize, false);
}
if (isset($p[1])) {
$new['y'] = $this->mpdf->ConvertSize(trim($p[1]), $this->mpdf->FontSize, $this->mpdf->FontSize, false);
}
if (isset($p[2])) {
if (preg_match('/^\s*[\.\-0-9]/', $p[2])) {
$new['blur'] = $this->mpdf->ConvertSize(trim($p[2]), $this->mpdf->blk[$this->mpdf->blklvl]['inner_width'], $this->mpdf->FontSize, false);
} else {
$new['col'] = $this->mpdf->ConvertColor(preg_replace('/\*/', ',', $p[2]));
}
if (isset($p[3])) {
$new['col'] = $this->mpdf->ConvertColor(preg_replace('/\*/', ',', $p[3]));
}
}
if (!isset($new['col']) || !$new['col']) {
$new['col'] = $this->mpdf->ConvertColor('#888888');
}
if (isset($new['y'])) {
array_unshift($sh, $new);
}
}
return $sh;
}
function parseCSSbackground($s)
{
$bg = array('c' => false, 'i' => false, 'r' => false, 'p' => false,);
/* -- BACKGROUNDS -- */
if (preg_match('/(-moz-)*(repeating-)*(linear|radial)-gradient\(.*\)/i', $s, $m)) {
$bg['i'] = $m[0];
} else
/* -- END BACKGROUNDS -- */
if (preg_match('/url\(/i', $s)) {
// If color, set and strip it off
// mPDF 5.6.05
if (preg_match('/^\s*(#[0-9a-fA-F]{3,6}|(rgba|rgb|device-cmyka|cmyka|device-cmyk|cmyk|hsla|hsl|spot)\(.*?\)|[a-zA-Z]{3,})\s+(url\(.*)/i', $s, $m)) {
$bg['c'] = strtolower($m[1]);
$s = $m[3];
}
/* -- BACKGROUNDS -- */
if (preg_match('/url\([\'\"]{0,1}(.*?)[\'\"]{0,1}\)\s*(.*)/i', $s, $m)) {
$bg['i'] = $m[1];
$s = strtolower($m[2]);
if (preg_match('/(repeat-x|repeat-y|no-repeat|repeat)/', $s, $m)) {
$bg['r'] = $m[1];
}
// Remove repeat, attachment (discarded) and also any inherit
$s = preg_replace('/(repeat-x|repeat-y|no-repeat|repeat|scroll|fixed|inherit)/', '', $s);
$bits = preg_split('/\s+/', trim($s));
// These should be Position x1 or x2
if (count($bits) == 1) {
if (preg_match('/bottom/', $bits[0])) {
$bg['p'] = '50% 100%';
} else if (preg_match('/top/', $bits[0])) {
$bg['p'] = '50% 0%';
} else {
$bg['p'] = $bits[0] . ' 50%';
}
} else if (count($bits) == 2) {
// Can be either right center or center right
if (preg_match('/(top|bottom)/', $bits[0]) || preg_match('/(left|right)/', $bits[1])) {
$bg['p'] = $bits[1] . ' ' . $bits[0];
} else {
$bg['p'] = $bits[0] . ' ' . $bits[1];
}
}
if ($bg['p']) {
$bg['p'] = preg_replace('/(left|top)/', '0%', $bg['p']);
$bg['p'] = preg_replace('/(right|bottom)/', '100%', $bg['p']);
$bg['p'] = preg_replace('/(center)/', '50%', $bg['p']);
if (!preg_match('/[\-]{0,1}\d+(in|cm|mm|pt|pc|em|ex|px|%)* [\-]{0,1}\d+(in|cm|mm|pt|pc|em|ex|px|%)*/', $bg['p'])) {
$bg['p'] = false;
}
}
}
/* -- END BACKGROUNDS -- */
} else if (preg_match('/^\s*(#[0-9a-fA-F]{3,6}|(rgba|rgb|device-cmyka|cmyka|device-cmyk|cmyk|hsla|hsl|spot)\(.*?\)|[a-zA-Z]{3,})/i', $s, $m)) {
$bg['c'] = strtolower($m[1]);
} // mPDF 5.6.05
return ($bg);
}
function expand24($mp)
{
$prop = preg_split('/\s+/', trim($mp));
if (count($prop) == 1) {
return array('T' => $prop[0], 'R' => $prop[0], 'B' => $prop[0], 'L' => $prop[0]);
}
if (count($prop) == 2) {
return array('T' => $prop[0], 'R' => $prop[1], 'B' => $prop[0], 'L' => $prop[1]);
}
if (count($prop) == 3) {
return array('T' => $prop[0], 'R' => $prop[1], 'B' => $prop[2], 'L' => $prop[1]);
}
if (count($prop) == 4) {
return array('T' => $prop[0], 'R' => $prop[1], 'B' => $prop[2], 'L' => $prop[3]);
}
return array();
}
/* -- BORDER-RADIUS -- */
function border_radius_expand($val, $k)
{
$b = array();
if ($k == 'BORDER-RADIUS') {
$hv = explode('/', trim($val));
$prop = preg_split('/\s+/', trim($hv[0]));
if (count($prop) == 1) {
$b['TL-H'] = $b['TR-H'] = $b['BR-H'] = $b['BL-H'] = $prop[0];
} else if (count($prop) == 2) {
$b['TL-H'] = $b['BR-H'] = $prop[0];
$b['TR-H'] = $b['BL-H'] = $prop[1];
} else if (count($prop) == 3) {
$b['TL-H'] = $prop[0];
$b['TR-H'] = $b['BL-H'] = $prop[1];
$b['BR-H'] = $prop[2];
} else if (count($prop) == 4) {
$b['TL-H'] = $prop[0];
$b['TR-H'] = $prop[1];
$b['BR-H'] = $prop[2];
$b['BL-H'] = $prop[3];
}
if (count($hv) == 2) {
$prop = preg_split('/\s+/', trim($hv[1]));
if (count($prop) == 1) {
$b['TL-V'] = $b['TR-V'] = $b['BR-V'] = $b['BL-V'] = $prop[0];
} else if (count($prop) == 2) {
$b['TL-V'] = $b['BR-V'] = $prop[0];
$b['TR-V'] = $b['BL-V'] = $prop[1];
} else if (count($prop) == 3) {
$b['TL-V'] = $prop[0];
$b['TR-V'] = $b['BL-V'] = $prop[1];
$b['BR-V'] = $prop[2];
} else if (count($prop) == 4) {
$b['TL-V'] = $prop[0];
$b['TR-V'] = $prop[1];
$b['BR-V'] = $prop[2];
$b['BL-V'] = $prop[3];
}
} else {
$b['TL-V'] = $b['TL-H'];
$b['TR-V'] = $b['TR-H'];
$b['BL-V'] = $b['BL-H'];
$b['BR-V'] = $b['BR-H'];
}
return $b;
}
// Parse 2
$h = 0;
$v = 0;
$prop = preg_split('/\s+/', trim($val));
if (count($prop) == 1) {
$h = $v = $val;
} else {
$h = $prop[0];
$v = $prop[1];
}
if ($h == 0 || $v == 0) {
$h = $v = 0;
}
if ($k == 'BORDER-TOP-LEFT-RADIUS') {
$b['TL-H'] = $h;
$b['TL-V'] = $v;
} else if ($k == 'BORDER-TOP-RIGHT-RADIUS') {
$b['TR-H'] = $h;
$b['TR-V'] = $v;
} else if ($k == 'BORDER-BOTTOM-LEFT-RADIUS') {
$b['BL-H'] = $h;
$b['BL-V'] = $v;
} else if ($k == 'BORDER-BOTTOM-RIGHT-RADIUS') {
$b['BR-H'] = $h;
$b['BR-V'] = $v;
}
return $b;
}
/* -- END BORDER-RADIUS -- */
function _mergeCSS($p, &$t)
{
// Save Cascading CSS e.g. "div.topic p" at this block level
if (isset($p) && $p) {
if ($t) {
$t = $this->array_merge_recursive_unique($t, $p);
} else {
$t = $p;
}
}
}
// for CSS handling
function array_merge_recursive_unique($array1, $array2)
{
$arrays = func_get_args();
$narrays = count($arrays);
$ret = $arrays[0];
for ($i = 1; $i < $narrays; $i ++) {
foreach ($arrays[$i] as $key => $value) {
if (((string) $key) === ((string) intval($key))) { // integer or string as integer key - append
$ret[] = $value;
} else { // string key - merge
if (is_array($value) && isset($ret[$key])) {
$ret[$key] = $this->array_merge_recursive_unique($ret[$key], $value);
} else {
$ret[$key] = $value;
}
}
}
}
return $ret;
}
function _mergeFullCSS($p, &$t, $tag, $classes, $id, $lang)
{ // mPDF 6
if (isset($p[$tag])) {
$this->_mergeCSS($p[$tag], $t);
}
// STYLESHEET CLASS e.g. .smallone{} .redletter{}
foreach ($classes AS $class) {
if (isset($p['CLASS>>' . $class])) {
$this->_mergeCSS($p['CLASS>>' . $class], $t);
}
}
// STYLESHEET nth-child SELECTOR e.g. tr:nth-child(odd) td:nth-child(2n+1)
if ($tag == 'TR' && isset($p) && $p) {
foreach ($p AS $k => $val) {
if (preg_match('/' . $tag . '>>SELECTORNTHCHILD>>(.*)/', $k, $m)) {
$select = false;
if ($tag == 'TR') {
$row = $this->mpdf->row;
$thnr = (isset($this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['is_thead']) ? count($this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['is_thead']) : 0);
$tfnr = (isset($this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['is_tfoot']) ? count($this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['is_tfoot']) : 0);
if ($this->mpdf->tabletfoot) {
$row -= $thnr;
} else if (!$this->mpdf->tablethead) {
$row -= ($thnr + $tfnr);
}
if (preg_match('/(([\-+]?\d*)?N([\-+]\d+)?|[\-+]?\d+|ODD|EVEN)/', $m[1], $a)) { // mPDF 5.7.4
$select = $this->_nthchild($a, $row);
}
} else if ($tag == 'TD' || $tag == 'TH') {
if (preg_match('/(([\-+]?\d*)?N([\-+]\d+)?|[\-+]?\d+|ODD|EVEN)/', $m[1], $a)) { // mPDF 5.7.4
$select = $this->_nthchild($a, $this->mpdf->col);
}
}
if ($select) {
$this->_mergeCSS($p[$tag . '>>SELECTORNTHCHILD>>' . $m[1]], $t);
}
}
}
}
// STYLESHEET CLASS e.g. [lang=fr]{} or :lang(fr)
if (isset($lang) && isset($p['LANG>>' . $lang])) {
$this->_mergeCSS($p['LANG>>' . $lang], $t);
}
// STYLESHEET CLASS e.g. #smallone{} #redletter{}
if (isset($id) && isset($p['ID>>' . $id])) {
$this->_mergeCSS($p['ID>>' . $id], $t);
}
// STYLESHEET CLASS e.g. .smallone{} .redletter{}
foreach ($classes AS $class) {
if (isset($p[$tag . '>>CLASS>>' . $class])) {
$this->_mergeCSS($p[$tag . '>>CLASS>>' . $class], $t);
}
}
// STYLESHEET CLASS e.g. [lang=fr]{} or :lang(fr)
if (isset($lang) && isset($p[$tag . '>>LANG>>' . $lang])) {
$this->_mergeCSS($p[$tag . '>>LANG>>' . $lang], $t);
}
// STYLESHEET CLASS e.g. #smallone{} #redletter{}
if (isset($id) && isset($p[$tag . '>>ID>>' . $id])) {
$this->_mergeCSS($p[$tag . '>>ID>>' . $id], $t);
}
}
function setBorderDominance($prop, $val)
{
if (isset($prop['BORDER-LEFT']) && $prop['BORDER-LEFT']) {
$this->cell_border_dominance_L = $val;
}
if (isset($prop['BORDER-RIGHT']) && $prop['BORDER-RIGHT']) {
$this->cell_border_dominance_R = $val;
}
if (isset($prop['BORDER-TOP']) && $prop['BORDER-TOP']) {
$this->cell_border_dominance_T = $val;
}
if (isset($prop['BORDER-BOTTOM']) && $prop['BORDER-BOTTOM']) {
$this->cell_border_dominance_B = $val;
}
}
function _set_mergedCSS(&$m, &$p, $d = true, $bd = false)
{
if (isset($m)) {
if ((isset($m['depth']) && $m['depth'] > 1) || $d == false) { // include check for 'depth'
if ($bd) {
$this->setBorderDominance($m, $bd);
} // *TABLES*
if (is_array($m)) {
$p = array_merge($p, $m);
$this->_mergeBorders($p, $m);
}
}
}
}
function _mergeBorders(&$b, &$a)
{ // Merges $a['BORDER-TOP-STYLE'] to $b['BORDER-TOP'] etc.
foreach (array('TOP', 'RIGHT', 'BOTTOM', 'LEFT') AS $side) {
foreach (array('STYLE', 'WIDTH', 'COLOR') AS $el) {
if (isset($a['BORDER-' . $side . '-' . $el])) { // e.g. $b['BORDER-TOP-STYLE']
$s = trim($a['BORDER-' . $side . '-' . $el]);
if (isset($b['BORDER-' . $side])) { // e.g. $b['BORDER-TOP']
$p = trim($b['BORDER-' . $side]);
} else {
$p = '';
}
if ($el == 'STYLE') {
if ($p) {
$b['BORDER-' . $side] = preg_replace('/(\S+)\s+(\S+)\s+(\S+)/', '\\1 ' . $s . ' \\3', $p);
} else {
$b['BORDER-' . $side] = '0px ' . $s . ' #000000';
}
} else if ($el == 'WIDTH') {
if ($p) {
$b['BORDER-' . $side] = preg_replace('/(\S+)\s+(\S+)\s+(\S+)/', $s . ' \\2 \\3', $p);
} else {
$b['BORDER-' . $side] = $s . ' none #000000';
}
} else if ($el == 'COLOR') {
if ($p) {
$b['BORDER-' . $side] = preg_replace('/(\S+)\s+(\S+)\s+(\S+)/', '\\1 \\2 ' . $s, $p);
} else {
$b['BORDER-' . $side] = '0px none ' . $s;
}
}
}
}
}
}
function MergeCSS($inherit, $tag, $attr)
{
$p = array();
$zp = array();
if (empty($attr)) $attr = array();
$classes = array();
if (isset($attr['CLASS'])) {
$classes = preg_split('/\s+/', $attr['CLASS']);
}
if (!isset($attr['ID'])) {
$attr['ID'] = '';
}
// mPDF 6
$shortlang = '';
if (!isset($attr['LANG'])) {
$attr['LANG'] = '';
} else {
$attr['LANG'] = strtolower($attr['LANG']);
if (strlen($attr['LANG']) == 5) {
$shortlang = substr($attr['LANG'], 0, 2);
}
}
//===============================================
/* -- TABLES -- */
// Set Inherited properties
if ($inherit == 'TOPTABLE') { // $tag = TABLE
//===============================================
// Save Cascading CSS e.g. "div.topic p" at this block level
if (isset($this->mpdf->blk[$this->mpdf->blklvl]['cascadeCSS'])) {
$this->tablecascadeCSS[0] = $this->mpdf->blk[$this->mpdf->blklvl]['cascadeCSS'];
} else {
$this->tablecascadeCSS[0] = $this->cascadeCSS;
}
}
//===============================================
// Set Inherited properties
if ($inherit == 'TOPTABLE' || $inherit == 'TABLE') {
//Cascade everything from last level that is not an actual property, or defined by current tag/attributes
if (isset($this->tablecascadeCSS[$this->tbCSSlvl - 1]) && is_array($this->tablecascadeCSS[$this->tbCSSlvl - 1])) {
foreach ($this->tablecascadeCSS[$this->tbCSSlvl - 1] AS $k => $v) {
$this->tablecascadeCSS[$this->tbCSSlvl][$k] = $v;
}
}
$this->_mergeFullCSS($this->cascadeCSS, $this->tablecascadeCSS[$this->tbCSSlvl], $tag, $classes, $attr['ID'], $attr['LANG']);
//===============================================
// Cascading forward CSS e.g. "table.topic td" for this table in $this->tablecascadeCSS
//===============================================
// STYLESHEET TAG e.g. table
$this->_mergeFullCSS($this->tablecascadeCSS[$this->tbCSSlvl - 1], $this->tablecascadeCSS[$this->tbCSSlvl], $tag, $classes, $attr['ID'], $attr['LANG']);
//===============================================
}
/* -- END TABLES -- */
//===============================================
// Set Inherited properties
if ($inherit == 'BLOCK') {
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1]['cascadeCSS']) && is_array($this->mpdf->blk[$this->mpdf->blklvl - 1]['cascadeCSS'])) {
foreach ($this->mpdf->blk[$this->mpdf->blklvl - 1]['cascadeCSS'] AS $k => $v) {
$this->mpdf->blk[$this->mpdf->blklvl]['cascadeCSS'][$k] = $v;
}
}
//===============================================
// Save Cascading CSS e.g. "div.topic p" at this block level
$this->_mergeFullCSS($this->cascadeCSS, $this->mpdf->blk[$this->mpdf->blklvl]['cascadeCSS'], $tag, $classes, $attr['ID'], $attr['LANG']);
//===============================================
// Cascading forward CSS
//===============================================
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1])) {
$this->_mergeFullCSS($this->mpdf->blk[$this->mpdf->blklvl - 1]['cascadeCSS'], $this->mpdf->blk[$this->mpdf->blklvl]['cascadeCSS'], $tag, $classes, $attr['ID'], $attr['LANG']);
}
//===============================================
// Block properties which are inherited
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1]['margin_collapse']) && $this->mpdf->blk[$this->mpdf->blklvl - 1]['margin_collapse']) {
$p['MARGIN-COLLAPSE'] = 'COLLAPSE';
} // custom tag, but follows CSS principle that border-collapse is inherited
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1]['line_height']) && $this->mpdf->blk[$this->mpdf->blklvl - 1]['line_height']) {
$p['LINE-HEIGHT'] = $this->mpdf->blk[$this->mpdf->blklvl - 1]['line_height'];
}
// mPDF 6
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1]['line_stacking_strategy']) && $this->mpdf->blk[$this->mpdf->blklvl - 1]['line_stacking_strategy']) {
$p['LINE-STACKING-STRATEGY'] = $this->mpdf->blk[$this->mpdf->blklvl - 1]['line_stacking_strategy'];
}
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1]['line_stacking_shift']) && $this->mpdf->blk[$this->mpdf->blklvl - 1]['line_stacking_shift']) {
$p['LINE-STACKING-SHIFT'] = $this->mpdf->blk[$this->mpdf->blklvl - 1]['line_stacking_shift'];
}
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1]['direction']) && $this->mpdf->blk[$this->mpdf->blklvl - 1]['direction']) {
$p['DIRECTION'] = $this->mpdf->blk[$this->mpdf->blklvl - 1]['direction'];
}
// mPDF 6 Lists
if ($tag == 'LI') {
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1]['list_style_type']) && $this->mpdf->blk[$this->mpdf->blklvl - 1]['list_style_type']) {
$p['LIST-STYLE-TYPE'] = $this->mpdf->blk[$this->mpdf->blklvl - 1]['list_style_type'];
}
}
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1]['list_style_image']) && $this->mpdf->blk[$this->mpdf->blklvl - 1]['list_style_image']) {
$p['LIST-STYLE-IMAGE'] = $this->mpdf->blk[$this->mpdf->blklvl - 1]['list_style_image'];
}
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1]['list_style_position']) && $this->mpdf->blk[$this->mpdf->blklvl - 1]['list_style_position']) {
$p['LIST-STYLE-POSITION'] = $this->mpdf->blk[$this->mpdf->blklvl - 1]['list_style_position'];
}
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1]['align']) && $this->mpdf->blk[$this->mpdf->blklvl - 1]['align']) {
if ($this->mpdf->blk[$this->mpdf->blklvl - 1]['align'] == 'L') {
$p['TEXT-ALIGN'] = 'left';
} else if ($this->mpdf->blk[$this->mpdf->blklvl - 1]['align'] == 'J') {
$p['TEXT-ALIGN'] = 'justify';
} else if ($this->mpdf->blk[$this->mpdf->blklvl - 1]['align'] == 'R') {
$p['TEXT-ALIGN'] = 'right';
} else if ($this->mpdf->blk[$this->mpdf->blklvl - 1]['align'] == 'C') {
$p['TEXT-ALIGN'] = 'center';
}
}
if ($this->mpdf->ColActive || $this->mpdf->keep_block_together) {
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1]['bgcolor']) && $this->mpdf->blk[$this->mpdf->blklvl - 1]['bgcolor']) { // Doesn't officially inherit, but default value is transparent (?=inherited)
$cor = $this->mpdf->blk[$this->mpdf->blklvl - 1]['bgcolorarray'];
$p['BACKGROUND-COLOR'] = $this->mpdf->_colAtoString($cor);
}
}
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1]['text_indent']) && ($this->mpdf->blk[$this->mpdf->blklvl - 1]['text_indent'] || $this->mpdf->blk[$this->mpdf->blklvl - 1]['text_indent'] === 0)) {
$p['TEXT-INDENT'] = $this->mpdf->blk[$this->mpdf->blklvl - 1]['text_indent'];
}
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1]['InlineProperties'])) {
$biilp = $this->mpdf->blk[$this->mpdf->blklvl - 1]['InlineProperties'];
$this->inlinePropsToCSS($biilp, $p); // mPDF 5.7.1
} else {
$biilp = null;
}
}
//===============================================
//===============================================
// INLINE HTML ATTRIBUTES e.g. .. ALIGN="CENTER">
// mPDF 6 (added)
if (isset($attr['DIR']) and $attr['DIR'] != '') {
$p['DIRECTION'] = $attr['DIR'];
}
// mPDF 6 (moved)
if (isset($attr['LANG']) and $attr['LANG'] != '') {
$p['LANG'] = $attr['LANG'];
}
if (isset($attr['COLOR']) and $attr['COLOR'] != '') {
$p['COLOR'] = $attr['COLOR'];
}
if ($tag != 'INPUT') {
if (isset($attr['WIDTH']) and $attr['WIDTH'] != '') {
$p['WIDTH'] = $attr['WIDTH'];
}
if (isset($attr['HEIGHT']) and $attr['HEIGHT'] != '') {
$p['HEIGHT'] = $attr['HEIGHT'];
}
}
if ($tag == 'FONT') {
if (isset($attr['FACE'])) {
$p['FONT-FAMILY'] = $attr['FACE'];
}
if (isset($attr['SIZE']) and $attr['SIZE'] != '') {
$s = '';
if ($attr['SIZE'] === '+1') {
$s = '120%';
} else if ($attr['SIZE'] === '-1') {
$s = '86%';
} else if ($attr['SIZE'] === '1') {
$s = 'XX-SMALL';
} else if ($attr['SIZE'] == '2') {
$s = 'X-SMALL';
} else if ($attr['SIZE'] == '3') {
$s = 'SMALL';
} else if ($attr['SIZE'] == '4') {
$s = 'MEDIUM';
} else if ($attr['SIZE'] == '5') {
$s = 'LARGE';
} else if ($attr['SIZE'] == '6') {
$s = 'X-LARGE';
} else if ($attr['SIZE'] == '7') {
$s = 'XX-LARGE';
}
if ($s)
$p['FONT-SIZE'] = $s;
}
}
if (isset($attr['VALIGN']) and $attr['VALIGN'] != '') {
$p['VERTICAL-ALIGN'] = $attr['VALIGN'];
}
if (isset($attr['VSPACE']) and $attr['VSPACE'] != '') {
$p['MARGIN-TOP'] = $attr['VSPACE'];
$p['MARGIN-BOTTOM'] = $attr['VSPACE'];
}
if (isset($attr['HSPACE']) and $attr['HSPACE'] != '') {
$p['MARGIN-LEFT'] = $attr['HSPACE'];
$p['MARGIN-RIGHT'] = $attr['HSPACE'];
}
//===============================================
//===============================================
// DEFAULT for this TAG set in DefaultCSS
if (isset($this->mpdf->defaultCSS[$tag])) {
$zp = $this->fixCSS($this->mpdf->defaultCSS[$tag]);
if (is_array($zp)) { // Default overwrites Inherited
$p = array_merge($p, $zp); // !! Note other way round !!
$this->_mergeBorders($p, $zp);
}
}
//===============================================
/* -- TABLES -- */
// mPDF 5.7.3
// cellSpacing overwrites TABLE default but not specific CSS set on table
if ($tag == 'TABLE' && isset($attr['CELLSPACING'])) {
$p['BORDER-SPACING-H'] = $p['BORDER-SPACING-V'] = $attr['CELLSPACING'];
}
// cellPadding overwrites TD/TH default but not specific CSS set on cell
if (($tag == 'TD' || $tag == 'TH') && isset($this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['cell_padding']) && ($this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['cell_padding'] || $this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['cell_padding'] === '0')) { // mPDF 5.7.3
$p['PADDING-LEFT'] = $this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['cell_padding'];
$p['PADDING-RIGHT'] = $this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['cell_padding'];
$p['PADDING-TOP'] = $this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['cell_padding'];
$p['PADDING-BOTTOM'] = $this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['cell_padding'];
}
/* -- END TABLES -- */
//===============================================
// STYLESHEET TAG e.g. h1 p div table
if (isset($this->CSS[$tag]) && $this->CSS[$tag]) {
$zp = $this->CSS[$tag];
if ($tag == 'TD' || $tag == 'TH') {
$this->setBorderDominance($zp, 9);
} // *TABLES* // *TABLES-ADVANCED-BORDERS*
if (is_array($zp)) {
$p = array_merge($p, $zp);
$this->_mergeBorders($p, $zp);
}
}
//===============================================
// STYLESHEET CLASS e.g. .smallone{} .redletter{}
foreach ($classes AS $class) {
$zp = array();
if (isset($this->CSS['CLASS>>' . $class]) && $this->CSS['CLASS>>' . $class]) {
$zp = $this->CSS['CLASS>>' . $class];
}
if ($tag == 'TD' || $tag == 'TH') {
$this->setBorderDominance($zp, 9);
} // *TABLES* // *TABLES-ADVANCED-BORDERS*
if (is_array($zp)) {
$p = array_merge($p, $zp);
$this->_mergeBorders($p, $zp);
}
}
//===============================================
/* -- TABLES -- */
// STYLESHEET nth-child SELECTOR e.g. tr:nth-child(odd) td:nth-child(2n+1)
if ($tag == 'TR' || $tag == 'TD' || $tag == 'TH') {
foreach ($this->CSS AS $k => $val) {
if (preg_match('/' . $tag . '>>SELECTORNTHCHILD>>(.*)/', $k, $m)) {
$select = false;
if ($tag == 'TR') {
$row = $this->mpdf->row;
$thnr = (isset($this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['is_thead']) ? count($this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['is_thead']) : 0);
$tfnr = (isset($this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['is_tfoot']) ? count($this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['is_tfoot']) : 0);
if ($this->mpdf->tabletfoot) {
$row -= $thnr;
} else if (!$this->mpdf->tablethead) {
$row -= ($thnr + $tfnr);
}
if (preg_match('/(([\-+]?\d*)?N([\-+]\d+)?|[\-+]?\d+|ODD|EVEN)/', $m[1], $a)) { // mPDF 5.7.4
$select = $this->_nthchild($a, $row);
}
} else if ($tag == 'TD' || $tag == 'TH') {
if (preg_match('/(([\-+]?\d*)?N([\-+]\d+)?|[\-+]?\d+|ODD|EVEN)/', $m[1], $a)) { // mPDF 5.7.4
$select = $this->_nthchild($a, $this->mpdf->col);
}
}
if ($select) {
$zp = $this->CSS[$tag . '>>SELECTORNTHCHILD>>' . $m[1]];
if ($tag == 'TD' || $tag == 'TH') {
$this->setBorderDominance($zp, 9);
}
if (is_array($zp)) {
$p = array_merge($p, $zp);
$this->_mergeBorders($p, $zp);
}
}
}
}
}
/* -- END TABLES -- */
//===============================================
// STYLESHEET LANG e.g. [lang=fr]{} or :lang(fr)
if (isset($attr['LANG'])) {
if (isset($this->CSS['LANG>>' . $attr['LANG']]) && $this->CSS['LANG>>' . $attr['LANG']]) {
$zp = $this->CSS['LANG>>' . $attr['LANG']];
if ($tag == 'TD' || $tag == 'TH') {
$this->setBorderDominance($zp, 9);
} // *TABLES* // *TABLES-ADVANCED-BORDERS*
if (is_array($zp)) {
$p = array_merge($p, $zp);
$this->_mergeBorders($p, $zp);
}
} else if (isset($this->CSS['LANG>>' . $shortlang]) && $this->CSS['LANG>>' . $shortlang]) {
$zp = $this->CSS['LANG>>' . $shortlang];
if ($tag == 'TD' || $tag == 'TH') {
$this->setBorderDominance($zp, 9);
} // *TABLES* // *TABLES-ADVANCED-BORDERS*
if (is_array($zp)) {
$p = array_merge($p, $zp);
$this->_mergeBorders($p, $zp);
}
}
}
//===============================================
// STYLESHEET ID e.g. #smallone{} #redletter{}
if (isset($attr['ID']) && isset($this->CSS['ID>>' . $attr['ID']]) && $this->CSS['ID>>' . $attr['ID']]) {
$zp = $this->CSS['ID>>' . $attr['ID']];
if ($tag == 'TD' || $tag == 'TH') {
$this->setBorderDominance($zp, 9);
} // *TABLES* // *TABLES-ADVANCED-BORDERS*
if (is_array($zp)) {
$p = array_merge($p, $zp);
$this->_mergeBorders($p, $zp);
}
}
//===============================================
// STYLESHEET CLASS e.g. p.smallone{} div.redletter{}
foreach ($classes AS $class) {
$zp = array();
if (isset($this->CSS[$tag . '>>CLASS>>' . $class]) && $this->CSS[$tag . '>>CLASS>>' . $class]) {
$zp = $this->CSS[$tag . '>>CLASS>>' . $class];
}
if ($tag == 'TD' || $tag == 'TH') {
$this->setBorderDominance($zp, 9);
} // *TABLES* // *TABLES-ADVANCED-BORDERS*
if (is_array($zp)) {
$p = array_merge($p, $zp);
$this->_mergeBorders($p, $zp);
}
}
//===============================================
// STYLESHEET LANG e.g. [lang=fr]{} or :lang(fr)
if (isset($attr['LANG'])) {
if (isset($this->CSS[$tag . '>>LANG>>' . $attr['LANG']]) && $this->CSS[$tag . '>>LANG>>' . $attr['LANG']]) {
$zp = $this->CSS[$tag . '>>LANG>>' . $attr['LANG']];
if ($tag == 'TD' || $tag == 'TH') {
$this->setBorderDominance($zp, 9);
} // *TABLES* // *TABLES-ADVANCED-BORDERS*
if (is_array($zp)) {
$p = array_merge($p, $zp);
$this->_mergeBorders($p, $zp);
}
} else if (isset($this->CSS[$tag . '>>LANG>>' . $shortlang]) && $this->CSS[$tag . '>>LANG>>' . $shortlang]) {
$zp = $this->CSS[$tag . '>>LANG>>' . $shortlang];
if ($tag == 'TD' || $tag == 'TH') {
$this->setBorderDominance($zp, 9);
} // *TABLES* // *TABLES-ADVANCED-BORDERS*
if (is_array($zp)) {
$p = array_merge($p, $zp);
$this->_mergeBorders($p, $zp);
}
}
}
//===============================================
// STYLESHEET CLASS e.g. p#smallone{} div#redletter{}
if (isset($attr['ID']) && isset($this->CSS[$tag . '>>ID>>' . $attr['ID']]) && $this->CSS[$tag . '>>ID>>' . $attr['ID']]) {
$zp = $this->CSS[$tag . '>>ID>>' . $attr['ID']];
if ($tag == 'TD' || $tag == 'TH') {
$this->setBorderDominance($zp, 9);
} // *TABLES* // *TABLES-ADVANCED-BORDERS*
if (is_array($zp)) {
$p = array_merge($p, $zp);
$this->_mergeBorders($p, $zp);
}
}
//===============================================
// Cascaded e.g. div.class p only works for block level
if ($inherit == 'BLOCK') {
if (isset($this->mpdf->blk[$this->mpdf->blklvl - 1])) { // mPDF 6
$this->_set_mergedCSS($this->mpdf->blk[$this->mpdf->blklvl - 1]['cascadeCSS'][$tag], $p);
foreach ($classes AS $class) {
$this->_set_mergedCSS($this->mpdf->blk[$this->mpdf->blklvl - 1]['cascadeCSS']['CLASS>>' . $class], $p);
}
$this->_set_mergedCSS($this->mpdf->blk[$this->mpdf->blklvl - 1]['cascadeCSS']['ID>>' . $attr['ID']], $p);
foreach ($classes AS $class) {
$this->_set_mergedCSS($this->mpdf->blk[$this->mpdf->blklvl - 1]['cascadeCSS'][$tag . '>>CLASS>>' . $class], $p);
}
$this->_set_mergedCSS($this->mpdf->blk[$this->mpdf->blklvl - 1]['cascadeCSS'][$tag . '>>ID>>' . $attr['ID']], $p);
}
} else if ($inherit == 'INLINE') {
$this->_set_mergedCSS($this->mpdf->blk[$this->mpdf->blklvl]['cascadeCSS'][$tag], $p);
foreach ($classes AS $class) {
$this->_set_mergedCSS($this->mpdf->blk[$this->mpdf->blklvl]['cascadeCSS']['CLASS>>' . $class], $p);
}
$this->_set_mergedCSS($this->mpdf->blk[$this->mpdf->blklvl]['cascadeCSS']['ID>>' . $attr['ID']], $p);
foreach ($classes AS $class) {
$this->_set_mergedCSS($this->mpdf->blk[$this->mpdf->blklvl]['cascadeCSS'][$tag . '>>CLASS>>' . $class], $p);
}
$this->_set_mergedCSS($this->mpdf->blk[$this->mpdf->blklvl]['cascadeCSS'][$tag . '>>ID>>' . $attr['ID']], $p);
}
/* -- TABLES -- */ else if ($inherit == 'TOPTABLE' || $inherit == 'TABLE') { // NB looks at $this->tablecascadeCSS-1 for cascading CSS
if (isset($this->tablecascadeCSS[$this->tbCSSlvl - 1])) { // mPDF 6
// false, 9 = don't check for 'depth' and do set border dominance
$this->_set_mergedCSS($this->tablecascadeCSS[$this->tbCSSlvl - 1][$tag], $p, false, 9);
foreach ($classes AS $class) {
$this->_set_mergedCSS($this->tablecascadeCSS[$this->tbCSSlvl - 1]['CLASS>>' . $class], $p, false, 9);
}
// STYLESHEET nth-child SELECTOR e.g. tr:nth-child(odd) td:nth-child(2n+1)
if ($tag == 'TR' || $tag == 'TD' || $tag == 'TH') {
foreach ($this->tablecascadeCSS[$this->tbCSSlvl - 1] AS $k => $val) {
if (preg_match('/' . $tag . '>>SELECTORNTHCHILD>>(.*)/', $k, $m)) {
$select = false;
if ($tag == 'TR') {
$row = $this->mpdf->row;
$thnr = (isset($this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['is_thead']) ? count($this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['is_thead']) : 0);
$tfnr = (isset($this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['is_tfoot']) ? count($this->mpdf->table[$this->mpdf->tableLevel][$this->mpdf->tbctr[$this->mpdf->tableLevel]]['is_tfoot']) : 0);
if ($this->mpdf->tabletfoot) {
$row -= $thnr;
} else if (!$this->mpdf->tablethead) {
$row -= ($thnr + $tfnr);
}
if (preg_match('/(([\-+]?\d*)?N([\-+]\d+)?|[\-+]?\d+|ODD|EVEN)/', $m[1], $a)) { // mPDF 5.7.4
$select = $this->_nthchild($a, $row);
}
} else if ($tag == 'TD' || $tag == 'TH') {
if (preg_match('/(([\-+]?\d*)?N([\-+]\d+)?|[\-+]?\d+|ODD|EVEN)/', $m[1], $a)) { // mPDF 5.7.4
$select = $this->_nthchild($a, $this->mpdf->col);
}
}
if ($select) {
$this->_set_mergedCSS($this->tablecascadeCSS[$this->tbCSSlvl - 1][$tag . '>>SELECTORNTHCHILD>>' . $m[1]], $p, false, 9);
}
}
}
}
}
$this->_set_mergedCSS($this->tablecascadeCSS[$this->tbCSSlvl - 1]['ID>>' . $attr['ID']], $p, false, 9);
foreach ($classes AS $class) {
$this->_set_mergedCSS($this->tablecascadeCSS[$this->tbCSSlvl - 1][$tag . '>>CLASS>>' . $class], $p, false, 9);
}
$this->_set_mergedCSS($this->tablecascadeCSS[$this->tbCSSlvl - 1][$tag . '>>ID>>' . $attr['ID']], $p, false, 9);
}
/* -- END TABLES -- */
//===============================================
//===============================================
// INLINE STYLE e.g. style="CSS:property"
if (isset($attr['STYLE'])) {
$zp = $this->readInlineCSS($attr['STYLE']);
if ($tag == 'TD' || $tag == 'TH') {
$this->setBorderDominance($zp, 9);
} // *TABLES* // *TABLES-ADVANCED-BORDERS*
if (is_array($zp)) {
$p = array_merge($p, $zp);
$this->_mergeBorders($p, $zp);
}
}
//===============================================
//===============================================
return $p;
}
// Convert inline Properties back to CSS
function inlinePropsToCSS($bilp, &$p)
{
if (isset($bilp['family']) && $bilp['family']) {
$p['FONT-FAMILY'] = $bilp['family'];
}
if (isset($bilp['I']) && $bilp['I']) {
$p['FONT-STYLE'] = 'italic';
}
if (isset($bilp['sizePt']) && $bilp['sizePt']) {
$p['FONT-SIZE'] = $bilp['sizePt'] . 'pt';
}
if (isset($bilp['B']) && $bilp['B']) {
$p['FONT-WEIGHT'] = 'bold';
}
if (isset($bilp['colorarray']) && $bilp['colorarray']) {
$cor = $bilp['colorarray'];
$p['COLOR'] = $this->mpdf->_colAtoString($cor);
}
if (isset($bilp['lSpacingCSS']) && $bilp['lSpacingCSS']) {
$p['LETTER-SPACING'] = $bilp['lSpacingCSS'];
}
if (isset($bilp['wSpacingCSS']) && $bilp['wSpacingCSS']) {
$p['WORD-SPACING'] = $bilp['wSpacingCSS'];
}
if (isset($bilp['textparam']) && $bilp['textparam']) {
if (isset($bilp['textparam']['hyphens'])) {
if ($bilp['textparam']['hyphens'] == 2) {
$p['HYPHENS'] = 'none';
}
if ($bilp['textparam']['hyphens'] == 1) {
$p['HYPHENS'] = 'auto';
}
if ($bilp['textparam']['hyphens'] == 0) {
$p['HYPHENS'] = 'manual';
}
}
if (isset($bilp['textparam']['outline-s']) && !$bilp['textparam']['outline-s']) {
$p['TEXT-OUTLINE'] = 'none';
}
if (isset($bilp['textparam']['outline-COLOR']) && $bilp['textparam']['outline-COLOR']) {
$p['TEXT-OUTLINE-COLOR'] = $this->mpdf->_colAtoString($bilp['textparam']['outline-COLOR']);
}
if (isset($bilp['textparam']['outline-WIDTH']) && $bilp['textparam']['outline-WIDTH']) {
$p['TEXT-OUTLINE-WIDTH'] = $bilp['textparam']['outline-WIDTH'] . 'mm';
}
}
if (isset($bilp['textvar']) && $bilp['textvar']) {
// CSS says text-decoration is not inherited, but IE7 does??
if ($bilp['textvar'] & FD_LINETHROUGH) {
if ($bilp['textvar'] & FD_UNDERLINE) {
$p['TEXT-DECORATION'] = 'underline line-through';
} else {
$p['TEXT-DECORATION'] = 'line-through';
}
} else if ($bilp['textvar'] & FD_UNDERLINE) {
$p['TEXT-DECORATION'] = 'underline';
} else {
$p['TEXT-DECORATION'] = 'none';
}
if ($bilp['textvar'] & FA_SUPERSCRIPT) {
$p['VERTICAL-ALIGN'] = 'super';
} else if ($bilp['textvar'] & FA_SUBSCRIPT) {
$p['VERTICAL-ALIGN'] = 'sub';
} else {
$p['VERTICAL-ALIGN'] = 'baseline';
}
if ($bilp['textvar'] & FT_CAPITALIZE) {
$p['TEXT-TRANSFORM'] = 'capitalize';
} else if ($bilp['textvar'] & FT_UPPERCASE) {
$p['TEXT-TRANSFORM'] = 'uppercase';
} else if ($bilp['textvar'] & FT_LOWERCASE) {
$p['TEXT-TRANSFORM'] = 'lowercase';
} else {
$p['TEXT-TRANSFORM'] = 'none';
}
if ($bilp['textvar'] & FC_KERNING) {
$p['FONT-KERNING'] = 'normal';
} // ignore 'auto' as default already applied
//if (isset($bilp[ 'OTLtags' ]) && $bilp[ 'OTLtags' ]['Plus'] contains 'kern'
else {
$p['FONT-KERNING'] = 'none';
}
if ($bilp['textvar'] & FA_SUPERSCRIPT) {
$p['FONT-VARIANT-POSITION'] = 'super';
}
//if (isset($bilp[ 'OTLtags' ]) && $bilp[ 'OTLtags' ]['Plus'] contains 'sups' / 'subs'
else if ($bilp['textvar'] & FA_SUBSCRIPT) {
$p['FONT-VARIANT-POSITION'] = 'sub';
} else {
$p['FONT-VARIANT-POSITION'] = 'normal';
}
if ($bilp['textvar'] & FC_SMALLCAPS) {
$p['FONT-VARIANT-CAPS'] = 'small-caps';
}
}
if (isset($bilp['fontLanguageOverride'])) {
if ($bilp['fontLanguageOverride']) {
$p['FONT-LANGUAGE-OVERRIDE'] = $bilp['fontLanguageOverride'];
} else {
$p['FONT-LANGUAGE-OVERRIDE'] = 'normal';
}
}
// All the variations of font-variant-* we are going to set as font-feature-settings...
if (isset($bilp['OTLtags']) && $bilp['OTLtags']) {
$ffs = array();
if (isset($bilp['OTLtags']['Minus']) && $bilp['OTLtags']['Minus']) {
$f = preg_split('/\s+/', trim($bilp['OTLtags']['Minus']));
foreach ($f AS $ff) {
$ffs[] = "'" . $ff . "' 0";
}
}
if (isset($bilp['OTLtags']['FFMinus']) && $bilp['OTLtags']['FFMinus']) {
$f = preg_split('/\s+/', trim($bilp['OTLtags']['FFMinus']));
foreach ($f AS $ff) {
$ffs[] = "'" . $ff . "' 0";
}
}
if (isset($bilp['OTLtags']['Plus']) && $bilp['OTLtags']['Plus']) {
$f = preg_split('/\s+/', trim($bilp['OTLtags']['Plus']));
foreach ($f AS $ff) {
$ffs[] = "'" . $ff . "' 1";
}
}
if (isset($bilp['OTLtags']['FFPlus']) && $bilp['OTLtags']['FFPlus']) { // May contain numeric value e.g. salt4
$f = preg_split('/\s+/', trim($bilp['OTLtags']['FFPlus']));
foreach ($f AS $ff) {
if (strlen($ff) > 4) {
$ffs[] = "'" . substr($ff, 0, 4) . "' " . substr($ff, 4);
} else {
$ffs[] = "'" . $ff . "' 1";
}
}
}
$p['FONT-FEATURE-SETTINGS'] = implode(', ', $ffs);
}
}
function PreviewBlockCSS($tag, $attr)
{
// Looks ahead from current block level to a new level
$p = array();
$zp = array();
$oldcascadeCSS = $this->mpdf->blk[$this->mpdf->blklvl]['cascadeCSS'];
$classes = array();
if (isset($attr['CLASS'])) {
$classes = preg_split('/\s+/', $attr['CLASS']);
}
//===============================================
// DEFAULT for this TAG set in DefaultCSS
if (isset($this->mpdf->defaultCSS[$tag])) {
$zp = $this->fixCSS($this->mpdf->defaultCSS[$tag]);
if (is_array($zp)) {
$p = array_merge($zp, $p);
} // Inherited overwrites default
}
// STYLESHEET TAG e.g. h1 p div table
if (isset($this->CSS[$tag])) {
$zp = $this->CSS[$tag];
if (is_array($zp)) {
$p = array_merge($p, $zp);
}
}
// STYLESHEET CLASS e.g. .smallone{} .redletter{}
foreach ($classes AS $class) {
$zp = array();
if (isset($this->CSS['CLASS>>' . $class])) {
$zp = $this->CSS['CLASS>>' . $class];
}
if (is_array($zp)) {
$p = array_merge($p, $zp);
}
}
// STYLESHEET ID e.g. #smallone{} #redletter{}
if (isset($attr['ID']) && isset($this->CSS['ID>>' . $attr['ID']])) {
$zp = $this->CSS['ID>>' . $attr['ID']];
if (is_array($zp)) {
$p = array_merge($p, $zp);
}
}
// STYLESHEET CLASS e.g. p.smallone{} div.redletter{}
foreach ($classes AS $class) {
$zp = array();
if (isset($this->CSS[$tag . '>>CLASS>>' . $class])) {
$zp = $this->CSS[$tag . '>>CLASS>>' . $class];
}
if (is_array($zp)) {
$p = array_merge($p, $zp);
}
}
// STYLESHEET CLASS e.g. p#smallone{} div#redletter{}
if (isset($attr['ID']) && isset($this->CSS[$tag . '>>ID>>' . $attr['ID']])) {
$zp = $this->CSS[$tag . '>>ID>>' . $attr['ID']];
if (is_array($zp)) {
$p = array_merge($p, $zp);
}
}
//===============================================
// STYLESHEET TAG e.g. div h1 div p
$this->_set_mergedCSS($oldcascadeCSS[$tag], $p);
// STYLESHEET CLASS e.g. .smallone{} .redletter{}
foreach ($classes AS $class) {
$this->_set_mergedCSS($oldcascadeCSS['CLASS>>' . $class], $p);
}
// STYLESHEET CLASS e.g. #smallone{} #redletter{}
if (isset($attr['ID'])) {
$this->_set_mergedCSS($oldcascadeCSS['ID>>' . $attr['ID']], $p);
}
// STYLESHEET CLASS e.g. div.smallone{} p.redletter{}
foreach ($classes AS $class) {
$this->_set_mergedCSS($oldcascadeCSS[$tag . '>>CLASS>>' . $class], $p);
}
// STYLESHEET CLASS e.g. div#smallone{} p#redletter{}
if (isset($attr['ID'])) {
$this->_set_mergedCSS($oldcascadeCSS[$tag . '>>ID>>' . $attr['ID']], $p);
}
//===============================================
// INLINE STYLE e.g. style="CSS:property"
if (isset($attr['STYLE'])) {
$zp = $this->readInlineCSS($attr['STYLE']);
if (is_array($zp)) {
$p = array_merge($p, $zp);
}
}
//===============================================
return $p;
}
// mPDF 5.7.4 nth-child
function _nthchild($f, $c)
{
// $f is formual e.g. 2N+1 spilt into a preg_match array
// $c is the comparator value e.g row or column number
$c += 1;
$select = false;
$a = 1;
$b = 1;
if ($f[0] == 'ODD') {
$a = 2;
$b = 1;
} else if ($f[0] == 'EVEN') {
$a = 2;
$b = 0;
} else if (count($f) == 2) {
$a = 0;
$b = $f[1] + 0;
} // e.g. (+6)
else if (count($f) == 3) { // e.g. (2N)
if ($f[2] == '') {
$a = 1;
} else if ($f[2] == '-') {
$a = -1;
} else {
$a = $f[2] + 0;
}
$b = 0;
} else if (count($f) == 4) { // e.g. (2N+6)
if ($f[2] == '') {
$a = 1;
} else if ($f[2] == '-') {
$a = -1;
} else {
$a = $f[2] + 0;
}
$b = $f[3] + 0;
} else {
return false;
}
if ($a > 0) {
if (((($c % $a) - $b) % $a) == 0 && $c >= $b) {
$select = true;
}
} else if ($a == 0) {
if ($c == $b) {
$select = true;
}
} else { // if ($a<0)
if (((($c % $a) - $b) % $a) == 0 && $c <= $b) {
$select = true;
}
}
return $select;
}
}