diff --git a/extra/html-extra/HtmlExtension.php b/extra/html-extra/HtmlExtension.php index d8a6c0036d1..08395096df7 100644 --- a/extra/html-extra/HtmlExtension.php +++ b/extra/html-extra/HtmlExtension.php @@ -12,8 +12,12 @@ namespace Twig\Extra\Html; use Symfony\Component\Mime\MimeTypes; +use Twig\Environment; use Twig\Error\RuntimeError; use Twig\Extension\AbstractExtension; +use Twig\Extension\CoreExtension; +use Twig\Extension\EscaperExtension; +use Twig\Runtime\EscaperRuntime; use Twig\TwigFilter; use Twig\TwigFunction; @@ -30,6 +34,7 @@ public function getFilters(): array { return [ new TwigFilter('data_uri', [$this, 'dataUri']), + new TwigFilter('html_attr_merge', [self::class, 'htmlAttrMerge']), ]; } @@ -38,6 +43,7 @@ public function getFunctions(): array return [ new TwigFunction('html_classes', [self::class, 'htmlClasses']), new TwigFunction('html_cva', [self::class, 'htmlCva']), + new TwigFunction('html_attr', [self::class, 'htmlAttr'], ['needs_environment' => true, 'is_safe' => ['html']]), ]; } @@ -124,4 +130,77 @@ public static function htmlCva(array|string $base = [], array $variants = [], ar { return new Cva($base, $variants, $compoundVariants, $defaultVariant); } + + public static function htmlAttrMerge(...$arrays): array + { + $result = []; + + foreach ($arrays as $argNumber => $array) { + if (!$array) { + continue; + } + + if (!is_iterable($array)) { + throw new RuntimeError(sprintf('The "attr_merge" filter only works with arrays or "Traversable", got "%s" for argument %d.', \gettype($array), $argNumber + 1)); + } + + $array = CoreExtension::toArray($array); + + foreach (['class', 'style', 'data', 'aria'] as $deepMergeKey) { + if (isset($array[$deepMergeKey])) { + $value = $array[$deepMergeKey]; + unset($array[$deepMergeKey]); + + if (!is_iterable($value)) { + $value = (array) $value; + } + + $value = CoreExtension::toArray($value); + + $result[$deepMergeKey] = array_merge($result[$deepMergeKey] ?? [], $value); + } + } + + $result = array_merge($result, $array); + } + + return $result; + } + + public static function htmlAttr(Environment $env, ...$args): string + { + $attr = self::htmlAttrMerge(...$args); + + if (isset($attr['class'])) { + $attr['class'] = trim(implode(' ', $attr['class'])); + } + + if (isset($attr['style'])) { + $style = ''; + foreach ($attr['style'] as $name => $value) { + if (is_numeric($name)) { + $style .= $value.'; '; + } else { + $style .= $name.': '.$value.'; '; + } + } + $attr['style'] = trim($style); + } + + if (isset($attr['data'])) { + foreach ($attr['data'] as $name => $value) { + $attr['data-'.$name] = $value; + } + unset($attr['data']); + } + + $result = ''; + $runtime = $env->getRuntime(EscaperRuntime::class); + + foreach ($attr as $name => $value) { + $result .= $runtime->escape($name, 'html_attr').'="'.$runtime->escape($value).'" '; + } + + return trim($result); + } }