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

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.         $sourceTagAttributes['type'] = $thumb->getMimeType();
  183.         $sourceCallback $options['sourceCallback'] ?? null;
  184.         if ($sourceCallback) {
  185.             $sourceTagAttributes $sourceCallback($sourceTagAttributes);
  186.         }
  187.         return '<source ' array_to_html_attribute_string($sourceTagAttributes) . ' />';
  188.     }
  189.     /**
  190.      * Get generated HTML for displaying the thumbnail image in a HTML document.
  191.      *
  192.      * @param array $options Custom configuration
  193.      *
  194.      * @return string
  195.      */
  196.     public function getHtml($options = [])
  197.     {
  198.         /** @var Image $image */
  199.         $image $this->getAsset();
  200.         $thumbConfig $this->getConfig();
  201.         $pictureTagAttributes $options['pictureAttributes'] ?? []; // this is used for the html5 <picture> element
  202.         if ((isset($options['lowQualityPlaceholder']) && $options['lowQualityPlaceholder']) && !Tool::isFrontendRequestByAdmin()) {
  203.             $previewDataUri $image->getLowQualityPreviewDataUri();
  204.             if (!$previewDataUri) {
  205.                 // use a 1x1 transparent GIF as a fallback if no LQIP exists
  206.                 $previewDataUri '';
  207.             }
  208.             // this gets used in getImagTag() later
  209.             $options['previewDataUri'] = $previewDataUri;
  210.         }
  211.         $isAutoFormat $thumbConfig instanceof Image\Thumbnail\Config strtolower($thumbConfig->getFormat()) === 'source' false;
  212.         if ($isAutoFormat) {
  213.             // ensure the default image is not WebP
  214.             $this->pathReference = [];
  215.         }
  216.         $pictureCallback $options['pictureCallback'] ?? null;
  217.         if ($pictureCallback) {
  218.             $pictureTagAttributes $pictureCallback($pictureTagAttributes);
  219.         }
  220.         $html '<picture ' array_to_html_attribute_string($pictureTagAttributes) . '>' "\n";
  221.         if ($thumbConfig instanceof Image\Thumbnail\Config) {
  222.             $mediaConfigs $thumbConfig->getMedias();
  223.             // currently only max-width is supported, the key of the media is WIDTHw (eg. 400w) according to the srcset specification
  224.             ksort($mediaConfigsSORT_NUMERIC);
  225.             array_push($mediaConfigs$thumbConfig->getItems()); //add the default config at the end - picturePolyfill v4
  226.             foreach ($mediaConfigs as $mediaQuery => $config) {
  227.                 $sourceHtml $this->getSourceTagHtml($thumbConfig$mediaQuery$image$options);
  228.                 if (!empty($sourceHtml)) {
  229.                     if ($isAutoFormat) {
  230.                         foreach ($thumbConfig->getAutoFormatThumbnailConfigs() as $autoFormatConfig) {
  231.                             $autoFormatThumbnailHtml $this->getSourceTagHtml($autoFormatConfig$mediaQuery$image$options);
  232.                             if (!empty($autoFormatThumbnailHtml)) {
  233.                                 $html .= "\t" $autoFormatThumbnailHtml "\n";
  234.                             }
  235.                         }
  236.                     }
  237.                     $html .= "\t" $sourceHtml "\n";
  238.                 }
  239.             }
  240.         }
  241.         if (!($options['disableImgTag'] ?? null)) {
  242.             $html .= "\t" $this->getImageTag($options) . "\n";
  243.         }
  244.         $html .= '</picture>' "\n";
  245.         if (isset($options['useDataSrc']) && $options['useDataSrc']) {
  246.             $html preg_replace('/ src(set)?=/i'' data-src$1='$html);
  247.         }
  248.         return $html;
  249.     }
  250.     /**
  251.      * @param array $options
  252.      * @param array $removeAttributes
  253.      *
  254.      * @return string
  255.      */
  256.     public function getImageTag(array $options = [], array $removeAttributes = []): string
  257.     {
  258.         /** @var Image $image */
  259.         $image $this->getAsset();
  260.         $attributes $options['imgAttributes'] ?? [];
  261.         $callback $options['imgCallback'] ?? null;
  262.         if (isset($options['previewDataUri'])) {
  263.             $attributes['src'] = $options['previewDataUri'];
  264.         } else {
  265.             $path $this->getPath(true);
  266.             $attributes['src'] = $this->addCacheBuster($path$options$image);
  267.         }
  268.         if (!isset($options['disableWidthHeightAttributes'])) {
  269.             if ($this->getWidth()) {
  270.                 $attributes['width'] = $this->getWidth();
  271.             }
  272.             if ($this->getHeight()) {
  273.                 $attributes['height'] = $this->getHeight();
  274.             }
  275.         }
  276.         $altText = !empty($options['alt']) ? $options['alt'] : (!empty($attributes['alt']) ? $attributes['alt'] : '');
  277.         $titleText = !empty($options['title']) ? $options['title'] : (!empty($attributes['title']) ? $attributes['title'] : '');
  278.         if (empty($titleText) && (!isset($options['disableAutoTitle']) || !$options['disableAutoTitle'])) {
  279.             if ($image->getMetadata('title')) {
  280.                 $titleText $image->getMetadata('title');
  281.             }
  282.         }
  283.         if (empty($altText) && (!isset($options['disableAutoAlt']) || !$options['disableAutoAlt'])) {
  284.             if ($image->getMetadata('alt')) {
  285.                 $altText $image->getMetadata('alt');
  286.             } elseif (isset($options['defaultalt'])) {
  287.                 $altText $options['defaultalt'];
  288.             } else {
  289.                 $altText $titleText;
  290.             }
  291.         }
  292.         // get copyright from asset
  293.         if ($image->getMetadata('copyright') && (!isset($options['disableAutoCopyright']) || !$options['disableAutoCopyright'])) {
  294.             if (!empty($altText)) {
  295.                 $altText .= ' | ';
  296.             }
  297.             if (!empty($titleText)) {
  298.                 $titleText .= ' | ';
  299.             }
  300.             $altText .= ('© ' $image->getMetadata('copyright'));
  301.             $titleText .= ('© ' $image->getMetadata('copyright'));
  302.         }
  303.         $attributes['alt'] = $altText;
  304.         if (!empty($titleText)) {
  305.             $attributes['title'] = $titleText;
  306.         }
  307.         if (!isset($attributes['loading'])) {
  308.             $attributes['loading'] = 'lazy';
  309.         }
  310.         foreach ($removeAttributes as $attribute) {
  311.             unset($attributes[$attribute]);
  312.         }
  313.         if ($callback) {
  314.             $attributes $callback($attributes);
  315.         }
  316.         $thumbConfig $this->getConfig();
  317.         if ($thumbConfig) {
  318.             $srcsetAttribute = isset($options['previewDataUri']) ? 'data-srcset' 'srcset';
  319.             $attributes[$srcsetAttribute] = $this->getSrcset($thumbConfig$image$options);
  320.         }
  321.         $htmlImgTag '';
  322.         if (!empty($attributes)) {
  323.             $htmlImgTag '<img ' array_to_html_attribute_string($attributes) . ' />';
  324.         }
  325.         return $htmlImgTag;
  326.     }
  327.     /**
  328.      * @param string $name
  329.      * @param int $highRes
  330.      *
  331.      * @return Thumbnail
  332.      *
  333.      * @throws \Exception
  334.      */
  335.     public function getMedia($name$highRes 1)
  336.     {
  337.         $thumbConfig $this->getConfig();
  338.         $mediaConfigs $thumbConfig->getMedias();
  339.         if (isset($mediaConfigs[$name])) {
  340.             $thumbConfigRes = clone $thumbConfig;
  341.             $thumbConfigRes->selectMedia($name);
  342.             $thumbConfigRes->setHighResolution($highRes);
  343.             $thumbConfigRes->setMedias([]);
  344.             /** @var Image $asset */
  345.             $asset $this->getAsset();
  346.             $thumb $asset->getThumbnail($thumbConfigRes);
  347.             return $thumb;
  348.         } else {
  349.             throw new \Exception("Media query '" $name "' doesn't exist in thumbnail configuration: " $thumbConfig->getName());
  350.         }
  351.     }
  352.     /**
  353.      * Get a thumbnail image configuration.
  354.      *
  355.      * @param string|array|Thumbnail\Config $selector Name, array or object describing a thumbnail configuration.
  356.      *
  357.      * @return Thumbnail\Config
  358.      *
  359.      * @throws NotFoundException
  360.      */
  361.     private function createConfig($selector)
  362.     {
  363.         $thumbnailConfig Thumbnail\Config::getByAutoDetect($selector);
  364.         if (!empty($selector) && $thumbnailConfig === null) {
  365.             throw new NotFoundException('Thumbnail definition "' . (is_string($selector) ? $selector '') . '" does not exist');
  366.         }
  367.         return $thumbnailConfig;
  368.     }
  369.     /**
  370.      * Get value that can be directly used ina srcset HTML attribute for images.
  371.      *
  372.      * @param Image\Thumbnail\Config $thumbConfig
  373.      * @param Image $image
  374.      * @param array $options
  375.      * @param string|null $mediaQuery Can be empty string if no media queries are defined.
  376.      *
  377.      * @return string Relative paths to different thunbnail images with 1x and 2x resolution
  378.      */
  379.     private function getSrcset(Image\Thumbnail\Config $thumbConfigImage $image, array $options, ?string $mediaQuery null): string
  380.     {
  381.         $srcSetValues = [];
  382.         foreach ([12] as $highRes) {
  383.             $thumbConfigRes = clone $thumbConfig;
  384.             if ($mediaQuery) {
  385.                 $thumbConfigRes->selectMedia($mediaQuery);
  386.             }
  387.             $thumbConfigRes->setHighResolution($highRes);
  388.             $thumb $image->getThumbnail($thumbConfigRestrue);
  389.             $descriptor $highRes 'x';
  390.             $srcSetValues[] = $this->addCacheBuster($thumb ' ' $descriptor$options$image);
  391.             if ($this->useOriginalFile($this->asset->getFilename()) && $this->getConfig()->isSvgTargetFormatPossible()) {
  392.                 break;
  393.             }
  394.         }
  395.         return implode(', '$srcSetValues);
  396.     }
  397. }