vendor/contao/core-bundle/src/Image/Studio/Figure.php line 299

Open in your IDE?
  1. <?php
  2. declare(strict_types=1);
  3. /*
  4.  * This file is part of Contao.
  5.  *
  6.  * (c) Leo Feyer
  7.  *
  8.  * @license LGPL-3.0-or-later
  9.  */
  10. namespace Contao\CoreBundle\Image\Studio;
  11. use Contao\Controller;
  12. use Contao\CoreBundle\File\Metadata;
  13. use Contao\File;
  14. use Contao\StringUtil;
  15. use Contao\Template;
  16. /**
  17.  * A Figure object holds image and metadata ready to be applied to a
  18.  * template's context. If you are using the legacy PHP templates, you can still
  19.  * use the provided legacy helper methods to manually apply the data to them.
  20.  *
  21.  * Wherever possible, the actual data is only requested/built on demand.
  22.  *
  23.  * @final This class will be made final in Contao 5.
  24.  */
  25. class Figure
  26. {
  27.     private ImageResult $image;
  28.     /**
  29.      * @var Metadata|(\Closure(self):Metadata|null)|null
  30.      */
  31.     private $metadata;
  32.     /**
  33.      * @var array<string, string|null>|(\Closure(self):array<string, string|null>)|null
  34.      */
  35.     private $linkAttributes;
  36.     /**
  37.      * @var LightboxResult|(\Closure(self):LightboxResult|null)|null
  38.      */
  39.     private $lightbox;
  40.     /**
  41.      * @var array<string, mixed>|(\Closure(self):array<string, mixed>)|null
  42.      */
  43.     private $options;
  44.     /**
  45.      * Creates a figure container.
  46.      *
  47.      * All arguments but the main image result can also be set via a Closure
  48.      * that only returns the value on demand.
  49.      *
  50.      * @param Metadata|(\Closure(self):Metadata|null)|null                                $metadata       Metadata container
  51.      * @param array<string, string|null>|(\Closure(self):array<string, string|null>)|null $linkAttributes Link attributes
  52.      * @param LightboxResult|(\Closure(self):LightboxResult|null)|null                    $lightbox       Lightbox
  53.      * @param array<string, mixed>|(\Closure(self):array<string, mixed>)|null             $options        Template options
  54.      */
  55.     public function __construct(ImageResult $image$metadata null$linkAttributes null$lightbox null$options null)
  56.     {
  57.         $this->image $image;
  58.         $this->metadata $metadata;
  59.         $this->linkAttributes $linkAttributes;
  60.         $this->lightbox $lightbox;
  61.         $this->options $options;
  62.     }
  63.     /**
  64.      * Returns the image result of the main resource.
  65.      */
  66.     public function getImage(): ImageResult
  67.     {
  68.         return $this->image;
  69.     }
  70.     /**
  71.      * Returns true if a lightbox result can be obtained.
  72.      */
  73.     public function hasLightbox(): bool
  74.     {
  75.         $this->resolveIfClosure($this->lightbox);
  76.         return $this->lightbox instanceof LightboxResult;
  77.     }
  78.     /**
  79.      * Returns the lightbox result (if available).
  80.      */
  81.     public function getLightbox(): LightboxResult
  82.     {
  83.         if (!$this->hasLightbox()) {
  84.             throw new \LogicException('This result container does not include a lightbox.');
  85.         }
  86.         /** @var LightboxResult */
  87.         return $this->lightbox;
  88.     }
  89.     public function hasMetadata(): bool
  90.     {
  91.         $this->resolveIfClosure($this->metadata);
  92.         return $this->metadata instanceof Metadata;
  93.     }
  94.     /**
  95.      * Returns the main resource's metadata.
  96.      */
  97.     public function getMetadata(): Metadata
  98.     {
  99.         if (!$this->hasMetadata()) {
  100.             throw new \LogicException('This result container does not include metadata.');
  101.         }
  102.         /** @var Metadata */
  103.         return $this->metadata;
  104.     }
  105.     public function getSchemaOrgData(): array
  106.     {
  107.         $imageIdentifier $this->getImage()->getImageSrc();
  108.         if ($this->hasMetadata() && $this->getMetadata()->has(Metadata::VALUE_UUID)) {
  109.             $imageIdentifier '#/schema/image/'.$this->getMetadata()->getUuid();
  110.         }
  111.         $jsonLd = [
  112.             '@type' => 'ImageObject',
  113.             'identifier' => $imageIdentifier,
  114.             'contentUrl' => $this->getImage()->getImageSrc(),
  115.         ];
  116.         if (!$this->hasMetadata()) {
  117.             ksort($jsonLd);
  118.             return $jsonLd;
  119.         }
  120.         $jsonLd array_merge($this->getMetadata()->getSchemaOrgData('ImageObject'), $jsonLd);
  121.         ksort($jsonLd);
  122.         return $jsonLd;
  123.     }
  124.     /**
  125.      * Returns a key-value list of all link attributes. This excludes "href" by
  126.      * default.
  127.      */
  128.     public function getLinkAttributes(bool $includeHref false): array
  129.     {
  130.         $this->resolveIfClosure($this->linkAttributes);
  131.         if (null === $this->linkAttributes) {
  132.             $this->linkAttributes = [];
  133.         }
  134.         // Generate the href attribute
  135.         if (!\array_key_exists('href'$this->linkAttributes)) {
  136.             $this->linkAttributes['href'] = (
  137.                 function () {
  138.                     if ($this->hasLightbox()) {
  139.                         return $this->getLightbox()->getLinkHref();
  140.                     }
  141.                     if ($this->hasMetadata()) {
  142.                         return $this->getMetadata()->getUrl();
  143.                     }
  144.                     return '';
  145.                 }
  146.             )();
  147.         }
  148.         // Add rel attribute "noreferrer noopener" to external links
  149.         if (
  150.             !empty($this->linkAttributes['href'])
  151.             && !\array_key_exists('rel'$this->linkAttributes)
  152.             && preg_match('#^https?://#'$this->linkAttributes['href'])
  153.         ) {
  154.             $this->linkAttributes['rel'] = 'noreferrer noopener';
  155.         }
  156.         // Add lightbox attributes
  157.         if (!\array_key_exists('data-lightbox'$this->linkAttributes) && $this->hasLightbox()) {
  158.             $lightbox $this->getLightbox();
  159.             $this->linkAttributes['data-lightbox'] = $lightbox->getGroupIdentifier();
  160.         }
  161.         // Allow removing attributes by setting them to null
  162.         $linkAttributes array_filter($this->linkAttributes, static fn ($attribute): bool => null !== $attribute);
  163.         // Optionally strip the href attribute
  164.         return $includeHref $linkAttributes array_diff_key($linkAttributes, ['href' => null]);
  165.     }
  166.     /**
  167.      * Returns the "href" link attribute.
  168.      */
  169.     public function getLinkHref(): string
  170.     {
  171.         return $this->getLinkAttributes(true)['href'] ?? '';
  172.     }
  173.     /**
  174.      * Returns a key-value list of template options.
  175.      */
  176.     public function getOptions(): array
  177.     {
  178.         $this->resolveIfClosure($this->options);
  179.         return $this->options ?? [];
  180.     }
  181.     /**
  182.      * Compiles an opinionated data set to be applied to a Contao template.
  183.      *
  184.      * Note: Do not use this method when building new templates from scratch or
  185.      *       when using Twig templates! Instead, add this object to your
  186.      *       template's context and directly access the specific data you need.
  187.      *
  188.      * @param string|array|null $margin              Set margins that will compose the inline CSS for the "margin" key
  189.      * @param string|null       $floating            Set/determine values for the "float_class" and "addBefore" keys
  190.      * @param bool              $includeFullMetadata Make all metadata available in the first dimension of the returned data set (key-value pairs)
  191.      */
  192.     public function getLegacyTemplateData($margin nullstring $floating nullbool $includeFullMetadata true): array
  193.     {
  194.         // Create a key-value list of the metadata and apply some renaming and
  195.         // formatting transformations to fit the legacy templates.
  196.         $createLegacyMetadataMapping = static function (Metadata $metadata): array {
  197.             if ($metadata->empty()) {
  198.                 return [];
  199.             }
  200.             $mapping $metadata->all();
  201.             // Handle special chars
  202.             foreach ([Metadata::VALUE_ALTMetadata::VALUE_TITLE] as $key) {
  203.                 if (isset($mapping[$key])) {
  204.                     $mapping[$key] = StringUtil::specialchars($mapping[$key]);
  205.                 }
  206.             }
  207.             // Rename certain keys (as used in the Contao templates)
  208.             if (isset($mapping[Metadata::VALUE_TITLE])) {
  209.                 $mapping['imageTitle'] = $mapping[Metadata::VALUE_TITLE];
  210.             }
  211.             if (isset($mapping[Metadata::VALUE_URL])) {
  212.                 $mapping['imageUrl'] = $mapping[Metadata::VALUE_URL];
  213.             }
  214.             unset($mapping[Metadata::VALUE_TITLE], $mapping[Metadata::VALUE_URL]);
  215.             return $mapping;
  216.         };
  217.         // Create a CSS margin property from an array or serialized string
  218.         $createMargin = static function ($margin): string {
  219.             if (!$margin) {
  220.                 return '';
  221.             }
  222.             $values array_merge(
  223.                 ['top' => '''right' => '''bottom' => '''left' => '''unit' => ''],
  224.                 StringUtil::deserialize($margintrue)
  225.             );
  226.             return Controller::generateMargin($values);
  227.         };
  228.         $image $this->getImage();
  229.         $originalSize $image->getOriginalDimensions()->getSize();
  230.         $fileInfoImageSize = (new File($image->getImageSrc(true)))->imageSize;
  231.         $linkAttributes $this->getLinkAttributes();
  232.         $metadata $this->hasMetadata() ? $this->getMetadata() : new Metadata([]);
  233.         // Primary image and metadata
  234.         $templateData array_merge(
  235.             [
  236.                 'picture' => [
  237.                     'img' => $image->getImg(),
  238.                     'sources' => $image->getSources(),
  239.                     'alt' => StringUtil::specialchars($metadata->getAlt()),
  240.                 ],
  241.                 'width' => $originalSize->getWidth(),
  242.                 'height' => $originalSize->getHeight(),
  243.                 'arrSize' => $fileInfoImageSize,
  244.                 'imgSize' => !empty($fileInfoImageSize) ? sprintf(' width="%d" height="%d"'$fileInfoImageSize[0], $fileInfoImageSize[1]) : '',
  245.                 'singleSRC' => $image->getFilePath(),
  246.                 'src' => $image->getImageSrc(),
  247.                 'fullsize' => ('_blank' === ($linkAttributes['target'] ?? null)) || $this->hasLightbox(),
  248.                 'margin' => $createMargin($margin),
  249.                 'addBefore' => 'below' !== $floating,
  250.                 'addImage' => true,
  251.             ],
  252.             $includeFullMetadata $createLegacyMetadataMapping($metadata) : []
  253.         );
  254.         // Link attributes and title
  255.         if ('' !== ($href $this->getLinkHref())) {
  256.             $templateData['href'] = $href;
  257.             $templateData['attributes'] = ''// always define attributes key if href is set
  258.             // Use link "title" attribute for "linkTitle" as it is already output explicitly in image.html5 (see #3385)
  259.             if (\array_key_exists('title'$linkAttributes)) {
  260.                 $templateData['linkTitle'] = $linkAttributes['title'];
  261.                 unset($linkAttributes['title']);
  262.             } else {
  263.                 // Map "imageTitle" to "linkTitle"
  264.                 $templateData['linkTitle'] = ($templateData['imageTitle'] ?? null) ?? StringUtil::specialchars($metadata->getTitle());
  265.                 unset($templateData['imageTitle']);
  266.             }
  267.         } elseif ($metadata->has(Metadata::VALUE_TITLE)) {
  268.             $templateData['picture']['title'] = StringUtil::specialchars($metadata->getTitle());
  269.         }
  270.         if (!empty($linkAttributes)) {
  271.             $htmlAttributes array_map(
  272.                 static fn (string $attributestring $value) => sprintf('%s="%s"'$attribute$value),
  273.                 array_keys($linkAttributes),
  274.                 $linkAttributes
  275.             );
  276.             $templateData['attributes'] = ' '.implode(' '$htmlAttributes);
  277.         }
  278.         // Lightbox
  279.         if ($this->hasLightbox()) {
  280.             $lightbox $this->getLightbox();
  281.             if ($lightbox->hasImage()) {
  282.                 $lightboxImage $lightbox->getImage();
  283.                 $templateData['lightboxPicture'] = [
  284.                     'img' => $lightboxImage->getImg(),
  285.                     'sources' => $lightboxImage->getSources(),
  286.                 ];
  287.             }
  288.         }
  289.         // Other
  290.         if ($floating) {
  291.             $templateData['floatClass'] = " float_$floating";
  292.         }
  293.         if (isset($this->getOptions()['attr']['class'])) {
  294.             $templateData['floatClass'] = ($templateData['floatClass'] ?? '').' '.$this->getOptions()['attr']['class'];
  295.         }
  296.         // Add arbitrary template options
  297.         return array_merge($templateData$this->getOptions());
  298.     }
  299.     /**
  300.      * Applies the legacy template data to an existing template. This will
  301.      * prevent overriding the "href" property if already present and use
  302.      * "imageHref" instead.
  303.      *
  304.      * Note: Do not use this method when building new templates from scratch or
  305.      *       when using Twig templates! Instead, add this object to your
  306.      *       template's context and directly access the specific data you need.
  307.      *
  308.      * @param Template|object   $template            The template to apply the data to
  309.      * @param string|array|null $margin              Set margins that will compose the inline CSS for the template's "margin" property
  310.      * @param string|null       $floating            Set/determine values for the template's "float_class" and "addBefore" properties
  311.      * @param bool              $includeFullMetadata Make all metadata entries directly available in the template
  312.      */
  313.     public function applyLegacyTemplateData(object $template$margin nullstring $floating nullbool $includeFullMetadata true): void
  314.     {
  315.         $new $this->getLegacyTemplateData($margin$floating$includeFullMetadata);
  316.         $existing $template instanceof Template $template->getData() : get_object_vars($template);
  317.         // Do not override the "href" key (see #6468)
  318.         if (isset($new['href'], $existing['href'])) {
  319.             $new['imageHref'] = $new['href'];
  320.             unset($new['href']);
  321.         }
  322.         // Allow accessing Figure methods in a legacy template context
  323.         $new['figure'] = $this;
  324.         // Apply data
  325.         if ($template instanceof Template) {
  326.             $template->setData(array_replace($existing$new));
  327.             return;
  328.         }
  329.         foreach ($new as $key => $value) {
  330.             $template->$key $value;
  331.         }
  332.     }
  333.     /**
  334.      * Evaluates closures to retrieve the value.
  335.      *
  336.      * @param mixed $property
  337.      */
  338.     private function resolveIfClosure(&$property): void
  339.     {
  340.         if ($property instanceof \Closure) {
  341.             $property $property($this);
  342.         }
  343.     }
  344. }