1308 lines
		
	
	
		
			39 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			1308 lines
		
	
	
		
			39 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| /**
 | |
|  * CSSTidy - CSS Parser and Optimiser
 | |
|  *
 | |
|  * CSS Optimising Class
 | |
|  * This class optimises CSS data generated by csstidy.
 | |
|  *
 | |
|  * Copyright 2005, 2006, 2007 Florian Schmitz
 | |
|  *
 | |
|  * This file is part of CSSTidy.
 | |
|  *
 | |
|  *   CSSTidy is free software; you can redistribute it and/or modify
 | |
|  *   it under the terms of the GNU Lesser General Public License as published by
 | |
|  *   the Free Software Foundation; either version 2.1 of the License, or
 | |
|  *   (at your option) any later version.
 | |
|  *
 | |
|  *   CSSTidy is distributed in the hope that it will be useful,
 | |
|  *   but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  *   GNU Lesser General Public License for more details.
 | |
|  *
 | |
|  *   You should have received a copy of the GNU Lesser General Public License
 | |
|  *   along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  *
 | |
|  * @license http://opensource.org/licenses/lgpl-license.php GNU Lesser General Public License
 | |
|  * @package csstidy
 | |
|  * @author Florian Schmitz (floele at gmail dot com) 2005-2007
 | |
|  * @author Brett Zamir (brettz9 at yahoo dot com) 2007
 | |
|  * @author Nikolay Matsievsky (speed at webo dot name) 2009-2010
 | |
|  * @author Cedric Morin (cedric at yterium dot com) 2010-2012
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * CSS Optimising Class
 | |
|  *
 | |
|  * This class optimises CSS data generated by csstidy.
 | |
|  *
 | |
|  * @package csstidy
 | |
|  * @author Florian Schmitz (floele at gmail dot com) 2005-2006
 | |
|  * @version 1.0
 | |
|  */
 | |
