vendor/contao/core-bundle/src/Twig/Extension/ContaoExtension.php line 83

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\Twig\Extension;
  11. use Contao\BackendTemplateTrait;
  12. use Contao\CoreBundle\InsertTag\ChunkedText;
  13. use Contao\CoreBundle\Twig\Inheritance\DynamicExtendsTokenParser;
  14. use Contao\CoreBundle\Twig\Inheritance\DynamicIncludeTokenParser;
  15. use Contao\CoreBundle\Twig\Inheritance\TemplateHierarchyInterface;
  16. use Contao\CoreBundle\Twig\Interop\ContaoEscaper;
  17. use Contao\CoreBundle\Twig\Interop\ContaoEscaperNodeVisitor;
  18. use Contao\CoreBundle\Twig\Interop\PhpTemplateProxyNode;
  19. use Contao\CoreBundle\Twig\Interop\PhpTemplateProxyNodeVisitor;
  20. use Contao\CoreBundle\Twig\Runtime\FigureRendererRuntime;
  21. use Contao\CoreBundle\Twig\Runtime\InsertTagRuntime;
  22. use Contao\CoreBundle\Twig\Runtime\LegacyTemplateFunctionsRuntime;
  23. use Contao\CoreBundle\Twig\Runtime\PictureConfigurationRuntime;
  24. use Contao\CoreBundle\Twig\Runtime\SchemaOrgRuntime;
  25. use Contao\FrontendTemplateTrait;
  26. use Contao\Template;
  27. use Symfony\Component\Filesystem\Path;
  28. use Twig\Environment;
  29. use Twig\Extension\AbstractExtension;
  30. use Twig\Extension\CoreExtension;
  31. use Twig\Extension\EscaperExtension;
  32. use Twig\TwigFilter;
  33. use Twig\TwigFunction;
  34. /**
  35.  * @experimental
  36.  */
  37. final class ContaoExtension extends AbstractExtension
  38. {
  39.     private Environment $environment;
  40.     private TemplateHierarchyInterface $hierarchy;
  41.     private array $contaoEscaperFilterRules = [];
  42.     public function __construct(Environment $environmentTemplateHierarchyInterface $hierarchy)
  43.     {
  44.         $this->environment $environment;
  45.         $this->hierarchy $hierarchy;
  46.         $contaoEscaper = new ContaoEscaper();
  47.         /** @var EscaperExtension $escaperExtension */
  48.         $escaperExtension $environment->getExtension(EscaperExtension::class);
  49.         $escaperExtension->setEscaper('contao_html', [$contaoEscaper'escapeHtml']);
  50.         $escaperExtension->setEscaper('contao_html_attr', [$contaoEscaper'escapeHtmlAttr']);
  51.         // Use our escaper on all templates in the "@Contao" and "@Contao_*"
  52.         // namespaces, as well as the existing bundle templates we're already
  53.         // shipping.
  54.         $this->addContaoEscaperRule('%^@Contao(_[a-zA-Z0-9_-]*)?/%');
  55.         $this->addContaoEscaperRule('%^@Contao(Core|Installation)/%');
  56.     }
  57.     /**
  58.      * Adds a Contao escaper rule.
  59.      *
  60.      * If a template name matches any of the defined rules, it will be processed
  61.      * with the "contao_html" escaper strategy. Make sure your rule will only
  62.      * match templates with input encoded contexts!
  63.      */
  64.     public function addContaoEscaperRule(string $regularExpression): void
  65.     {
  66.         if (\in_array($regularExpression$this->contaoEscaperFilterRulestrue)) {
  67.             return;
  68.         }
  69.         $this->contaoEscaperFilterRules[] = $regularExpression;
  70.     }
  71.     public function getNodeVisitors(): array
  72.     {
  73.         return [
  74.             // Enables the "contao_twig" escaper for Contao templates with
  75.             // input encoding
  76.             new ContaoEscaperNodeVisitor(
  77.                 fn () => $this->contaoEscaperFilterRules
  78.             ),
  79.             // Allows rendering PHP templates with the legacy framework by
  80.             // installing proxy nodes
  81.             new PhpTemplateProxyNodeVisitor(self::class),
  82.             // Triggers PHP deprecations if deprecated constructs are found in
  83.             // the parsed templates.
  84.             new DeprecationsNodeVisitor(),
  85.         ];
  86.     }
  87.     public function getTokenParsers(): array
  88.     {
  89.         return [
  90.             // Overwrite the parsers for the "extends" and "include" tags to
  91.             // additionally support the Contao template hierarchy
  92.             new DynamicExtendsTokenParser($this->hierarchy),
  93.             new DynamicIncludeTokenParser($this->hierarchy),
  94.         ];
  95.     }
  96.     public function getFunctions(): array
  97.     {
  98.         $includeFunctionCallable $this->getTwigIncludeFunction()->getCallable();
  99.         return [
  100.             // Overwrite the "include" function to additionally support the
  101.             // Contao template hierarchy
  102.             new TwigFunction(
  103.                 'include',
  104.                 function (Environment $env$context$template$variables = [], $withContext true$ignoreMissing false$sandboxed false /* we need named arguments here */) use ($includeFunctionCallable) {
  105.                     $args \func_get_args();
  106.                     $args[2] = DynamicIncludeTokenParser::adjustTemplateName((string) $template$this->hierarchy);
  107.                     return $includeFunctionCallable(...$args);
  108.                 },
  109.                 ['needs_environment' => true'needs_context' => true'is_safe' => ['all']]
  110.             ),
  111.             new TwigFunction(
  112.                 'contao_figure',
  113.                 [FigureRendererRuntime::class, 'render'],
  114.                 ['is_safe' => ['html']]
  115.             ),
  116.             new TwigFunction(
  117.                 'picture_config',
  118.                 [PictureConfigurationRuntime::class, 'fromArray']
  119.             ),
  120.             new TwigFunction(
  121.                 'insert_tag',
  122.                 [InsertTagRuntime::class, 'renderInsertTag'],
  123.             ),
  124.             new TwigFunction(
  125.                 'add_schema_org',
  126.                 [SchemaOrgRuntime::class, 'add']
  127.             ),
  128.             new TwigFunction(
  129.                 'contao_sections',
  130.                 [LegacyTemplateFunctionsRuntime::class, 'renderLayoutSections'],
  131.                 ['needs_context' => true'is_safe' => ['html']]
  132.             ),
  133.             new TwigFunction(
  134.                 'contao_section',
  135.                 [LegacyTemplateFunctionsRuntime::class, 'renderLayoutSection'],
  136.                 ['needs_context' => true'is_safe' => ['html']]
  137.             ),
  138.             new TwigFunction(
  139.                 'render_contao_backend_template',
  140.                 [LegacyTemplateFunctionsRuntime::class, 'renderContaoBackendTemplate'],
  141.                 ['is_safe' => ['html']]
  142.             ),
  143.         ];
  144.     }
  145.     public function getFilters(): array
  146.     {
  147.         $escaperFilter = static function (Environment $env$string$strategy 'html'$charset null$autoescape false) {
  148.             if ($string instanceof ChunkedText) {
  149.                 $parts = [];
  150.                 foreach ($string as [$type$chunk]) {
  151.                     $parts[] = ChunkedText::TYPE_RAW === $type
  152.                         $chunk
  153.                         twig_escape_filter($env$chunk$strategy$charset);
  154.                 }
  155.                 return implode(''$parts);
  156.             }
  157.             return twig_escape_filter($env$string$strategy$charset$autoescape);
  158.         };
  159.         return [
  160.             // Overwrite the "escape" filter to additionally support chunked text
  161.             new TwigFilter(
  162.                 'escape',
  163.                 $escaperFilter,
  164.                 ['needs_environment' => true'is_safe_callback' => 'twig_escape_filter_is_safe']
  165.             ),
  166.             new TwigFilter(
  167.                 'e',
  168.                 $escaperFilter,
  169.                 ['needs_environment' => true'is_safe_callback' => 'twig_escape_filter_is_safe']
  170.             ),
  171.             new TwigFilter(
  172.                 'insert_tag',
  173.                 [InsertTagRuntime::class, 'replaceInsertTags']
  174.             ),
  175.             new TwigFilter(
  176.                 'insert_tag_raw',
  177.                 [InsertTagRuntime::class, 'replaceInsertTagsChunkedRaw']
  178.             ),
  179.         ];
  180.     }
  181.     /**
  182.      * @see PhpTemplateProxyNode
  183.      * @see PhpTemplateProxyNodeVisitor
  184.      *
  185.      * @internal
  186.      */
  187.     public function renderLegacyTemplate(string $name, array $blocks, array $context): string
  188.     {
  189.         $template Path::getFilenameWithoutExtension($name);
  190.         $partialTemplate = new class($template) extends Template {
  191.             use BackendTemplateTrait;
  192.             use FrontendTemplateTrait;
  193.             public function setBlocks(array $blocks): void
  194.             {
  195.                 $this->arrBlocks array_map(static fn ($block) => \is_array($block) ? $block : [$block], $blocks);
  196.             }
  197.             public function parse(): string
  198.             {
  199.                 return $this->inherit();
  200.             }
  201.             protected function renderTwigSurrogateIfExists(): ?string
  202.             {
  203.                 return null;
  204.             }
  205.         };
  206.         $partialTemplate->setData($context);
  207.         $partialTemplate->setBlocks($blocks);
  208.         return $partialTemplate->parse();
  209.     }
  210.     private function getTwigIncludeFunction(): TwigFunction
  211.     {
  212.         foreach ($this->environment->getExtension(CoreExtension::class)->getFunctions() as $function) {
  213.             if ('include' === $function->getName()) {
  214.                 return $function;
  215.             }
  216.         }
  217.         throw new \RuntimeException(sprintf('The %s class was expected to register the "include" Twig function but did not.'CoreExtension::class));
  218.     }
  219. }