178a6aeb1SAndreas Gohr<?php 278a6aeb1SAndreas Gohr/** 378a6aeb1SAndreas Gohr * DokuWiki StyleSheet creator 478a6aeb1SAndreas Gohr * 578a6aeb1SAndreas Gohr * @license GPL 2 (http://www.gnu.org/licenses/gpl.html) 678a6aeb1SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 778a6aeb1SAndreas Gohr */ 878a6aeb1SAndreas Gohr 9d0a27cb0SAndreas Gohrif(!defined('DOKU_INC')) define('DOKU_INC',dirname(__FILE__).'/../../'); 101c2d1019SAndreas Gohrif(!defined('NOSESSION')) define('NOSESSION',true); // we do not use a session or authentication here (better caching) 1198bda4fdSAndreas Gohrif(!defined('DOKU_DISABLE_GZIP_OUTPUT')) define('DOKU_DISABLE_GZIP_OUTPUT',1); // we gzip ourself here 126c47a78cSAnika Henkeif(!defined('NL')) define('NL',"\n"); 1378a6aeb1SAndreas Gohrrequire_once(DOKU_INC.'inc/init.php'); 1478a6aeb1SAndreas Gohr 1578a6aeb1SAndreas Gohr// Main (don't run when UNIT test) 1678a6aeb1SAndreas Gohrif(!defined('SIMPLE_TEST')){ 1778a6aeb1SAndreas Gohr header('Content-Type: text/css; charset=utf-8'); 1878a6aeb1SAndreas Gohr css_out(); 1978a6aeb1SAndreas Gohr} 2078a6aeb1SAndreas Gohr 2178a6aeb1SAndreas Gohr 2278a6aeb1SAndreas Gohr// ---------------------- functions ------------------------------ 2378a6aeb1SAndreas Gohr 2478a6aeb1SAndreas Gohr/** 2578a6aeb1SAndreas Gohr * Output all needed Styles 2678a6aeb1SAndreas Gohr * 2778a6aeb1SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 2878a6aeb1SAndreas Gohr */ 2978a6aeb1SAndreas Gohrfunction css_out(){ 3078a6aeb1SAndreas Gohr global $conf; 3178a6aeb1SAndreas Gohr global $lang; 3209edb711SAndreas Gohr global $config_cascade; 33bfd0f597STom N Harris global $INPUT; 3409edb711SAndreas Gohr 3529f2dfdcSAndreas Gohr if ($INPUT->str('s') == 'feed') { 366c47a78cSAnika Henke $mediatypes = array('feed'); 376c47a78cSAnika Henke $type = 'feed'; 386c47a78cSAnika Henke } else { 396c47a78cSAnika Henke $mediatypes = array('screen', 'all', 'print'); 406c47a78cSAnika Henke $type = ''; 41615960feSTom N Harris } 4278a6aeb1SAndreas Gohr 43bfd0f597STom N Harris $tpl = trim(preg_replace('/[^\w-]+/','',$INPUT->str('t'))); 44124af657SAndreas Gohr if($tpl){ 45124af657SAndreas Gohr $tplinc = DOKU_INC.'lib/tpl/'.$tpl.'/'; 46124af657SAndreas Gohr $tpldir = DOKU_BASE.'lib/tpl/'.$tpl.'/'; 47124af657SAndreas Gohr }else{ 4881aca18eSAndreas Gohr $tplinc = tpl_incdir(); 4981aca18eSAndreas Gohr $tpldir = tpl_basedir(); 50124af657SAndreas Gohr } 51124af657SAndreas Gohr 520e6f9f08SAnika Henke // used style.ini file 530e6f9f08SAnika Henke $styleini = css_styleini($tplinc); 540e6f9f08SAnika Henke 5578a6aeb1SAndreas Gohr // The generated script depends on some dynamic options 566c47a78cSAnika Henke $cache = new cache('styles'.$_SERVER['HTTP_HOST'].$_SERVER['SERVER_PORT'].DOKU_BASE.$tplinc.$type,'.css'); 5778a6aeb1SAndreas Gohr 58519b3173SAndreas Gohr // load template styles 59519b3173SAndreas Gohr $tplstyles = array(); 600e6f9f08SAnika Henke if ($styleini) { 610ac69508SAnika Henke foreach($styleini['stylesheets'] as $file => $mode) { 62124af657SAndreas Gohr $tplstyles[$mode][$tplinc.$file] = $tpldir; 63519b3173SAndreas Gohr } 64519b3173SAndreas Gohr } 65519b3173SAndreas Gohr 66dbf794bfSMichael Hamann // if old 'default' userstyle setting exists, make it 'screen' userstyle for backwards compatibility 67dbf794bfSMichael Hamann if (isset($config_cascade['userstyle']['default'])) { 68dbf794bfSMichael Hamann $config_cascade['userstyle']['screen'] = $config_cascade['userstyle']['default']; 69dbf794bfSMichael Hamann } 70dbf794bfSMichael Hamann 7178a6aeb1SAndreas Gohr // Array of needed files and their web locations, the latter ones 7278a6aeb1SAndreas Gohr // are needed to fix relative paths in the stylesheets 7378a6aeb1SAndreas Gohr $files = array(); 7414977bd2SMichael Hamann 7514977bd2SMichael Hamann $cache_files = getConfigFiles('main'); 760ac69508SAnika Henke $cache_files[] = $tplinc.'style.ini'; 770ac69508SAnika Henke $cache_files[] = $tplinc.'style.local.ini'; 7814977bd2SMichael Hamann $cache_files[] = __FILE__; 7914977bd2SMichael Hamann 8014977bd2SMichael Hamann foreach($mediatypes as $mediatype) { 8114977bd2SMichael Hamann $files[$mediatype] = array(); 82318cd03eSAnika Henke // load core styles 8314977bd2SMichael Hamann $files[$mediatype][DOKU_INC.'lib/styles/'.$mediatype.'.css'] = DOKU_BASE.'lib/styles/'; 8443576758SAndreas Gohr // load jQuery-UI theme 856c47a78cSAnika Henke if ($mediatype == 'screen') { 8614977bd2SMichael Hamann $files[$mediatype][DOKU_INC.'lib/scripts/jquery/jquery-ui-theme/smoothness.css'] = DOKU_BASE.'lib/scripts/jquery/jquery-ui-theme/'; 876c47a78cSAnika Henke } 88318cd03eSAnika Henke // load plugin styles 8914977bd2SMichael Hamann $files[$mediatype] = array_merge($files[$mediatype], css_pluginstyles($mediatype)); 90318cd03eSAnika Henke // load template styles 91318cd03eSAnika Henke if (isset($tplstyles[$mediatype])) { 9214977bd2SMichael Hamann $files[$mediatype] = array_merge($files[$mediatype], $tplstyles[$mediatype]); 9309edb711SAndreas Gohr } 94318cd03eSAnika Henke // load user styles 95318cd03eSAnika Henke if(isset($config_cascade['userstyle'][$mediatype])){ 9614977bd2SMichael Hamann $files[$mediatype][$config_cascade['userstyle'][$mediatype]] = DOKU_BASE; 97318cd03eSAnika Henke } 98318cd03eSAnika Henke // load rtl styles 996c47a78cSAnika Henke // note: this adds the rtl styles only to the 'screen' media type 1006c47a78cSAnika Henke // @deprecated 2012-04-09: rtl will cease to be a mode of its own, 1016c47a78cSAnika Henke // please use "[dir=rtl]" in any css file in all, screen or print mode instead 102318cd03eSAnika Henke if ($mediatype=='screen') { 10378a6aeb1SAndreas Gohr if($lang['direction'] == 'rtl'){ 10414977bd2SMichael Hamann if (isset($tplstyles['rtl'])) $files[$mediatype] = array_merge($files[$mediatype], $tplstyles['rtl']); 105c5c68de9SMichael Hamann if (isset($config_cascade['userstyle']['rtl'])) $files[$mediatype][$config_cascade['userstyle']['rtl']] = DOKU_BASE; 10678a6aeb1SAndreas Gohr } 10778a6aeb1SAndreas Gohr } 10878a6aeb1SAndreas Gohr 10914977bd2SMichael Hamann $cache_files = array_merge($cache_files, array_keys($files[$mediatype])); 11014977bd2SMichael Hamann } 1116619f42eSAdrian Lang 11238f56bffSBen Coburn // check cache age & handle conditional request 1136619f42eSAdrian Lang // This may exit if a cache can be used 1146619f42eSAdrian Lang http_cached($cache->cache, 1156619f42eSAdrian Lang $cache->useCache(array('files' => $cache_files))); 11678a6aeb1SAndreas Gohr 1173899c2ecSMichael Hamann // start output buffering 1183899c2ecSMichael Hamann ob_start(); 1193899c2ecSMichael Hamann 1206c47a78cSAnika Henke // build the stylesheet 12114977bd2SMichael Hamann foreach ($mediatypes as $mediatype) { 12278a6aeb1SAndreas Gohr 123d15166e5SAndreas Gohr // print the default classes for interwiki links and file downloads 1246c47a78cSAnika Henke if ($mediatype == 'screen') { 125*cacfb606SAnika Henke print '@media screen {'; 1261c2d1019SAndreas Gohr css_interwiki(); 127d15166e5SAndreas Gohr css_filetypes(); 128*cacfb606SAnika Henke print '}'; 12978a6aeb1SAndreas Gohr } 13078a6aeb1SAndreas Gohr 1316c47a78cSAnika Henke // load files 1326c47a78cSAnika Henke $css_content = ''; 13314977bd2SMichael Hamann foreach($files[$mediatype] as $file => $location){ 1346c47a78cSAnika Henke $css_content .= css_loadfile($file, $location); 1356c47a78cSAnika Henke } 1366c47a78cSAnika Henke switch ($mediatype) { 1376c47a78cSAnika Henke case 'screen': 1386c47a78cSAnika Henke print NL.'@media screen { /* START screen styles */'.NL.$css_content.NL.'} /* /@media END screen styles */'.NL; 1396c47a78cSAnika Henke break; 1406c47a78cSAnika Henke case 'print': 1416c47a78cSAnika Henke print NL.'@media print { /* START print styles */'.NL.$css_content.NL.'} /* /@media END print styles */'.NL; 1426c47a78cSAnika Henke break; 1436c47a78cSAnika Henke case 'all': 1446c47a78cSAnika Henke case 'feed': 1456c47a78cSAnika Henke default: 1466c47a78cSAnika Henke print NL.'/* START rest styles */ '.NL.$css_content.NL.'/* END rest styles */'.NL; 1476c47a78cSAnika Henke break; 1486c47a78cSAnika Henke } 1496c47a78cSAnika Henke } 15078a6aeb1SAndreas Gohr // end output buffering and get contents 15178a6aeb1SAndreas Gohr $css = ob_get_contents(); 15278a6aeb1SAndreas Gohr ob_end_clean(); 15378a6aeb1SAndreas Gohr 1546e69c1baSAndreas Gohr // apply style replacements 155124af657SAndreas Gohr $css = css_applystyle($css,$tplinc); 1566e69c1baSAndreas Gohr 157f7d780b9SGabriel Birke // place all @import statements at the top of the file 158f7d780b9SGabriel Birke $css = css_moveimports($css); 159f7d780b9SGabriel Birke 16078a6aeb1SAndreas Gohr // compress whitespace and comments 16178a6aeb1SAndreas Gohr if($conf['compress']){ 16278a6aeb1SAndreas Gohr $css = css_compress($css); 16378a6aeb1SAndreas Gohr } 16478a6aeb1SAndreas Gohr 165809d3ba5SAndreas Gohr // embed small images right into the stylesheet 166809d3ba5SAndreas Gohr if($conf['cssdatauri']){ 167809d3ba5SAndreas Gohr $base = preg_quote(DOKU_BASE,'#'); 168809d3ba5SAndreas Gohr $css = preg_replace_callback('#(url\([ \'"]*)('.$base.')(.*?(?:\.(png|gif)))#i','css_datauri',$css); 169809d3ba5SAndreas Gohr } 170809d3ba5SAndreas Gohr 1716619f42eSAdrian Lang http_cached_finish($cache->cache, $css); 17278a6aeb1SAndreas Gohr} 17378a6aeb1SAndreas Gohr 17478a6aeb1SAndreas Gohr/** 1756e69c1baSAndreas Gohr * Does placeholder replacements in the style according to 1766e69c1baSAndreas Gohr * the ones defined in a templates style.ini file 1776e69c1baSAndreas Gohr * 1786e69c1baSAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 1796e69c1baSAndreas Gohr */ 180124af657SAndreas Gohrfunction css_applystyle($css,$tplinc){ 1810e6f9f08SAnika Henke $styleini = css_styleini($tplinc); 1820e6f9f08SAnika Henke 1830e6f9f08SAnika Henke if($styleini){ 1840ac69508SAnika Henke $css = strtr($css,$styleini['replacements']); 1856e69c1baSAndreas Gohr } 1866e69c1baSAndreas Gohr return $css; 1876e69c1baSAndreas Gohr} 1886e69c1baSAndreas Gohr 1896e69c1baSAndreas Gohr/** 1900ac69508SAnika Henke * Get contents of merged style.ini and style.local.ini as an array. 1910ac69508SAnika Henke * 1920ac69508SAnika Henke * @author Anika Henke <anika@selfthinker.org> 1930e6f9f08SAnika Henke */ 1940e6f9f08SAnika Henkefunction css_styleini($tplinc) { 1950ac69508SAnika Henke $styleini = array(); 1960ac69508SAnika Henke 1970ac69508SAnika Henke foreach (array($tplinc.'style.ini', $tplinc.'style.local.ini') as $ini) { 1980ac69508SAnika Henke $tmp = (@file_exists($ini)) ? parse_ini_file($ini, true) : array(); 1990ac69508SAnika Henke 2000ac69508SAnika Henke foreach($tmp as $key => $value) { 2010ac69508SAnika Henke if(array_key_exists($key, $styleini) && is_array($value)) { 2020ac69508SAnika Henke $styleini[$key] = array_merge($styleini[$key], $tmp[$key]); 2030ac69508SAnika Henke } else { 2040ac69508SAnika Henke $styleini[$key] = $value; 2050ac69508SAnika Henke } 2060ac69508SAnika Henke } 2070ac69508SAnika Henke } 2080ac69508SAnika Henke return $styleini; 2090e6f9f08SAnika Henke} 2100e6f9f08SAnika Henke 2110e6f9f08SAnika Henke/** 2121c2d1019SAndreas Gohr * Prints classes for interwikilinks 2131c2d1019SAndreas Gohr * 2141c2d1019SAndreas Gohr * Interwiki links have two classes: 'interwiki' and 'iw_$name>' where 2151c2d1019SAndreas Gohr * $name is the identifier given in the config. All Interwiki links get 2161c2d1019SAndreas Gohr * an default style with a default icon. If a special icon is available 2171c2d1019SAndreas Gohr * for an interwiki URL it is set in it's own class. Both classes can be 2181c2d1019SAndreas Gohr * overwritten in the template or userstyles. 2191c2d1019SAndreas Gohr * 2201c2d1019SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 2211c2d1019SAndreas Gohr */ 2221c2d1019SAndreas Gohrfunction css_interwiki(){ 2231c2d1019SAndreas Gohr 2241c2d1019SAndreas Gohr // default style 2251c2d1019SAndreas Gohr echo 'a.interwiki {'; 2261c2d1019SAndreas Gohr echo ' background: transparent url('.DOKU_BASE.'lib/images/interwiki.png) 0px 1px no-repeat;'; 2277b4ea081Smarklundeberg echo ' padding: 1px 0px 1px 16px;'; 2281c2d1019SAndreas Gohr echo '}'; 2291c2d1019SAndreas Gohr 2301c2d1019SAndreas Gohr // additional styles when icon available 2311c2d1019SAndreas Gohr $iwlinks = getInterwiki(); 2321c2d1019SAndreas Gohr foreach(array_keys($iwlinks) as $iw){ 2339d2ddea4SAndreas Gohr $class = preg_replace('/[^_\-a-z0-9]+/i','_',$iw); 2341c2d1019SAndreas Gohr if(@file_exists(DOKU_INC.'lib/images/interwiki/'.$iw.'.png')){ 2359d2ddea4SAndreas Gohr echo "a.iw_$class {"; 2361c2d1019SAndreas Gohr echo ' background-image: url('.DOKU_BASE.'lib/images/interwiki/'.$iw.'.png)'; 2371c2d1019SAndreas Gohr echo '}'; 2381c2d1019SAndreas Gohr }elseif(@file_exists(DOKU_INC.'lib/images/interwiki/'.$iw.'.gif')){ 2399d2ddea4SAndreas Gohr echo "a.iw_$class {"; 2401c2d1019SAndreas Gohr echo ' background-image: url('.DOKU_BASE.'lib/images/interwiki/'.$iw.'.gif)'; 2411c2d1019SAndreas Gohr echo '}'; 2421c2d1019SAndreas Gohr } 2431c2d1019SAndreas Gohr } 244d15166e5SAndreas Gohr} 2451c2d1019SAndreas Gohr 246d15166e5SAndreas Gohr/** 247d15166e5SAndreas Gohr * Prints classes for file download links 248d15166e5SAndreas Gohr * 249d15166e5SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 250d15166e5SAndreas Gohr */ 251d15166e5SAndreas Gohrfunction css_filetypes(){ 252d15166e5SAndreas Gohr 253d15166e5SAndreas Gohr // default style 254035e07f1SKate Arzamastseva echo '.mediafile {'; 255d15166e5SAndreas Gohr echo ' background: transparent url('.DOKU_BASE.'lib/images/fileicons/file.png) 0px 1px no-repeat;'; 2565b77caf4SAndreas Gohr echo ' padding-left: 18px;'; 2575b77caf4SAndreas Gohr echo ' padding-bottom: 1px;'; 258d15166e5SAndreas Gohr echo '}'; 259d15166e5SAndreas Gohr 260d15166e5SAndreas Gohr // additional styles when icon available 26127bf7924STom N Harris // scan directory for all icons 26227bf7924STom N Harris $exts = array(); 26327bf7924STom N Harris if($dh = opendir(DOKU_INC.'lib/images/fileicons')){ 26427bf7924STom N Harris while(false !== ($file = readdir($dh))){ 26527bf7924STom N Harris if(preg_match('/([_\-a-z0-9]+(?:\.[_\-a-z0-9]+)*?)\.(png|gif)/i',$file,$match)){ 26627bf7924STom N Harris $ext = strtolower($match[1]); 26727bf7924STom N Harris $type = '.'.strtolower($match[2]); 26827bf7924STom N Harris if($ext!='file' && (!isset($exts[$ext]) || $type=='.png')){ 26927bf7924STom N Harris $exts[$ext] = $type; 270d15166e5SAndreas Gohr } 271d15166e5SAndreas Gohr } 2721c2d1019SAndreas Gohr } 27327bf7924STom N Harris closedir($dh); 27427bf7924STom N Harris } 27527bf7924STom N Harris foreach($exts as $ext=>$type){ 27627bf7924STom N Harris $class = preg_replace('/[^_\-a-z0-9]+/','_',$ext); 277035e07f1SKate Arzamastseva echo ".mf_$class {"; 27827bf7924STom N Harris echo ' background-image: url('.DOKU_BASE.'lib/images/fileicons/'.$ext.$type.')'; 27927bf7924STom N Harris echo '}'; 28027bf7924STom N Harris } 28127bf7924STom N Harris} 2821c2d1019SAndreas Gohr 2831c2d1019SAndreas Gohr/** 28478a6aeb1SAndreas Gohr * Loads a given file and fixes relative URLs with the 28578a6aeb1SAndreas Gohr * given location prefix 28678a6aeb1SAndreas Gohr */ 28778a6aeb1SAndreas Gohrfunction css_loadfile($file,$location=''){ 28878a6aeb1SAndreas Gohr if(!@file_exists($file)) return ''; 28978a6aeb1SAndreas Gohr $css = io_readFile($file); 29078a6aeb1SAndreas Gohr if(!$location) return $css; 29178a6aeb1SAndreas Gohr 292809d3ba5SAndreas Gohr $css = preg_replace('#(url\([ \'"]*)(?!/|data:|http://|https://| |\'|")#','\\1'.$location,$css); 293809d3ba5SAndreas Gohr $css = preg_replace('#(@import\s+[\'"])(?!/|data:|http://|https://)#', '\\1'.$location, $css); 294809d3ba5SAndreas Gohr 29578a6aeb1SAndreas Gohr return $css; 29678a6aeb1SAndreas Gohr} 29778a6aeb1SAndreas Gohr 298809d3ba5SAndreas Gohr/** 299809d3ba5SAndreas Gohr * Converte local image URLs to data URLs if the filesize is small 300809d3ba5SAndreas Gohr * 301809d3ba5SAndreas Gohr * Callback for preg_replace_callback 302809d3ba5SAndreas Gohr */ 303809d3ba5SAndreas Gohrfunction css_datauri($match){ 30428f4004cSAndreas Gohr global $conf; 30528f4004cSAndreas Gohr 306809d3ba5SAndreas Gohr $pre = unslash($match[1]); 307809d3ba5SAndreas Gohr $base = unslash($match[2]); 308809d3ba5SAndreas Gohr $url = unslash($match[3]); 309809d3ba5SAndreas Gohr $ext = unslash($match[4]); 310809d3ba5SAndreas Gohr 311809d3ba5SAndreas Gohr $local = DOKU_INC.$url; 312809d3ba5SAndreas Gohr $size = @filesize($local); 31328f4004cSAndreas Gohr if($size && $size < $conf['cssdatauri']){ 314809d3ba5SAndreas Gohr $data = base64_encode(file_get_contents($local)); 315809d3ba5SAndreas Gohr } 316809d3ba5SAndreas Gohr if($data){ 317809d3ba5SAndreas Gohr $url = 'data:image/'.$ext.';base64,'.$data; 318809d3ba5SAndreas Gohr }else{ 319809d3ba5SAndreas Gohr $url = $base.$url; 320809d3ba5SAndreas Gohr } 321809d3ba5SAndreas Gohr return $pre.$url; 322809d3ba5SAndreas Gohr} 323809d3ba5SAndreas Gohr 32415c394afSAndreas Gohr 32578a6aeb1SAndreas Gohr/** 32678a6aeb1SAndreas Gohr * Returns a list of possible Plugin Styles (no existance check here) 32778a6aeb1SAndreas Gohr * 32878a6aeb1SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 32978a6aeb1SAndreas Gohr */ 330318cd03eSAnika Henkefunction css_pluginstyles($mediatype='screen'){ 331208c0215SAndreas Gohr global $lang; 33278a6aeb1SAndreas Gohr $list = array(); 33378a6aeb1SAndreas Gohr $plugins = plugin_list(); 33478a6aeb1SAndreas Gohr foreach ($plugins as $p){ 335318cd03eSAnika Henke $list[DOKU_PLUGIN."$p/$mediatype.css"] = DOKU_BASE."lib/plugins/$p/"; 336318cd03eSAnika Henke // alternative for screen.css 337318cd03eSAnika Henke if ($mediatype=='screen') { 33878a6aeb1SAndreas Gohr $list[DOKU_PLUGIN."$p/style.css"] = DOKU_BASE."lib/plugins/$p/"; 33978a6aeb1SAndreas Gohr } 3406c47a78cSAnika Henke // @deprecated 2012-04-09: rtl will cease to be a mode of its own, 3416c47a78cSAnika Henke // please use "[dir=rtl]" in any css file in all, screen or print mode instead 342208c0215SAndreas Gohr if($lang['direction'] == 'rtl'){ 343208c0215SAndreas Gohr $list[DOKU_PLUGIN."$p/rtl.css"] = DOKU_BASE."lib/plugins/$p/"; 344208c0215SAndreas Gohr } 34578a6aeb1SAndreas Gohr } 34678a6aeb1SAndreas Gohr return $list; 34778a6aeb1SAndreas Gohr} 34878a6aeb1SAndreas Gohr 34978a6aeb1SAndreas Gohr/** 350f7d780b9SGabriel Birke * Move all @import statements in a combined stylesheet to the top so they 351f7d780b9SGabriel Birke * aren't ignored by the browser. 352f7d780b9SGabriel Birke * 353f7d780b9SGabriel Birke * @author Gabriel Birke <birke@d-scribe.de> 354f7d780b9SGabriel Birke */ 355f7d780b9SGabriel Birkefunction css_moveimports($css) 356f7d780b9SGabriel Birke{ 357f7d780b9SGabriel Birke if(!preg_match_all('/@import\s+(?:url\([^)]+\)|"[^"]+")\s*[^;]*;\s*/', $css, $matches, PREG_OFFSET_CAPTURE)) { 358f7d780b9SGabriel Birke return $css; 359f7d780b9SGabriel Birke } 360f7d780b9SGabriel Birke $newCss = ""; 361f7d780b9SGabriel Birke $imports = ""; 362f7d780b9SGabriel Birke $offset = 0; 363f7d780b9SGabriel Birke foreach($matches[0] as $match) { 364f7d780b9SGabriel Birke $newCss .= substr($css, $offset, $match[1] - $offset); 365f7d780b9SGabriel Birke $imports .= $match[0]; 366f7d780b9SGabriel Birke $offset = $match[1] + strlen($match[0]); 367f7d780b9SGabriel Birke } 368f7d780b9SGabriel Birke $newCss .= substr($css, $offset); 369f7d780b9SGabriel Birke return $imports.$newCss; 370f7d780b9SGabriel Birke} 371f7d780b9SGabriel Birke 372f7d780b9SGabriel Birke/** 37378a6aeb1SAndreas Gohr * Very simple CSS optimizer 37478a6aeb1SAndreas Gohr * 37578a6aeb1SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 37678a6aeb1SAndreas Gohr */ 37778a6aeb1SAndreas Gohrfunction css_compress($css){ 378fd7c2db0SAndreas Gohr //strip comments through a callback 379fd7c2db0SAndreas Gohr $css = preg_replace_callback('#(/\*)(.*?)(\*/)#s','css_comment_cb',$css); 380fd7c2db0SAndreas Gohr 381247c1c5dSAndreas Gohr //strip (incorrect but common) one line comments 382fd7c2db0SAndreas Gohr $css = preg_replace('/(?<!:)\/\/.*$/m','',$css); 383247c1c5dSAndreas Gohr 38478a6aeb1SAndreas Gohr // strip whitespaces 38578a6aeb1SAndreas Gohr $css = preg_replace('![\r\n\t ]+!',' ',$css); 386f5379589SChristopher Smith $css = preg_replace('/ ?([;,{}\/]) ?/','\\1',$css); 387f5379589SChristopher Smith $css = preg_replace('/ ?: /',':',$css); 38878a6aeb1SAndreas Gohr 38978a6aeb1SAndreas Gohr // shorten colors 39078a6aeb1SAndreas Gohr $css = preg_replace("/#([0-9a-fA-F]{1})\\1([0-9a-fA-F]{1})\\2([0-9a-fA-F]{1})\\3/", "#\\1\\2\\3",$css); 39178a6aeb1SAndreas Gohr 39278a6aeb1SAndreas Gohr return $css; 39378a6aeb1SAndreas Gohr} 39478a6aeb1SAndreas Gohr 395c00aef76SAndreas Gohr/** 396c00aef76SAndreas Gohr * Callback for css_compress() 397c00aef76SAndreas Gohr * 398c00aef76SAndreas Gohr * Keeps short comments (< 5 chars) to maintain typical browser hacks 399c00aef76SAndreas Gohr * 400c00aef76SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 401c00aef76SAndreas Gohr */ 402c00aef76SAndreas Gohrfunction css_comment_cb($matches){ 403c00aef76SAndreas Gohr if(strlen($matches[2]) > 4) return ''; 404c00aef76SAndreas Gohr return $matches[0]; 405c00aef76SAndreas Gohr} 40678a6aeb1SAndreas Gohr 407e3776c06SMichael Hamann//Setup VIM: ex: et ts=4 : 408