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 43afb2c082SAndreas Gohr // decide from where to get the template 44bfd0f597STom N Harris $tpl = trim(preg_replace('/[^\w-]+/','',$INPUT->str('t'))); 45afb2c082SAndreas Gohr if(!$tpl) $tpl = $conf['template']; 460e6f9f08SAnika Henke 4778a6aeb1SAndreas Gohr // The generated script depends on some dynamic options 48afb2c082SAndreas Gohr $cache = new cache('styles'.$_SERVER['HTTP_HOST'].$_SERVER['SERVER_PORT'].DOKU_BASE.$tpl.$type,'.css'); 4978a6aeb1SAndreas Gohr 50afb2c082SAndreas Gohr // load styl.ini 51afb2c082SAndreas Gohr $styleini = css_styleini($tpl); 52afb2c082SAndreas Gohr 53dbf794bfSMichael Hamann // if old 'default' userstyle setting exists, make it 'screen' userstyle for backwards compatibility 54dbf794bfSMichael Hamann if (isset($config_cascade['userstyle']['default'])) { 55dbf794bfSMichael Hamann $config_cascade['userstyle']['screen'] = $config_cascade['userstyle']['default']; 56dbf794bfSMichael Hamann } 57dbf794bfSMichael Hamann 58afb2c082SAndreas Gohr // cache influencers 59afb2c082SAndreas Gohr $tplinc = tpl_basedir($tpl); 60afb2c082SAndreas Gohr $cache_files = getConfigFiles('main'); 61afb2c082SAndreas Gohr $cache_files[] = $tplinc.'style.ini'; 62*724f2999SAnika Henke $cache_files[] = $tplinc.'style.local.ini'; // @deprecated 63afb2c082SAndreas Gohr $cache_files[] = DOKU_CONF."tpl/$tpl/style.ini"; 64afb2c082SAndreas Gohr $cache_files[] = __FILE__; 65afb2c082SAndreas Gohr 6678a6aeb1SAndreas Gohr // Array of needed files and their web locations, the latter ones 6778a6aeb1SAndreas Gohr // are needed to fix relative paths in the stylesheets 6878a6aeb1SAndreas Gohr $files = array(); 6914977bd2SMichael Hamann foreach($mediatypes as $mediatype) { 7014977bd2SMichael Hamann $files[$mediatype] = array(); 71318cd03eSAnika Henke // load core styles 7214977bd2SMichael Hamann $files[$mediatype][DOKU_INC.'lib/styles/'.$mediatype.'.css'] = DOKU_BASE.'lib/styles/'; 7343576758SAndreas Gohr // load jQuery-UI theme 746c47a78cSAnika Henke if ($mediatype == 'screen') { 7514977bd2SMichael Hamann $files[$mediatype][DOKU_INC.'lib/scripts/jquery/jquery-ui-theme/smoothness.css'] = DOKU_BASE.'lib/scripts/jquery/jquery-ui-theme/'; 766c47a78cSAnika Henke } 77318cd03eSAnika Henke // load plugin styles 7814977bd2SMichael Hamann $files[$mediatype] = array_merge($files[$mediatype], css_pluginstyles($mediatype)); 79318cd03eSAnika Henke // load template styles 80afb2c082SAndreas Gohr if (isset($styleini['stylesheets'][$mediatype])) { 81afb2c082SAndreas Gohr $files[$mediatype] = array_merge($files[$mediatype], $styleini['stylesheets'][$mediatype]); 8209edb711SAndreas Gohr } 83318cd03eSAnika Henke // load user styles 84318cd03eSAnika Henke if(isset($config_cascade['userstyle'][$mediatype])){ 8514977bd2SMichael Hamann $files[$mediatype][$config_cascade['userstyle'][$mediatype]] = DOKU_BASE; 86318cd03eSAnika Henke } 87318cd03eSAnika Henke // load rtl styles 886c47a78cSAnika Henke // note: this adds the rtl styles only to the 'screen' media type 896c47a78cSAnika Henke // @deprecated 2012-04-09: rtl will cease to be a mode of its own, 906c47a78cSAnika Henke // please use "[dir=rtl]" in any css file in all, screen or print mode instead 91318cd03eSAnika Henke if ($mediatype=='screen') { 9278a6aeb1SAndreas Gohr if($lang['direction'] == 'rtl'){ 93afb2c082SAndreas Gohr if (isset($styleini['stylesheets']['rtl'])) $files[$mediatype] = array_merge($files[$mediatype], $styleini['stylesheets']['rtl']); 94c5c68de9SMichael Hamann if (isset($config_cascade['userstyle']['rtl'])) $files[$mediatype][$config_cascade['userstyle']['rtl']] = DOKU_BASE; 9578a6aeb1SAndreas Gohr } 9678a6aeb1SAndreas Gohr } 9778a6aeb1SAndreas Gohr 9814977bd2SMichael Hamann $cache_files = array_merge($cache_files, array_keys($files[$mediatype])); 9914977bd2SMichael Hamann } 1006619f42eSAdrian Lang 10138f56bffSBen Coburn // check cache age & handle conditional request 1026619f42eSAdrian Lang // This may exit if a cache can be used 1036619f42eSAdrian Lang http_cached($cache->cache, 1046619f42eSAdrian Lang $cache->useCache(array('files' => $cache_files))); 10578a6aeb1SAndreas Gohr 1063899c2ecSMichael Hamann // start output buffering 1073899c2ecSMichael Hamann ob_start(); 1083899c2ecSMichael Hamann 1096c47a78cSAnika Henke // build the stylesheet 11014977bd2SMichael Hamann foreach ($mediatypes as $mediatype) { 11178a6aeb1SAndreas Gohr 112d15166e5SAndreas Gohr // print the default classes for interwiki links and file downloads 1136c47a78cSAnika Henke if ($mediatype == 'screen') { 114cacfb606SAnika Henke print '@media screen {'; 1151c2d1019SAndreas Gohr css_interwiki(); 116d15166e5SAndreas Gohr css_filetypes(); 117cacfb606SAnika Henke print '}'; 11878a6aeb1SAndreas Gohr } 11978a6aeb1SAndreas Gohr 1206c47a78cSAnika Henke // load files 1216c47a78cSAnika Henke $css_content = ''; 12214977bd2SMichael Hamann foreach($files[$mediatype] as $file => $location){ 12372a66eb7SAndreas Gohr $display = str_replace(fullpath(DOKU_INC), '', fullpath($file)); 12472a66eb7SAndreas Gohr $css_content .= "\n/* XXXXXXXXX $display XXXXXXXXX */\n"; 1256c47a78cSAnika Henke $css_content .= css_loadfile($file, $location); 1266c47a78cSAnika Henke } 1276c47a78cSAnika Henke switch ($mediatype) { 1286c47a78cSAnika Henke case 'screen': 1296c47a78cSAnika Henke print NL.'@media screen { /* START screen styles */'.NL.$css_content.NL.'} /* /@media END screen styles */'.NL; 1306c47a78cSAnika Henke break; 1316c47a78cSAnika Henke case 'print': 1326c47a78cSAnika Henke print NL.'@media print { /* START print styles */'.NL.$css_content.NL.'} /* /@media END print styles */'.NL; 1336c47a78cSAnika Henke break; 1346c47a78cSAnika Henke case 'all': 1356c47a78cSAnika Henke case 'feed': 1366c47a78cSAnika Henke default: 1376c47a78cSAnika Henke print NL.'/* START rest styles */ '.NL.$css_content.NL.'/* END rest styles */'.NL; 1386c47a78cSAnika Henke break; 1396c47a78cSAnika Henke } 1406c47a78cSAnika Henke } 14178a6aeb1SAndreas Gohr // end output buffering and get contents 14278a6aeb1SAndreas Gohr $css = ob_get_contents(); 14378a6aeb1SAndreas Gohr ob_end_clean(); 14478a6aeb1SAndreas Gohr 1456e69c1baSAndreas Gohr // apply style replacements 146afb2c082SAndreas Gohr $css = css_applystyle($css, $styleini['replacements']); 1476e69c1baSAndreas Gohr 14872a66eb7SAndreas Gohr // parse less 14972a66eb7SAndreas Gohr $css = css_parseless($css); 150d4a1ece8SAndreas Gohr 151d4a1ece8SAndreas Gohr // place all remaining @import statements at the top of the file 152f7d780b9SGabriel Birke $css = css_moveimports($css); 153f7d780b9SGabriel Birke 15478a6aeb1SAndreas Gohr // compress whitespace and comments 15578a6aeb1SAndreas Gohr if($conf['compress']){ 15678a6aeb1SAndreas Gohr $css = css_compress($css); 15778a6aeb1SAndreas Gohr } 15878a6aeb1SAndreas Gohr 159809d3ba5SAndreas Gohr // embed small images right into the stylesheet 160809d3ba5SAndreas Gohr if($conf['cssdatauri']){ 161809d3ba5SAndreas Gohr $base = preg_quote(DOKU_BASE,'#'); 162809d3ba5SAndreas Gohr $css = preg_replace_callback('#(url\([ \'"]*)('.$base.')(.*?(?:\.(png|gif)))#i','css_datauri',$css); 163809d3ba5SAndreas Gohr } 164809d3ba5SAndreas Gohr 1656619f42eSAdrian Lang http_cached_finish($cache->cache, $css); 16678a6aeb1SAndreas Gohr} 16778a6aeb1SAndreas Gohr 16878a6aeb1SAndreas Gohr/** 16972a66eb7SAndreas Gohr * Uses phpless to parse LESS in our CSS 17072a66eb7SAndreas Gohr * 17172a66eb7SAndreas Gohr * most of this function is error handling to show a nice useful error when 17272a66eb7SAndreas Gohr * LESS compilation fails 17372a66eb7SAndreas Gohr * 17472a66eb7SAndreas Gohr * @param $css 17572a66eb7SAndreas Gohr * @return string 17672a66eb7SAndreas Gohr */ 17772a66eb7SAndreas Gohrfunction css_parseless($css) { 17872a66eb7SAndreas Gohr $less = new lessc(); 17972a66eb7SAndreas Gohr try { 18072a66eb7SAndreas Gohr return $less->compile($css); 18172a66eb7SAndreas Gohr } catch(Exception $e) { 18272a66eb7SAndreas Gohr // get exception message 18372a66eb7SAndreas Gohr $msg = str_replace(array("\n", "\r", "'"), array(), $e->getMessage()); 18472a66eb7SAndreas Gohr 18572a66eb7SAndreas Gohr // try to use line number to find affected file 18672a66eb7SAndreas Gohr if(preg_match('/line: (\d+)$/', $msg, $m)){ 18772a66eb7SAndreas Gohr $msg = substr($msg, 0, -1* strlen($m[0])); //remove useless linenumber 18872a66eb7SAndreas Gohr $lno = $m[1]; 18972a66eb7SAndreas Gohr 19072a66eb7SAndreas Gohr // walk upwards to last include 19172a66eb7SAndreas Gohr $lines = explode("\n", $css); 19272a66eb7SAndreas Gohr for($i=$lno-1; $i>=0; $i--){ 19372a66eb7SAndreas Gohr if(preg_match('/\/(\* XXXXXXXXX )(.*?)( XXXXXXXXX \*)\//', $lines[$i], $m)){ 19472a66eb7SAndreas Gohr // we found it, add info to message 19572a66eb7SAndreas Gohr $msg .= ' in '.$m[2].' at line '.($lno-$i); 19672a66eb7SAndreas Gohr break; 19772a66eb7SAndreas Gohr } 19872a66eb7SAndreas Gohr } 19972a66eb7SAndreas Gohr } 20072a66eb7SAndreas Gohr 20172a66eb7SAndreas Gohr // something went wrong 20272a66eb7SAndreas Gohr $error = 'A fatal error occured during compilation of the CSS files. '. 20372a66eb7SAndreas Gohr 'If you recently installed a new plugin or template it '. 20472a66eb7SAndreas Gohr 'might be broken and you should try disabling it again. ['.$msg.']'; 20572a66eb7SAndreas Gohr 20672a66eb7SAndreas Gohr echo ".dokuwiki:before { 20772a66eb7SAndreas Gohr content: '$error'; 20872a66eb7SAndreas Gohr background-color: red; 20972a66eb7SAndreas Gohr display: block; 21072a66eb7SAndreas Gohr background-color: #fcc; 21172a66eb7SAndreas Gohr border-color: #ebb; 21272a66eb7SAndreas Gohr color: #000; 21372a66eb7SAndreas Gohr padding: 0.5em; 21472a66eb7SAndreas Gohr }"; 21572a66eb7SAndreas Gohr 21672a66eb7SAndreas Gohr exit; 21772a66eb7SAndreas Gohr } 21872a66eb7SAndreas Gohr} 21972a66eb7SAndreas Gohr 22072a66eb7SAndreas Gohr/** 2216e69c1baSAndreas Gohr * Does placeholder replacements in the style according to 2226e69c1baSAndreas Gohr * the ones defined in a templates style.ini file 2236e69c1baSAndreas Gohr * 224d4a1ece8SAndreas Gohr * This also adds the ini defined placeholders as less variables 225d4a1ece8SAndreas Gohr * (sans the surrounding __ and with a ini_ prefix) 226d4a1ece8SAndreas Gohr * 2276e69c1baSAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 2286e69c1baSAndreas Gohr */ 229afb2c082SAndreas Gohrfunction css_applystyle($css, $replacements) { 230cbe37079SAndreas Gohr // we convert ini replacements to LESS variable names 231cbe37079SAndreas Gohr // and build a list of variable: value; pairs 232d4a1ece8SAndreas Gohr $less = ''; 233afb2c082SAndreas Gohr foreach((array) $replacements as $key => $value) { 234cbe37079SAndreas Gohr $lkey = trim($key, '_'); 235cbe37079SAndreas Gohr $lkey = '@ini_'.$lkey; 236cbe37079SAndreas Gohr $less .= "$lkey: $value;\n"; 237cbe37079SAndreas Gohr 238afb2c082SAndreas Gohr $replacements[$key] = $lkey; 239d4a1ece8SAndreas Gohr } 240c51b334eSAndreas Gohr 241cbe37079SAndreas Gohr // we now replace all old ini replacements with LESS variables 242afb2c082SAndreas Gohr $css = strtr($css, $replacements); 243cbe37079SAndreas Gohr 244cbe37079SAndreas Gohr // now prepend the list of LESS variables as the very first thing 245c51b334eSAndreas Gohr $css = $less.$css; 2466e69c1baSAndreas Gohr return $css; 2476e69c1baSAndreas Gohr} 2486e69c1baSAndreas Gohr 2496e69c1baSAndreas Gohr/** 250afb2c082SAndreas Gohr * Load style ini contents 2510ac69508SAnika Henke * 252afb2c082SAndreas Gohr * Loads and merges style.ini files from template and config and prepares 253afb2c082SAndreas Gohr * the stylesheet modes 254afb2c082SAndreas Gohr * 255afb2c082SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 256afb2c082SAndreas Gohr * @param string $tpl the used template 257afb2c082SAndreas Gohr * @return array with keys 'stylesheets' and 'replacements' 2580e6f9f08SAnika Henke */ 259afb2c082SAndreas Gohrfunction css_styleini($tpl) { 260afb2c082SAndreas Gohr $stylesheets = array(); // mode, file => base 261afb2c082SAndreas Gohr $replacements = array(); // placeholder => value 2620ac69508SAnika Henke 263afb2c082SAndreas Gohr // load template's style.ini 264afb2c082SAndreas Gohr $incbase = tpl_incdir($tpl); 265afb2c082SAndreas Gohr $webbase = tpl_basedir($tpl); 266afb2c082SAndreas Gohr $ini = $incbase.'style.ini'; 267afb2c082SAndreas Gohr if(file_exists($ini)){ 268afb2c082SAndreas Gohr $data = parse_ini_file($ini, true); 2690ac69508SAnika Henke 270afb2c082SAndreas Gohr // stylesheets 271afb2c082SAndreas Gohr if(is_array($data['stylesheets'])) foreach($data['stylesheets'] as $file => $mode){ 272afb2c082SAndreas Gohr $stylesheets[$mode][$incbase.$file] = $webbase; 273afb2c082SAndreas Gohr } 274afb2c082SAndreas Gohr 275afb2c082SAndreas Gohr // replacements 276afb2c082SAndreas Gohr if(is_array($data['replacements'])){ 277afb2c082SAndreas Gohr $replacements = array_merge($replacements, $data['replacements']); 2780ac69508SAnika Henke } 2790ac69508SAnika Henke } 280afb2c082SAndreas Gohr 281568ffd7eSAndreas Gohr // load template's style.local.ini 282568ffd7eSAndreas Gohr // @deprecated 2013-08-03 283568ffd7eSAndreas Gohr $ini = $incbase.'style.local.ini'; 284568ffd7eSAndreas Gohr if(file_exists($ini)){ 285568ffd7eSAndreas Gohr $data = parse_ini_file($ini, true); 286568ffd7eSAndreas Gohr 287568ffd7eSAndreas Gohr // stylesheets 288568ffd7eSAndreas Gohr if(is_array($data['stylesheets'])) foreach($data['stylesheets'] as $file => $mode){ 289568ffd7eSAndreas Gohr $stylesheets[$mode][$incbase.$file] = $webbase; 290568ffd7eSAndreas Gohr } 291568ffd7eSAndreas Gohr 292568ffd7eSAndreas Gohr // replacements 293568ffd7eSAndreas Gohr if(is_array($data['replacements'])){ 294568ffd7eSAndreas Gohr $replacements = array_merge($replacements, $data['replacements']); 295568ffd7eSAndreas Gohr } 296568ffd7eSAndreas Gohr } 297568ffd7eSAndreas Gohr 298afb2c082SAndreas Gohr // load configs's style.ini 299afb2c082SAndreas Gohr $webbase = DOKU_BASE; 3008c5aad7bSAnika Henke $ini = DOKU_CONF."tpl/$tpl/style.ini"; 3018c5aad7bSAnika Henke $incbase = dirname($ini).'/'; 302afb2c082SAndreas Gohr if(file_exists($ini)){ 303afb2c082SAndreas Gohr $data = parse_ini_file($ini, true); 304afb2c082SAndreas Gohr 305afb2c082SAndreas Gohr // stylesheets 306afb2c082SAndreas Gohr if(is_array($data['stylesheets'])) foreach($data['stylesheets'] as $file => $mode){ 307afb2c082SAndreas Gohr $stylesheets[$mode][$incbase.$file] = $webbase; 3080ac69508SAnika Henke } 309afb2c082SAndreas Gohr 310afb2c082SAndreas Gohr // replacements 311afb2c082SAndreas Gohr if(is_array($data['replacements'])){ 312afb2c082SAndreas Gohr $replacements = array_merge($replacements, $data['replacements']); 313afb2c082SAndreas Gohr } 314afb2c082SAndreas Gohr } 315afb2c082SAndreas Gohr 316afb2c082SAndreas Gohr return array( 317afb2c082SAndreas Gohr 'stylesheets' => $stylesheets, 318afb2c082SAndreas Gohr 'replacements' => $replacements 319afb2c082SAndreas Gohr ); 3200e6f9f08SAnika Henke} 3210e6f9f08SAnika Henke 3220e6f9f08SAnika Henke/** 3231c2d1019SAndreas Gohr * Prints classes for interwikilinks 3241c2d1019SAndreas Gohr * 3251c2d1019SAndreas Gohr * Interwiki links have two classes: 'interwiki' and 'iw_$name>' where 3261c2d1019SAndreas Gohr * $name is the identifier given in the config. All Interwiki links get 3271c2d1019SAndreas Gohr * an default style with a default icon. If a special icon is available 3281c2d1019SAndreas Gohr * for an interwiki URL it is set in it's own class. Both classes can be 3291c2d1019SAndreas Gohr * overwritten in the template or userstyles. 3301c2d1019SAndreas Gohr * 3311c2d1019SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 3321c2d1019SAndreas Gohr */ 3331c2d1019SAndreas Gohrfunction css_interwiki(){ 3341c2d1019SAndreas Gohr 3351c2d1019SAndreas Gohr // default style 3361c2d1019SAndreas Gohr echo 'a.interwiki {'; 3371c2d1019SAndreas Gohr echo ' background: transparent url('.DOKU_BASE.'lib/images/interwiki.png) 0px 1px no-repeat;'; 3387b4ea081Smarklundeberg echo ' padding: 1px 0px 1px 16px;'; 3391c2d1019SAndreas Gohr echo '}'; 3401c2d1019SAndreas Gohr 3411c2d1019SAndreas Gohr // additional styles when icon available 3421c2d1019SAndreas Gohr $iwlinks = getInterwiki(); 3431c2d1019SAndreas Gohr foreach(array_keys($iwlinks) as $iw){ 3449d2ddea4SAndreas Gohr $class = preg_replace('/[^_\-a-z0-9]+/i','_',$iw); 3451c2d1019SAndreas Gohr if(@file_exists(DOKU_INC.'lib/images/interwiki/'.$iw.'.png')){ 3469d2ddea4SAndreas Gohr echo "a.iw_$class {"; 3471c2d1019SAndreas Gohr echo ' background-image: url('.DOKU_BASE.'lib/images/interwiki/'.$iw.'.png)'; 3481c2d1019SAndreas Gohr echo '}'; 3491c2d1019SAndreas Gohr }elseif(@file_exists(DOKU_INC.'lib/images/interwiki/'.$iw.'.gif')){ 3509d2ddea4SAndreas Gohr echo "a.iw_$class {"; 3511c2d1019SAndreas Gohr echo ' background-image: url('.DOKU_BASE.'lib/images/interwiki/'.$iw.'.gif)'; 3521c2d1019SAndreas Gohr echo '}'; 3531c2d1019SAndreas Gohr } 3541c2d1019SAndreas Gohr } 355d15166e5SAndreas Gohr} 3561c2d1019SAndreas Gohr 357d15166e5SAndreas Gohr/** 358d15166e5SAndreas Gohr * Prints classes for file download links 359d15166e5SAndreas Gohr * 360d15166e5SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 361d15166e5SAndreas Gohr */ 362d15166e5SAndreas Gohrfunction css_filetypes(){ 363d15166e5SAndreas Gohr 364d15166e5SAndreas Gohr // default style 365035e07f1SKate Arzamastseva echo '.mediafile {'; 366d15166e5SAndreas Gohr echo ' background: transparent url('.DOKU_BASE.'lib/images/fileicons/file.png) 0px 1px no-repeat;'; 3675b77caf4SAndreas Gohr echo ' padding-left: 18px;'; 3685b77caf4SAndreas Gohr echo ' padding-bottom: 1px;'; 369d15166e5SAndreas Gohr echo '}'; 370d15166e5SAndreas Gohr 371d15166e5SAndreas Gohr // additional styles when icon available 37227bf7924STom N Harris // scan directory for all icons 37327bf7924STom N Harris $exts = array(); 37427bf7924STom N Harris if($dh = opendir(DOKU_INC.'lib/images/fileicons')){ 37527bf7924STom N Harris while(false !== ($file = readdir($dh))){ 37627bf7924STom N Harris if(preg_match('/([_\-a-z0-9]+(?:\.[_\-a-z0-9]+)*?)\.(png|gif)/i',$file,$match)){ 37727bf7924STom N Harris $ext = strtolower($match[1]); 37827bf7924STom N Harris $type = '.'.strtolower($match[2]); 37927bf7924STom N Harris if($ext!='file' && (!isset($exts[$ext]) || $type=='.png')){ 38027bf7924STom N Harris $exts[$ext] = $type; 381d15166e5SAndreas Gohr } 382d15166e5SAndreas Gohr } 3831c2d1019SAndreas Gohr } 38427bf7924STom N Harris closedir($dh); 38527bf7924STom N Harris } 38627bf7924STom N Harris foreach($exts as $ext=>$type){ 38727bf7924STom N Harris $class = preg_replace('/[^_\-a-z0-9]+/','_',$ext); 388035e07f1SKate Arzamastseva echo ".mf_$class {"; 38927bf7924STom N Harris echo ' background-image: url('.DOKU_BASE.'lib/images/fileicons/'.$ext.$type.')'; 39027bf7924STom N Harris echo '}'; 39127bf7924STom N Harris } 39227bf7924STom N Harris} 3931c2d1019SAndreas Gohr 3941c2d1019SAndreas Gohr/** 39578a6aeb1SAndreas Gohr * Loads a given file and fixes relative URLs with the 39678a6aeb1SAndreas Gohr * given location prefix 39778a6aeb1SAndreas Gohr */ 39878a6aeb1SAndreas Gohrfunction css_loadfile($file,$location=''){ 39978a6aeb1SAndreas Gohr if(!@file_exists($file)) return ''; 40078a6aeb1SAndreas Gohr $css = io_readFile($file); 40178a6aeb1SAndreas Gohr if(!$location) return $css; 40278a6aeb1SAndreas Gohr 403809d3ba5SAndreas Gohr $css = preg_replace('#(url\([ \'"]*)(?!/|data:|http://|https://| |\'|")#','\\1'.$location,$css); 404809d3ba5SAndreas Gohr $css = preg_replace('#(@import\s+[\'"])(?!/|data:|http://|https://)#', '\\1'.$location, $css); 405809d3ba5SAndreas Gohr 40678a6aeb1SAndreas Gohr return $css; 40778a6aeb1SAndreas Gohr} 40878a6aeb1SAndreas Gohr 409809d3ba5SAndreas Gohr/** 410809d3ba5SAndreas Gohr * Converte local image URLs to data URLs if the filesize is small 411809d3ba5SAndreas Gohr * 412809d3ba5SAndreas Gohr * Callback for preg_replace_callback 413809d3ba5SAndreas Gohr */ 414809d3ba5SAndreas Gohrfunction css_datauri($match){ 41528f4004cSAndreas Gohr global $conf; 41628f4004cSAndreas Gohr 417809d3ba5SAndreas Gohr $pre = unslash($match[1]); 418809d3ba5SAndreas Gohr $base = unslash($match[2]); 419809d3ba5SAndreas Gohr $url = unslash($match[3]); 420809d3ba5SAndreas Gohr $ext = unslash($match[4]); 421809d3ba5SAndreas Gohr 422809d3ba5SAndreas Gohr $local = DOKU_INC.$url; 423809d3ba5SAndreas Gohr $size = @filesize($local); 42428f4004cSAndreas Gohr if($size && $size < $conf['cssdatauri']){ 425809d3ba5SAndreas Gohr $data = base64_encode(file_get_contents($local)); 426809d3ba5SAndreas Gohr } 427809d3ba5SAndreas Gohr if($data){ 42888833bacSflammy $url = '\'data:image/'.$ext.';base64,'.$data.'\''; 429809d3ba5SAndreas Gohr }else{ 430809d3ba5SAndreas Gohr $url = $base.$url; 431809d3ba5SAndreas Gohr } 432809d3ba5SAndreas Gohr return $pre.$url; 433809d3ba5SAndreas Gohr} 434809d3ba5SAndreas Gohr 43515c394afSAndreas Gohr 43678a6aeb1SAndreas Gohr/** 43778a6aeb1SAndreas Gohr * Returns a list of possible Plugin Styles (no existance check here) 43878a6aeb1SAndreas Gohr * 43978a6aeb1SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 44078a6aeb1SAndreas Gohr */ 441318cd03eSAnika Henkefunction css_pluginstyles($mediatype='screen'){ 442208c0215SAndreas Gohr global $lang; 44378a6aeb1SAndreas Gohr $list = array(); 44478a6aeb1SAndreas Gohr $plugins = plugin_list(); 44578a6aeb1SAndreas Gohr foreach ($plugins as $p){ 446318cd03eSAnika Henke $list[DOKU_PLUGIN."$p/$mediatype.css"] = DOKU_BASE."lib/plugins/$p/"; 447d4a1ece8SAndreas Gohr $list[DOKU_PLUGIN."$p/$mediatype.less"] = DOKU_BASE."lib/plugins/$p/"; 448318cd03eSAnika Henke // alternative for screen.css 449318cd03eSAnika Henke if ($mediatype=='screen') { 45078a6aeb1SAndreas Gohr $list[DOKU_PLUGIN."$p/style.css"] = DOKU_BASE."lib/plugins/$p/"; 451d4a1ece8SAndreas Gohr $list[DOKU_PLUGIN."$p/style.less"] = DOKU_BASE."lib/plugins/$p/"; 45278a6aeb1SAndreas Gohr } 4536c47a78cSAnika Henke // @deprecated 2012-04-09: rtl will cease to be a mode of its own, 4546c47a78cSAnika Henke // please use "[dir=rtl]" in any css file in all, screen or print mode instead 455208c0215SAndreas Gohr if($lang['direction'] == 'rtl'){ 456208c0215SAndreas Gohr $list[DOKU_PLUGIN."$p/rtl.css"] = DOKU_BASE."lib/plugins/$p/"; 457208c0215SAndreas Gohr } 45878a6aeb1SAndreas Gohr } 45978a6aeb1SAndreas Gohr return $list; 46078a6aeb1SAndreas Gohr} 46178a6aeb1SAndreas Gohr 46278a6aeb1SAndreas Gohr/** 463f7d780b9SGabriel Birke * Move all @import statements in a combined stylesheet to the top so they 464f7d780b9SGabriel Birke * aren't ignored by the browser. 465f7d780b9SGabriel Birke * 466f7d780b9SGabriel Birke * @author Gabriel Birke <birke@d-scribe.de> 467f7d780b9SGabriel Birke */ 468f7d780b9SGabriel Birkefunction css_moveimports($css) 469f7d780b9SGabriel Birke{ 470f7d780b9SGabriel Birke if(!preg_match_all('/@import\s+(?:url\([^)]+\)|"[^"]+")\s*[^;]*;\s*/', $css, $matches, PREG_OFFSET_CAPTURE)) { 471f7d780b9SGabriel Birke return $css; 472f7d780b9SGabriel Birke } 473f7d780b9SGabriel Birke $newCss = ""; 474f7d780b9SGabriel Birke $imports = ""; 475f7d780b9SGabriel Birke $offset = 0; 476f7d780b9SGabriel Birke foreach($matches[0] as $match) { 477f7d780b9SGabriel Birke $newCss .= substr($css, $offset, $match[1] - $offset); 478f7d780b9SGabriel Birke $imports .= $match[0]; 479f7d780b9SGabriel Birke $offset = $match[1] + strlen($match[0]); 480f7d780b9SGabriel Birke } 481f7d780b9SGabriel Birke $newCss .= substr($css, $offset); 482f7d780b9SGabriel Birke return $imports.$newCss; 483f7d780b9SGabriel Birke} 484f7d780b9SGabriel Birke 485f7d780b9SGabriel Birke/** 48678a6aeb1SAndreas Gohr * Very simple CSS optimizer 48778a6aeb1SAndreas Gohr * 48878a6aeb1SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 48978a6aeb1SAndreas Gohr */ 49078a6aeb1SAndreas Gohrfunction css_compress($css){ 491fd7c2db0SAndreas Gohr //strip comments through a callback 492fd7c2db0SAndreas Gohr $css = preg_replace_callback('#(/\*)(.*?)(\*/)#s','css_comment_cb',$css); 493fd7c2db0SAndreas Gohr 494247c1c5dSAndreas Gohr //strip (incorrect but common) one line comments 495fd7c2db0SAndreas Gohr $css = preg_replace('/(?<!:)\/\/.*$/m','',$css); 496247c1c5dSAndreas Gohr 49778a6aeb1SAndreas Gohr // strip whitespaces 49878a6aeb1SAndreas Gohr $css = preg_replace('![\r\n\t ]+!',' ',$css); 499f5379589SChristopher Smith $css = preg_replace('/ ?([;,{}\/]) ?/','\\1',$css); 500f5379589SChristopher Smith $css = preg_replace('/ ?: /',':',$css); 50178a6aeb1SAndreas Gohr 50278a6aeb1SAndreas Gohr // shorten colors 50378a6aeb1SAndreas 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); 50478a6aeb1SAndreas Gohr 50578a6aeb1SAndreas Gohr return $css; 50678a6aeb1SAndreas Gohr} 50778a6aeb1SAndreas Gohr 508c00aef76SAndreas Gohr/** 509c00aef76SAndreas Gohr * Callback for css_compress() 510c00aef76SAndreas Gohr * 511c00aef76SAndreas Gohr * Keeps short comments (< 5 chars) to maintain typical browser hacks 512c00aef76SAndreas Gohr * 513c00aef76SAndreas Gohr * @author Andreas Gohr <andi@splitbrain.org> 514c00aef76SAndreas Gohr */ 515c00aef76SAndreas Gohrfunction css_comment_cb($matches){ 516c00aef76SAndreas Gohr if(strlen($matches[2]) > 4) return ''; 517c00aef76SAndreas Gohr return $matches[0]; 518c00aef76SAndreas Gohr} 51978a6aeb1SAndreas Gohr 520e3776c06SMichael Hamann//Setup VIM: ex: et ts=4 : 521