vendor/contao/core-bundle/src/Routing/Page/PageRegistry.php line 54

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\Routing\Page;
  11. use Contao\PageModel;
  12. use Doctrine\DBAL\Connection;
  13. class PageRegistry
  14. {
  15.     private const DISABLE_CONTENT_COMPOSITION = ['redirect''forward''logout'];
  16.     private Connection $connection;
  17.     private ?array $urlPrefixes null;
  18.     private ?array $urlSuffixes null;
  19.     /**
  20.      * @var array<RouteConfig>
  21.      */
  22.     private array $routeConfigs = [];
  23.     /**
  24.      * @var array<DynamicRouteInterface>
  25.      */
  26.     private array $routeEnhancers = [];
  27.     /**
  28.      * @var array<ContentCompositionInterface|bool>
  29.      */
  30.     private array $contentComposition = [];
  31.     public function __construct(Connection $connection)
  32.     {
  33.         $this->connection $connection;
  34.     }
  35.     /**
  36.      * Returns the route for a page.
  37.      *
  38.      * If no path is configured (is null), the route will accept
  39.      * any parameters after the page alias (e.g. "en/page-alias/foo/bar.html").
  40.      *
  41.      * A route enhancer might enhance the route for a specific page.
  42.      */
  43.     public function getRoute(PageModel $pageModel): PageRoute
  44.     {
  45.         $type $pageModel->type;
  46.         $config $this->routeConfigs[$type] ?? new RouteConfig();
  47.         $defaults $config->getDefaults();
  48.         $requirements $config->getRequirements();
  49.         $options $config->getOptions();
  50.         $path $config->getPath();
  51.         if (false === $path) {
  52.             $path '';
  53.             $options['compiler_class'] = UnroutablePageRouteCompiler::class;
  54.         } elseif (null === $path) {
  55.             if ($this->isParameterless($pageModel)) {
  56.                 $path '/'.($pageModel->alias ?: $pageModel->id);
  57.             } else {
  58.                 $path '/'.($pageModel->alias ?: $pageModel->id).'{!parameters}';
  59.                 $defaults['parameters'] = '';
  60.                 $requirements['parameters'] = $pageModel->requireItem '/.+?' '(/.+?)?';
  61.             }
  62.         }
  63.         $route = new PageRoute($pageModel$path$defaults$requirements$options$config->getMethods());
  64.         if (null !== $config->getUrlSuffix()) {
  65.             $route->setUrlSuffix($config->getUrlSuffix());
  66.         }
  67.         if (!isset($this->routeEnhancers[$type])) {
  68.             return $route;
  69.         }
  70.         /** @var DynamicRouteInterface $enhancer */
  71.         $enhancer $this->routeEnhancers[$type];
  72.         $enhancer->configurePageRoute($route);
  73.         return $route;
  74.     }
  75.     public function getPathRegex(): array
  76.     {
  77.         $prefixes = [];
  78.         foreach ($this->routeConfigs as $type => $config) {
  79.             $regex $config->getPathRegex();
  80.             if (null !== $regex) {
  81.                 $prefixes[$type] = $regex;
  82.             }
  83.         }
  84.         return $prefixes;
  85.     }
  86.     public function supportsContentComposition(PageModel $pageModel): bool
  87.     {
  88.         if (!isset($this->contentComposition[$pageModel->type])) {
  89.             return !\in_array($pageModel->typeself::DISABLE_CONTENT_COMPOSITIONtrue);
  90.         }
  91.         $service $this->contentComposition[$pageModel->type];
  92.         if ($service instanceof ContentCompositionInterface) {
  93.             return $service->supportsContentComposition($pageModel);
  94.         }
  95.         return (bool) $service;
  96.     }
  97.     /**
  98.      * @return array<string>
  99.      */
  100.     public function getUrlPrefixes(): array
  101.     {
  102.         $this->initializePrefixAndSuffix();
  103.         return $this->urlPrefixes;
  104.     }
  105.     /**
  106.      * @return array<string>
  107.      */
  108.     public function getUrlSuffixes(): array
  109.     {
  110.         $this->initializePrefixAndSuffix();
  111.         return $this->urlSuffixes;
  112.     }
  113.     /**
  114.      * @param ContentCompositionInterface|bool $contentComposition
  115.      */
  116.     public function add(string $typeRouteConfig $configDynamicRouteInterface $routeEnhancer null$contentComposition true): self
  117.     {
  118.         // Override existing pages with the same identifier
  119.         $this->routeConfigs[$type] = $config;
  120.         if (null !== $routeEnhancer) {
  121.             $this->routeEnhancers[$type] = $routeEnhancer;
  122.         }
  123.         if (null !== $contentComposition) {
  124.             $this->contentComposition[$type] = $contentComposition;
  125.         }
  126.         $this->urlPrefixes $this->urlSuffixes null;
  127.         return $this;
  128.     }
  129.     public function remove(string $type): self
  130.     {
  131.         unset(
  132.             $this->routeConfigs[$type],
  133.             $this->routeEnhancers[$type],
  134.             $this->contentComposition[$type]
  135.         );
  136.         $this->urlPrefixes $this->urlSuffixes null;
  137.         return $this;
  138.     }
  139.     public function keys(): array
  140.     {
  141.         return array_keys($this->routeConfigs);
  142.     }
  143.     /**
  144.      * Checks whether this is a routable page type (see #3415).
  145.      */
  146.     public function isRoutable(PageModel $page): bool
  147.     {
  148.         $type $page->type;
  149.         // Any legacy page without route config is routable by default
  150.         if (!isset($this->routeConfigs[$type])) {
  151.             return true;
  152.         }
  153.         // Check if page controller is routable
  154.         return false !== $this->routeConfigs[$type]->getPath();
  155.     }
  156.     /**
  157.      * @return array<string>
  158.      */
  159.     public function getUnroutableTypes(): array
  160.     {
  161.         $types = [];
  162.         foreach ($this->routeConfigs as $type => $config) {
  163.             if (false === $config->getPath()) {
  164.                 $types[] = $type;
  165.             }
  166.         }
  167.         return $types;
  168.     }
  169.     private function initializePrefixAndSuffix(): void
  170.     {
  171.         if (null !== $this->urlPrefixes || null !== $this->urlSuffixes) {
  172.             return;
  173.         }
  174.         $results $this->connection->fetchAllAssociative("SELECT urlPrefix, urlSuffix FROM tl_page WHERE type='root'");
  175.         $urlSuffixes = [
  176.             array_column($results'urlSuffix'),
  177.             array_filter(array_map(
  178.                 static fn (RouteConfig $config) => $config->getUrlSuffix(),
  179.                 $this->routeConfigs
  180.             )),
  181.         ];
  182.         foreach ($this->routeConfigs as $config) {
  183.             if (null !== ($suffix $config->getUrlSuffix())) {
  184.                 $urlSuffixes[] = [$suffix];
  185.             }
  186.         }
  187.         foreach ($this->routeEnhancers as $enhancer) {
  188.             $urlSuffixes[] = $enhancer->getUrlSuffixes();
  189.         }
  190.         $this->urlSuffixes array_values(array_unique(array_merge(...$urlSuffixes)));
  191.         $this->urlPrefixes array_values(array_unique(array_column($results'urlPrefix')));
  192.     }
  193.     private function isParameterless(PageModel $pageModel): bool
  194.     {
  195.         if ('redirect' === $pageModel->type) {
  196.             return true;
  197.         }
  198.         return 'forward' === $pageModel->type && !$pageModel->alwaysForward;
  199.     }
  200. }