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

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.     public function getFileSize(): ?int
  81.     {
  82.         $pathReference $this->getPathReference(false);
  83.         if ($pathReference['type'] === 'asset') {
  84.             return $this->asset->getFileSize();
  85.         } elseif (isset($pathReference['storagePath'])) {
  86.             return Tool\Storage::get('thumbnail')->fileSize($pathReference['storagePath']);
  87.         }
  88.         return null;
  89.     }
  90.     /**
  91.      * @return null|resource
  92.      */
  93.     public function getStream()
  94.     {
  95.         $pathReference $this->getPathReference(false);
  96.         if ($pathReference['type'] === 'asset') {
  97.             return $this->asset->getStream();
  98.         } elseif (isset($pathReference['storagePath'])) {
  99.             return Tool\Storage::get('thumbnail')->readStream($pathReference['storagePath']);
  100.         }
  101.         return null;
  102.     }
  103.     /**
  104.      * @param string $eventName
  105.      *
  106.      * @return bool
  107.      */
  108.     protected function hasListeners(string $eventName): bool
  109.     {
  110.         if (!isset(self::$hasListenersCache[$eventName])) {
  111.             self::$hasListenersCache[$eventName] = \Pimcore::getEventDispatcher()->hasListeners($eventName);
  112.         }
  113.         return self::$hasListenersCache[$eventName];
  114.     }
  115.     /**
  116.      * @param string $filename
  117.      *
  118.      * @return bool
  119.      */
  120.     protected function useOriginalFile($filename)
  121.     {
  122.         if ($this->getConfig()) {
  123.             if (!$this->getConfig()->isRasterizeSVG() && preg_match("@\.svgz?$@"$filename)) {
  124.                 return true;
  125.             }
  126.         }
  127.         return false;
  128.     }
  129.     /**
  130.      * @internal
  131.      *
  132.      * @param bool $deferredAllowed
  133.      */
  134.     public function generate($deferredAllowed true)
  135.     {
  136.         $deferred false;
  137.         $generated false;
  138.         if ($this->asset && empty($this->pathReference)) {
  139.             // if no correct thumbnail config is given use the original image as thumbnail
  140.             if (!$this->config) {
  141.                 $this->pathReference = [
  142.                     'type' => 'asset',
  143.                     'src' => $this->asset->getRealFullPath(),
  144.                 ];
  145.             } else {
  146.                 try {
  147.                     $deferred $deferredAllowed && $this->deferred;
  148.                     $this->pathReference Thumbnail\Processor::process($this->asset$this->confignull$deferred$generated);
  149.                 } catch (\Exception $e) {
  150.                     Logger::error("Couldn't create thumbnail of image " $this->asset->getRealFullPath());
  151.                     Logger::error($e->getMessage());
  152.                 }
  153.             }
  154.         }
  155.         if (empty($this->pathReference)) {
  156.             $this->pathReference = [
  157.                 'type' => 'error',
  158.                 'src' => '/bundles/pimcoreadmin/img/filetype-not-supported.svg',
  159.             ];
  160.         }
  161.         if ($this->hasListeners(AssetEvents::IMAGE_THUMBNAIL)) {
  162.             $event = new GenericEvent($this, [
  163.                 'deferred' => $deferred,
  164.                 'generated' => $generated,
  165.             ]);
  166.             \Pimcore::getEventDispatcher()->dispatch($eventAssetEvents::IMAGE_THUMBNAIL);
  167.         }
  168.     }
  169.     /**
  170.      * @return string Public path to thumbnail image.
  171.      */
  172.     public function __toString()
  173.     {
  174.         return $this->getPath(true);
  175.     }
  176.     /**
  177.      * @param string $path
  178.      * @param array $options
  179.      * @param Asset $asset
  180.      *
  181.      * @return string
  182.      */
  183.     private function addCacheBuster(string $path, array $optionsAsset $asset): string
  184.     {
  185.         if (isset($options['cacheBuster']) && $options['cacheBuster']) {
  186.             if (!str_starts_with($path'http')) {
  187.                 $path '/cache-buster-' $asset->getVersionCount() . $path;
  188.             }
  189.         }
  190.         return $path;
  191.     }
  192.     private function getSourceTagHtml(Image\Thumbnail\Config $thumbConfigstring $mediaQueryImage $image, array $options): string
  193.     {
  194.         $srcSetValues = [];
  195.         $sourceTagAttributes = [];
  196.         foreach ([12] as $highRes) {
  197.             $thumbConfigRes = clone $thumbConfig;
  198.             $thumbConfigRes->selectMedia($mediaQuery);
  199.             $thumbConfigRes->setHighResolution($highRes);
  200.             $thumb $image->getThumbnail($thumbConfigRestrue);
  201.             $descriptor $highRes 'x';
  202.             $srcSetValues[] = $this->addCacheBuster($thumb ' ' $descriptor$options$image);
  203.             if ($this->useOriginalFile($this->asset->getFilename()) && $this->getConfig()->isSvgTargetFormatPossible()) {
  204.                 break;
  205.             }
  206.         }
  207.         $sourceTagAttributes['srcset'] = implode(', '$srcSetValues);
  208.         if ($mediaQuery) {
  209.             $sourceTagAttributes['media'] = $mediaQuery;
  210.             $thumb->reset();
  211.         }
  212.         if (isset($options['previewDataUri'])) {
  213.             $sourceTagAttributes['data-srcset'] = $sourceTagAttributes['srcset'];
  214.             unset($sourceTagAttributes['srcset']);
  215.         }
  216.         $sourceTagAttributes['type'] = $thumb->getMimeType();
  217.         $sourceCallback $options['sourceCallback'] ?? null;
  218.         if ($sourceCallback) {
  219.             $sourceTagAttributes $sourceCallback($sourceTagAttributes);
  220.         }
  221.         return '<source ' array_to_html_attribute_string($sourceTagAttributes) . ' />';
  222.     }
  223.     /**
  224.      * Get generated HTML for displaying the thumbnail image in a HTML document.
  225.      *
  226.      * @param array $options Custom configuration
  227.      *
  228.      * @return string
  229.      */
  230.     public function getHtml($options = [])
  231.     {
  232.         /** @var Image $image */
  233.         $image $this->getAsset();
  234.         $thumbConfig $this->getConfig();
  235.         $pictureTagAttributes $options['pictureAttributes'] ?? []; // this is used for the html5 <picture> element
  236.         if ((isset($options['lowQualityPlaceholder']) && $options['lowQualityPlaceholder']) && !Tool::isFrontendRequestByAdmin()) {
  237.             $previewDataUri $image->getLowQualityPreviewDataUri();
  238.             if (!$previewDataUri) {
  239.                 // use a 1x1 transparent GIF as a fallback if no LQIP exists
  240.                 $previewDataUri '';
  241.             }
  242.             // this gets used in getImagTag() later
  243.             $options['previewDataUri'] = $previewDataUri;
  244.         }
  245.         $isAutoFormat $thumbConfig instanceof Image\Thumbnail\Config strtolower($thumbConfig->getFormat()) === 'source' false;
  246.         if ($isAutoFormat) {
  247.             // ensure the default image is not WebP
  248.             $this->pathReference = [];
  249.         }
  250.         $pictureCallback $options['pictureCallback'] ?? null;
  251.         if ($pictureCallback) {
  252.             $pictureTagAttributes $pictureCallback($pictureTagAttributes);
  253.         }
  254.         $html '<picture ' array_to_html_attribute_string($pictureTagAttributes) . '>' "\n";
  255.         if ($thumbConfig instanceof Image\Thumbnail\Config) {
  256.             $mediaConfigs $thumbConfig->getMedias();
  257.             // currently only max-width is supported, the key of the media is WIDTHw (eg. 400w) according to the srcset specification
  258.             ksort($mediaConfigsSORT_NUMERIC);
  259.             array_push($mediaConfigs$thumbConfig->getItems()); //add the default config at the end - picturePolyfill v4
  260.             foreach ($mediaConfigs as $mediaQuery => $config) {
  261.                 $sourceHtml $this->getSourceTagHtml($thumbConfig$mediaQuery$image$options);
  262.                 if (!empty($sourceHtml)) {
  263.                     if ($isAutoFormat) {
  264.                         foreach ($thumbConfig->getAutoFormatThumbnailConfigs() as $autoFormatConfig) {
  265.                             $autoFormatThumbnailHtml $this->getSourceTagHtml($autoFormatConfig$mediaQuery$image$options);
  266.                             if (!empty($autoFormatThumbnailHtml)) {
  267.                                 $html .= "\t" $autoFormatThumbnailHtml "\n";
  268.                             }
  269.                         }
  270.                     }
  271.                     $html .= "\t" $sourceHtml "\n";
  272.                 }
  273.             }
  274.         }
  275.         if (!($options['disableImgTag'] ?? null)) {
  276.             $html .= "\t" $this->getImageTag($options) . "\n";
  277.         }
  278.         $html .= '</picture>' "\n";
  279.         if (isset($options['useDataSrc']) && $options['useDataSrc']) {
  280.             $html preg_replace('/ src(set)?=/i'' data-src$1='$html);
  281.         }
  282.         return $html;
  283.     }
  284.     /**
  285.      * @param array $options
  286.      * @param array $removeAttributes
  287.      *
  288.      * @return string
  289.      */
  290.     public function getImageTag(array $options = [], array $removeAttributes = []): string
  291.     {
  292.         /** @var Image $image */
  293.         $image $this->getAsset();
  294.         $attributes $options['imgAttributes'] ?? [];
  295.         $callback $options['imgCallback'] ?? null;
  296.         if (isset($options['previewDataUri'])) {
  297.             $attributes['src'] = $options['previewDataUri'];
  298.         } else {
  299.             $path $this->getPath(true);
  300.             $attributes['src'] = $this->addCacheBuster($path$options$image);
  301.         }
  302.         if (!isset($options['disableWidthHeightAttributes'])) {
  303.             if ($this->getWidth()) {
  304.                 $attributes['width'] = $this->getWidth();
  305.             }
  306.             if ($this->getHeight()) {
  307.                 $attributes['height'] = $this->getHeight();
  308.             }
  309.         }
  310.         $altText = !empty($options['alt']) ? $options['alt'] : (!empty($attributes['alt']) ? $attributes['alt'] : '');
  311.         $titleText = !empty($options['title']) ? $options['title'] : (!empty($attributes['title']) ? $attributes['title'] : '');
  312.         if (empty($titleText) && (!isset($options['disableAutoTitle']) || !$options['disableAutoTitle'])) {
  313.             if ($image->getMetadata('title')) {
  314.                 $titleText $image->getMetadata('title');
  315.             }
  316.         }
  317.         if (empty($altText) && (!isset($options['disableAutoAlt']) || !$options['disableAutoAlt'])) {
  318.             if ($image->getMetadata('alt')) {
  319.                 $altText $image->getMetadata('alt');
  320.             } elseif (isset($options['defaultalt'])) {
  321.                 $altText $options['defaultalt'];
  322.             } else {
  323.                 $altText $titleText;
  324.             }
  325.         }
  326.         // get copyright from asset
  327.         if ($image->getMetadata('copyright') && (!isset($options['disableAutoCopyright']) || !$options['disableAutoCopyright'])) {
  328.             if (!empty($altText)) {
  329.                 $altText .= ' | ';
  330.             }
  331.             if (!empty($titleText)) {
  332.                 $titleText .= ' | ';
  333.             }
  334.             $altText .= ('© ' $image->getMetadata('copyright'));
  335.             $titleText .= ('© ' $image->getMetadata('copyright'));
  336.         }
  337.         $attributes['alt'] = $altText;
  338.         if (!empty($titleText)) {
  339.             $attributes['title'] = $titleText;
  340.         }
  341.         if (!isset($attributes['loading'])) {
  342.             $attributes['loading'] = 'lazy';
  343.         }
  344.         foreach ($removeAttributes as $attribute) {
  345.             unset($attributes[$attribute]);
  346.         }
  347.         if ($callback) {
  348.             $attributes $callback($attributes);
  349.         }
  350.         $htmlImgTag '';
  351.         if (!empty($attributes)) {
  352.             $htmlImgTag '<img ' array_to_html_attribute_string($attributes) . ' />';
  353.         }
  354.         return $htmlImgTag;
  355.     }
  356.     /**
  357.      * @param string $name
  358.      * @param int $highRes
  359.      *
  360.      * @return Thumbnail
  361.      *
  362.      * @throws \Exception
  363.      */
  364.     public function getMedia($name$highRes 1)
  365.     {
  366.         $thumbConfig $this->getConfig();
  367.         $mediaConfigs $thumbConfig->getMedias();
  368.         if (isset($mediaConfigs[$name])) {
  369.             $thumbConfigRes = clone $thumbConfig;
  370.             $thumbConfigRes->selectMedia($name);
  371.             $thumbConfigRes->setHighResolution($highRes);
  372.             $thumbConfigRes->setMedias([]);
  373.             /** @var Image $asset */
  374.             $asset $this->getAsset();
  375.             $thumb $asset->getThumbnail($thumbConfigRes);
  376.             return $thumb;
  377.         } else {
  378.             throw new \Exception("Media query '" $name "' doesn't exist in thumbnail configuration: " $thumbConfig->getName());
  379.         }
  380.     }
  381.     /**
  382.      * Get a thumbnail image configuration.
  383.      *
  384.      * @param string|array|Thumbnail\Config $selector Name, array or object describing a thumbnail configuration.
  385.      *
  386.      * @return Thumbnail\Config
  387.      *
  388.      * @throws NotFoundException
  389.      */
  390.     private function createConfig($selector)
  391.     {
  392.         $thumbnailConfig Thumbnail\Config::getByAutoDetect($selector);
  393.         if (!empty($selector) && $thumbnailConfig === null) {
  394.             throw new NotFoundException('Thumbnail definition "' . (is_string($selector) ? $selector '') . '" does not exist');
  395.         }
  396.         return $thumbnailConfig;
  397.     }
  398. }