342 lines
9.7 KiB
PHP
342 lines
9.7 KiB
PHP
<?php
|
|
/* Icinga Web 2 | (c) 2014 Icinga Development Team | GPLv2+ */
|
|
|
|
namespace Icinga\Web;
|
|
|
|
use Exception;
|
|
use Icinga\Application\Icinga;
|
|
use Icinga\Application\Logger;
|
|
use Icinga\Authentication\Auth;
|
|
use Icinga\Exception\IcingaException;
|
|
|
|
/**
|
|
* Send CSS for Web 2 and all loaded modules to the client
|
|
*/
|
|
class StyleSheet
|
|
{
|
|
/**
|
|
* The name of the default theme
|
|
*
|
|
* @var string
|
|
*/
|
|
const DEFAULT_THEME = 'Icinga';
|
|
|
|
/**
|
|
* The name of the default theme mode
|
|
*
|
|
* @var string
|
|
*/
|
|
const DEFAULT_MODE = 'none';
|
|
|
|
/**
|
|
* The themes that are compatible with the default theme
|
|
*
|
|
* @var array
|
|
*/
|
|
const THEME_WHITELIST = [
|
|
'colorblind',
|
|
'high-contrast',
|
|
'Winter'
|
|
];
|
|
|
|
/**
|
|
* Sequence that signals that a theme supports light mode
|
|
*
|
|
* @var string
|
|
*/
|
|
const LIGHT_MODE_IDENTIFIER = '@light-mode:';
|
|
|
|
/**
|
|
* Array of core LESS files Web 2 sends to the client
|
|
*
|
|
* @var string[]
|
|
*/
|
|
protected static $lessFiles = [
|
|
'../application/fonts/fontello-ifont/css/ifont-embedded.css',
|
|
'css/vendor/normalize.css',
|
|
'css/icinga/base.less',
|
|
'css/icinga/badges.less',
|
|
'css/icinga/configmenu.less',
|
|
'css/icinga/mixins.less',
|
|
'css/icinga/grid.less',
|
|
'css/icinga/nav.less',
|
|
'css/icinga/main.less',
|
|
'css/icinga/animation.less',
|
|
'css/icinga/layout.less',
|
|
'css/icinga/layout-structure.less',
|
|
'css/icinga/menu.less',
|
|
'css/icinga/tabs.less',
|
|
'css/icinga/forms.less',
|
|
'css/icinga/setup.less',
|
|
'css/icinga/widgets.less',
|
|
'css/icinga/login.less',
|
|
'css/icinga/about.less',
|
|
'css/icinga/controls.less',
|
|
'css/icinga/dev.less',
|
|
'css/icinga/spinner.less',
|
|
'css/icinga/compat.less',
|
|
'css/icinga/print.less',
|
|
'css/icinga/responsive.less',
|
|
'css/icinga/modal.less',
|
|
'css/icinga/audit.less',
|
|
'css/icinga/health.less',
|
|
'css/icinga/php-diff.less'
|
|
];
|
|
|
|
/**
|
|
* Application instance
|
|
*
|
|
* @var \Icinga\Application\EmbeddedWeb
|
|
*/
|
|
protected $app;
|
|
|
|
/** @var string[] Pre-compiled CSS files */
|
|
protected $cssFiles = [];
|
|
|
|
/**
|
|
* Less compiler
|
|
*
|
|
* @var LessCompiler
|
|
*/
|
|
protected $lessCompiler;
|
|
|
|
/**
|
|
* Path to the public directory
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $pubPath;
|
|
|
|
/**
|
|
* Create the StyleSheet
|
|
*/
|
|
public function __construct()
|
|
{
|
|
$app = Icinga::app();
|
|
$this->app = $app;
|
|
$this->lessCompiler = new LessCompiler();
|
|
$this->pubPath = $app->getBaseDir('public');
|
|
$this->collect();
|
|
}
|
|
|
|
/**
|
|
* Collect Web 2 and module LESS files and add them to the LESS compiler
|
|
*/
|
|
protected function collect()
|
|
{
|
|
foreach ($this->app->getLibraries() as $library) {
|
|
foreach ($library->getCssAssets() as $lessFile) {
|
|
if (substr($lessFile, -4) === '.css') {
|
|
$this->cssFiles[] = $lessFile;
|
|
} else {
|
|
$this->lessCompiler->addLessFile($lessFile);
|
|
}
|
|
}
|
|
}
|
|
|
|
foreach (self::$lessFiles as $lessFile) {
|
|
$this->lessCompiler->addLessFile($this->pubPath . '/' . $lessFile);
|
|
}
|
|
|
|
$mm = $this->app->getModuleManager();
|
|
|
|
foreach ($mm->getLoadedModules() as $moduleName => $module) {
|
|
if ($module->hasCss()) {
|
|
foreach ($module->getCssFiles() as $lessFilePath) {
|
|
$this->lessCompiler->addModuleLessFile($moduleName, $lessFilePath);
|
|
}
|
|
}
|
|
}
|
|
|
|
$themingConfig = $this->app->getConfig()->getSection('themes');
|
|
$defaultTheme = $themingConfig->get('default');
|
|
$theme = null;
|
|
if ($defaultTheme !== null && $defaultTheme !== self::DEFAULT_THEME) {
|
|
$theme = $defaultTheme;
|
|
}
|
|
|
|
if (! (bool) $themingConfig->get('disabled', false)) {
|
|
$auth = Auth::getInstance();
|
|
if ($auth->isAuthenticated()) {
|
|
$userTheme = $auth->getUser()->getPreferences()->getValue('icingaweb', 'theme');
|
|
if ($userTheme !== null) {
|
|
$theme = $userTheme;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($themePath = self::getThemeFile($theme)) {
|
|
if ($this->app->isCli() || is_file($themePath) && is_readable($themePath)) {
|
|
$this->lessCompiler->setTheme($themePath);
|
|
} else {
|
|
$themePath = null;
|
|
Logger::warning(sprintf(
|
|
'Theme "%s" set by user "%s" has not been found.',
|
|
$theme,
|
|
($user = Auth::getInstance()->getUser()) !== null ? $user->getUsername() : 'anonymous'
|
|
));
|
|
}
|
|
}
|
|
|
|
if (! $themePath || in_array($theme, self::THEME_WHITELIST, true)) {
|
|
$this->lessCompiler->addLessFile($this->pubPath . '/css/icinga/login-orbs.less');
|
|
}
|
|
|
|
$mode = 'none';
|
|
if ($user = Auth::getInstance()->getUser()) {
|
|
$file = $themePath !== null ? @file_get_contents($themePath) : false;
|
|
if (! $file || strpos($file, self::LIGHT_MODE_IDENTIFIER) !== false) {
|
|
$mode = $user->getPreferences()->getValue('icingaweb', 'theme_mode', self::DEFAULT_MODE);
|
|
}
|
|
}
|
|
|
|
$this->lessCompiler->setThemeMode($this->pubPath . '/css/modes/'. $mode . '.less');
|
|
}
|
|
|
|
/**
|
|
* Get all collected files
|
|
*
|
|
* @return string[]
|
|
*/
|
|
protected function getFiles(): array
|
|
{
|
|
return array_merge($this->cssFiles, $this->lessCompiler->getLessFiles());
|
|
}
|
|
|
|
/**
|
|
* Get the stylesheet for PDF export
|
|
*
|
|
* @return $this
|
|
*/
|
|
public static function forPdf()
|
|
{
|
|
$styleSheet = new self();
|
|
$styleSheet->lessCompiler->setTheme(null);
|
|
$styleSheet->lessCompiler->setThemeMode($styleSheet->pubPath . '/css/modes/none.less');
|
|
$styleSheet->lessCompiler->addLessFile($styleSheet->pubPath . '/css/pdf/pdfprint.less');
|
|
// TODO(el): Caching
|
|
return $styleSheet;
|
|
}
|
|
|
|
/**
|
|
* Render the stylesheet
|
|
*
|
|
* @param bool $minified Whether to compress the stylesheet
|
|
*
|
|
* @return string CSS
|
|
*/
|
|
public function render($minified = false)
|
|
{
|
|
if ($minified) {
|
|
$this->lessCompiler->compress();
|
|
}
|
|
|
|
$css = '';
|
|
foreach ($this->cssFiles as $cssFile) {
|
|
$css .= file_get_contents($cssFile);
|
|
}
|
|
|
|
return $css . $this->lessCompiler->render();
|
|
}
|
|
|
|
/**
|
|
* Send the stylesheet to the client
|
|
*
|
|
* Does not cache the stylesheet if the HTTP header Cache-Control or Pragma is set to no-cache.
|
|
*
|
|
* @param bool $minified Whether to compress the stylesheet
|
|
*/
|
|
public static function send($minified = false)
|
|
{
|
|
$styleSheet = new self();
|
|
|
|
$request = $styleSheet->app->getRequest();
|
|
$response = $styleSheet->app->getResponse();
|
|
$response->setHeader('Cache-Control', 'private,no-cache,must-revalidate', true);
|
|
|
|
$noCache = $request->getHeader('Cache-Control') === 'no-cache' || $request->getHeader('Pragma') === 'no-cache';
|
|
|
|
$collectedFiles = $styleSheet->getFiles();
|
|
if (! $noCache && FileCache::etagMatchesFiles($collectedFiles)) {
|
|
$response
|
|
->setHttpResponseCode(304)
|
|
->sendHeaders();
|
|
return;
|
|
}
|
|
|
|
$etag = FileCache::etagForFiles($collectedFiles);
|
|
|
|
$response->setHeader('ETag', $etag, true)
|
|
->setHeader('Content-Type', 'text/css', true);
|
|
|
|
$cacheFile = 'icinga-' . $etag . ($minified ? '.min' : '') . '.css';
|
|
$cache = FileCache::instance();
|
|
|
|
if (! $noCache && $cache->has($cacheFile)) {
|
|
$response->setBody($cache->get($cacheFile));
|
|
} else {
|
|
$css = $styleSheet->render($minified);
|
|
$response->setBody($css);
|
|
$cache->store($cacheFile, $css);
|
|
}
|
|
|
|
$response->sendResponse();
|
|
}
|
|
|
|
/**
|
|
* Render the stylesheet
|
|
*
|
|
* @return string
|
|
*/
|
|
public function __toString()
|
|
{
|
|
try {
|
|
return $this->render();
|
|
} catch (Exception $e) {
|
|
Logger::error($e);
|
|
return IcingaException::describe($e);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get the path to the current LESS theme file
|
|
*
|
|
* @param $theme
|
|
*
|
|
* @return string|null Return null if self::DEFAULT_THEME is set as theme, path otherwise
|
|
*/
|
|
public static function getThemeFile($theme)
|
|
{
|
|
$app = Icinga::app();
|
|
|
|
if ($theme && $theme !== self::DEFAULT_THEME) {
|
|
if (Hook::has('ThemeLoader')) {
|
|
try {
|
|
$path = Hook::first('ThemeLoader')->getThemeFile($theme);
|
|
} catch (Exception $e) {
|
|
Logger::error('Failed to call ThemeLoader hook: %s', $e);
|
|
$path = null;
|
|
}
|
|
|
|
if ($path !== null) {
|
|
return $path;
|
|
}
|
|
}
|
|
|
|
if (($pos = strpos($theme, '/')) !== false) {
|
|
$moduleName = substr($theme, 0, $pos);
|
|
$theme = substr($theme, $pos + 1);
|
|
if ($app->getModuleManager()->hasLoaded($moduleName)) {
|
|
$module = $app->getModuleManager()->getModule($moduleName);
|
|
|
|
return $module->getCssDir() . '/themes/' . $theme . '.less';
|
|
}
|
|
} else {
|
|
return $app->getBaseDir('public') . '/css/themes/' . $theme . '.less';
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
}
|