1
0
mirror of https://github.com/mrclay/minify.git synced 2025-08-09 15:46:34 +02:00

Now substring and str_slice methods match exactly their Javascript counterparts behavior.

Minor cleaning/formatting.
Tabs to spaces (Clay seems to use spaces instead of tabs)
This commit is contained in:
tubalmartin
2012-03-16 20:48:40 +01:00
parent ff45c76484
commit 73dd77569e

View File

@@ -13,18 +13,18 @@
*/ */
/*! /*!
* YUI Compressor * YUI Compressor
* http://developer.yahoo.com/yui/compressor/ * http://developer.yahoo.com/yui/compressor/
* Author: Julien Lecomte - http://www.julienlecomte.net/ * Author: Julien Lecomte - http://www.julienlecomte.net/
* Copyright (c) 2011 Yahoo! Inc. All rights reserved. * Copyright (c) 2011 Yahoo! Inc. All rights reserved.
* The copyrights embodied in the content of this file are licensed * The copyrights embodied in the content of this file are licensed
* by Yahoo! Inc. under the BSD (revised) open source license. * by Yahoo! Inc. under the BSD (revised) open source license.
*/ */
class CSSmin class CSSmin
{ {
private $comments = array(); private $comments;
private $preserved_tokens = array(); private $preserved_tokens;
/** /**
* @param bool $raisePhpSettingsLimits if true, raisePhpSettingLimits() will * @param bool $raisePhpSettingsLimits if true, raisePhpSettingLimits() will
@@ -43,92 +43,86 @@ class CSSmin
* @param int|bool $linebreak_pos * @param int|bool $linebreak_pos
* @return string * @return string
*/ */
public function run($css, $linebreak_pos = FALSE) public function run($css, $linebreak_pos = FALSE)
{ {
// Try to increase the memory limit for this script
ini_set('memory_limit', '128M');
// Try to increase the PCRE limits
ini_set('pcre.backtrack_limit', 1000 * 1000);
ini_set('pcre.recursion_limit', 500 * 1000);
$this->comments = array(); $this->comments = array();
$this->preserved_tokens = array(); $this->preserved_tokens = array();
$start_index = 0; $start_index = 0;
$length = strlen($css); $length = strlen($css);
$css = $this->extract_data_urls($css); $css = $this->extract_data_urls($css);
// collect all comment blocks... // collect all comment blocks...
while (($start_index = $this->index_of($css, '/*', $start_index)) >= 0) { while (($start_index = $this->index_of($css, '/*', $start_index)) >= 0) {
$end_index = $this->index_of($css, '*/', $start_index + 2); $end_index = $this->index_of($css, '*/', $start_index + 2);
if ($end_index < 0) { if ($end_index < 0) {
$end_index = $length; $end_index = $length;
} }
$this->comments[] = $this->str_slice($css, $start_index + 2, $end_index); $this->comments[] = $this->str_slice($css, $start_index + 2, $end_index);
$css = $this->str_slice($css, 0, $start_index + 2) . '___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_' . (count($this->comments) - 1) . '___' . $this->str_slice($css, $end_index); $css = $this->str_slice($css, 0, $start_index + 2) . '___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_' . (count($this->comments) - 1) . '___' . $this->str_slice($css, $end_index);
$start_index += 2; $start_index += 2;
} }
// preserve strings so their content doesn't get accidentally minified // preserve strings so their content doesn't get accidentally minified
$css = preg_replace_callback('/(?:"(?:[^\\\\"]|\\\\.|\\\\)*")|'."(?:'(?:[^\\\\']|\\\\.|\\\\)*')/", array($this, 'callback_one'), $css); $css = preg_replace_callback('/(?:"(?:[^\\\\"]|\\\\.|\\\\)*")|'."(?:'(?:[^\\\\']|\\\\.|\\\\)*')/", array($this, 'callback_one'), $css);
// Let's divide css code in chunks of 25.000 chars aprox. // Let's divide css code in chunks of 25.000 chars aprox.
// Reason: PHP's PCRE functions like preg_replace have a "backtrack limit" of 100.000 chars by default (php < 5.3.7) // Reason: PHP's PCRE functions like preg_replace have a "backtrack limit" of 100.000 chars by default (php < 5.3.7)
// so if we're dealing with really long strings and a (sub)pattern matches a number of chars greater than // so if we're dealing with really long strings and a (sub)pattern matches a number of chars greater than
// the backtrack limit number (i.e. /(.*)/s) PCRE functions may fail silently returning NULL and // the backtrack limit number (i.e. /(.*)/s) PCRE functions may fail silently returning NULL and
// $css would be empty. // $css would be empty.
$charset = ''; $charset = '';
$charset_regexp = '/@charset [^;]+;/i'; $charset_regexp = '/@charset [^;]+;/i';
$css_chunks = array(); $css_chunks = array();
$css_chunk_length = 25000; // aprox size, not exact $css_chunk_length = 25000; // aprox size, not exact
$start_index = 0; $start_index = 0;
$i = $css_chunk_length; // save initial iterations $i = $css_chunk_length; // save initial iterations
$l = strlen($css); $l = strlen($css);
// if the number of characters is 25000 or less, do not chunk // if the number of characters is 25000 or less, do not chunk
if ($l <= $css_chunk_length) { if ($l <= $css_chunk_length) {
$css_chunks[] = $css; $css_chunks[] = $css;
} else { } else {
// chunk css code securely // chunk css code securely
while ($i < $l) { while ($i < $l) {
$i += 50; // save iterations. 500 checks for a closing curly brace } $i += 50; // save iterations. 500 checks for a closing curly brace }
if ($l - $start_index <= $css_chunk_length || $i >= $l) { if ($l - $start_index <= $css_chunk_length || $i >= $l) {
$css_chunks[] = $this->str_slice($css, $start_index); $css_chunks[] = $this->str_slice($css, $start_index);
break; break;
} }
if ($css[$i - 1] === '}' && $i - $start_index > $css_chunk_length) { if ($css[$i - 1] === '}' && $i - $start_index > $css_chunk_length) {
// If there are two ending curly braces }} separated or not by spaces, // If there are two ending curly braces }} separated or not by spaces,
// join them in the same chunk (i.e. @media blocks) // join them in the same chunk (i.e. @media blocks)
$next_chunk = substr($css, $i); $next_chunk = substr($css, $i);
if (preg_match('/^\s*\}/', $next_chunk)) { if (preg_match('/^\s*\}/', $next_chunk)) {
$i = $i + $this->index_of($next_chunk, '}') + 1; $i = $i + $this->index_of($next_chunk, '}') + 1;
} }
$css_chunks[] = $this->str_slice($css, $start_index, $i); $css_chunks[] = $this->str_slice($css, $start_index, $i);
$start_index = $i; $start_index = $i;
} }
} }
} }
// Minify each chunk // Minify each chunk
for ($i = 0, $n = count($css_chunks); $i < $n; $i++) { for ($i = 0, $n = count($css_chunks); $i < $n; $i++) {
$css_chunks[$i] = $this->minify($css_chunks[$i], $linebreak_pos); $css_chunks[$i] = $this->minify($css_chunks[$i], $linebreak_pos);
// If there is a @charset in a css chunk... // If there is a @charset in a css chunk...
if (empty($charset) && preg_match($charset_regexp, $css_chunks[$i], $matches)) { if (empty($charset) && preg_match($charset_regexp, $css_chunks[$i], $matches)) {
// delete all of them no matter the chunk // delete all of them no matter the chunk
$css_chunks[$i] = preg_replace($charset_regexp, '', $css_chunks[$i]); $css_chunks[$i] = preg_replace($charset_regexp, '', $css_chunks[$i]);
$charset = $matches[0]; $charset = $matches[0];
} }
} }
// Update the first chunk and put the charset to the top of the file. // Update the first chunk and put the charset to the top of the file.
$css_chunks[0] = $charset . $css_chunks[0]; $css_chunks[0] = $charset . $css_chunks[0];
return implode('', $css_chunks); return implode('', $css_chunks);
} }
/** /**
* Get the minimum PHP setting values suggested for CSSmin * Get the minimum PHP setting values suggested for CSSmin
@@ -165,139 +159,139 @@ class CSSmin
* @param int|bool $linebreak_pos * @param int|bool $linebreak_pos
* @return string * @return string
*/ */
private function minify($css, $linebreak_pos) private function minify($css, $linebreak_pos)
{ {
// strings are safe, now wrestle the comments // strings are safe, now wrestle the comments
for ($i = 0, $max = count($this->comments); $i < $max; $i++) { for ($i = 0, $max = count($this->comments); $i < $max; $i++) {
$token = $this->comments[$i]; $token = $this->comments[$i];
$placeholder = '/___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_' . $i . '___/'; $placeholder = '/___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_' . $i . '___/';
// ! in the first position of the comment means preserve // ! in the first position of the comment means preserve
// so push to the preserved tokens keeping the ! // so push to the preserved tokens keeping the !
if (substr($token, 0, 1) === '!') { if (substr($token, 0, 1) === '!') {
$this->preserved_tokens[] = $token; $this->preserved_tokens[] = $token;
$css = preg_replace($placeholder, '___YUICSSMIN_PRESERVED_TOKEN_' . (count($this->preserved_tokens) - 1) . '___', $css, 1); $css = preg_replace($placeholder, '___YUICSSMIN_PRESERVED_TOKEN_' . (count($this->preserved_tokens) - 1) . '___', $css, 1);
continue; continue;
} }
// \ in the last position looks like hack for Mac/IE5 // \ in the last position looks like hack for Mac/IE5
// shorten that to /*\*/ and the next one to /**/ // shorten that to /*\*/ and the next one to /**/
if (substr($token, (strlen($token) - 1), 1) === '\\') { if (substr($token, (strlen($token) - 1), 1) === '\\') {
$this->preserved_tokens[] = '\\'; $this->preserved_tokens[] = '\\';
$css = preg_replace($placeholder, '___YUICSSMIN_PRESERVED_TOKEN_' . (count($this->preserved_tokens) - 1) . '___', $css, 1); $css = preg_replace($placeholder, '___YUICSSMIN_PRESERVED_TOKEN_' . (count($this->preserved_tokens) - 1) . '___', $css, 1);
$i = $i + 1; // attn: advancing the loop $i = $i + 1; // attn: advancing the loop
$this->preserved_tokens[] = ''; $this->preserved_tokens[] = '';
$css = preg_replace('/___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_' . $i . '___/', '___YUICSSMIN_PRESERVED_TOKEN_' . (count($this->preserved_tokens) - 1) . '___', $css, 1); $css = preg_replace('/___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_' . $i . '___/', '___YUICSSMIN_PRESERVED_TOKEN_' . (count($this->preserved_tokens) - 1) . '___', $css, 1);
continue; continue;
} }
// keep empty comments after child selectors (IE7 hack) // keep empty comments after child selectors (IE7 hack)
// e.g. html >/**/ body // e.g. html >/**/ body
if (strlen($token) === 0) { if (strlen($token) === 0) {
$start_index = $this->index_of($css, $this->str_slice($placeholder, 1, -1)); $start_index = $this->index_of($css, $this->str_slice($placeholder, 1, -1));
if ($start_index > 2) { if ($start_index > 2) {
if (substr($css, $start_index - 3, 1) === '>') { if (substr($css, $start_index - 3, 1) === '>') {
$this->preserved_tokens[] = ''; $this->preserved_tokens[] = '';
$css = preg_replace($placeholder, '___YUICSSMIN_PRESERVED_TOKEN_' . (count($this->preserved_tokens) - 1) . '___', $css, 1); $css = preg_replace($placeholder, '___YUICSSMIN_PRESERVED_TOKEN_' . (count($this->preserved_tokens) - 1) . '___', $css, 1);
} }
} }
} }
// in all other cases kill the comment // in all other cases kill the comment
$css = preg_replace('/\/\*' . $this->str_slice($placeholder, 1, -1) . '\*\//', '', $css, 1); $css = preg_replace('/\/\*' . $this->str_slice($placeholder, 1, -1) . '\*\//', '', $css, 1);
} }
// Normalize all whitespace strings to single spaces. Easier to work with that way. // Normalize all whitespace strings to single spaces. Easier to work with that way.
$css = preg_replace('/\s+/', ' ', $css); $css = preg_replace('/\s+/', ' ', $css);
// Remove the spaces before the things that should not have spaces before them. // Remove the spaces before the things that should not have spaces before them.
// But, be careful not to turn "p :link {...}" into "p:link{...}" // But, be careful not to turn "p :link {...}" into "p:link{...}"
// Swap out any pseudo-class colons with the token, and then swap back. // Swap out any pseudo-class colons with the token, and then swap back.
$css = preg_replace_callback('/(?:^|\})(?:(?:[^\{\:])+\:)+(?:[^\{]*\{)/', array($this, 'callback_two'), $css); $css = preg_replace_callback('/(?:^|\})(?:(?:[^\{\:])+\:)+(?:[^\{]*\{)/', array($this, 'callback_two'), $css);
$css = preg_replace('/\s+([\!\{\}\;\:\>\+\(\)\],])/', '$1', $css); $css = preg_replace('/\s+([\!\{\}\;\:\>\+\(\)\],])/', '$1', $css);
$css = preg_replace('/___YUICSSMIN_PSEUDOCLASSCOLON___/', ':', $css); $css = preg_replace('/___YUICSSMIN_PSEUDOCLASSCOLON___/', ':', $css);
// retain space for special IE6 cases // retain space for special IE6 cases
$css = preg_replace('/\:first\-(line|letter)(\{|,)/', ':first-$1 $2', $css); $css = preg_replace('/\:first\-(line|letter)(\{|,)/', ':first-$1 $2', $css);
// no space after the end of a preserved comment // no space after the end of a preserved comment
$css = preg_replace('/\*\/ /', '*/', $css); $css = preg_replace('/\*\/ /', '*/', $css);
// Put the space back in some cases, to support stuff like // Put the space back in some cases, to support stuff like
// @media screen and (-webkit-min-device-pixel-ratio:0){ // @media screen and (-webkit-min-device-pixel-ratio:0){
$css = preg_replace('/\band\(/i', 'and (', $css); $css = preg_replace('/\band\(/i', 'and (', $css);
// Remove the spaces after the things that should not have spaces after them. // Remove the spaces after the things that should not have spaces after them.
$css = preg_replace('/([\!\{\}\:;\>\+\(\[,])\s+/', '$1', $css); $css = preg_replace('/([\!\{\}\:;\>\+\(\[,])\s+/', '$1', $css);
// remove unnecessary semicolons // remove unnecessary semicolons
$css = preg_replace('/;+\}/', '}', $css); $css = preg_replace('/;+\}/', '}', $css);
// Replace 0(px,em,%) with 0. // Replace 0(px,em,%) with 0.
$css = preg_replace('/([\s\:])(0)(?:px|em|%|in|cm|mm|pc|pt|ex)/i', '$1$2', $css); $css = preg_replace('/([\s\:])(0)(?:px|em|%|in|cm|mm|pc|pt|ex)/i', '$1$2', $css);
// Replace 0 0 0 0; with 0. // Replace 0 0 0 0; with 0.
$css = preg_replace('/\:0 0 0 0(;|\})/', ':0$1', $css); $css = preg_replace('/\:0 0 0 0(;|\})/', ':0$1', $css);
$css = preg_replace('/\:0 0 0(;|\})/', ':0$1', $css); $css = preg_replace('/\:0 0 0(;|\})/', ':0$1', $css);
$css = preg_replace('/\:0 0(;|\})/', ':0$1', $css); $css = preg_replace('/\:0 0(;|\})/', ':0$1', $css);
// Replace background-position:0; with background-position:0 0; // Replace background-position:0; with background-position:0 0;
// same for transform-origin // same for transform-origin
$css = preg_replace_callback('/(background\-position|transform\-origin|webkit\-transform\-origin|moz\-transform\-origin|o-transform\-origin|ms\-transform\-origin)\:0(;|\})/i', array($this, 'callback_three'), $css); $css = preg_replace_callback('/(background\-position|transform\-origin|webkit\-transform\-origin|moz\-transform\-origin|o-transform\-origin|ms\-transform\-origin)\:0(;|\})/i', array($this, 'callback_three'), $css);
// Replace 0.6 to .6, but only when preceded by : or a white-space // Replace 0.6 to .6, but only when preceded by : or a white-space
$css = preg_replace('/(\:|\s)0+\.(\d+)/', '$1.$2', $css); $css = preg_replace('/(\:|\s)0+\.(\d+)/', '$1.$2', $css);
// Shorten colors from rgb(51,102,153) to #336699 // Shorten colors from rgb(51,102,153) to #336699
// This makes it more likely that it'll get further compressed in the next step. // This makes it more likely that it'll get further compressed in the next step.
$css = preg_replace_callback('/rgb\s*\(\s*([0-9,\s]+)\s*\)/i', array($this, 'callback_four'), $css); $css = preg_replace_callback('/rgb\s*\(\s*([0-9,\s]+)\s*\)/i', array($this, 'callback_four'), $css);
// Shorten colors from #AABBCC to #ABC. // Shorten colors from #AABBCC to #ABC.
$css = $this->compress_hex_colors($css); $css = $this->compress_hex_colors($css);
// border: none -> border:0 // border: none -> border:0
$css = preg_replace_callback('/(border|border\-top|border\-right|border\-bottom|border\-right|outline|background)\:none(;|\})/i', array($this, 'callback_five'), $css); $css = preg_replace_callback('/(border|border\-top|border\-right|border\-bottom|border\-right|outline|background)\:none(;|\})/i', array($this, 'callback_five'), $css);
// shorter opacity IE filter // shorter opacity IE filter
$css = preg_replace('/progid\:DXImageTransform\.Microsoft\.Alpha\(Opacity\=/i', 'alpha(opacity=', $css); $css = preg_replace('/progid\:DXImageTransform\.Microsoft\.Alpha\(Opacity\=/i', 'alpha(opacity=', $css);
// Remove empty rules. // Remove empty rules.
$css = preg_replace('/[^\};\{\/]+\{\}/', '', $css); $css = preg_replace('/[^\};\{\/]+\{\}/', '', $css);
// Some source control tools don't like it when files containing lines longer // Some source control tools don't like it when files containing lines longer
// than, say 8000 characters, are checked in. The linebreak option is used in // than, say 8000 characters, are checked in. The linebreak option is used in
// that case to split long lines after a specific column. // that case to split long lines after a specific column.
if ($linebreak_pos !== FALSE && (int) $linebreak_pos >= 0) { if ($linebreak_pos !== FALSE && (int) $linebreak_pos >= 0) {
$linebreak_pos = (int) $linebreak_pos; $linebreak_pos = (int) $linebreak_pos;
$start_index = $i = 0; $start_index = $i = 0;
while ($i < strlen($css)) { while ($i < strlen($css)) {
$i++; $i++;
if ($css[$i - 1] === '}' && $i - $start_index > $linebreak_pos) { if ($css[$i - 1] === '}' && $i - $start_index > $linebreak_pos) {
$css = $this->str_slice($css, 0, $i) . "\n" . $this->str_slice($css, $i); $css = $this->str_slice($css, 0, $i) . "\n" . $this->str_slice($css, $i);
$start_index = $i; $start_index = $i;
} }
} }
} }
// Replace multiple semi-colons in a row by a single one // Replace multiple semi-colons in a row by a single one
// See SF bug #1980989 // See SF bug #1980989
$css = preg_replace('/;;+/', ';', $css); $css = preg_replace('/;;+/', ';', $css);
// restore preserved comments and strings // restore preserved comments and strings
for ($i = 0, $max = count($this->preserved_tokens); $i < $max; $i++) { for ($i = 0, $max = count($this->preserved_tokens); $i < $max; $i++) {
$css = preg_replace('/___YUICSSMIN_PRESERVED_TOKEN_' . $i . '___/', $this->preserved_tokens[$i], $css, 1); $css = preg_replace('/___YUICSSMIN_PRESERVED_TOKEN_' . $i . '___/', $this->preserved_tokens[$i], $css, 1);
} }
// Trim the final string (for any leading or trailing white spaces) // Trim the final string (for any leading or trailing white spaces)
$css = preg_replace('/^\s+|\s+$/', '', $css); $css = preg_replace('/^\s+|\s+$/', '', $css);
return $css; return $css;
} }
/** /**
* Utility method to replace all data urls with tokens before we start * Utility method to replace all data urls with tokens before we start
* compressing, to avoid performance issues running some of the subsequent * compressing, to avoid performance issues running some of the subsequent
* regexes against large strings chunks. * regexes against large strings chunks.
@@ -305,271 +299,262 @@ class CSSmin
* @param string $css * @param string $css
* @return string * @return string
*/ */
private function extract_data_urls($css) private function extract_data_urls($css)
{ {
// Leave data urls alone to increase parse performance. // Leave data urls alone to increase parse performance.
$max_index = strlen($css) - 1; $max_index = strlen($css) - 1;
$append_index = $index = $last_index = $offset = 0; $append_index = $index = $last_index = $offset = 0;
$sb = array(); $sb = array();
$pattern = '/url\(\s*(["\']?)data\:/'; $pattern = '/url\(\s*(["\']?)data\:/';
// Since we need to account for non-base64 data urls, we need to handle // Since we need to account for non-base64 data urls, we need to handle
// ' and ) being part of the data string. Hence switching to indexOf, // ' and ) being part of the data string. Hence switching to indexOf,
// to determine whether or not we have matching string terminators and // to determine whether or not we have matching string terminators and
// handling sb appends directly, instead of using matcher.append* methods. // handling sb appends directly, instead of using matcher.append* methods.
while (preg_match($pattern, $css, $m, 0, $offset)) { while (preg_match($pattern, $css, $m, 0, $offset)) {
$index = $this->index_of($css, $m[0], $offset); $index = $this->index_of($css, $m[0], $offset);
$last_index = $index + strlen($m[0]); $last_index = $index + strlen($m[0]);
$start_index = $index + 4; // "url(".length() $start_index = $index + 4; // "url(".length()
$end_index = $last_index - 1; $end_index = $last_index - 1;
$terminator = $m[1]; // ', " or empty (not quoted) $terminator = $m[1]; // ', " or empty (not quoted)
$found_terminator = FALSE; $found_terminator = FALSE;
if (strlen($terminator) === 0) { if (strlen($terminator) === 0) {
$terminator = ')'; $terminator = ')';
} }
while ($found_terminator === FALSE && $end_index+1 <= $max_index) { while ($found_terminator === FALSE && $end_index+1 <= $max_index) {
$end_index = $this->index_of($css, $terminator, $end_index + 1); $end_index = $this->index_of($css, $terminator, $end_index + 1);
// endIndex == 0 doesn't really apply here // endIndex == 0 doesn't really apply here
if ($end_index > 0 && substr($css, $end_index - 1, 1) !== '\\') { if ($end_index > 0 && substr($css, $end_index - 1, 1) !== '\\') {
$found_terminator = TRUE; $found_terminator = TRUE;
if (')' != $terminator) { if (')' != $terminator) {
$end_index = $this->index_of($css, ')', $end_index); $end_index = $this->index_of($css, ')', $end_index);
} }
} }
} }
// Enough searching, start moving stuff over to the buffer // Enough searching, start moving stuff over to the buffer
$sb[] = $this->substring($css, $append_index, $index); $sb[] = $this->substring($css, $append_index, $index);
if ($found_terminator) { if ($found_terminator) {
$token = $this->substring($css, $start_index, $end_index); $token = $this->substring($css, $start_index, $end_index);
$token = preg_replace('/\s+/', '', $token); $token = preg_replace('/\s+/', '', $token);
$this->preserved_tokens[] = $token; $this->preserved_tokens[] = $token;
$preserver = 'url(___YUICSSMIN_PRESERVED_TOKEN_' . (count($this->preserved_tokens) - 1) . '___)'; $preserver = 'url(___YUICSSMIN_PRESERVED_TOKEN_' . (count($this->preserved_tokens) - 1) . '___)';
$sb[] = $preserver; $sb[] = $preserver;
$append_index = $end_index + 1; $append_index = $end_index + 1;
} else { } else {
// No end terminator found, re-add the whole match. Should we throw/warn here? // No end terminator found, re-add the whole match. Should we throw/warn here?
$sb[] = $this->substring($css, $index, $last_index); $sb[] = $this->substring($css, $index, $last_index);
$append_index = $last_index; $append_index = $last_index;
} }
$offset = $last_index; $offset = $last_index;
} }
$sb[] = $this->substring($css, $append_index); $sb[] = $this->substring($css, $append_index);
return implode('', $sb); return implode('', $sb);
} }
/** /**
* Utility method to compress hex color values of the form #AABBCC to #ABC. * Utility method to compress hex color values of the form #AABBCC to #ABC.
* *
* DOES NOT compress CSS ID selectors which match the above pattern (which would break things). * DOES NOT compress CSS ID selectors which match the above pattern (which would break things).
* e.g. #AddressForm { ... } * e.g. #AddressForm { ... }
* *
* DOES NOT compress IE filters, which have hex color values (which would break things). * DOES NOT compress IE filters, which have hex color values (which would break things).
* e.g. filter: chroma(color="#FFFFFF"); * e.g. filter: chroma(color="#FFFFFF");
* *
* DOES NOT compress invalid hex values. * DOES NOT compress invalid hex values.
* e.g. background-color: #aabbccdd * e.g. background-color: #aabbccdd
* *
* @param string $css * @param string $css
* @return string * @return string
*/ */
private function compress_hex_colors($css) private function compress_hex_colors($css)
{ {
// Look for hex colors inside { ... } (to avoid IDs) and which don't have a =, or a " in front of them (to avoid filters) // Look for hex colors inside { ... } (to avoid IDs) and which don't have a =, or a " in front of them (to avoid filters)
$pattern = '/(\=\s*?["\']?)?#([0-9a-f])([0-9a-f])([0-9a-f])([0-9a-f])([0-9a-f])([0-9a-f])(\}|[^0-9a-f{][^{]*?\})/i'; $pattern = '/(\=\s*?["\']?)?#([0-9a-f])([0-9a-f])([0-9a-f])([0-9a-f])([0-9a-f])([0-9a-f])(\}|[^0-9a-f{][^{]*?\})/i';
$_index = $index = $last_index = $offset = 0; $_index = $index = $last_index = $offset = 0;
$sb = array(); $sb = array();
while (preg_match($pattern, $css, $m, 0, $offset)) { while (preg_match($pattern, $css, $m, 0, $offset)) {
$index = $this->index_of($css, $m[0], $offset); $index = $this->index_of($css, $m[0], $offset);
$last_index = $index + strlen($m[0]); $last_index = $index + strlen($m[0]);
$is_filter = (bool) $m[1]; $is_filter = (bool) $m[1];
$sb[] = $this->substring($css, $_index, $index); $sb[] = $this->substring($css, $_index, $index);
if ($is_filter) { if ($is_filter) {
// Restore, maintain case, otherwise filter will break // Restore, maintain case, otherwise filter will break
$sb[] = $m[1] . '#' . $m[2] . $m[3] . $m[4] . $m[5] . $m[6] . $m[7]; $sb[] = $m[1] . '#' . $m[2] . $m[3] . $m[4] . $m[5] . $m[6] . $m[7];
} else { } else {
if (strtolower($m[2]) == strtolower($m[3]) && if (strtolower($m[2]) == strtolower($m[3]) &&
strtolower($m[4]) == strtolower($m[5]) && strtolower($m[4]) == strtolower($m[5]) &&
strtolower($m[6]) == strtolower($m[7])) { strtolower($m[6]) == strtolower($m[7])) {
// Compress. // Compress.
$sb[] = '#' . strtolower($m[3] . $m[5] . $m[7]); $sb[] = '#' . strtolower($m[3] . $m[5] . $m[7]);
} else { } else {
// Non compressible color, restore but lower case. // Non compressible color, restore but lower case.
$sb[] = '#' . strtolower($m[2] . $m[3] . $m[4] . $m[5] . $m[6] . $m[7]); $sb[] = '#' . strtolower($m[2] . $m[3] . $m[4] . $m[5] . $m[6] . $m[7]);
} }
} }
$_index = $offset = $last_index - strlen($m[8]); $_index = $offset = $last_index - strlen($m[8]);
} }
$sb[] = $this->substring($css, $_index); $sb[] = $this->substring($css, $_index);
return implode('', $sb); return implode('', $sb);
} }
/* CALLBACKS /* CALLBACKS
* --------------------------------------------------------------------------------------------- * ---------------------------------------------------------------------------------------------
*/ */
private function callback_one($matches) private function callback_one($matches)
{ {
$match = $matches[0]; $match = $matches[0];
$quote = substr($match, 0, 1); $quote = substr($match, 0, 1);
// Must use addcslashes in PHP to avoid parsing of backslashes // Must use addcslashes in PHP to avoid parsing of backslashes
$match = addcslashes($this->str_slice($match, 1, -1), '\\'); $match = addcslashes($this->str_slice($match, 1, -1), '\\');
// maybe the string contains a comment-like substring? // maybe the string contains a comment-like substring?
// one, maybe more? put'em back then // one, maybe more? put'em back then
if (($pos = $this->index_of($match, '___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_')) >= 0) { if (($pos = $this->index_of($match, '___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_')) >= 0) {
for ($i = 0, $max = count($this->comments); $i < $max; $i++) { for ($i = 0, $max = count($this->comments); $i < $max; $i++) {
$match = preg_replace('/___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_' . $i . '___/', $this->comments[$i], $match, 1); $match = preg_replace('/___YUICSSMIN_PRESERVE_CANDIDATE_COMMENT_' . $i . '___/', $this->comments[$i], $match, 1);
} }
} }
// minify alpha opacity in filter strings // minify alpha opacity in filter strings
$match = preg_replace('/progid\:DXImageTransform\.Microsoft\.Alpha\(Opacity\=/i', 'alpha(opacity=', $match); $match = preg_replace('/progid\:DXImageTransform\.Microsoft\.Alpha\(Opacity\=/i', 'alpha(opacity=', $match);
$this->preserved_tokens[] = $match; $this->preserved_tokens[] = $match;
return $quote . '___YUICSSMIN_PRESERVED_TOKEN_' . (count($this->preserved_tokens) - 1) . '___' . $quote; return $quote . '___YUICSSMIN_PRESERVED_TOKEN_' . (count($this->preserved_tokens) - 1) . '___' . $quote;
} }
private function callback_two($matches) private function callback_two($matches)
{ {
return preg_replace('/\:/', '___YUICSSMIN_PSEUDOCLASSCOLON___', $matches[0]); return preg_replace('/\:/', '___YUICSSMIN_PSEUDOCLASSCOLON___', $matches[0]);
} }
private function callback_three($matches) private function callback_three($matches)
{ {
return strtolower($matches[1]) . ':0 0' . $matches[2]; return strtolower($matches[1]) . ':0 0' . $matches[2];
} }
private function callback_four($matches) private function callback_four($matches)
{ {
$rgbcolors = explode(',', $matches[1]); $rgbcolors = explode(',', $matches[1]);
for ($i = 0; $i < count($rgbcolors); $i++) { for ($i = 0; $i < count($rgbcolors); $i++) {
$rgbcolors[$i] = base_convert(strval(intval($rgbcolors[$i], 10)), 10, 16); $rgbcolors[$i] = base_convert(strval(intval($rgbcolors[$i], 10)), 10, 16);
if (strlen($rgbcolors[$i]) === 1) { if (strlen($rgbcolors[$i]) === 1) {
$rgbcolors[$i] = '0' . $rgbcolors[$i]; $rgbcolors[$i] = '0' . $rgbcolors[$i];
} }
} }
return '#' . implode('', $rgbcolors); return '#' . implode('', $rgbcolors);
} }
private function callback_five($matches) private function callback_five($matches)
{ {
return strtolower($matches[1]) . ':0' . $matches[2]; return strtolower($matches[1]) . ':0' . $matches[2];
} }
/* HELPERS /* HELPERS
* --------------------------------------------------------------------------------------------- * ---------------------------------------------------------------------------------------------
*/ */
/** /**
* PHP port of Javascript's "indexOf" function * PHP port of Javascript's "indexOf" function for strings only
* Author: Tubal Martin http://blog.margenn.com * Author: Tubal Martin http://blog.margenn.com
* *
* @param string $haystack * @param string $haystack
* @param string $needle * @param string $needle
* @param int $offset index (optional) * @param int $offset index (optional)
* @return int * @return int
*/ */
private function index_of($haystack, $needle, $offset = 0) private function index_of($haystack, $needle, $offset = 0)
{ {
$index = strpos($haystack, $needle, $offset); $index = strpos($haystack, $needle, $offset);
return ($index !== FALSE) ? $index : -1; return ($index !== FALSE) ? $index : -1;
} }
/** /**
* PHP port of Javascript's "substring" function * PHP port of Javascript's "substring" function
* Author: Tubal Martin http://blog.margenn.com * Author: Tubal Martin http://blog.margenn.com
* * Tests: http://margenn.com/tubal/substring/
* @param string $str *
* @param int $from index * @param string $str
* @param int|bool $to index (optional) * @param int $from index
* @param int|bool $to index (optional)
* @return string * @return string
*/ */
private function substring($str, $from, $to = FALSE) private function substring($str, $from = 0, $to = FALSE)
{ {
if ($from < 0) { if ($to !== FALSE) {
$from = 0; if ($from == $to || ($from <= 0 && $to < 0)) {
} return '';
}
if ($to !== FALSE) { if ($from > $to) {
$from_copy = $from;
$from = $to;
$to = $from_copy;
}
}
if ($from === $to || $to < 0) { if ($from < 0) {
return ''; $from = 0;
} }
if ($from > $to) { $substring = ($to === FALSE) ? substr($str, $from) : substr($str, $from, $to - $from);
$from_copy = $from; return ($substring === FALSE) ? '' : $substring;
$from = $to; }
$to = $from_copy;
}
$substring = substr($str, $from, $to - $from);
return ($substring === FALSE) ? '' : $substring;
}
$substring = substr($str, $from);
return ($substring === FALSE) ? '' : $substring;
}
/** /**
* PHP port of Javascript's "slice" function * PHP port of Javascript's "slice" function for strings only
* Author: Tubal Martin http://blog.margenn.com * Author: Tubal Martin http://blog.margenn.com
* Tests: http://margenn.com/tubal/str_slice/ * Tests: http://margenn.com/tubal/str_slice/
* *
* @param string $str * @param string $str
* @param int $start index * @param int $start index
* @param int|bool $end index (optional) * @param int|bool $end index (optional)
* @return string * @return string
*/ */
private function str_slice($str, $start, $end = FALSE) private function str_slice($str, $start = 0, $end = FALSE)
{ {
if ($start < 0 || $end <= 0) { if ($end !== FALSE && ($start < 0 || $end <= 0)) {
$max = strlen($str);
if ($end === FALSE) { if ($start < 0) {
$slice = substr($str, $start); if (($start = $max + $start) < 0) {
return ($slice === FALSE) ? '' : $slice; return '';
} }
}
$max = strlen($str); if ($end < 0) {
if (($end = $max + $end) < 0) {
return '';
}
}
if ($start < 0) { if ($end <= $start) {
if (($start = $max + $start) < 0) { return '';
return ''; }
} }
}
if ($end < 0) { $slice = ($end === FALSE) ? substr($str, $start) : substr($str, $start, $end - $start);
if (($end = $max + $end) < 0) { return ($slice === FALSE) ? '' : $slice;
return ''; }
}
}
if ($end <= $start) {
return '';
}
}
$slice = substr($str, $start, $end - $start);
return ($slice === FALSE) ? '' : $slice;
}
/** /**
* Convert strings like "64M" to int values * Convert strings like "64M" to int values