| class csstidy_optimise {
 | |
| 
 | |
| 	/**
 | |
| 	 * csstidy object
 | |
| 	 * @var object
 | |
| 	 */
 | |
| 	public $parser;
 | |
| 
 | |
| 	/**
 | |
| 	 * Constructor
 | |
| 	 * @param array $css contains the class csstidy
 | |
| 	 * @access private
 | |
| 	 * @version 1.0
 | |
| 	 */
 | |
| 	public function __construct($css) {
 | |
| 		$this->parser = $css;
 | |
| 		$this->css = & $css->css;
 | |
| 		$this->sub_value = & $css->sub_value;
 | |
| 		$this->at = & $css->at;
 | |
| 		$this->selector = & $css->selector;
 | |
| 		$this->property = & $css->property;
 | |
| 		$this->value = & $css->value;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Optimises $css after parsing
 | |
| 	 * @access public
 | |
| 	 * @version 1.0
 | |
| 	 */
 | |
| 	public function postparse() {
 | |
| 
 | |
| 		if ($this->parser->get_cfg('reverse_left_and_right') > 0) {
 | |
| 
 | |
| 			foreach ($this->css as $medium => $selectors) {
 | |
| 				if (is_array($selectors)) {
 | |
| 					foreach ($selectors as $selector => $properties) {
 | |
| 						$this->css[$medium][$selector] = $this->reverse_left_and_right($this->css[$medium][$selector]);
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 		if ($this->parser->get_cfg('preserve_css')) {
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		if ((int)$this->parser->get_cfg('merge_selectors') === 2) {
 | |
| 			foreach ($this->css as $medium => $value) {
 | |
| 				if (is_array($value)) {
 | |
| 					$this->merge_selectors($this->css[$medium]);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if ($this->parser->get_cfg('discard_invalid_selectors')) {
 | |
| 			foreach ($this->css as $medium => $value) {
 | |
| 				if (is_array($value)) {
 | |
| 					$this->discard_invalid_selectors($this->css[$medium]);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if ($this->parser->get_cfg('optimise_shorthands') > 0) {
 | |
| 			foreach ($this->css as $medium => $value) {
 | |
| 				if (is_array($value)) {
 | |
| 					foreach ($value as $selector => $value1) {
 | |
| 						$this->css[$medium][$selector] = $this->merge_4value_shorthands($this->css[$medium][$selector]);
 | |
| 						$this->css[$medium][$selector] = $this->merge_4value_radius_shorthands($this->css[$medium][$selector]);
 | |
| 
 | |
| 						if ($this->parser->get_cfg('optimise_shorthands') < 2) {
 | |
| 							continue;
 | |
| 						}
 | |
| 
 | |
| 						$this->css[$medium][$selector] = $this->merge_font($this->css[$medium][$selector]);
 | |
| 
 | |
| 						if ($this->parser->get_cfg('optimise_shorthands') < 3) {
 | |
| 							continue;
 | |
| 						}
 | |
| 
 | |
| 						$this->css[$medium][$selector] = $this->merge_bg($this->css[$medium][$selector]);
 | |
| 						if (empty($this->css[$medium][$selector])) {
 | |
| 							unset($this->css[$medium][$selector]);
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Optimises values
 | |
| 	 * @access public
 | |
| 	 * @version 1.0
 | |
| 	 */
 | |
| 	public function value() {
 | |
| 		$shorthands = & $this->parser->data['csstidy']['shorthands'];
 | |
| 
 | |
| 		// optimise shorthand properties
 | |
| 		if (isset($shorthands[$this->property])) {
 | |
| 			$temp = $this->shorthand($this->value); // FIXME - move
 | |
| 			if ($temp != $this->value) {
 | |
| 				$this->parser->log('Optimised shorthand notation (' . $this->property . '): Changed "' . $this->value . '" to "' . $temp . '"', 'Information');
 | |
| 			}
 | |
| 			$this->value = $temp;
 | |
| 		}
 | |
| 
 | |
| 		// Remove whitespace at ! important
 | |
| 		if ($this->value != $this->compress_important($this->value)) {
 | |
| 			$this->parser->log('Optimised !important', 'Information');
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Optimises shorthands
 | |
| 	 * @access public
 | |
| 	 * @version 1.0
 | |
| 	 */
 | |
| 	public function shorthands() {
 | |
| 		$shorthands = & $this->parser->data['csstidy']['shorthands'];
 | |
| 
 | |
| 		if (!$this->parser->get_cfg('optimise_shorthands') || $this->parser->get_cfg('preserve_css')) {
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		if ($this->property === 'font' && $this->parser->get_cfg('optimise_shorthands') > 1) {
 | |
| 			$this->css[$this->at][$this->selector]['font']='';
 | |
| 			$this->parser->merge_css_blocks($this->at, $this->selector, $this->dissolve_short_font($this->value));
 | |
| 		}
 | |
| 		if ($this->property === 'background' && $this->parser->get_cfg('optimise_shorthands') > 2) {
 | |
| 			$this->css[$this->at][$this->selector]['background']='';
 | |
| 			$this->parser->merge_css_blocks($this->at, $this->selector, $this->dissolve_short_bg($this->value));
 | |
| 		}
 | |
| 		if (isset($shorthands[$this->property])) {
 | |
| 			$this->parser->merge_css_blocks($this->at, $this->selector, $this->dissolve_4value_shorthands($this->property, $this->value));
 | |
| 			if (is_array($shorthands[$this->property])) {
 | |
| 				$this->css[$this->at][$this->selector][$this->property] = '';
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Optimises a sub-value
 | |
| 	 * @access public
 | |
| 	 * @version 1.0
 | |
| 	 */
 | |
| 	public function subvalue() {
 | |
| 		$replace_colors = & $this->parser->data['csstidy']['replace_colors'];
 | |
| 
 | |
| 		$this->sub_value = trim($this->sub_value);
 | |
| 		if ($this->sub_value == '') { // caution : '0'
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		$important = '';
 | |
| 		if ($this->parser->is_important($this->sub_value)) {
 | |
| 			$important = '!important';
 | |
| 		}
 | |
| 		$this->sub_value = $this->parser->gvw_important($this->sub_value);
 | |
| 
 | |
| 		// Compress font-weight
 | |
| 		if ($this->property === 'font-weight' && $this->parser->get_cfg('compress_font-weight')) {
 | |
| 			if ($this->sub_value === 'bold') {
 | |
| 				$this->sub_value = '700';
 | |
| 				$this->parser->log('Optimised font-weight: Changed "bold" to "700"', 'Information');
 | |
| 			} elseif ($this->sub_value === 'normal') {
 | |
| 				$this->sub_value = '400';
 | |
| 				$this->parser->log('Optimised font-weight: Changed "normal" to "400"', 'Information');
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		$temp = $this->compress_numbers($this->sub_value);
 | |
| 		if (strcasecmp($temp, $this->sub_value) !== 0) {
 | |
| 			if (strlen($temp) > strlen($this->sub_value)) {
 | |
| 				$this->parser->log('Fixed invalid number: Changed "' . $this->sub_value . '" to "' . $temp . '"', 'Warning');
 | |
| 			} else {
 | |
| 				$this->parser->log('Optimised number: Changed "' . $this->sub_value . '" to "' . $temp . '"', 'Information');
 | |
| 			}
 | |
| 			$this->sub_value = $temp;
 | |
| 		}
 | |
| 		if ($this->parser->get_cfg('compress_colors')) {
 | |
| 			$temp = $this->cut_color($this->sub_value);
 | |
| 			if ($temp !== $this->sub_value) {
 | |
| 				if (isset($replace_colors[$this->sub_value])) {
 | |
| 					$this->parser->log('Fixed invalid color name: Changed "' . $this->sub_value . '" to "' . $temp . '"', 'Warning');
 | |
| 				} else {
 | |
| 					$this->parser->log('Optimised color: Changed "' . $this->sub_value . '" to "' . $temp . '"', 'Information');
 | |
| 				}
 | |
| 				$this->sub_value = $temp;
 | |
| 			}
 | |
| 		}
 | |
| 		$this->sub_value .= $important;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Compresses shorthand values. Example: margin:1px 1px 1px 1px -> margin:1px
 | |
| 	 * @param string $value
 | |
| 	 * @access public
 | |
| 	 * @return string
 | |
| 	 * @version 1.0
 | |
| 	 */
 | |
| 	public function shorthand($value) {
 | |
| 		$important = '';
 | |
| 		if ($this->parser->is_important($value)) {
 | |
| 			$values = $this->parser->gvw_important($value);
 | |
| 			$important = '!important';
 | |
| 		}
 | |
| 		else
 | |
| 			$values = $value;
 | |
| 
 | |
| 		$values = explode(' ', $values);
 | |
| 		switch (count($values)) {
 | |
| 			case 4:
 | |
| 				if ($values[0] == $values[1] && $values[0] == $values[2] && $values[0] == $values[3]) {
 | |
| 					return $values[0] . $important;
 | |
| 				} elseif ($values[1] == $values[3] && $values[0] == $values[2]) {
 | |
| 					return $values[0] . ' ' . $values[1] . $important;
 | |
| 				} elseif ($values[1] == $values[3]) {
 | |
| 					return $values[0] . ' ' . $values[1] . ' ' . $values[2] . $important;
 | |
| 				}
 | |
| 				break;
 | |
| 
 | |
| 			case 3:
 | |
| 				if ($values[0] == $values[1] && $values[0] == $values[2]) {
 | |
| 					return $values[0] . $important;
 | |
| 				} elseif ($values[0] == $values[2]) {
 | |
| 					return $values[0] . ' ' . $values[1] . $important;
 | |
| 				}
 | |
| 				break;
 | |
| 
 | |
| 			case 2:
 | |
| 				if ($values[0] == $values[1]) {
 | |
| 					return $values[0] . $important;
 | |
| 				}
 | |
| 				break;
 | |
| 		}
 | |
| 
 | |
| 		return $value;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Removes unnecessary whitespace in ! important
 | |
| 	 * @param string $string
 | |
| 	 * @return string
 | |
| 	 * @access public
 | |
| 	 * @version 1.1
 | |
| 	 */
 | |
| 	public function compress_important(&$string) {
 | |
| 		if ($this->parser->is_important($string)) {
 | |
| 			$important = $this->parser->get_cfg('space_before_important') ? ' !important' : '!important';
 | |
| 			$string = $this->parser->gvw_important($string) . $important;
 | |
| 		}
 | |
| 		return $string;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Color compression function. Converts all rgb() values to #-values and uses the short-form if possible. Also replaces 4 color names by #-values.
 | |
| 	 * @param string $color
 | |
| 	 * @return string
 | |
| 	 * @version 1.1
 | |
| 	 */
 | |
| 	public function cut_color($color) {
 | |
| 		$replace_colors = & $this->parser->data['csstidy']['replace_colors'];
 | |
| 
 | |
| 		// if it's a string, don't touch !
 | |
| 		if (strncmp($color, "'", 1) == 0 || strncmp($color, '"', 1) == 0)
 | |
| 			return $color;
 | |
| 
 | |
| 		/* expressions complexes de type gradient */
 | |
| 		if (strpos($color, '(') !== false && strncmp($color, 'rgb(' ,4) != 0) {
 | |
| 			// on ne touche pas aux couleurs dans les expression ms, c'est trop sensible
 | |
| 			if (stripos($color, 'progid:') !== false)
 | |
| 				return $color;
 | |
| 			preg_match_all(",rgb\([^)]+\),i", $color, $matches, PREG_SET_ORDER);
 | |
| 			if (count($matches)) {
 | |
| 				foreach ($matches as $m) {
 | |
| 					$color = str_replace($m[0], $this->cut_color($m[0]), $color);
 | |
| 				}
 | |
| 			}
 | |
| 			preg_match_all(",#[0-9a-f]{6}(?=[^0-9a-f]),i", $color, $matches, PREG_SET_ORDER);
 | |
| 			if (count($matches)) {
 | |
| 				foreach ($matches as $m) {
 | |
| 					$color = str_replace($m[0],$this->cut_color($m[0]), $color);
 | |
| 				}
 | |
| 			}
 | |
| 			return $color;
 | |
| 		}
 | |
| 
 | |
| 		// rgb(0,0,0) -> #000000 (or #000 in this case later)
 | |
| 		if (strncasecmp($color, 'rgb(', 4)==0) {
 | |
| 			$color_tmp = substr($color, 4, strlen($color) - 5);
 | |
| 			$color_tmp = explode(',', $color_tmp);
 | |
| 			for ($i = 0; $i < count($color_tmp); $i++) {
 | |
| 				$color_tmp[$i] = trim($color_tmp[$i]);
 | |
| 				if (substr($color_tmp[$i], -1) === '%') {
 | |
| 					$color_tmp[$i] = round((255 * $color_tmp[$i]) / 100);
 | |
| 				}
 | |
| 				if ($color_tmp[$i] > 255)
 | |
| 					$color_tmp[$i] = 255;
 | |
| 			}
 | |
| 			$color = '#';
 | |
| 			for ($i = 0; $i < 3; $i++) {
 | |
| 				if ($color_tmp[$i] < 16) {
 | |
| 					$color .= '0' . dechex($color_tmp[$i]);
 | |
| 				} else {
 | |
| 					$color .= dechex($color_tmp[$i]);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// Fix bad color names
 | |
| 		if (isset($replace_colors[strtolower($color)])) {
 | |
| 			$color = $replace_colors[strtolower($color)];
 | |
| 		}
 | |
| 
 | |
| 		// #aabbcc -> #abc
 | |
| 		if (strlen($color) == 7) {
 | |
| 			$color_temp = strtolower($color);
 | |
| 			if ($color_temp[0] === '#' && $color_temp[1] == $color_temp[2] && $color_temp[3] == $color_temp[4] && $color_temp[5] == $color_temp[6]) {
 | |
| 				$color = '#' . $color[1] . $color[3] . $color[5];
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		switch (strtolower($color)) {
 | |
| 			/* color name -> hex code */
 | |
| 			case 'black': return '#000';
 | |
| 			case 'fuchsia': return '#f0f';
 | |
| 			case 'white': return '#fff';
 | |
| 			case 'yellow': return '#ff0';
 | |
| 
 | |
| 			/* hex code -> color name */
 | |
| 			case '#800000': return 'maroon';
 | |
| 			case '#ffa500': return 'orange';
 | |
| 			case '#808000': return 'olive';
 | |
| 			case '#800080': return 'purple';
 | |
| 			case '#008000': return 'green';
 | |
| 			case '#000080': return 'navy';
 | |
| 			case '#008080': return 'teal';
 | |
| 			case '#c0c0c0': return 'silver';
 | |
| 			case '#808080': return 'gray';
 | |
| 			case '#f00': return 'red';
 | |
| 		}
 | |
| 
 | |
| 		return $color;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Compresses numbers (ie. 1.0 becomes 1 or 1.100 becomes 1.1 )
 | |
| 	 * @param string $subvalue
 | |
| 	 * @return string
 | |
| 	 * @version 1.2
 | |
| 	 */
 | |
| 	public function compress_numbers($subvalue) {
 | |
| 		$unit_values = & $this->parser->data['csstidy']['unit_values'];
 | |
| 		$color_values = & $this->parser->data['csstidy']['color_values'];
 | |
| 
 | |
| 		// for font:1em/1em sans-serif...;
 | |
| 		if ($this->property === 'font') {
 | |
| 			$temp = explode('/', $subvalue);
 | |
| 		} else {
 | |
| 			$temp = array($subvalue);
 | |
| 		}
 | |
| 
 | |
| 		for ($l = 0; $l < count($temp); $l++) {
 | |
| 			// if we are not dealing with a number at this point, do not optimise anything
 | |
| 			$number = $this->AnalyseCssNumber($temp[$l]);
 | |
| 			if ($number === false) {
 | |
| 				return $subvalue;
 | |
| 			}
 | |
| 
 | |
| 			// Fix bad colors
 | |
| 			if (in_array($this->property, $color_values)) {
 | |
| 				$temp[$l] = '#' . $temp[$l];
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			if (abs($number[0]) > 0) {
 | |
| 				if ($number[1] == '' && in_array($this->property, $unit_values, true)) {
 | |
| 					$number[1] = 'px';
 | |
| 				}
 | |
| 						} elseif ($number[1] != 's' && $number[1] != 'ms') {
 | |
| 								$number[1] = '';
 | |
| 						}
 | |
| 
 | |
| 			$temp[$l] = $number[0] . $number[1];
 | |
| 		}
 | |
| 
 | |
| 		return ((count($temp) > 1) ? $temp[0] . '/' . $temp[1] : $temp[0]);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Checks if a given string is a CSS valid number. If it is,
 | |
| 	 * an array containing the value and unit is returned
 | |
| 	 * @param string $string
 | |
| 	 * @return array ('unit' if unit is found or '' if no unit exists, number value) or false if no number
 | |
| 	 */
 | |
| 	public function AnalyseCssNumber($string) {
 | |
| 		// most simple checks first
 | |
| 		if (strlen($string) == 0 || ctype_alpha($string[0])) {
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		$units = & $this->parser->data['csstidy']['units'];
 | |
| 		$return = array(0, '');
 | |
| 
 | |
| 		$return[0] = floatval($string);
 | |
| 		if (abs($return[0]) > 0 && abs($return[0]) < 1) {
 | |
| 			if ($return[0] < 0) {
 | |
| 				$return[0] = '-' . ltrim(substr($return[0], 1), '0');
 | |
| 			} else {
 | |
| 				$return[0] = ltrim($return[0], '0');
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// Look for unit and split from value if exists
 | |
| 		foreach ($units as $unit) {
 | |
| 			$expectUnitAt = strlen($string) - strlen($unit);
 | |
| 			if (!($unitInString = stristr($string, $unit))) { // mb_strpos() fails with "false"
 | |
| 				continue;
 | |
| 			}
 | |
| 			$actualPosition = strpos($string, $unitInString);
 | |
| 			if ($expectUnitAt === $actualPosition) {
 | |
| 				$return[1] = $unit;
 | |
| 				$string = substr($string, 0, - strlen($unit));
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 		if (!is_numeric($string)) {
 | |
| 			return false;
 | |
| 		}
 | |
| 		return $return;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Merges selectors with same properties. Example: a{color:red} b{color:red} -> a,b{color:red}
 | |
| 	 * Very basic and has at least one bug. Hopefully there is a replacement soon.
 | |
| 	 * @param array $array
 | |
| 	 * @return array
 | |
| 	 * @access public
 | |
| 	 * @version 1.2
 | |
| 	 */
 | |
| 	public function merge_selectors(&$array) {
 | |
| 		$css = $array;
 | |
| 		foreach ($css as $key => $value) {
 | |
| 			if (!isset($css[$key])) {
 | |
| 				continue;
 | |
| 			}
 | |
| 			$newsel = '';
 | |
| 
 | |
| 			// Check if properties also exist in another selector
 | |
| 			$keys = array();
 | |
| 			// PHP bug (?) without $css = $array; here
 | |
| 			foreach ($css as $selector => $vali) {
 | |
| 				if ($selector == $key) {
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				if ($css[$key] === $vali) {
 | |
| 					$keys[] = $selector;
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			if (!empty($keys)) {
 | |
| 				$newsel = $key;
 | |
| 				unset($css[$key]);
 | |
| 				foreach ($keys as $selector) {
 | |
| 					unset($css[$selector]);
 | |
| 					$newsel .= ',' . $selector;
 | |
| 				}
 | |
| 				$css[$newsel] = $value;
 | |
| 			}
 | |
| 		}
 | |
| 		$array = $css;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Removes invalid selectors and their corresponding rule-sets as
 | |
| 	 * defined by 4.1.7 in REC-CSS2. This is a very rudimentary check
 | |
| 	 * and should be replaced by a full-blown parsing algorithm or
 | |
| 	 * regular expression
 | |
| 	 * @version 1.4
 | |
| 	 */
 | |
| 	public function discard_invalid_selectors(&$array) {
 | |
| 		$invalid = array('+' => true, '~' => true, ',' => true, '>' => true);
 | |
| 		foreach ($array as $selector => $decls) {
 | |
| 			$ok = true;
 | |
| 			$selectors = array_map('trim', explode(',', $selector));
 | |
| 			foreach ($selectors as $s) {
 | |
| 				$simple_selectors = preg_split('/\s*[+>~\s]\s*/', $s);
 | |
| 				foreach ($simple_selectors as $ss) {
 | |
| 					if ($ss === '')
 | |
| 						$ok = false;
 | |
| 					// could also check $ss for internal structure,
 | |
| 					// but that probably would be too slow
 | |
| 				}
 | |
| 			}
 | |
| 			if (!$ok)
 | |
| 				unset($array[$selector]);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Dissolves properties like padding:10px 10px 10px to padding-top:10px;padding-bottom:10px;...
 | |
| 	 * @param string $property
 | |
| 	 * @param string $value
 | |
| 	 * @param array|null $shorthands
 | |
| 	 * @return array
 | |
| 	 * @version 1.0
 | |
| 	 * @see merge_4value_shorthands()
 | |
| 	 */
 | |
| 	public function dissolve_4value_shorthands($property, $value, $shorthands = null) {
 | |
| 		if (is_null($shorthands)) {
 | |
| 			$shorthands = & $this->parser->data['csstidy']['shorthands'];
 | |
| 		}
 | |
| 		if (!is_array($shorthands[$property])) {
 | |
| 			$return[$property] = $value;
 | |
| 			return $return;
 | |
| 		}
 | |
| 
 | |
| 		$important = '';
 | |
| 		if ($this->parser->is_important($value)) {
 | |
| 			$value = $this->parser->gvw_important($value);
 | |
| 			$important = '!important';
 | |
| 		}
 | |
| 		$values = explode(' ', $value);
 | |
| 
 | |
| 
 | |
| 		$return = array();
 | |
| 		if (count($values) == 4) {
 | |
| 			for ($i = 0; $i < 4; $i++) {
 | |
| 				$return[$shorthands[$property][$i]] = $values[$i] . $important;
 | |
| 			}
 | |
| 		} elseif (count($values) == 3) {
 | |
| 			$return[$shorthands[$property][0]] = $values[0] . $important;
 | |
| 			$return[$shorthands[$property][1]] = $values[1] . $important;
 | |
| 			$return[$shorthands[$property][3]] = $values[1] . $important;
 | |
| 			$return[$shorthands[$property][2]] = $values[2] . $important;
 | |
| 		} elseif (count($values) == 2) {
 | |
| 			for ($i = 0; $i < 4; $i++) {
 | |
| 				$return[$shorthands[$property][$i]] = (($i % 2 != 0)) ? $values[1] . $important : $values[0] . $important;
 | |
| 			}
 | |
| 		} else {
 | |
| 			for ($i = 0; $i < 4; $i++) {
 | |
| 				$return[$shorthands[$property][$i]] = $values[0] . $important;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return $return;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Dissolves radius properties like
 | |
| 	 * border-radius:10px 10px 10px / 1px 2px
 | |
| 	 * to border-top-left:10px 1px;border-top-right:10px 2x;...
 | |
| 	 * @param string $property
 | |
| 	 * @param string $value
 | |
| 	 * @return array
 | |
| 	 * @version 1.0
 | |
| 	 * @use dissolve_4value_shorthands()
 | |
| 	 * @see merge_4value_radius_shorthands()
 | |
| 	 */
 | |
| 	public function dissolve_4value_radius_shorthands($property, $value) {
 | |
| 		$shorthands = & $this->parser->data['csstidy']['radius_shorthands'];
 | |
| 		if (!is_array($shorthands[$property])) {
 | |
| 			$return[$property] = $value;
 | |
| 			return $return;
 | |
| 		}
 | |
| 
 | |
| 		if (strpos($value, '/') !== false) {
 | |
| 			$values = $this->explode_ws('/', $value);
 | |
| 			if (count($values) == 2) {
 | |
| 				$r[0] = $this->dissolve_4value_shorthands($property, trim($values[0]), $shorthands);
 | |
| 				$r[1] = $this->dissolve_4value_shorthands($property, trim($values[1]), $shorthands);
 | |
| 				$return = array();
 | |
| 				foreach ($r[0] as $p=>$v) {
 | |
| 					$return[$p] = $v;
 | |
| 					if ($r[1][$p] !== $v) {
 | |
| 						$return[$p] .= ' ' . $r[1][$p];
 | |
| 					}
 | |
| 				}
 | |
| 				return $return;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		$return = $this->dissolve_4value_shorthands($property, $value, $shorthands);
 | |
| 		return $return;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Explodes a string as explode() does, however, not if $sep is escaped or within a string.
 | |
| 	 * @param string $sep seperator
 | |
| 	 * @param string $string
 | |
| 	 * @param bool $explode_in_parenthesis
 | |
| 	 * @return array
 | |
| 	 * @version 1.0
 | |
| 	 */
 | |
| 	public function explode_ws($sep, $string, $explode_in_parenthesis = false) {
 | |
| 		$status = 'st';
 | |
| 		$to = '';
 | |
| 
 | |
| 		$output = array(
 | |
| 			0 => '',
 | |
| 		);
 | |
| 		$num = 0;
 | |
| 		for ($i = 0, $len = strlen($string); $i < $len; $i++) {
 | |
| 			switch ($status) {
 | |
| 				case 'st':
 | |
| 					if ($string[$i] == $sep && !$this->parser->escaped($string, $i)) {
 | |
| 						++$num;
 | |
| 					} elseif ($string[$i] === '"' || $string[$i] === '\'' || (!$explode_in_parenthesis && $string[$i] === '(') && !$this->parser->escaped($string, $i)) {
 | |
| 						$status = 'str';
 | |
| 						$to = ($string[$i] === '(') ? ')' : $string[$i];
 | |
| 						(isset($output[$num])) ? $output[$num] .= $string[$i] : $output[$num] = $string[$i];
 | |
| 					} else {
 | |
| 						(isset($output[$num])) ? $output[$num] .= $string[$i] : $output[$num] = $string[$i];
 | |
| 					}
 | |
| 					break;
 | |
| 
 | |
| 				case 'str':
 | |
| 					if ($string[$i] == $to && !$this->parser->escaped($string, $i)) {
 | |
| 						$status = 'st';
 | |
| 					}
 | |
| 					(isset($output[$num])) ? $output[$num] .= $string[$i] : $output[$num] = $string[$i];
 | |
| 					break;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return $output;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Merges Shorthand properties again, the opposite of dissolve_4value_shorthands()
 | |
| 	 * @param array $array
 | |
| 	 * @param array|null $shorthands
 | |
| 	 * @return array
 | |
| 	 * @version 1.2
 | |
| 	 * @see dissolve_4value_shorthands()
 | |
| 	 */
 | |
| 	public function merge_4value_shorthands($array, $shorthands = null) {
 | |
| 		$return = $array;
 | |
| 		if (is_null($shorthands)) {
 | |
| 			$shorthands = & $this->parser->data['csstidy']['shorthands'];
 | |
| 		}
 | |
| 
 | |
| 		foreach ($shorthands as $key => $value) {
 | |
| 			if ($value !== 0 && isset($array[$value[0]]) && isset($array[$value[1]])
 | |
| 							&& isset($array[$value[2]]) && isset($array[$value[3]])) {
 | |
| 				$return[$key] = '';
 | |
| 
 | |
| 				$important = '';
 | |
| 				for ($i = 0; $i < 4; $i++) {
 | |
| 					$val = $array[$value[$i]];
 | |
| 					if ($this->parser->is_important($val)) {
 | |
| 						$important = '!important';
 | |
| 						$return[$key] .= $this->parser->gvw_important($val) . ' ';
 | |
| 					} else {
 | |
| 						$return[$key] .= $val . ' ';
 | |
| 					}
 | |
| 					unset($return[$value[$i]]);
 | |
| 				}
 | |
| 				$return[$key] = $this->shorthand(trim($return[$key] . $important));
 | |
| 			}
 | |
| 		}
 | |
| 		return $return;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Merges Shorthand properties again, the opposite of dissolve_4value_shorthands()
 | |
| 	 * @param array $array
 | |
| 	 * @return array
 | |
| 	 * @version 1.2
 | |
| 	 * @use merge_4value_shorthands()
 | |
| 	 * @see dissolve_4value_radius_shorthands()
 | |
| 	 */
 | |
| 	public function merge_4value_radius_shorthands($array) {
 | |
| 		$return = $array;
 | |
| 		$shorthands = & $this->parser->data['csstidy']['radius_shorthands'];
 | |
| 
 | |
| 		foreach ($shorthands as $key => $value) {
 | |
| 			if (isset($array[$value[0]]) && isset($array[$value[1]])
 | |
| 							&& isset($array[$value[2]]) && isset($array[$value[3]]) && $value !== 0) {
 | |
| 				$return[$key] = '';
 | |
| 				$a = array();
 | |
| 				for ($i = 0; $i < 4; $i++) {
 | |
| 					$v = $this->explode_ws(' ', trim($array[$value[$i]]));
 | |
| 					$a[0][$value[$i]] = reset($v);
 | |
| 					$a[1][$value[$i]] = end($v);
 | |
| 				}
 | |
| 				$r = array();
 | |
| 				$r[0] = $this->merge_4value_shorthands($a[0], $shorthands);
 | |
| 				$r[1] = $this->merge_4value_shorthands($a[1], $shorthands);
 | |
| 
 | |
| 				if (isset($r[0][$key]) and isset($r[1][$key])) {
 | |
| 					$return[$key] = $r[0][$key];
 | |
| 					if ($r[1][$key] !== $r[0][$key]) {
 | |
| 						$return[$key] .= ' / ' . $r[1][$key];
 | |
| 					}
 | |
| 					for ($i = 0; $i < 4; $i++) {
 | |
| 						unset($return[$value[$i]]);
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		return $return;
 | |
| 	}
 | |
| 	/**
 | |
| 	 * Dissolve background property
 | |
| 	 * @param string $str_value
 | |
| 	 * @return array
 | |
| 	 * @version 1.0
 | |
| 	 * @see merge_bg()
 | |
| 	 * @todo full CSS 3 compliance
 | |
| 	 */
 | |
| 	public function dissolve_short_bg($str_value) {
 | |
| 		// don't try to explose background gradient !
 | |
| 		if (stripos($str_value, 'gradient(')!== false)
 | |
| 			return array('background'=>$str_value);
 | |
| 
 | |
| 		$background_prop_default = & $this->parser->data['csstidy']['background_prop_default'];
 | |
| 		$repeat = array('repeat', 'repeat-x', 'repeat-y', 'no-repeat', 'space');
 | |
| 		$attachment = array('scroll', 'fixed', 'local');
 | |
| 		$clip = array('border', 'padding');
 | |
| 		$origin = array('border', 'padding', 'content');
 | |
| 		$pos = array('top', 'center', 'bottom', 'left', 'right');
 | |
| 		$important = '';
 | |
| 		$return = array('background-image' => null, 'background-size' => null, 'background-repeat' => null, 'background-position' => null, 'background-attachment' => null, 'background-clip' => null, 'background-origin' => null, 'background-color' => null);
 | |
| 
 | |
| 		if ($this->parser->is_important($str_value)) {
 | |
| 			$important = ' !important';
 | |
| 			$str_value = $this->parser->gvw_important($str_value);
 | |
| 		}
 | |
| 
 | |
| 		$str_value = $this->explode_ws(',', $str_value);
 | |
| 		for ($i = 0; $i < count($str_value); $i++) {
 | |
| 			$have['clip'] = false;
 | |
| 			$have['pos'] = false;
 | |
| 			$have['color'] = false;
 | |
| 			$have['bg'] = false;
 | |
| 
 | |
| 			if (is_array($str_value[$i])) {
 | |
| 				$str_value[$i] = $str_value[$i][0];
 | |
| 			}
 | |
| 			$str_value[$i] = $this->explode_ws(' ', trim($str_value[$i]));
 | |
| 
 | |
| 			for ($j = 0; $j < count($str_value[$i]); $j++) {
 | |
| 				if ($have['bg'] === false && (substr($str_value[$i][$j], 0, 4) === 'url(' || $str_value[$i][$j] === 'none')) {
 | |
| 					$return['background-image'] .= $str_value[$i][$j] . ',';
 | |
| 					$have['bg'] = true;
 | |
| 				} elseif (in_array($str_value[$i][$j], $repeat, true)) {
 | |
| 					$return['background-repeat'] .= $str_value[$i][$j] . ',';
 | |
| 				} elseif (in_array($str_value[$i][$j], $attachment, true)) {
 | |
| 					$return['background-attachment'] .= $str_value[$i][$j] . ',';
 | |
| 				} elseif (in_array($str_value[$i][$j], $clip, true) && !$have['clip']) {
 | |
| 					$return['background-clip'] .= $str_value[$i][$j] . ',';
 | |
| 					$have['clip'] = true;
 | |
| 				} elseif (in_array($str_value[$i][$j], $origin, true)) {
 | |
| 					$return['background-origin'] .= $str_value[$i][$j] . ',';
 | |
| 				} elseif ($str_value[$i][$j][0] === '(') {
 | |
| 					$return['background-size'] .= substr($str_value[$i][$j], 1, -1) . ',';
 | |
| 				} elseif (in_array($str_value[$i][$j], $pos, true) || is_numeric($str_value[$i][$j][0]) || $str_value[$i][$j][0] === null || $str_value[$i][$j][0] === '-' || $str_value[$i][$j][0] === '.') {
 | |
| 					$return['background-position'] .= $str_value[$i][$j];
 | |
| 					if (!$have['pos'])
 | |
| 						$return['background-position'] .= ' '; else
 | |
| 						$return['background-position'].= ',';
 | |
| 					$have['pos'] = true;
 | |
| 				} elseif (!$have['color']) {
 | |
| 					$return['background-color'] .= $str_value[$i][$j] . ',';
 | |
| 					$have['color'] = true;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		foreach ($background_prop_default as $bg_prop => $default_value) {
 | |
| 			if ($return[$bg_prop] !== null) {
 | |
| 				$return[$bg_prop] = substr($return[$bg_prop], 0, -1) . $important;
 | |
| 			}
 | |
| 			else
 | |
| 				$return[$bg_prop] = $default_value . $important;
 | |
| 		}
 | |
| 		return $return;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Merges all background properties
 | |
| 	 * @param array $input_css
 | |
| 	 * @return array
 | |
| 	 * @version 1.0
 | |
| 	 * @see dissolve_short_bg()
 | |
| 	 * @todo full CSS 3 compliance
 | |
| 	 */
 | |
| 	public function merge_bg($input_css) {
 | |
| 		$background_prop_default = & $this->parser->data['csstidy']['background_prop_default'];
 | |
| 		// Max number of background images. CSS3 not yet fully implemented
 | |
| 		$number_of_values = @max(count($this->explode_ws(',', $input_css['background-image'])), count($this->explode_ws(',', $input_css['background-color'])), 1);
 | |
| 		// Array with background images to check if BG image exists
 | |
| 		$bg_img_array = @$this->explode_ws(',', $this->parser->gvw_important($input_css['background-image']));
 | |
| 		$new_bg_value = '';
 | |
| 		$important = '';
 | |
| 
 | |
| 		// if background properties is here and not empty, don't try anything
 | |
| 		if (isset($input_css['background']) && $input_css['background'])
 | |
| 			return $input_css;
 | |
| 
 | |
| 		for ($i = 0; $i < $number_of_values; $i++) {
 | |
| 			foreach ($background_prop_default as $bg_property => $default_value) {
 | |
| 				// Skip if property does not exist
 | |
| 				if (!isset($input_css[$bg_property])) {
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				$cur_value = $input_css[$bg_property];
 | |
| 				// skip all optimisation if gradient() somewhere
 | |
| 				if (stripos($cur_value, 'gradient(') !== false)
 | |
| 					return $input_css;
 | |
| 
 | |
| 				// Skip some properties if there is no background image
 | |
| 				if ((!isset($bg_img_array[$i]) || $bg_img_array[$i] === 'none')
 | |
| 								&& ($bg_property === 'background-size' || $bg_property === 'background-position'
 | |
| 								|| $bg_property === 'background-attachment' || $bg_property === 'background-repeat')) {
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				// Remove !important
 | |
| 				if ($this->parser->is_important($cur_value)) {
 | |
| 					$important = ' !important';
 | |
| 					$cur_value = $this->parser->gvw_important($cur_value);
 | |
| 				}
 | |
| 
 | |
| 				// Do not add default values
 | |
| 				if ($cur_value === $default_value) {
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				$temp = $this->explode_ws(',', $cur_value);
 | |
| 
 | |
| 				if (isset($temp[$i])) {
 | |
| 					if ($bg_property === 'background-size') {
 | |
| 						$new_bg_value .= '(' . $temp[$i] . ') ';
 | |
| 					} else {
 | |
| 						$new_bg_value .= $temp[$i] . ' ';
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			$new_bg_value = trim($new_bg_value);
 | |
| 			if ($i != $number_of_values - 1)
 | |
| 				$new_bg_value .= ',';
 | |
| 		}
 | |
| 
 | |
| 		// Delete all background-properties
 | |
| 		foreach ($background_prop_default as $bg_property => $default_value) {
 | |
| 			unset($input_css[$bg_property]);
 | |
| 		}
 | |
| 
 | |
| 		// Add new background property
 | |
| 		if ($new_bg_value !== '')
 | |
| 			$input_css['background'] = $new_bg_value . $important;
 | |
| 		elseif(isset ($input_css['background']))
 | |
| 			$input_css['background'] = 'none';
 | |
| 
 | |
| 		return $input_css;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Dissolve font property
 | |
| 	 * @param string $str_value
 | |
| 	 * @return array
 | |
| 	 * @version 1.3
 | |
| 	 * @see merge_font()
 | |
| 	 */
 | |
| 	public function dissolve_short_font($str_value) {
 | |
| 		$font_prop_default = & $this->parser->data['csstidy']['font_prop_default'];
 | |
| 		$font_weight = array('normal', 'bold', 'bolder', 'lighter', 100, 200, 300, 400, 500, 600, 700, 800, 900);
 | |
| 		$font_variant = array('normal', 'small-caps');
 | |
| 		$font_style = array('normal', 'italic', 'oblique');
 | |
| 		$important = '';
 | |
| 		$return = array('font-style' => null, 'font-variant' => null, 'font-weight' => null, 'font-size' => null, 'line-height' => null, 'font-family' => null);
 | |
| 
 | |
| 		if ($this->parser->is_important($str_value)) {
 | |
| 			$important = '!important';
 | |
| 			$str_value = $this->parser->gvw_important($str_value);
 | |
| 		}
 | |
| 
 | |
| 		$have['style'] = false;
 | |
| 		$have['variant'] = false;
 | |
| 		$have['weight'] = false;
 | |
| 		$have['size'] = false;
 | |
| 		// Detects if font-family consists of several words w/o quotes
 | |
| 		$multiwords = false;
 | |
| 
 | |
| 		// Workaround with multiple font-family
 | |
| 		$str_value = $this->explode_ws(',', trim($str_value));
 | |
| 
 | |
| 		$str_value[0] = $this->explode_ws(' ', trim($str_value[0]));
 | |
| 
 | |
| 		for ($j = 0; $j < count($str_value[0]); $j++) {
 | |
| 			if ($have['weight'] === false && in_array($str_value[0][$j], $font_weight)) {
 | |
| 				$return['font-weight'] = $str_value[0][$j];
 | |
| 				$have['weight'] = true;
 | |
| 			} elseif ($have['variant'] === false && in_array($str_value[0][$j], $font_variant)) {
 | |
| 				$return['font-variant'] = $str_value[0][$j];
 | |
| 				$have['variant'] = true;
 | |
| 			} elseif ($have['style'] === false && in_array($str_value[0][$j], $font_style)) {
 | |
| 				$return['font-style'] = $str_value[0][$j];
 | |
| 				$have['style'] = true;
 | |
| 			} elseif ($have['size'] === false && (is_numeric($str_value[0][$j][0]) || $str_value[0][$j][0] === null || $str_value[0][$j][0] === '.')) {
 | |
| 				$size = $this->explode_ws('/', trim($str_value[0][$j]));
 | |
| 				$return['font-size'] = $size[0];
 | |
| 				if (isset($size[1])) {
 | |
| 					$return['line-height'] = $size[1];
 | |
| 				} else {
 | |
| 					$return['line-height'] = ''; // don't add 'normal' !
 | |
| 				}
 | |
| 				$have['size'] = true;
 | |
| 			} else {
 | |
| 				if (isset($return['font-family'])) {
 | |
| 					$return['font-family'] .= ' ' . $str_value[0][$j];
 | |
| 					$multiwords = true;
 | |
| 				} else {
 | |
| 					$return['font-family'] = $str_value[0][$j];
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		// add quotes if we have several qords in font-family
 | |
| 		if ($multiwords !== false) {
 | |
| 			$return['font-family'] = '"' . $return['font-family'] . '"';
 | |
| 		}
 | |
| 		$i = 1;
 | |
| 		while (isset($str_value[$i])) {
 | |
| 			$return['font-family'] .= ',' . trim($str_value[$i]);
 | |
| 			$i++;
 | |
| 		}
 | |
| 
 | |
| 		// Fix for 100 and more font-size
 | |
| 		if ($have['size'] === false && isset($return['font-weight']) &&
 | |
| 						is_numeric($return['font-weight'][0])) {
 | |
| 			$return['font-size'] = $return['font-weight'];
 | |
| 			unset($return['font-weight']);
 | |
| 		}
 | |
| 
 | |
| 		foreach ($font_prop_default as $font_prop => $default_value) {
 | |
| 			if ($return[$font_prop] !== null) {
 | |
| 				$return[$font_prop] = $return[$font_prop] . $important;
 | |
| 			}
 | |
| 			else
 | |
| 				$return[$font_prop] = $default_value . $important;
 | |
| 		}
 | |
| 		return $return;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Merges all fonts properties
 | |
| 	 * @param array $input_css
 | |
| 	 * @return array
 | |
| 	 * @version 1.3
 | |
| 	 * @see dissolve_short_font()
 | |
| 	 */
 | |
| 	public function merge_font($input_css) {
 | |
| 		$font_prop_default = & $this->parser->data['csstidy']['font_prop_default'];
 | |
| 		$new_font_value = '';
 | |
| 		$important = '';
 | |
| 		// Skip if not font-family and font-size set
 | |
| 		if (isset($input_css['font-family']) && isset($input_css['font-size']) && $input_css['font-family'] != 'inherit') {
 | |
| 			// fix several words in font-family - add quotes
 | |
| 			if (isset($input_css['font-family'])) {
 | |
| 				$families = explode(',', $input_css['font-family']);
 | |
| 				$result_families = array();
 | |
| 				foreach ($families as $family) {
 | |
| 					$family = trim($family);
 | |
| 					$len = strlen($family);
 | |
| 					if (strpos($family, ' ') &&
 | |
| 									!(($family[0] === '"' && $family[$len - 1] === '"') ||
 | |
| 									($family[0] === "'" && $family[$len - 1] === "'"))) {
 | |
| 						$family = '"' . $family . '"';
 | |
| 					}
 | |
| 					$result_families[] = $family;
 | |
| 				}
 | |
| 				$input_css['font-family'] = implode(',', $result_families);
 | |
| 			}
 | |
| 			foreach ($font_prop_default as $font_property => $default_value) {
 | |
| 
 | |
| 				// Skip if property does not exist
 | |
| 				if (!isset($input_css[$font_property])) {
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				$cur_value = $input_css[$font_property];
 | |
| 
 | |
| 				// Skip if default value is used
 | |
| 				if ($cur_value === $default_value) {
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				// Remove !important
 | |
| 				if ($this->parser->is_important($cur_value)) {
 | |
| 					$important = '!important';
 | |
| 					$cur_value = $this->parser->gvw_important($cur_value);
 | |
| 				}
 | |
| 
 | |
| 				$new_font_value .= $cur_value;
 | |
| 				// Add delimiter
 | |
| 				$new_font_value .= ( $font_property === 'font-size' &&
 | |
| 								isset($input_css['line-height'])) ? '/' : ' ';
 | |
| 			}
 | |
| 
 | |
| 			$new_font_value = trim($new_font_value);
 | |
| 
 | |
| 			// Delete all font-properties
 | |
| 			foreach ($font_prop_default as $font_property => $default_value) {
 | |
| 				if ($font_property !== 'font' || !$new_font_value)
 | |
| 					unset($input_css[$font_property]);
 | |
| 			}
 | |
| 
 | |
| 			// Add new font property
 | |
| 			if ($new_font_value !== '') {
 | |
| 				$input_css['font'] = $new_font_value . $important;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return $input_css;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reverse left vs right in a list of properties/values
 | |
| 	 * @param array $array
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function reverse_left_and_right($array) {
 | |
| 		$return = array();
 | |
| 
 | |
| 		// change left <-> right in properties name and values
 | |
| 		foreach ($array as $propertie => $value) {
 | |
| 
 | |
| 			if (method_exists($this, $m = 'reverse_left_and_right_' . str_replace('-','_',trim($propertie)))) {
 | |
| 				$value = $this->$m($value);
 | |
| 			}
 | |
| 
 | |
| 			// simple replacement for properties
 | |
| 			$propertie = str_ireplace(array('left', 'right' ,"\x1"), array("\x1", 'left', 'right') , $propertie);
 | |
| 			// be careful for values, not modifying protected or quoted valued
 | |
| 			foreach (array('left' => "\x1", 'right' => 'left', "\x1" => 'right') as $v => $r) {
 | |
| 				if (strpos($value, $v) !== false) {
 | |
| 					// attraper les left et right separes du reste (pas au milieu d'un mot)
 | |
| 					if (in_array($v, array('left', 'right') )) {
 | |
| 						$value = preg_replace(",\\b$v\\b,", "\x0" , $value);
 | |
| 					}
 | |
| 					else {
 | |
| 						$value = str_replace($v, "\x0" , $value);
 | |
| 					}
 | |
| 					$value = $this->explode_ws("\x0", $value . ' ', true);
 | |
| 					$value = rtrim(implode($r, $value));
 | |
| 					$value = str_replace("\x0" , $v, $value);
 | |
| 				}
 | |
| 			}
 | |
| 			$return[$propertie] = $value;
 | |
| 		}
 | |
| 
 | |
| 		return $return;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reversing 4 values shorthands properties
 | |
| 	 * @param string $value
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function reverse_left_and_right_4value_shorthands($property, $value) {
 | |
| 		$shorthands = & $this->parser->data['csstidy']['shorthands'];
 | |
| 		if (isset($shorthands[$property])) {
 | |
| 			$property_right = $shorthands[$property][1];
 | |
| 			$property_left = $shorthands[$property][3];
 | |
| 			$v = $this->dissolve_4value_shorthands($property, $value);
 | |
| 			if ($v[$property_left] !== $v[$property_right]) {
 | |
| 				$r = $v[$property_right];
 | |
| 				$v[$property_right] = $v[$property_left];
 | |
| 				$v[$property_left] = $r;
 | |
| 				$v = $this->merge_4value_shorthands($v);
 | |
| 				if (isset($v[$property])) {
 | |
| 					return $v[$property];
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		return $value;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reversing 4 values radius shorthands properties
 | |
| 	 * @param string $value
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function reverse_left_and_right_4value_radius_shorthands($property, $value) {
 | |
| 		$shorthands = & $this->parser->data['csstidy']['radius_shorthands'];
 | |
| 		if (isset($shorthands[$property])) {
 | |
| 			$v = $this->dissolve_4value_radius_shorthands($property, $value);
 | |
| 			if ($v[$shorthands[$property][0]] !== $v[$shorthands[$property][1]]
 | |
| 			  or $v[$shorthands[$property][2]] !== $v[$shorthands[$property][3]]) {
 | |
| 				$r = array(
 | |
| 					$shorthands[$property][0] => $v[$shorthands[$property][1]],
 | |
| 					$shorthands[$property][1] => $v[$shorthands[$property][0]],
 | |
| 					$shorthands[$property][2] => $v[$shorthands[$property][3]],
 | |
| 					$shorthands[$property][3] => $v[$shorthands[$property][2]],
 | |
| 				);
 | |
| 				$v = $this->merge_4value_radius_shorthands($r);
 | |
| 				if (isset($v[$property])) {
 | |
| 					return $v[$property];
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		return $value;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reversing margin shorthands
 | |
| 	 * @param string $value
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function reverse_left_and_right_margin($value) {
 | |
| 		return $this->reverse_left_and_right_4value_shorthands('margin', $value);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reversing padding shorthands
 | |
| 	 * @param string $value
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function reverse_left_and_right_padding($value) {
 | |
| 		return $this->reverse_left_and_right_4value_shorthands('padding', $value);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reversing border-color shorthands
 | |
| 	 * @param string $value
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function reverse_left_and_right_border_color($value) {
 | |
| 		return $this->reverse_left_and_right_4value_shorthands('border-color', $value);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reversing border-style shorthands
 | |
| 	 * @param string $value
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function reverse_left_and_right_border_style($value) {
 | |
| 		return $this->reverse_left_and_right_4value_shorthands('border-style', $value);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reversing border-width shorthands
 | |
| 	 * @param string $value
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function reverse_left_and_right_border_width($value) {
 | |
| 		return $this->reverse_left_and_right_4value_shorthands('border-width', $value);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reversing border-radius shorthands
 | |
| 	 * @param string $value
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function reverse_left_and_right_border_radius($value) {
 | |
| 		return $this->reverse_left_and_right_4value_radius_shorthands('border-radius', $value);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reversing border-radius shorthands
 | |
| 	 * @param string $value
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function reverse_left_and_right__moz_border_radius($value) {
 | |
| 		return $this->reverse_left_and_right_4value_radius_shorthands('border-radius', $value);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reversing border-radius shorthands
 | |
| 	 * @param string $value
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function reverse_left_and_right__webkit_border_radius($value) {
 | |
| 		return $this->reverse_left_and_right_4value_radius_shorthands('border-radius', $value);
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Reversing background shorthands
 | |
| 	 * @param string $value
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function reverse_left_and_right_background($value) {
 | |
| 		$values = $this->dissolve_short_bg($value);
 | |
| 		if (isset($values['background-position']) and $values['background-position']) {
 | |
| 			$v = $this->reverse_left_and_right_background_position($values['background-position']);
 | |
| 			if ($v !== $values['background-position']) {
 | |
| 				if ($value == $values['background-position']) {
 | |
| 					return $v;
 | |
| 				}
 | |
| 				else {
 | |
| 					$values['background-position'] = $v;
 | |
| 					$x = $this->merge_bg($values);
 | |
| 					if (isset($x['background'])) {
 | |
| 						return $x['background'];
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		return $value;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reversing background position shorthands
 | |
| 	 * @param string $value
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function reverse_left_and_right_background_position_x($value) {
 | |
| 		return $this->reverse_left_and_right_background_position($value);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reversing background position shorthands
 | |
| 	 * @param string $value
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function reverse_left_and_right_background_position($value) {
 | |
| 		// multiple background case
 | |
| 		if (strpos($value, ',') !== false) {
 | |
| 			$values = $this->explode_ws(',', $value);
 | |
| 			if (count($values) > 1) {
 | |
| 				foreach ($values as $k=>$v) {
 | |
| 					$values[$k] = $this->reverse_left_and_right_background_position($v);
 | |
| 				}
 | |
| 				return implode(',', $values);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// if no explicit left or right value
 | |
| 		if (stripos($value, 'left') === false and stripos($value, 'right') === false) {
 | |
| 			$values = $this->explode_ws(' ', trim($value));
 | |
| 			$values = array_map('trim', $values);
 | |
| 			$values = array_filter($values, function ($v) { return strlen($v);});
 | |
| 			$values = array_values($values);
 | |
| 			if (count($values) == 1) {
 | |
| 				if (in_array($value, array('center', 'top', 'bottom', 'inherit', 'initial', 'unset'))) {
 | |
| 					return $value;
 | |
| 				}
 | |
| 				return "left $value";
 | |
| 			}
 | |
| 			if ($values[1] == 'top' or $values[1] == 'bottom') {
 | |
| 				if ($values[0] === 'center') {
 | |
| 					return $value;
 | |
| 				}
 | |
| 				return 'left ' . implode(' ', $values);
 | |
| 			}
 | |
| 			else {
 | |
| 				$last = array_pop($values);
 | |
| 				if ($last === 'center') {
 | |
| 					return $value;
 | |
| 				}
 | |
| 				return implode(' ', $values) . ' left ' . $last;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return $value;
 | |
| 	}
 | |
| 
 | |
| }
 |