summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMorris Jobke <hey@morrisjobke.de>2018-10-25 16:04:23 +0200
committerGitHub <noreply@github.com>2018-10-25 16:04:23 +0200
commit6ad7f329385f73a731f69d9f00cf5b038f3eac04 (patch)
treec85fa80469f653b4a9390ae8ea4a0dc3f699ea5b
parent0a2476fcb3823864bceb116d79d48a8278ea4981 (diff)
parentd21ded67a70672cfa56a48051fa752b431118604 (diff)
downloadnextcloud-server-6ad7f329385f73a731f69d9f00cf5b038f3eac04.tar.gz
nextcloud-server-6ad7f329385f73a731f69d9f00cf5b038f3eac04.zip
Merge pull request #12016 from nextcloud/wip/noid/icon-base64
Directly embed icons into the icon-vars css file
-rw-r--r--apps/accessibility/lib/Controller/AccessibilityController.php4
-rw-r--r--core/Controller/SvgController.php18
-rw-r--r--lib/private/Template/IconsCacher.php119
-rw-r--r--tests/Core/Controller/SvgControllerTest.php4
-rw-r--r--tests/lib/Template/IconsCacherTest.php14
5 files changed, 122 insertions, 37 deletions
diff --git a/apps/accessibility/lib/Controller/AccessibilityController.php b/apps/accessibility/lib/Controller/AccessibilityController.php
index d679e37ed99..5a5ff5b8a2f 100644
--- a/apps/accessibility/lib/Controller/AccessibilityController.php
+++ b/apps/accessibility/lib/Controller/AccessibilityController.php
@@ -166,8 +166,8 @@ class AccessibilityController extends Controller {
$appWebRoot = substr($this->appRoot, strlen($this->serverRoot) - strlen(\OC::$WEBROOT));
$css = $this->rebaseUrls($css, $appWebRoot . '/css');
- if (in_array('themedark', $userValues) && $this->iconsCacher->getCachedCSS() && $this->iconsCacher->getCachedCSS()->getSize() > 0) {
- $iconsCss = $this->invertSvgIconsColor($this->iconsCacher->getCachedCSS()->getContent());
+ if (in_array('themedark', $userValues) && $this->iconsCacher->getCachedList() && $this->iconsCacher->getCachedList()->getSize() > 0) {
+ $iconsCss = $this->invertSvgIconsColor($this->iconsCacher->getCachedList()->getContent());
$css = $css . $iconsCss;
}
diff --git a/core/Controller/SvgController.php b/core/Controller/SvgController.php
index f7159dd9fe1..bbf4e61c60c 100644
--- a/core/Controller/SvgController.php
+++ b/core/Controller/SvgController.php
@@ -31,6 +31,7 @@ use OCP\AppFramework\Http\NotFoundResponse;
use OCP\AppFramework\Utility\ITimeFactory;
use OCP\App\IAppManager;
use OCP\IRequest;
+use OC\Template\IconsCacher;
class SvgController extends Controller {
@@ -43,15 +44,20 @@ class SvgController extends Controller {
/** @var IAppManager */
protected $appManager;
+ /** @var IconsCacher */
+ private $iconsCacher;
+
public function __construct(string $appName,
IRequest $request,
ITimeFactory $timeFactory,
- IAppManager $appManager) {
+ IAppManager $appManager,
+ IconsCacher $iconsCacher) {
parent::__construct($appName, $request);
$this->serverRoot = \OC::$SERVERROOT;
$this->timeFactory = $timeFactory;
$this->appManager = $appManager;
+ $this->iconsCacher = $iconsCacher;
}
/**
@@ -114,17 +120,11 @@ class SvgController extends Controller {
$svg = file_get_contents($path);
- if (is_null($svg)) {
+ if ($svg === null) {
return new NotFoundResponse();
}
- // 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);
+ $svg = $this->iconsCacher->colorizeSvg($svg, $color);
$response = new DataDisplayResponse($svg, Http::STATUS_OK, ['Content-Type' => 'image/svg+xml']);
diff --git a/lib/private/Template/IconsCacher.php b/lib/private/Template/IconsCacher.php
index f3660442dc5..c1a78a567f9 100644
--- a/lib/private/Template/IconsCacher.php
+++ b/lib/private/Template/IconsCacher.php
@@ -47,15 +47,18 @@ class IconsCacher {
protected $urlGenerator;
/** @var string */
- private $iconVarRE = '/--(icon-[a-zA-Z0-9-]+):\s?url\(["\']([a-zA-Z0-9-_\~\/\.\?\=]+)[^;]+;/m';
+ 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';
+
/**
* @param ILogger $logger
* @param Factory $appDataFactory
* @param IURLGenerator $urlGenerator
+ * @throws \OCP\Files\NotPermittedException
*/
public function __construct(ILogger $logger,
Factory $appDataFactory,
@@ -71,7 +74,7 @@ class IconsCacher {
}
}
- private function getIconsFromCss(string $css): array{
+ private function getIconsFromCss(string $css): array {
preg_match_all($this->iconVarRE, $css, $matches, PREG_SET_ORDER);
$icons = [];
foreach ($matches as $icon) {
@@ -80,45 +83,113 @@ class IconsCacher {
return $icons;
}
+
/**
- * Parse and cache css
- *
* @param string $css
+ * @return string
+ * @throws NotFoundException
+ * @throws \OCP\Files\NotPermittedException
*/
- public function setIconsCss(string $css) {
+ public function setIconsCss(string $css): string {
- $cachedFile = $this->getCachedCSS();
+ $cachedFile = $this->getCachedList();
if (!$cachedFile) {
$currentData = '';
+ $cachedFile = $this->folder->newFile($this->iconList);
} else {
$currentData = $cachedFile->getContent();
}
- // remove :root
- $currentData = str_replace([':root {', '}'], '', $currentData);
+ $cachedVarsCssFile = $this->getCachedCSS();
+ if (!$cachedVarsCssFile) {
+ $cachedVarsCssFile = $this->folder->newFile($this->fileName);
+ }
$icons = $this->getIconsFromCss($currentData . $css);
$data = '';
+ $list = '';
foreach ($icons as $icon => $url) {
- $data .= "--$icon: url('$url');";
- }
-
- if (strlen($data) > 0) {
- if (!$cachedFile) {
- $cachedFile = $this->folder->newFile($this->fileName);
+ $list .= "--$icon: url('$url');";
+ list($location,$color) = $this->parseUrl($url);
+ $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 . ');';
+ }
- $data = ":root {
- $data
- }";
- $cachedFile->putContent($data);
+ if (\strlen($data) > 0 && \strlen($list) > 0) {
+ $data = ":root {\n$data\n}";
+ $cachedVarsCssFile->putContent($data);
+ $list = ":root {\n$list\n}";
+ $cachedFile->putContent($list);
}
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'), \strpos($cleanUrl, '?')-\strlen('core'));
+ $parts = \explode('/', $cleanUrl);
+ $color = \array_pop($parts);
+ $cleanUrl = \implode('/', $parts);
+ $location = \OC::$SERVERROOT . '/core/img/' . $cleanUrl . '.svg';
+ } elseif (\strpos($url, $base) === 0) {
+ $cleanUrl = \substr($cleanUrl, 0, \strpos($cleanUrl, '?'));
+ $parts = \explode('/', $cleanUrl);
+ $app = \array_shift($parts);
+ $color = \array_pop($parts);
+ $cleanUrl = \implode('/', $parts);
+ $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 {
+ // 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
*/
@@ -130,6 +201,18 @@ class IconsCacher {
}
}
+ /**
+ * Get icon-vars list template
+ * @return ISimpleFile|boolean
+ */
+ public function getCachedList() {
+ try {
+ return $this->folder->getFile($this->iconList);
+ } catch (NotFoundException $e) {
+ return false;
+ }
+ }
+
public function injectCss() {
// Only inject once
foreach (\OC_Util::$headers as $header) {
diff --git a/tests/Core/Controller/SvgControllerTest.php b/tests/Core/Controller/SvgControllerTest.php
index 7a31d02b90f..2cac635c896 100644
--- a/tests/Core/Controller/SvgControllerTest.php
+++ b/tests/Core/Controller/SvgControllerTest.php
@@ -26,6 +26,7 @@ namespace Tests\Core\Controller;
use OC\AppFramework\Http;
use OC\Core\Controller\SvgController;
+use OC\Template\IconsCacher;
use OCP\App\IAppManager;
use OCP\AppFramework\Utility\ITimeFactory;
use OCP\IRequest;
@@ -89,7 +90,8 @@ class SvgControllerTest extends TestCase {
$request = $this->getMockBuilder(IRequest::class)->getMock();
$timeFactory = $this->getMockBuilder(ITimeFactory::class)->getMock();
$appManager = $this->getMockBuilder(IAppManager::class)->getMock();
- $this->svgController = new SvgController('core', $request, $timeFactory, $appManager);
+ $iconsCacher = $this->getMockBuilder(IconsCacher::class)->disableOriginalConstructor()->setMethods(['__construct'])->getMock();
+ $this->svgController = new SvgController('core', $request, $timeFactory, $appManager, $iconsCacher);
}
/**
diff --git a/tests/lib/Template/IconsCacherTest.php b/tests/lib/Template/IconsCacherTest.php
index d6a9908989c..33735e3a45e 100644
--- a/tests/lib/Template/IconsCacherTest.php
+++ b/tests/lib/Template/IconsCacherTest.php
@@ -104,7 +104,7 @@ class IconsCacherTest extends \Test\TestCase {
public function testSetIconsFromValidCss() {
$css = "
icon.test {
- --icon-test: url('/svg/core/actions/add/000?v=1');
+ --icon-test: url('/index.php/svg/core/actions/add/000?v=1');
background-image: var(--icon-test);
}
";
@@ -116,10 +116,10 @@ class IconsCacherTest extends \Test\TestCase {
";
$iconsFile = $this->createMock(ISimpleFile::class);
- $this->folder->expects($this->once())
+ $this->folder->expects($this->exactly(2))
->method('getFile')
->willReturn($iconsFile);
-
+
$actual = $this->iconsCacher->setIconsCss($css);
$this->assertEquals($expected, $actual);
}
@@ -127,7 +127,7 @@ class IconsCacherTest extends \Test\TestCase {
public function testSetIconsFromValidCssMultipleTimes() {
$css = "
icon.test {
- --icon-test: url('/svg/core/actions/add/000?v=1');
+ --icon-test: url('/index.php/svg/core/actions/add/000?v=1');
background-image: var(--icon-test);
}
";
@@ -139,14 +139,14 @@ class IconsCacherTest extends \Test\TestCase {
";
$iconsFile = $this->createMock(ISimpleFile::class);
- $this->folder->expects($this->exactly(3))
+ $this->folder->expects($this->exactly(6))
->method('getFile')
->willReturn($iconsFile);
-
+
$actual = $this->iconsCacher->setIconsCss($css);
$actual = $this->iconsCacher->setIconsCss($actual);
$actual = $this->iconsCacher->setIconsCss($actual);
$this->assertEquals($expected, $actual);
}
-} \ No newline at end of file
+}