|
|
- <?php
-
- declare(strict_types=1);
-
- /**
- * @copyright Copyright (c) 2018, John Molakvoæ (skjnldsv@protonmail.com)
- *
- * @author Christoph Wurst <christoph@winzerhof-wurst.at>
- * @author Joas Schilling <coding@schilljs.com>
- * @author John Molakvoæ (skjnldsv) <skjnldsv@protonmail.com>
- * @author Julius Härtl <jus@bitgrid.net>
- * @author Robin Appelman <robin@icewind.nl>
- * @author Roeland Jago Douma <roeland@famdouma.nl>
- *
- * @license GNU AGPL version 3 or any later version
- *
- * This program is free software: you can redistribute it and/or modify
- * it under the terms of the GNU Affero General Public License as
- * published by the Free Software Foundation, either version 3 of the
- * License, or (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU Affero General Public License for more details.
- *
- * You should have received a copy of the GNU Affero General Public License
- * along with this program. If not, see <http://www.gnu.org/licenses/>.
- *
- */
-
- namespace OC\Template;
-
- use OC\Files\AppData\Factory;
- use OCP\AppFramework\Utility\ITimeFactory;
- use OCP\Files\IAppData;
- use OCP\Files\NotFoundException;
- use OCP\Files\SimpleFS\ISimpleFile;
- use OCP\Files\SimpleFS\ISimpleFolder;
- use OCP\ILogger;
- use OCP\IURLGenerator;
-
- class IconsCacher {
-
- /** @var ILogger */
- protected $logger;
-
- /** @var IAppData */
- protected $appData;
-
- /** @var ISimpleFolder */
- private $folder;
-
- /** @var IURLGenerator */
- protected $urlGenerator;
-
- /** @var ITimeFactory */
- protected $timeFactory;
-
- /** @var string */
- private $iconVarRE = '/--(icon-[a-zA-Z0-9-]+):\s?url\(["\']?([a-zA-Z0-9-_\~\/\.\?\&\=\:\;\+\,]+)[^;]+;/m';
-
- /** @var string */
- private $fileName = 'icons-vars.css';
-
- private $iconList = 'icons-list.template';
-
- private $cachedCss;
- private $cachedList;
-
- /**
- * @param ILogger $logger
- * @param Factory $appDataFactory
- * @param IURLGenerator $urlGenerator
- * @param ITimeFactory $timeFactory
- * @throws \OCP\Files\NotPermittedException
- */
- public function __construct(ILogger $logger,
- Factory $appDataFactory,
- IURLGenerator $urlGenerator,
- ITimeFactory $timeFactory) {
- $this->logger = $logger;
- $this->appData = $appDataFactory->get('css');
- $this->urlGenerator = $urlGenerator;
- $this->timeFactory = $timeFactory;
-
- try {
- $this->folder = $this->appData->getFolder('icons');
- } catch (NotFoundException $e) {
- $this->folder = $this->appData->newFolder('icons');
- }
- }
-
- private function getIconsFromCss(string $css): array {
- preg_match_all($this->iconVarRE, $css, $matches, PREG_SET_ORDER);
- $icons = [];
- foreach ($matches as $icon) {
- $icons[$icon[1]] = $icon[2];
- }
-
- return $icons;
- }
-
- /**
- * @param string $css
- * @return string
- * @throws NotFoundException
- * @throws \OCP\Files\NotPermittedException
- */
- public function setIconsCss(string $css): string {
- $cachedFile = $this->getCachedList();
- if (!$cachedFile) {
- $currentData = '';
- $cachedFile = $this->folder->newFile($this->iconList);
- } else {
- $currentData = $cachedFile->getContent();
- }
-
- $cachedVarsCssFile = $this->getCachedCSS();
- if (!$cachedVarsCssFile) {
- $cachedVarsCssFile = $this->folder->newFile($this->fileName);
- }
-
- $icons = $this->getIconsFromCss($currentData . $css);
-
- $data = '';
- $list = '';
- foreach ($icons as $icon => $url) {
- $list .= "--$icon: url('$url');";
- list($location,$color) = $this->parseUrl($url);
- $svg = false;
- if ($location !== '' && \file_exists($location)) {
- $svg = \file_get_contents($location);
- }
- if ($svg === false) {
- $this->logger->debug('Failed to get icon file ' . $location);
- $data .= "--$icon: url('$url');";
- continue;
- }
- $encode = base64_encode($this->colorizeSvg($svg, $color));
- $data .= '--' . $icon . ': url(data:image/svg+xml;base64,' . $encode . ');';
- }
-
- if (\strlen($data) > 0 && \strlen($list) > 0) {
- $data = ":root {\n$data\n}";
- $cachedVarsCssFile->putContent($data);
- $list = ":root {\n$list\n}";
- $cachedFile->putContent($list);
- $this->cachedList = null;
- $this->cachedCss = null;
- }
-
- return preg_replace($this->iconVarRE, '', $css);
- }
-
- /**
- * @param $url
- * @return array
- */
- private function parseUrl($url): array {
- $location = '';
- $color = '';
- $base = $this->getRoutePrefix() . '/svg/';
- $cleanUrl = \substr($url, \strlen($base));
- if (\strpos($url, $base . 'core') === 0) {
- $cleanUrl = \substr($cleanUrl, \strlen('core'));
- if (\preg_match('/\/([a-zA-Z0-9-_\~\/\.\=\:\;\+\,]+)\?color=([0-9a-fA-F]{3,6})/', $cleanUrl, $matches)) {
- list(,$cleanUrl,$color) = $matches;
- $location = \OC::$SERVERROOT . '/core/img/' . $cleanUrl . '.svg';
- }
- } elseif (\strpos($url, $base) === 0) {
- if (\preg_match('/([A-z0-9\_\-]+)\/([a-zA-Z0-9-_\~\/\.\=\:\;\+\,]+)\?color=([0-9a-fA-F]{3,6})/', $cleanUrl, $matches)) {
- list(,$app,$cleanUrl, $color) = $matches;
- $location = \OC_App::getAppPath($app) . '/img/' . $cleanUrl . '.svg';
- if ($app === 'settings') {
- $location = \OC::$SERVERROOT . '/settings/img/' . $cleanUrl . '.svg';
- }
- }
- }
- return [
- $location,
- $color
- ];
- }
-
- /**
- * @param $svg
- * @param $color
- * @return string
- */
- public function colorizeSvg($svg, $color): string {
- if (!preg_match('/^[0-9a-f]{3,6}$/i', $color)) {
- // Prevent not-sane colors from being written into the SVG
- $color = '000';
- }
-
- // add fill (fill is not present on black elements)
- $fillRe = '/<((circle|rect|path)((?!fill)[a-z0-9 =".\-#():;,])+)\/>/mi';
- $svg = preg_replace($fillRe, '<$1 fill="#' . $color . '"/>', $svg);
-
- // replace any fill or stroke colors
- $svg = preg_replace('/stroke="#([a-z0-9]{3,6})"/mi', 'stroke="#' . $color . '"', $svg);
- $svg = preg_replace('/fill="#([a-z0-9]{3,6})"/mi', 'fill="#' . $color . '"', $svg);
- return $svg;
- }
-
- private function getRoutePrefix() {
- $frontControllerActive = (\OC::$server->getConfig()->getSystemValue('htaccess.IgnoreFrontController', false) === true || getenv('front_controller_active') === 'true');
- $prefix = \OC::$WEBROOT . '/index.php';
- if ($frontControllerActive) {
- $prefix = \OC::$WEBROOT;
- }
- return $prefix;
- }
-
- /**
- * Get icons css file
- * @return ISimpleFile|boolean
- */
- public function getCachedCSS() {
- try {
- if (!$this->cachedCss) {
- $this->cachedCss = $this->folder->getFile($this->fileName);
- }
- return $this->cachedCss;
- } catch (NotFoundException $e) {
- return false;
- }
- }
-
- /**
- * Get icon-vars list template
- * @return ISimpleFile|boolean
- */
- public function getCachedList() {
- try {
- if (!$this->cachedList) {
- $this->cachedList = $this->folder->getFile($this->iconList);
- }
- return $this->cachedList;
- } catch (NotFoundException $e) {
- return false;
- }
- }
-
- /**
- * Add the icons cache css into the header
- */
- public function injectCss() {
- $mtime = $this->timeFactory->getTime();
- $file = $this->getCachedList();
- if ($file) {
- $mtime = $file->getMTime();
- }
- // Only inject once
- foreach (\OC_Util::$headers as $header) {
- if (
- array_key_exists('attributes', $header) &&
- array_key_exists('href', $header['attributes']) &&
- strpos($header['attributes']['href'], $this->fileName) !== false) {
- return;
- }
- }
- $linkToCSS = $this->urlGenerator->linkToRoute('core.Css.getCss', ['appName' => 'icons', 'fileName' => $this->fileName, 'v' => $mtime]);
- \OC_Util::addHeader('link', ['rel' => 'stylesheet', 'href' => $linkToCSS], null, true);
- }
- }
|