vendor/contao/core-bundle/src/Image/ImageSizes.php line 123

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;
  11. use Contao\BackendUser;
  12. use Contao\CoreBundle\Event\ContaoCoreEvents;
  13. use Contao\CoreBundle\Event\ImageSizesEvent;
  14. use Contao\CoreBundle\Framework\ContaoFramework;
  15. use Contao\StringUtil;
  16. use Doctrine\DBAL\Connection;
  17. use Symfony\Component\EventDispatcher\EventDispatcherInterface;
  18. use Symfony\Contracts\Service\ResetInterface;
  19. use Symfony\Contracts\Translation\TranslatorInterface;
  20. class ImageSizes implements ResetInterface
  21. {
  22.     private Connection $connection;
  23.     private EventDispatcherInterface $eventDispatcher;
  24.     private ContaoFramework $framework;
  25.     private TranslatorInterface $translator;
  26.     private array $predefinedSizes = [];
  27.     private ?array $options null;
  28.     /**
  29.      * @internal
  30.      */
  31.     public function __construct(Connection $connectionEventDispatcherInterface $eventDispatcherContaoFramework $frameworkTranslatorInterface $translator)
  32.     {
  33.         $this->connection $connection;
  34.         $this->eventDispatcher $eventDispatcher;
  35.         $this->framework $framework;
  36.         $this->translator $translator;
  37.     }
  38.     /**
  39.      * Sets the predefined image sizes.
  40.      */
  41.     public function setPredefinedSizes(array $predefinedSizes): void
  42.     {
  43.         $this->predefinedSizes $predefinedSizes;
  44.     }
  45.     /**
  46.      * Returns the image sizes as options suitable for widgets.
  47.      *
  48.      * @return array<string, array<string>>
  49.      */
  50.     public function getAllOptions(): array
  51.     {
  52.         $this->loadOptions();
  53.         $event = new ImageSizesEvent($this->options);
  54.         $this->eventDispatcher->dispatch($eventContaoCoreEvents::IMAGE_SIZES_ALL);
  55.         return $event->getImageSizes();
  56.     }
  57.     /**
  58.      * Returns the image sizes for the given user suitable for widgets.
  59.      *
  60.      * @return array<string, array<string>>
  61.      */
  62.     public function getOptionsForUser(BackendUser $user): array
  63.     {
  64.         $this->loadOptions();
  65.         if ($user->isAdmin) {
  66.             $event = new ImageSizesEvent($this->options$user);
  67.         } else {
  68.             $options array_map(
  69.                 static fn ($val) => is_numeric($val) ? (int) $val $val,
  70.                 StringUtil::deserialize($user->imageSizestrue)
  71.             );
  72.             $event = new ImageSizesEvent($this->filterOptions($options), $user);
  73.         }
  74.         $this->eventDispatcher->dispatch($eventContaoCoreEvents::IMAGE_SIZES_USER);
  75.         return $event->getImageSizes();
  76.     }
  77.     public function reset(): void
  78.     {
  79.         $this->options null;
  80.     }
  81.     /**
  82.      * Loads the options from the database.
  83.      */
  84.     private function loadOptions(): void
  85.     {
  86.         if (null !== $this->options) {
  87.             return;
  88.         }
  89.         // The framework is required to have the TL_CROP options available
  90.         $this->framework->initialize();
  91.         // Backwards compatibility
  92.         $this->options $GLOBALS['TL_CROP'] ?? [];
  93.         if (
  94.             !== \count($this->options)
  95.             || !== \count($this->options['image_sizes'] ?? [])
  96.             || !== \count($this->options['relative'] ?? [])
  97.             || 10 !== \count($this->options['exact'] ?? [])
  98.         ) {
  99.             trigger_deprecation('contao/core-bundle''4.13''Using $GLOBALS[\'TL_CROP\'] has been deprecated and will be removed in Contao 5.0. Use the "contao.image.sizes" service instead.');
  100.         }
  101.         $rows $this->connection->fetchAllAssociative(
  102.             'SELECT
  103.                 s.id, s.name, s.width, s.height, t.name as theme
  104.             FROM
  105.                 tl_image_size s
  106.             LEFT JOIN
  107.                 tl_theme t ON s.pid=t.id
  108.             ORDER BY
  109.                 s.pid, s.name'
  110.         );
  111.         $options = [];
  112.         foreach ($this->predefinedSizes as $name => $imageSize) {
  113.             $options['image_sizes'][$name] = sprintf(
  114.                 '%s (%sx%s)',
  115.                 $this->translator->trans(substr($name1), [], 'image_sizes') ?: substr($name1),
  116.                 $imageSize['width'] ?? '',
  117.                 $imageSize['height'] ?? ''
  118.             );
  119.         }
  120.         foreach ($rows as $imageSize) {
  121.             // Prefix theme names that are numeric or collide with existing group names
  122.             if (is_numeric($imageSize['theme']) || \in_array($imageSize['theme'], ['exact''relative''image_sizes'], true)) {
  123.                 $imageSize['theme'] = 'Theme '.$imageSize['theme'];
  124.             }
  125.             if (!isset($options[$imageSize['theme']])) {
  126.                 $options[$imageSize['theme']] = [];
  127.             }
  128.             $options[$imageSize['theme']][$imageSize['id']] = sprintf(
  129.                 '%s (%sx%s)',
  130.                 $imageSize['name'],
  131.                 $imageSize['width'],
  132.                 $imageSize['height']
  133.             );
  134.         }
  135.         $this->options array_merge_recursive($options$this->options);
  136.     }
  137.     /**
  138.      * Filters the options by the given allowed sizes and returns the result.
  139.      *
  140.      * @return array<string, array<string>>
  141.      */
  142.     private function filterOptions(array $allowedSizes): array
  143.     {
  144.         if (empty($allowedSizes)) {
  145.             return [];
  146.         }
  147.         $filteredSizes = [];
  148.         foreach ($this->options as $group => $sizes) {
  149.             if ('relative' === $group || 'exact' === $group) {
  150.                 $this->filterResizeModes($sizes$allowedSizes$filteredSizes$group);
  151.             } else {
  152.                 $this->filterImageSizes($sizes$allowedSizes$filteredSizes$group);
  153.             }
  154.         }
  155.         return $filteredSizes;
  156.     }
  157.     private function filterImageSizes(array $sizes, array $allowedSizes, array &$filteredSizesstring $group): void
  158.     {
  159.         foreach ($sizes as $key => $size) {
  160.             if (\in_array($key$allowedSizestrue)) {
  161.                 $filteredSizes[$group][$key] = $size;
  162.             }
  163.         }
  164.     }
  165.     private function filterResizeModes(array $sizes, array $allowedSizes, array &$filteredSizesstring $group): void
  166.     {
  167.         foreach ($sizes as $size) {
  168.             if (\in_array($size$allowedSizestrue)) {
  169.                 $filteredSizes[$group][] = $size;
  170.             }
  171.         }
  172.     }
  173. }