vendor/pimcore/pimcore/models/Asset/Image/Thumbnail.php line 128

Open in your IDE?
  1. <?php
  2. /**
  3.  * Pimcore
  4.  *
  5.  * This source file is available under two different licenses:
  6.  * - GNU General Public License version 3 (GPLv3)
  7.  * - Pimcore Commercial License (PCL)
  8.  * Full copyright and license information is available in
  9.  * LICENSE.md which is distributed with this source code.
  10.  *
  11.  *  @copyright  Copyright (c) Pimcore GmbH (http://www.pimcore.org)
  12.  *  @license    http://www.pimcore.org/license     GPLv3 and PCL
  13.  */
  14. namespace Pimcore\Model\Asset\Image;
  15. use Pimcore\Event\AssetEvents;
  16. use Pimcore\Event\FrontendEvents;
  17. use Pimcore\Logger;
  18. use Pimcore\Model\Asset;
  19. use Pimcore\Model\Asset\Image;
  20. use Pimcore\Model\Asset\Thumbnail\ImageThumbnailTrait;
  21. use Pimcore\Model\Exception\NotFoundException;
  22. use Pimcore\Tool;
  23. use Symfony\Component\EventDispatcher\GenericEvent;
  24. final class Thumbnail
  25. {
  26.     use ImageThumbnailTrait;
  27.     /**
  28.      * @internal
  29.      *
  30.      * @var bool[]
  31.      */
  32.     protected static $hasListenersCache = [];
  33.     /**
  34.      * @param Image $asset
  35.      * @param string|array|Thumbnail\Config|null $config
  36.      * @param bool $deferred
  37.      */
  38.     public function __construct($asset$config null$deferred true)
  39.     {
  40.         $this->asset $asset;
  41.         $this->deferred $deferred;
  42.         $this->config $this->createConfig($config);
  43.     }
  44.     /**
  45.      * @param bool $deferredAllowed
  46.      * @param bool $cacheBuster
  47.      *
  48.      * @return string
  49.      */
  50.     public function getPath($deferredAllowed true$cacheBuster false)
  51.     {
  52.         $pathReference null;
  53.         if ($this->getConfig()) {
  54.             if ($this->useOriginalFile($this->asset->getFilename()) && $this->getConfig()->isSvgTargetFormatPossible()) {
  55.                 // we still generate the raster image, to get the final size of the thumbnail
  56.                 // we use getRealFullPath() here, to avoid double encoding (getFullPath() returns already encoded path)
  57.                 $pathReference = [
  58.                     'src' => $this->asset->getRealFullPath(),
  59.                     'type' => 'asset',
  60.                 ];
  61.             }
  62.         }
  63.         if (!$pathReference) {
  64.             $pathReference $this->getPathReference($deferredAllowed);
  65.         }
  66.         $path $this->convertToWebPath($pathReference);
  67.         if ($cacheBuster) {
  68.             $path $this->addCacheBuster($path, ['cacheBuster' => true], $this->getAsset());
  69.         }
  70.         if ($this->hasListeners(FrontendEvents::ASSET_IMAGE_THUMBNAIL)) {
  71.             $event = new GenericEvent($this, [
  72.                 'pathReference' => $pathReference,
  73.                 'frontendPath' => $path,
  74.             ]);
  75.             \Pimcore::getEventDispatcher()->dispatch($eventFrontendEvents::ASSET_IMAGE_THUMBNAIL);
  76.             $path $event->getArgument('frontendPath');
  77.         }
  78.         return $path;
  79.     }
  80.     /**
  81.      * @param string $eventName
  82.      *
  83.      * @return bool
  84.      */
  85.     protected function hasListeners(string $eventName): bool
  86.     {
  87.         if (!isset(self::$hasListenersCache[$eventName])) {
  88.             self::$hasListenersCache[$eventName] = \Pimcore::getEventDispatcher()->hasListeners($eventName);
  89.         }
  90.         return self::$hasListenersCache[$eventName];
  91.     }
  92.     /**
  93.      * @param string $filename
  94.      *
  95.      * @return bool
  96.      */
  97.     protected function useOriginalFile($filename)
  98.     {
  99.         if ($this->getConfig()) {
  100.             if (!$this->getConfig()->isRasterizeSVG() && preg_match("@\.svgz?$@"$filename)) {
  101.                 return true;
  102.             }
  103.         }
  104.         return false;
  105.     }
  106.     /**
  107.      * @internal
  108.      *
  109.      * @param bool $deferredAllowed
  110.      */
  111.     public function generate($deferredAllowed true)
  112.     {
  113.         $deferred false;
  114.         $generated false;
  115.         if ($this->asset && empty($this->pathReference)) {
  116.             // if no correct thumbnail config is given use the original image as thumbnail
  117.             if (!$this->config) {
  118.                 $this->pathReference = [
  119.                     'type' => 'asset',
  120.                     'src' => $this->asset->getRealFullPath(),
  121.                 ];
  122.             } else {
  123.                 try {
  124.                     $deferred $deferredAllowed && $this->deferred;
  125.                     $this->pathReference Thumbnail\Processor::process($this->asset$this->confignull$deferred$generated);
  126.                 } catch (\Exception $e) {
  127.                     Logger::error("Couldn't create thumbnail of image " $this->asset->getRealFullPath());
  128.                     Logger::error($e->getMessage());
  129.                 }
  130.             }
  131.         }
  132.         if (empty($this->pathReference)) {
  133.             $this->pathReference = [
  134.                 'type' => 'error',
  135.                 'src' => '/bundles/pimcoreadmin/img/filetype-not-supported.svg',
  136.             ];
  137.         }
  138.         if ($this->hasListeners(AssetEvents::IMAGE_THUMBNAIL)) {
  139.             $event = new GenericEvent($this, [
  140.                 'deferred' => $deferred,
  141.                 'generated' => $generated,
  142.             ]);
  143.             \Pimcore::getEventDispatcher()->dispatch($eventAssetEvents::IMAGE_THUMBNAIL);
  144.         }
  145.     }
  146.     /**
  147.      * @return string Public path to thumbnail image.
  148.      */
  149.     public function __toString()
  150.     {
  151.         return $this->getPath(true);
  152.     }
  153.     /**
  154.      * @param string $path
  155.      * @param array $options
  156.      * @param Asset $asset
  157.      *
  158.      * @return string
  159.      */
  160.     private function addCacheBuster(string $path, array $optionsAsset $asset): string
  161.     {
  162.         if (isset($options['cacheBuster']) && $options['cacheBuster']) {
  163.             if (!str_starts_with($path'http')) {
  164.                 $path '/cache-buster-' $asset->getVersionCount() . $path;
  165.             }
  166.         }
  167.         return $path;
  168.     }
  169.     private function getSourceTagHtml(Image\Thumbnail\Config $thumbConfigstring $mediaQueryImage $image, array $options): string
  170.     {
  171.         $sourceTagAttributes = [];
  172.         $sourceTagAttributes['srcset'] = $this->getSrcset($thumbConfig$image$options$mediaQuery);
  173.         $thumb $image->getThumbnail($thumbConfigtrue);
  174.         if ($mediaQuery) {
  175.             $sourceTagAttributes['media'] = $mediaQuery;
  176.             $thumb->reset();
  177.         }
  178.         if (isset($options['previewDataUri'])) {
  179.             $sourceTagAttributes['data-srcset'] = $sourceTagAttributes['srcset'];
  180.             unset($sourceTagAttributes['srcset']);
  181.         }
  182.         if (!isset($options['disableWidthHeightAttributes'])) {
  183.             if ($thumb->getWidth()) {
  184.                 $sourceTagAttributes['width'] = $thumb->getWidth();
  185.             }
  186.             if ($thumb->getHeight()) {
  187.                 $sourceTagAttributes['height'] = $thumb->getHeight();
  188.             }
  189.         }
  190.         $sourceTagAttributes['type'] = $thumb->getMimeType();
  191.         $sourceCallback $options['sourceCallback'] ?? null;
  192.         if ($sourceCallback) {
  193.             $sourceTagAttributes $sourceCallback($sourceTagAttributes);
  194.         }
  195.         return '<source ' array_to_html_attribute_string($sourceTagAttributes) . ' />';
  196.     }
  197.     /**
  198.      * Get generated HTML for displaying the thumbnail image in a HTML document.
  199.      *
  200.      * @param array $options Custom configuration
  201.      *
  202.      * @return string
  203.      */
  204.     public function getHtml($options = [])
  205.     {
  206.         /** @var Image $image */
  207.         $image $this->getAsset();
  208.         $thumbConfig $this->getConfig();
  209.         $pictureTagAttributes $options['pictureAttributes'] ?? []; // this is used for the html5 <picture> element
  210.         if ((isset($options['lowQualityPlaceholder']) && $options['lowQualityPlaceholder']) && !Tool::isFrontendRequestByAdmin()) {
  211.             $previewDataUri $image->getLowQualityPreviewDataUri();
  212.             if (!$previewDataUri) {
  213.                 // use a 1x1 transparent GIF as a fallback if no LQIP exists
  214.                 $previewDataUri '';
  215.             }
  216.             // this gets used in getImagTag() later
  217.             $options['previewDataUri'] = $previewDataUri;
  218.         }
  219.         $isAutoFormat $thumbConfig instanceof Image\Thumbnail\Config strtolower($thumbConfig->getFormat()) === 'source' false;
  220.         if ($isAutoFormat) {
  221.             // ensure the default image is not WebP
  222.             $this->pathReference = [];
  223.         }
  224.         $pictureCallback $options['pictureCallback'] ?? null;
  225.         if ($pictureCallback) {
  226.             $pictureTagAttributes $pictureCallback($pictureTagAttributes);
  227.         }
  228.         $html '<picture ' array_to_html_attribute_string($pictureTagAttributes) . '>' "\n";
  229.         if ($thumbConfig instanceof Image\Thumbnail\Config) {
  230.             $mediaConfigs $thumbConfig->getMedias();
  231.             // currently only max-width is supported, the key of the media is WIDTHw (eg. 400w) according to the srcset specification
  232.             ksort($mediaConfigsSORT_NUMERIC);
  233.             array_push($mediaConfigs$thumbConfig->getItems()); //add the default config at the end - picturePolyfill v4
  234.             foreach ($mediaConfigs as $mediaQuery => $config) {
  235.                 $sourceHtml $this->getSourceTagHtml($thumbConfig$mediaQuery$image$options);
  236.                 if (!empty($sourceHtml)) {
  237.                     if ($isAutoFormat) {
  238.                         foreach ($thumbConfig->getAutoFormatThumbnailConfigs() as $autoFormatConfig) {
  239.                             $autoFormatThumbnailHtml $this->getSourceTagHtml($autoFormatConfig$mediaQuery$image$options);
  240.                             if (!empty($autoFormatThumbnailHtml)) {
  241.                                 $html .= "\t" $autoFormatThumbnailHtml "\n";
  242.                             }
  243.                         }
  244.                     }
  245.                     $html .= "\t" $sourceHtml "\n";
  246.                 }
  247.             }
  248.         }
  249.         if (!($options['disableImgTag'] ?? null)) {
  250.             $html .= "\t" $this->getImageTag($options) . "\n";
  251.         }
  252.         $html .= '</picture>' "\n";
  253.         if (isset($options['useDataSrc']) && $options['useDataSrc']) {
  254.             $html preg_replace('/ src(set)?=/i'' data-src$1='$html);
  255.         }
  256.         return $html;
  257.     }
  258.     /**
  259.      * @param array $options
  260.      * @param array $removeAttributes
  261.      *
  262.      * @return string
  263.      */
  264.     public function getImageTag(array $options = [], array $removeAttributes = []): string
  265.     {
  266.         /** @var Image $image */
  267.         $image $this->getAsset();
  268.         $attributes $options['imgAttributes'] ?? [];
  269.         $callback $options['imgCallback'] ?? null;
  270.         if (isset($options['previewDataUri'])) {
  271.             $attributes['src'] = $options['previewDataUri'];
  272.         } else {
  273.             $path $this->getPath(true);
  274.             $attributes['src'] = $this->addCacheBuster($path$options$image);
  275.         }
  276.         if (!isset($options['disableWidthHeightAttributes'])) {
  277.             if ($this->getWidth()) {
  278.                 $attributes['width'] = $this->getWidth();
  279.             }
  280.             if ($this->getHeight()) {
  281.                 $attributes['height'] = $this->getHeight();
  282.             }
  283.         }
  284.         $altText = !empty($options['alt']) ? $options['alt'] : (!empty($attributes['alt']) ? $attributes['alt'] : '');
  285.         $titleText = !empty($options['title']) ? $options['title'] : (!empty($attributes['title']) ? $attributes['title'] : '');
  286.         if (empty($titleText) && (!isset($options['disableAutoTitle']) || !$options['disableAutoTitle'])) {
  287.             if ($image->getMetadata('title')) {
  288.                 $titleText $image->getMetadata('title');
  289.             }
  290.         }
  291.         if (empty($altText) && (!isset($options['disableAutoAlt']) || !$options['disableAutoAlt'])) {
  292.             if ($image->getMetadata('alt')) {
  293.                 $altText $image->getMetadata('alt');
  294.             } elseif (isset($options['defaultalt'])) {
  295.                 $altText $options['defaultalt'];
  296.             } else {
  297.                 $altText $titleText;
  298.             }
  299.         }
  300.         // get copyright from asset
  301.         if ($image->getMetadata('copyright') && (!isset($options['disableAutoCopyright']) || !$options['disableAutoCopyright'])) {
  302.             if (!empty($altText)) {
  303.                 $altText .= ' | ';
  304.             }
  305.             if (!empty($titleText)) {
  306.                 $titleText .= ' | ';
  307.             }
  308.             $altText .= ('© ' $image->getMetadata('copyright'));
  309.             $titleText .= ('© ' $image->getMetadata('copyright'));
  310.         }
  311.         $attributes['alt'] = $altText;
  312.         if (!empty($titleText)) {
  313.             $attributes['title'] = $titleText;
  314.         }
  315.         if (!isset($attributes['loading'])) {
  316.             $attributes['loading'] = 'lazy';
  317.         }
  318.         foreach ($removeAttributes as $attribute) {
  319.             unset($attributes[$attribute]);
  320.         }
  321.         if ($callback) {
  322.             $attributes $callback($attributes);
  323.         }
  324.         $thumbConfig $this->getConfig();
  325.         if ($thumbConfig) {
  326.             $srcsetAttribute = isset($options['previewDataUri']) ? 'data-srcset' 'srcset';
  327.             $attributes[$srcsetAttribute] = $this->getSrcset($thumbConfig$image$options);
  328.         }
  329.         $htmlImgTag '';
  330.         if (!empty($attributes)) {
  331.             $htmlImgTag '<img ' array_to_html_attribute_string($attributes) . ' />';
  332.         }
  333.         return $htmlImgTag;
  334.     }
  335.     /**
  336.      * @param string $name
  337.      * @param int $highRes
  338.      *
  339.      * @return Thumbnail
  340.      *
  341.      * @throws \Exception
  342.      */
  343.     public function getMedia($name$highRes 1)
  344.     {
  345.         $thumbConfig $this->getConfig();
  346.         $mediaConfigs $thumbConfig->getMedias();
  347.         if (isset($mediaConfigs[$name])) {
  348.             $thumbConfigRes = clone $thumbConfig;
  349.             $thumbConfigRes->selectMedia($name);
  350.             $thumbConfigRes->setHighResolution($highRes);
  351.             $thumbConfigRes->setMedias([]);
  352.             /** @var Image $asset */
  353.             $asset $this->getAsset();
  354.             $thumb $asset->getThumbnail($thumbConfigRes);
  355.             return $thumb;
  356.         } else {
  357.             throw new \Exception("Media query '" $name "' doesn't exist in thumbnail configuration: " $thumbConfig->getName());
  358.         }
  359.     }
  360.     /**
  361.      * Get a thumbnail image configuration.
  362.      *
  363.      * @param string|array|Thumbnail\Config $selector Name, array or object describing a thumbnail configuration.
  364.      *
  365.      * @return Thumbnail\Config
  366.      *
  367.      * @throws NotFoundException
  368.      */
  369.     private function createConfig($selector)
  370.     {
  371.         $thumbnailConfig Thumbnail\Config::getByAutoDetect($selector);
  372.         if (!empty($selector) && $thumbnailConfig === null) {
  373.             throw new NotFoundException('Thumbnail definition "' . (is_string($selector) ? $selector '') . '" does not exist');
  374.         }
  375.         return $thumbnailConfig;
  376.     }
  377.     /**
  378.      * Get value that can be directly used ina srcset HTML attribute for images.
  379.      *
  380.      * @param Image\Thumbnail\Config $thumbConfig
  381.      * @param Image $image
  382.      * @param array $options
  383.      * @param string|null $mediaQuery Can be empty string if no media queries are defined.
  384.      *
  385.      * @return string Relative paths to different thunbnail images with 1x and 2x resolution
  386.      */
  387.     private function getSrcset(Image\Thumbnail\Config $thumbConfigImage $image, array $options, ?string $mediaQuery null): string
  388.     {
  389.         $srcSetValues = [];
  390.         foreach ([12] as $highRes) {
  391.             $thumbConfigRes = clone $thumbConfig;
  392.             if ($mediaQuery) {
  393.                 $thumbConfigRes->selectMedia($mediaQuery);
  394.             }
  395.             $thumbConfigRes->setHighResolution($highRes);
  396.             $thumb $image->getThumbnail($thumbConfigRestrue);
  397.             $descriptor $highRes 'x';
  398.             // encode comma in thumbnail path as srcset is a comma separated list
  399.             $srcSetValues[] = str_replace(',''%2C'$this->addCacheBuster($thumb ' ' $descriptor$options$image));
  400.             if ($this->useOriginalFile($this->asset->getFilename()) && $this->getConfig()->isSvgTargetFormatPossible()) {
  401.                 break;
  402.             }
  403.         }
  404.         return implode(', '$srcSetValues);
  405.     }
  406. }