2016-08-30 10:03:06 +03:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* @copyright Copyright (c) 2016 Julius Härtl <jus@bitgrid.net>
|
|
|
|
*
|
2017-11-06 17:56:42 +03:00
|
|
|
* @author Jan-Christoph Borchardt <hey@jancborchardt.net>
|
|
|
|
* @author Julius Haertl <jus@bitgrid.net>
|
2016-08-30 10:03:06 +03:00
|
|
|
* @author Julius Härtl <jus@bitgrid.net>
|
|
|
|
*
|
|
|
|
* @license GNU AGPL version 3 or any later version
|
|
|
|
*
|
2017-11-06 17:56:42 +03:00
|
|
|
* 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.
|
2016-08-30 10:03:06 +03:00
|
|
|
*
|
2017-11-06 17:56:42 +03:00
|
|
|
* 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.
|
2016-08-30 10:03:06 +03:00
|
|
|
*
|
2017-11-06 17:56:42 +03:00
|
|
|
* You should have received a copy of the GNU Affero General Public License
|
2019-12-03 21:57:53 +03:00
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2016-08-30 10:03:06 +03:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace OCA\Theming;
|
|
|
|
|
|
|
|
use Imagick;
|
|
|
|
use ImagickPixel;
|
2016-11-18 12:49:03 +03:00
|
|
|
use OCP\App\AppPathNotFoundException;
|
2017-05-15 00:52:14 +03:00
|
|
|
use OCP\Files\SimpleFS\ISimpleFile;
|
2016-08-30 10:03:06 +03:00
|
|
|
|
|
|
|
class IconBuilder {
|
|
|
|
/** @var ThemingDefaults */
|
|
|
|
private $themingDefaults;
|
|
|
|
/** @var Util */
|
|
|
|
private $util;
|
2018-06-05 17:59:05 +03:00
|
|
|
/** @var ImageManager */
|
|
|
|
private $imageManager;
|
2016-08-30 10:03:06 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* IconBuilder constructor.
|
|
|
|
*
|
|
|
|
* @param ThemingDefaults $themingDefaults
|
|
|
|
* @param Util $util
|
2018-06-05 17:59:05 +03:00
|
|
|
* @param ImageManager $imageManager
|
2016-08-30 10:03:06 +03:00
|
|
|
*/
|
|
|
|
public function __construct(
|
|
|
|
ThemingDefaults $themingDefaults,
|
2018-06-05 17:59:05 +03:00
|
|
|
Util $util,
|
|
|
|
ImageManager $imageManager
|
2016-08-30 10:03:06 +03:00
|
|
|
) {
|
|
|
|
$this->themingDefaults = $themingDefaults;
|
|
|
|
$this->util = $util;
|
2018-06-05 17:59:05 +03:00
|
|
|
$this->imageManager = $imageManager;
|
2016-08-30 10:03:06 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-08-30 12:51:48 +03:00
|
|
|
* @param $app string app name
|
2016-11-05 22:11:49 +03:00
|
|
|
* @return string|false image blob
|
2016-08-30 10:03:06 +03:00
|
|
|
*/
|
|
|
|
public function getFavicon($app) {
|
2018-06-05 17:59:05 +03:00
|
|
|
if (!$this->imageManager->shouldReplaceIcons()) {
|
2017-09-23 12:35:45 +03:00
|
|
|
return false;
|
|
|
|
}
|
2017-05-15 00:48:51 +03:00
|
|
|
try {
|
2017-09-12 13:05:03 +03:00
|
|
|
$favicon = new Imagick();
|
|
|
|
$favicon->setFormat("ico");
|
|
|
|
$icon = $this->renderAppIcon($app, 128);
|
2017-05-15 00:48:51 +03:00
|
|
|
if ($icon === false) {
|
|
|
|
return false;
|
|
|
|
}
|
2017-09-12 12:32:53 +03:00
|
|
|
$icon->setImageFormat("png32");
|
2017-09-12 13:05:03 +03:00
|
|
|
|
|
|
|
$clone = clone $icon;
|
|
|
|
$clone->scaleImage(16,0);
|
|
|
|
$favicon->addImage($clone);
|
|
|
|
|
|
|
|
$clone = clone $icon;
|
|
|
|
$clone->scaleImage(32,0);
|
|
|
|
$favicon->addImage($clone);
|
|
|
|
|
|
|
|
$clone = clone $icon;
|
|
|
|
$clone->scaleImage(64,0);
|
|
|
|
$favicon->addImage($clone);
|
|
|
|
|
|
|
|
$clone = clone $icon;
|
|
|
|
$clone->scaleImage(128,0);
|
|
|
|
$favicon->addImage($clone);
|
|
|
|
|
|
|
|
$data = $favicon->getImagesBlob();
|
|
|
|
$favicon->destroy();
|
2017-05-15 00:48:51 +03:00
|
|
|
$icon->destroy();
|
2017-09-12 13:05:03 +03:00
|
|
|
$clone->destroy();
|
2017-05-15 00:48:51 +03:00
|
|
|
return $data;
|
|
|
|
} catch (\ImagickException $e) {
|
2016-11-05 22:11:49 +03:00
|
|
|
return false;
|
|
|
|
}
|
2016-08-30 10:03:06 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-08-30 12:51:48 +03:00
|
|
|
* @param $app string app name
|
2016-11-05 22:11:49 +03:00
|
|
|
* @return string|false image blob
|
2016-08-30 10:03:06 +03:00
|
|
|
*/
|
|
|
|
public function getTouchIcon($app) {
|
2017-05-15 00:48:51 +03:00
|
|
|
try {
|
|
|
|
$icon = $this->renderAppIcon($app, 512);
|
|
|
|
if ($icon === false) {
|
|
|
|
return false;
|
|
|
|
}
|
2017-09-12 12:32:53 +03:00
|
|
|
$icon->setImageFormat("png32");
|
2017-05-15 00:48:51 +03:00
|
|
|
$data = $icon->getImageBlob();
|
|
|
|
$icon->destroy();
|
|
|
|
return $data;
|
|
|
|
} catch (\ImagickException $e) {
|
2016-11-05 22:11:49 +03:00
|
|
|
return false;
|
|
|
|
}
|
2016-08-30 10:03:06 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Render app icon on themed background color
|
|
|
|
* fallback to logo
|
|
|
|
*
|
2016-08-30 12:51:48 +03:00
|
|
|
* @param $app string app name
|
2016-12-04 15:02:17 +03:00
|
|
|
* @param $size int size of the icon in px
|
2016-11-05 22:11:49 +03:00
|
|
|
* @return Imagick|false
|
2016-08-30 10:03:06 +03:00
|
|
|
*/
|
2016-12-04 15:02:17 +03:00
|
|
|
public function renderAppIcon($app, $size) {
|
2017-05-15 00:52:14 +03:00
|
|
|
$appIcon = $this->util->getAppIcon($app);
|
2020-04-10 15:19:56 +03:00
|
|
|
if ($appIcon === false) {
|
2016-11-18 12:49:03 +03:00
|
|
|
return false;
|
|
|
|
}
|
2017-05-15 00:52:14 +03:00
|
|
|
if ($appIcon instanceof ISimpleFile) {
|
|
|
|
$appIconContent = $appIcon->getContent();
|
|
|
|
$mime = $appIcon->getMimeType();
|
|
|
|
} else {
|
|
|
|
$appIconContent = file_get_contents($appIcon);
|
|
|
|
$mime = mime_content_type($appIcon);
|
|
|
|
}
|
2016-11-18 12:49:03 +03:00
|
|
|
|
2020-04-10 15:19:56 +03:00
|
|
|
if ($appIconContent === false || $appIconContent === "") {
|
2016-11-05 22:11:49 +03:00
|
|
|
return false;
|
|
|
|
}
|
2016-08-30 10:03:06 +03:00
|
|
|
|
2017-03-28 02:37:47 +03:00
|
|
|
$color = $this->themingDefaults->getColorPrimary();
|
2016-09-18 21:15:06 +03:00
|
|
|
|
2016-08-30 10:03:06 +03:00
|
|
|
// generate background image with rounded corners
|
|
|
|
$background = '<?xml version="1.0" encoding="UTF-8"?>' .
|
|
|
|
'<svg xmlns="http://www.w3.org/2000/svg" version="1.1" xmlns:cc="http://creativecommons.org/ns#" width="512" height="512" xmlns:xlink="http://www.w3.org/1999/xlink">' .
|
2016-12-04 15:03:02 +03:00
|
|
|
'<rect x="0" y="0" rx="100" ry="100" width="512" height="512" style="fill:' . $color . ';" />' .
|
2016-08-30 10:03:06 +03:00
|
|
|
'</svg>';
|
|
|
|
// resize svg magic as this seems broken in Imagemagick
|
2020-04-10 15:19:56 +03:00
|
|
|
if ($mime === "image/svg+xml" || substr($appIconContent, 0, 4) === "<svg") {
|
|
|
|
if (substr($appIconContent, 0, 5) !== "<?xml") {
|
2016-09-18 21:15:06 +03:00
|
|
|
$svg = "<?xml version=\"1.0\"?>".$appIconContent;
|
2016-10-14 15:57:58 +03:00
|
|
|
} else {
|
|
|
|
$svg = $appIconContent;
|
2016-09-18 21:15:06 +03:00
|
|
|
}
|
2016-08-30 10:03:06 +03:00
|
|
|
$tmp = new Imagick();
|
|
|
|
$tmp->readImageBlob($svg);
|
|
|
|
$x = $tmp->getImageWidth();
|
|
|
|
$y = $tmp->getImageHeight();
|
|
|
|
$res = $tmp->getImageResolution();
|
|
|
|
$tmp->destroy();
|
|
|
|
|
2020-04-10 15:19:56 +03:00
|
|
|
if ($x>$y) {
|
2016-10-14 22:42:25 +03:00
|
|
|
$max = $x;
|
|
|
|
} else {
|
|
|
|
$max = $y;
|
|
|
|
}
|
|
|
|
|
2016-08-30 10:03:06 +03:00
|
|
|
// convert svg to resized image
|
|
|
|
$appIconFile = new Imagick();
|
2016-10-14 22:42:25 +03:00
|
|
|
$resX = (int)(512 * $res['x'] / $max * 2.53);
|
|
|
|
$resY = (int)(512 * $res['y'] / $max * 2.53);
|
2016-08-30 10:03:06 +03:00
|
|
|
$appIconFile->setResolution($resX, $resY);
|
|
|
|
$appIconFile->setBackgroundColor(new ImagickPixel('transparent'));
|
|
|
|
$appIconFile->readImageBlob($svg);
|
2017-09-12 12:35:02 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* invert app icons for bright primary colors
|
|
|
|
* the default nextcloud logo will not be inverted to black
|
|
|
|
*/
|
|
|
|
if ($this->util->invertTextColor($color)
|
|
|
|
&& !$appIcon instanceof ISimpleFile
|
|
|
|
&& $app !== "core"
|
|
|
|
) {
|
|
|
|
$appIconFile->negateImage(false);
|
|
|
|
}
|
2016-10-14 22:42:25 +03:00
|
|
|
$appIconFile->scaleImage(512, 512, true);
|
2016-08-30 10:03:06 +03:00
|
|
|
} else {
|
|
|
|
$appIconFile = new Imagick();
|
|
|
|
$appIconFile->setBackgroundColor(new ImagickPixel('transparent'));
|
2017-05-15 00:52:14 +03:00
|
|
|
$appIconFile->readImageBlob($appIconContent);
|
2016-08-30 10:03:06 +03:00
|
|
|
$appIconFile->scaleImage(512, 512, true);
|
|
|
|
}
|
|
|
|
// offset for icon positioning
|
|
|
|
$border_w = (int)($appIconFile->getImageWidth() * 0.05);
|
|
|
|
$border_h = (int)($appIconFile->getImageHeight() * 0.05);
|
|
|
|
$innerWidth = (int)($appIconFile->getImageWidth() - $border_w * 2);
|
|
|
|
$innerHeight = (int)($appIconFile->getImageHeight() - $border_h * 2);
|
|
|
|
$appIconFile->adaptiveResizeImage($innerWidth, $innerHeight);
|
|
|
|
// center icon
|
|
|
|
$offset_w = 512 / 2 - $innerWidth / 2;
|
|
|
|
$offset_h = 512 / 2 - $innerHeight / 2;
|
|
|
|
|
|
|
|
$finalIconFile = new Imagick();
|
2016-12-04 15:04:19 +03:00
|
|
|
$finalIconFile->setBackgroundColor(new ImagickPixel('transparent'));
|
2016-08-30 10:03:06 +03:00
|
|
|
$finalIconFile->readImageBlob($background);
|
|
|
|
$finalIconFile->setImageVirtualPixelMethod(Imagick::VIRTUALPIXELMETHOD_TRANSPARENT);
|
|
|
|
$finalIconFile->setImageArtifact('compose:args', "1,0,-0.5,0.5");
|
|
|
|
$finalIconFile->compositeImage($appIconFile, Imagick::COMPOSITE_ATOP, $offset_w, $offset_h);
|
2016-12-04 15:02:17 +03:00
|
|
|
$finalIconFile->setImageFormat('png24');
|
2016-12-09 13:51:02 +03:00
|
|
|
if (defined("Imagick::INTERPOLATE_BICUBIC") === true) {
|
|
|
|
$filter = Imagick::INTERPOLATE_BICUBIC;
|
|
|
|
} else {
|
|
|
|
$filter = Imagick::FILTER_LANCZOS;
|
|
|
|
}
|
|
|
|
$finalIconFile->resizeImage($size, $size, $filter, 1, false);
|
2016-08-30 10:03:06 +03:00
|
|
|
|
|
|
|
$appIconFile->destroy();
|
|
|
|
return $finalIconFile;
|
|
|
|
}
|
|
|
|
|
2016-09-18 21:15:06 +03:00
|
|
|
public function colorSvg($app, $image) {
|
2016-11-18 12:49:03 +03:00
|
|
|
try {
|
|
|
|
$imageFile = $this->util->getAppImage($app, $image);
|
|
|
|
} catch (AppPathNotFoundException $e) {
|
|
|
|
return false;
|
|
|
|
}
|
2016-09-18 21:15:06 +03:00
|
|
|
$svg = file_get_contents($imageFile);
|
2016-12-09 14:44:01 +03:00
|
|
|
if ($svg !== false && $svg !== "") {
|
2017-03-28 02:37:47 +03:00
|
|
|
$color = $this->util->elementColor($this->themingDefaults->getColorPrimary());
|
2016-11-05 22:11:49 +03:00
|
|
|
$svg = $this->util->colorizeSvg($svg, $color);
|
|
|
|
return $svg;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
2016-09-18 21:15:06 +03:00
|
|
|
}
|
2016-10-14 15:57:58 +03:00
|
|
|
}
|