app = $app; $this->lessCompiler = new LessCompiler($disableModes); $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) { $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 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(); } 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 * @param bool $disableModes Disable replacing compiled Less colors with CSS var() function calls and don't inject * light mode calls */ public static function send($minified = false, $disableModes = false) { $styleSheet = new self($disableModes); $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'; if (! $noCache && FileCache::etagMatchesFiles($styleSheet->lessCompiler->getLessFiles())) { $response ->setHttpResponseCode(304) ->sendHeaders(); return; } $etag = FileCache::etagForFiles($styleSheet->lessCompiler->getLessFiles()); $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 (($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; } }