2014-03-04 16:15:08 +01:00
|
|
|
<?php
|
2016-02-08 15:41:00 +01:00
|
|
|
/* Icinga Web 2 | (c) 2014 Icinga Development Team | GPLv2+ */
|
2014-03-04 16:15:08 +01:00
|
|
|
|
|
|
|
namespace Icinga\Web;
|
|
|
|
|
2015-11-27 16:40:17 +01:00
|
|
|
use Exception;
|
2014-03-04 16:15:08 +01:00
|
|
|
use Icinga\Application\Icinga;
|
2015-11-27 16:40:17 +01:00
|
|
|
use Icinga\Application\Logger;
|
2015-12-22 13:00:01 +01:00
|
|
|
use Icinga\Authentication\Auth;
|
2015-11-27 16:40:17 +01:00
|
|
|
use Icinga\Exception\IcingaException;
|
2014-03-04 16:15:08 +01:00
|
|
|
|
2015-11-27 16:40:17 +01:00
|
|
|
/**
|
|
|
|
* Send CSS for Web 2 and all loaded modules to the client
|
|
|
|
*/
|
2014-03-04 16:15:08 +01:00
|
|
|
class StyleSheet
|
|
|
|
{
|
2015-11-27 16:40:17 +01:00
|
|
|
/**
|
|
|
|
* The name of the default theme
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
const DEFAULT_THEME = 'Icinga';
|
|
|
|
|
2021-06-11 16:53:30 +02:00
|
|
|
/**
|
|
|
|
* The name of the default theme mode
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
const DEFAULT_MODE = 'dark';
|
|
|
|
|
2021-07-21 09:05:46 +02:00
|
|
|
/**
|
|
|
|
* The themes that are compatible with the default theme
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
const THEME_WHITELIST = [
|
|
|
|
'colorblind',
|
|
|
|
'high-contrast',
|
|
|
|
'solarized-dark',
|
|
|
|
'Winter'
|
|
|
|
];
|
|
|
|
|
2021-06-11 16:53:30 +02:00
|
|
|
/**
|
|
|
|
* RegEx pattern for matching full css @media query of theme mode
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
const REGEX_ALL_MODE_QUERY = '/@media\s*\(\s*min-height\s*:\s*@prefer-light-color-scheme\s*\)\s*,' .
|
|
|
|
'\s*\(\s*prefers-color-scheme\s*:\s*light\s*\)\s*and\s*\(\s*min-height\s*:\s*@enable-color-preference\s*\)/';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* RegEx pattern for matching css @media query of theme mode
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
const REGEX_AUTO_MODE_QUERY = '/@media.*prefers-color-scheme/';
|
|
|
|
|
2015-11-27 16:40:17 +01:00
|
|
|
/**
|
|
|
|
* Array of core LESS files Web 2 sends to the client
|
|
|
|
*
|
|
|
|
* @var string[]
|
|
|
|
*/
|
2020-11-10 15:27:59 +01:00
|
|
|
protected static $lessFiles = [
|
2015-01-29 11:25:37 +01:00
|
|
|
'../application/fonts/fontello-ifont/css/ifont-embedded.css',
|
2015-09-23 10:43:43 +02:00
|
|
|
'css/vendor/normalize.css',
|
2015-11-09 13:32:59 +01:00
|
|
|
'css/icinga/base.less',
|
2015-09-28 17:02:37 +02:00
|
|
|
'css/icinga/badges.less',
|
2015-08-21 16:45:05 +02:00
|
|
|
'css/icinga/mixins.less',
|
2015-09-29 17:02:59 +02:00
|
|
|
'css/icinga/grid.less',
|
2015-09-25 13:36:04 +02:00
|
|
|
'css/icinga/nav.less',
|
2015-09-27 16:00:19 +02:00
|
|
|
'css/icinga/main.less',
|
2015-02-25 13:51:26 +01:00
|
|
|
'css/icinga/animation.less',
|
2015-12-10 12:13:40 +01:00
|
|
|
'css/icinga/layout.less',
|
2014-03-04 16:15:08 +01:00
|
|
|
'css/icinga/layout-structure.less',
|
|
|
|
'css/icinga/menu.less',
|
|
|
|
'css/icinga/tabs.less',
|
2015-09-27 13:37:35 +02:00
|
|
|
'css/icinga/forms.less',
|
2014-10-06 16:04:58 +02:00
|
|
|
'css/icinga/setup.less',
|
2014-03-05 20:12:45 +01:00
|
|
|
'css/icinga/widgets.less',
|
2015-04-15 14:20:36 +02:00
|
|
|
'css/icinga/login.less',
|
2015-11-13 13:57:09 +01:00
|
|
|
'css/icinga/about.less',
|
2015-09-25 00:35:12 +02:00
|
|
|
'css/icinga/controls.less',
|
2015-09-30 18:52:52 +02:00
|
|
|
'css/icinga/dev.less',
|
2015-10-01 16:31:25 +02:00
|
|
|
'css/icinga/spinner.less',
|
2015-12-10 12:05:48 +01:00
|
|
|
'css/icinga/compat.less',
|
2016-01-11 11:12:24 +01:00
|
|
|
'css/icinga/print.less',
|
2019-10-21 11:28:18 +02:00
|
|
|
'css/icinga/responsive.less',
|
2021-03-25 17:47:30 +01:00
|
|
|
'css/icinga/modal.less',
|
2021-04-20 13:04:21 +02:00
|
|
|
'css/icinga/audit.less',
|
|
|
|
'css/icinga/health.less'
|
2020-11-10 15:27:59 +01:00
|
|
|
];
|
2014-03-04 16:15:08 +01:00
|
|
|
|
2015-11-27 16:40:17 +01:00
|
|
|
/**
|
|
|
|
* Application instance
|
|
|
|
*
|
|
|
|
* @var \Icinga\Application\EmbeddedWeb
|
|
|
|
*/
|
|
|
|
protected $app;
|
2014-03-06 10:21:32 +01:00
|
|
|
|
2015-11-27 16:40:17 +01:00
|
|
|
/**
|
|
|
|
* Less compiler
|
|
|
|
*
|
|
|
|
* @var LessCompiler
|
|
|
|
*/
|
|
|
|
protected $lessCompiler;
|
2014-03-27 08:32:02 +01:00
|
|
|
|
2015-11-27 16:40:17 +01:00
|
|
|
/**
|
|
|
|
* Path to the public directory
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $pubPath;
|
2014-09-05 10:55:25 +02:00
|
|
|
|
2015-11-27 16:40:17 +01:00
|
|
|
/**
|
|
|
|
* Create the StyleSheet
|
|
|
|
*/
|
|
|
|
public function __construct()
|
2014-11-13 16:35:26 +01:00
|
|
|
{
|
2015-11-27 16:40:17 +01:00
|
|
|
$app = Icinga::app();
|
|
|
|
$this->app = $app;
|
|
|
|
$this->lessCompiler = new LessCompiler();
|
2019-06-04 09:43:39 +02:00
|
|
|
$this->pubPath = $app->getBaseDir('public');
|
2015-11-27 16:40:17 +01:00
|
|
|
$this->collect();
|
2014-11-13 16:35:26 +01:00
|
|
|
}
|
|
|
|
|
2015-11-27 16:40:17 +01:00
|
|
|
/**
|
|
|
|
* Collect Web 2 and module LESS files and add them to the LESS compiler
|
|
|
|
*/
|
|
|
|
protected function collect()
|
2014-03-04 16:15:08 +01:00
|
|
|
{
|
2020-11-10 15:27:59 +01:00
|
|
|
foreach ($this->app->getLibraries() as $library) {
|
|
|
|
foreach ($library->getCssAssets() as $lessFile) {
|
|
|
|
$this->lessCompiler->addLessFile($lessFile);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-27 16:40:17 +01:00
|
|
|
foreach (self::$lessFiles as $lessFile) {
|
|
|
|
$this->lessCompiler->addLessFile($this->pubPath . '/' . $lessFile);
|
2014-09-02 16:24:55 +02:00
|
|
|
}
|
2015-11-27 16:40:17 +01:00
|
|
|
|
|
|
|
$mm = $this->app->getModuleManager();
|
|
|
|
|
|
|
|
foreach ($mm->getLoadedModules() as $moduleName => $module) {
|
2014-09-02 16:24:55 +02:00
|
|
|
if ($module->hasCss()) {
|
2015-11-27 16:40:17 +01:00
|
|
|
foreach ($module->getCssFiles() as $lessFilePath) {
|
|
|
|
$this->lessCompiler->addModuleLessFile($moduleName, $lessFilePath);
|
2015-08-06 15:14:29 +02:00
|
|
|
}
|
2014-09-02 16:24:55 +02:00
|
|
|
}
|
2019-09-25 09:53:53 +02:00
|
|
|
|
|
|
|
if ($module->requiresCss()) {
|
|
|
|
foreach ($module->getCssRequires() as $lessFilePath) {
|
|
|
|
$this->lessCompiler->addModuleRequire($moduleName, $lessFilePath);
|
|
|
|
}
|
|
|
|
}
|
2014-09-02 16:24:55 +02:00
|
|
|
}
|
|
|
|
|
2015-12-07 12:37:34 +01:00
|
|
|
$themingConfig = $this->app->getConfig()->getSection('themes');
|
2015-12-22 13:00:01 +01:00
|
|
|
$defaultTheme = $themingConfig->get('default');
|
2015-11-27 16:40:17 +01:00
|
|
|
$theme = null;
|
2016-01-22 16:34:31 +01:00
|
|
|
if ($defaultTheme !== null && $defaultTheme !== self::DEFAULT_THEME) {
|
|
|
|
$theme = $defaultTheme;
|
|
|
|
}
|
2015-11-27 16:40:17 +01:00
|
|
|
|
2016-01-22 16:34:31 +01:00
|
|
|
if (! (bool) $themingConfig->get('disabled', false)) {
|
2015-12-22 13:00:01 +01:00
|
|
|
$auth = Auth::getInstance();
|
|
|
|
if ($auth->isAuthenticated()) {
|
|
|
|
$userTheme = $auth->getUser()->getPreferences()->getValue('icingaweb', 'theme');
|
|
|
|
if ($userTheme !== null) {
|
|
|
|
$theme = $userTheme;
|
|
|
|
}
|
2015-11-27 16:40:17 +01:00
|
|
|
}
|
2014-09-02 16:24:55 +02:00
|
|
|
}
|
2014-03-27 08:32:02 +01:00
|
|
|
|
2021-06-11 16:53:30 +02:00
|
|
|
if ($themePath = self::getThemeFile($theme)) {
|
|
|
|
$this->lessCompiler->setTheme($themePath);
|
|
|
|
}
|
|
|
|
|
2021-07-21 09:05:46 +02:00
|
|
|
if (! $themePath || in_array($theme, self::THEME_WHITELIST, true)) {
|
|
|
|
$this->lessCompiler->addLessFile('css/icinga/login-orbs.less');
|
|
|
|
}
|
|
|
|
|
2021-06-11 16:53:30 +02:00
|
|
|
$mode = 'none';
|
|
|
|
if ($user = Auth::getInstance()->getUser()) {
|
|
|
|
$file = $themePath !== null ? file_get_contents($themePath) : '';
|
|
|
|
if ($file !== '' && ! preg_match(self::REGEX_ALL_MODE_QUERY, $file)) {
|
|
|
|
if (preg_match(self::REGEX_AUTO_MODE_QUERY, $file)) {
|
|
|
|
$mode = 'system';
|
2015-11-27 16:40:17 +01:00
|
|
|
}
|
|
|
|
} else {
|
2021-06-11 16:53:30 +02:00
|
|
|
$mode = $user->getPreferences()->getValue('icingaweb', 'theme_mode', self::DEFAULT_MODE);
|
2015-11-27 16:40:17 +01:00
|
|
|
}
|
|
|
|
}
|
2021-06-11 16:53:30 +02:00
|
|
|
|
|
|
|
$this->lessCompiler->setThemeMode($this->pubPath . '/css/modes/'. $mode . '.less');
|
2015-11-27 16:40:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the stylesheet for PDF export
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public static function forPdf()
|
|
|
|
{
|
|
|
|
$styleSheet = new self();
|
|
|
|
$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();
|
|
|
|
}
|
|
|
|
return $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();
|
2021-06-16 14:58:44 +02:00
|
|
|
$response->setHeader('Cache-Control', 'private,no-cache,must-revalidate', true);
|
2015-11-27 16:40:17 +01:00
|
|
|
|
|
|
|
$noCache = $request->getHeader('Cache-Control') === 'no-cache' || $request->getHeader('Pragma') === 'no-cache';
|
|
|
|
|
|
|
|
if (! $noCache && FileCache::etagMatchesFiles($styleSheet->lessCompiler->getLessFiles())) {
|
|
|
|
$response
|
|
|
|
->setHttpResponseCode(304)
|
|
|
|
->sendHeaders();
|
2014-09-02 16:24:55 +02:00
|
|
|
return;
|
2014-03-27 08:32:02 +01:00
|
|
|
}
|
2015-01-28 17:03:23 +01:00
|
|
|
|
2015-11-27 16:40:17 +01:00
|
|
|
$etag = FileCache::etagForFiles($styleSheet->lessCompiler->getLessFiles());
|
|
|
|
|
2017-08-22 09:33:28 +02:00
|
|
|
$response->setHeader('ETag', $etag, true)
|
2016-11-09 11:38:04 +01:00
|
|
|
->setHeader('Content-Type', 'text/css', true);
|
2015-11-27 16:40:17 +01:00
|
|
|
|
|
|
|
$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);
|
2014-03-04 16:15:08 +01:00
|
|
|
}
|
2015-11-27 16:40:17 +01:00
|
|
|
|
|
|
|
$response->sendResponse();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Render the stylesheet
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function __toString()
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
return $this->render();
|
|
|
|
} catch (Exception $e) {
|
|
|
|
Logger::error($e);
|
|
|
|
return IcingaException::describe($e);
|
2014-03-27 08:32:02 +01:00
|
|
|
}
|
2014-03-04 16:15:08 +01:00
|
|
|
}
|
2021-06-11 16:53:30 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 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 (($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;
|
|
|
|
}
|
2014-03-04 16:15:08 +01:00
|
|
|
}
|