From af13e98ca482262c814a27c0ab4579dd45611f79 Mon Sep 17 00:00:00 2001
From: Oliver Bartsch <bo@cedev.de>
Date: Tue, 13 Aug 2024 17:04:01 +0200
Subject: [PATCH] [TASK] Introduce TypolinkParameter object

This adds a new value object, containing
the resolved parameters of a typolink.

The object will be used as enriched value
for TCA type "link" properties in the
Record object. The enrichment will be added
with #103581.

Since this value is then passed to the view
instead of the plain typolink string, the
TypoLink ViewHelpers do also support the
new object.

Resolves: #104615
Related: #103581
Releases: main
Change-Id: Ifbac6d44ed05d5d793b951b38229891d6b219eb4
Reviewed-on: https://review.typo3.org/c/Packages/TYPO3.CMS/+/85619
Reviewed-by: Nikita Hovratov <nikita.h@live.de>
Tested-by: Oliver Bartsch <bo@cedev.de>
Tested-by: core-ci <typo3@b13.com>
Reviewed-by: Oliver Bartsch <bo@cedev.de>
Tested-by: Jochen Roth <rothjochen@gmail.com>
Reviewed-by: Jochen Roth <rothjochen@gmail.com>
Tested-by: Nikita Hovratov <nikita.h@live.de>
---
 .../ViewHelpers/Link/TypolinkViewHelper.php   | 22 +++---
 .../ViewHelpers/Uri/TypolinkViewHelper.php    | 23 ++++---
 .../Link/TypolinkViewHelperTest.php           | 26 +++++++
 .../Uri/TypolinkViewHelperTest.php            | 69 +++++++++++++++++--
 .../Classes/Typolink/TypolinkParameter.php    | 68 ++++++++++++++++++
 5 files changed, 184 insertions(+), 24 deletions(-)
 create mode 100644 typo3/sysext/frontend/Classes/Typolink/TypolinkParameter.php

diff --git a/typo3/sysext/fluid/Classes/ViewHelpers/Link/TypolinkViewHelper.php b/typo3/sysext/fluid/Classes/ViewHelpers/Link/TypolinkViewHelper.php
index 303d61e28edf..b0bd716d9889 100644
--- a/typo3/sysext/fluid/Classes/ViewHelpers/Link/TypolinkViewHelper.php
+++ b/typo3/sysext/fluid/Classes/ViewHelpers/Link/TypolinkViewHelper.php
@@ -20,6 +20,7 @@ namespace TYPO3\CMS\Fluid\ViewHelpers\Link;
 use TYPO3\CMS\Core\LinkHandling\TypoLinkCodecService;
 use TYPO3\CMS\Core\Utility\GeneralUtility;
 use TYPO3\CMS\Frontend\ContentObject\ContentObjectRenderer;
+use TYPO3\CMS\Frontend\Typolink\TypolinkParameter;
 use TYPO3Fluid\Fluid\Core\Rendering\RenderingContextInterface;
 use TYPO3Fluid\Fluid\Core\Variables\ScopedVariableProvider;
 use TYPO3Fluid\Fluid\Core\Variables\StandardVariableProvider;
@@ -101,7 +102,7 @@ final class TypolinkViewHelper extends AbstractViewHelper
 
     public function initializeArguments(): void
     {
-        $this->registerArgument('parameter', 'string', 'stdWrap.typolink style parameter string', true);
+        $this->registerArgument('parameter', 'mixed', 'stdWrap.typolink style parameter string', true);
         $this->registerArgument('target', 'string', 'Define where to display the linked URL', false, '');
         $this->registerArgument('class', 'string', 'Define classes for the link element', false, '');
         $this->registerArgument('title', 'string', 'Define the title for the link element', false, '');
@@ -123,23 +124,28 @@ final class TypolinkViewHelper extends AbstractViewHelper
     {
         $parameter = $arguments['parameter'] ?? '';
         $partsAs = $arguments['parts-as'] ?? 'typoLinkParts';
+        $typoLinkCodecService = GeneralUtility::makeInstance(TypoLinkCodecService::class);
+
+        if (!$parameter instanceof TypolinkParameter) {
+            $parameter = TypolinkParameter::createFromTypolinkParts(
+                is_scalar($parameter) ? $typoLinkCodecService->decode((string)$parameter) : []
+            );
+        }
 
-        $typoLinkCodec = GeneralUtility::makeInstance(TypoLinkCodecService::class);
-        $typoLinkConfiguration = $typoLinkCodec->decode((string)$parameter);
         // Merge the $parameter with other arguments
-        $mergedTypoLinkConfiguration = self::mergeTypoLinkConfiguration($typoLinkConfiguration, $arguments);
-        $typoLinkParameter = $typoLinkCodec->encode($mergedTypoLinkConfiguration);
+        $typolinkParameter = TypolinkParameter::createFromTypolinkParts(self::mergeTypoLinkConfiguration($parameter->toArray(), $arguments))->toArray();
 
         // expose internal typoLink configuration to Fluid child context
-        $variableProvider = new ScopedVariableProvider($renderingContext->getVariableProvider(), new StandardVariableProvider([$partsAs => $typoLinkConfiguration]));
+        $variableProvider = new ScopedVariableProvider($renderingContext->getVariableProvider(), new StandardVariableProvider([$partsAs => $typolinkParameter]));
         $renderingContext->setVariableProvider($variableProvider);
         // If no link has to be rendered, the inner content will be returned as such
         $content = (string)$renderChildrenClosure();
         // clean up exposed variables
         $renderingContext->setVariableProvider($variableProvider->getGlobalVariableProvider());
 
-        if ($parameter) {
-            $content = self::invokeContentObjectRenderer($arguments, $typoLinkParameter, $content);
+        $typolink = $typoLinkCodecService->encode($typolinkParameter);
+        if ($typolink !== '') {
+            $content = self::invokeContentObjectRenderer($arguments, $typolink, $content);
         }
         return $content;
     }
diff --git a/typo3/sysext/fluid/Classes/ViewHelpers/Uri/TypolinkViewHelper.php b/typo3/sysext/fluid/Classes/ViewHelpers/Uri/TypolinkViewHelper.php
index 7a346a1995d7..a6f47669abeb 100644
--- a/typo3/sysext/fluid/Classes/ViewHelpers/Uri/TypolinkViewHelper.php
+++ b/typo3/sysext/fluid/Classes/ViewHelpers/Uri/TypolinkViewHelper.php
@@ -20,6 +20,7 @@ namespace TYPO3\CMS\Fluid\ViewHelpers\Uri;
 use TYPO3\CMS\Core\LinkHandling\TypoLinkCodecService;
 use TYPO3\CMS\Core\Utility\GeneralUtility;
 use TYPO3\CMS\Frontend\ContentObject\ContentObjectRenderer;
+use TYPO3\CMS\Frontend\Typolink\TypolinkParameter;
 use TYPO3Fluid\Fluid\Core\Rendering\RenderingContextInterface;
 use TYPO3Fluid\Fluid\Core\ViewHelper\AbstractViewHelper;
 use TYPO3Fluid\Fluid\Core\ViewHelper\Traits\CompileWithRenderStatic;
@@ -64,7 +65,7 @@ final class TypolinkViewHelper extends AbstractViewHelper
 
     public function initializeArguments(): void
     {
-        $this->registerArgument('parameter', 'string', 'stdWrap.typolink style parameter string', true);
+        $this->registerArgument('parameter', 'mixed', 'stdWrap.typolink style parameter string', true);
         $this->registerArgument('additionalParams', 'string', 'stdWrap.typolink additionalParams', false, '');
         $this->registerArgument('language', 'string', 'link to a specific language - defaults to the current language, use a language ID or "current" to enforce a specific language');
         $this->registerArgument('addQueryString', 'string', 'If set, the current query parameters will be kept in the URL. If set to "untrusted", then ALL query parameters will be added. Be aware, that this might lead to problems when the generated link is cached.', false, false);
@@ -75,17 +76,19 @@ final class TypolinkViewHelper extends AbstractViewHelper
     public static function renderStatic(array $arguments, \Closure $renderChildrenClosure, RenderingContextInterface $renderingContext): string
     {
         $parameter = $arguments['parameter'] ?? '';
+        $typoLinkCodecService = GeneralUtility::makeInstance(TypoLinkCodecService::class);
 
-        $typoLinkCodec = GeneralUtility::makeInstance(TypoLinkCodecService::class);
-        $typoLinkConfiguration = $typoLinkCodec->decode((string)$parameter);
-        $mergedTypoLinkConfiguration = self::mergeTypoLinkConfiguration($typoLinkConfiguration, $arguments);
-        $typoLinkParameter = $typoLinkCodec->encode($mergedTypoLinkConfiguration);
-
-        $content = '';
-        if ($parameter) {
-            $content = self::invokeContentObjectRenderer($arguments, $typoLinkParameter);
+        if (!$parameter instanceof TypolinkParameter) {
+            $parameter = TypolinkParameter::createFromTypolinkParts(
+                is_scalar($parameter) ? $typoLinkCodecService->decode((string)$parameter) : []
+            );
         }
-        return $content;
+
+        // Merge the $parameter with other arguments and encode the typolink again
+        $typolink = $typoLinkCodecService->encode(
+            TypolinkParameter::createFromTypolinkParts(self::mergeTypoLinkConfiguration($parameter->toArray(), $arguments))->toArray()
+        );
+        return $typolink !== '' ? self::invokeContentObjectRenderer($arguments, $typolink) : '';
     }
 
     protected static function invokeContentObjectRenderer(array $arguments, string $typoLinkParameter): string
diff --git a/typo3/sysext/fluid/Tests/Functional/ViewHelpers/Link/TypolinkViewHelperTest.php b/typo3/sysext/fluid/Tests/Functional/ViewHelpers/Link/TypolinkViewHelperTest.php
index 8f7a54239b3b..c9316f5896f1 100644
--- a/typo3/sysext/fluid/Tests/Functional/ViewHelpers/Link/TypolinkViewHelperTest.php
+++ b/typo3/sysext/fluid/Tests/Functional/ViewHelpers/Link/TypolinkViewHelperTest.php
@@ -26,6 +26,7 @@ use TYPO3\CMS\Core\Routing\PageArguments;
 use TYPO3\CMS\Core\Site\Entity\Site;
 use TYPO3\CMS\Core\Tests\Functional\SiteHandling\SiteBasedTestTrait;
 use TYPO3\CMS\Fluid\Core\Rendering\RenderingContextFactory;
+use TYPO3\CMS\Frontend\Typolink\TypolinkParameter;
 use TYPO3\TestingFramework\Core\Functional\Framework\Frontend\InternalRequest;
 use TYPO3\TestingFramework\Core\Functional\FunctionalTestCase;
 use TYPO3Fluid\Fluid\View\TemplateView;
@@ -69,6 +70,10 @@ final class TypolinkViewHelperTest extends FunctionalTestCase
     public static function renderDataProvider(): array
     {
         return [
+            'empty link' => [
+                '<f:link.typolink parameter="">This is a testlink</f:link.typolink>',
+                'This is a testlink',
+            ],
             'link: default' => [
                 '<f:link.typolink parameter="1">This is a testlink</f:link.typolink>',
                 '<a href="/en/">This is a testlink</a>',
@@ -177,6 +182,13 @@ EOT
     public static function renderWithAssignedParametersDataProvider(): array
     {
         return [
+            'empty parameter' => [
+                '<f:link.typolink parameter="{parameter}">Link text</f:link.typolink>',
+                [
+                    'parameter' => '',
+                ],
+                'Link text',
+            ],
             'target _self' => [
                 '<f:link.typolink parameter="{parameter}" parts-as="typoLinkParts">Individual {typoLinkParts.target} {typoLinkParts.class} {typoLinkParts.title}</f:link.typolink>',
                 [
@@ -213,6 +225,20 @@ EOT
                 ],
                 '<a href="http://typo3.org/" target="_self">_self</a>',
             ],
+            'typolinkParameter object' => [
+                '<f:link.typolink parameter="{parameter}" parts-as="typoLinkParts">{typoLinkParts.target}</f:link.typolink>{typoLinkParts.target}',
+                [
+                    'parameter' => new TypolinkParameter('http://typo3.org/', '_self'),
+                ],
+                '<a href="http://typo3.org/" target="_self">_self</a>',
+            ],
+            'invalid parameter' => [
+                '<f:link.typolink parameter="{parameter}" parts-as="typoLinkParts">{typoLinkParts.target}</f:link.typolink>{typoLinkParts.target}',
+                [
+                    'parameter' => new \stdClass(),
+                ],
+                '',
+            ],
         ];
     }
 
diff --git a/typo3/sysext/fluid/Tests/Functional/ViewHelpers/Uri/TypolinkViewHelperTest.php b/typo3/sysext/fluid/Tests/Functional/ViewHelpers/Uri/TypolinkViewHelperTest.php
index 57c3eeb3b8b9..a8eaf223a0ee 100644
--- a/typo3/sysext/fluid/Tests/Functional/ViewHelpers/Uri/TypolinkViewHelperTest.php
+++ b/typo3/sysext/fluid/Tests/Functional/ViewHelpers/Uri/TypolinkViewHelperTest.php
@@ -19,10 +19,17 @@ namespace TYPO3\CMS\Fluid\Tests\Functional\ViewHelpers\Uri;
 
 use PHPUnit\Framework\Attributes\DataProvider;
 use PHPUnit\Framework\Attributes\Test;
+use TYPO3\CMS\Core\Core\SystemEnvironmentBuilder;
 use TYPO3\CMS\Core\Database\ConnectionPool;
+use TYPO3\CMS\Core\Http\ServerRequest;
+use TYPO3\CMS\Core\Routing\PageArguments;
+use TYPO3\CMS\Core\Site\Entity\Site;
 use TYPO3\CMS\Core\Tests\Functional\SiteHandling\SiteBasedTestTrait;
+use TYPO3\CMS\Fluid\Core\Rendering\RenderingContextFactory;
+use TYPO3\CMS\Frontend\Typolink\TypolinkParameter;
 use TYPO3\TestingFramework\Core\Functional\Framework\Frontend\InternalRequest;
 use TYPO3\TestingFramework\Core\Functional\FunctionalTestCase;
+use TYPO3Fluid\Fluid\View\TemplateView;
 
 final class TypolinkViewHelperTest extends FunctionalTestCase
 {
@@ -46,14 +53,18 @@ final class TypolinkViewHelperTest extends FunctionalTestCase
     protected function setUp(): void
     {
         parent::setUp();
-        $this->importCSVDataSet(__DIR__ . '/../../Fixtures/pages.csv');
-        $this->writeSiteConfiguration(
-            'test',
-            $this->buildSiteConfiguration(1, '/'),
+        $request = new ServerRequest('http://localhost/');
+        $request = $request->withAttribute('applicationType', SystemEnvironmentBuilder::REQUESTTYPE_FE);
+        $request = $request->withAttribute('routing', new PageArguments(1, '0', []));
+        $request = $request->withAttribute('site', new Site(
+            'site',
+            1,
             [
-                $this->buildDefaultLanguageConfiguration('EN', '/en/'),
+                'base' => 'http://localhost/',
+                'languages' => [],
             ]
-        );
+        ));
+        $GLOBALS['TYPO3_REQUEST'] = $request;
     }
 
     public static function renderDataProvider(): array
@@ -114,6 +125,14 @@ final class TypolinkViewHelperTest extends FunctionalTestCase
     #[Test]
     public function render(string $template, string $expected): void
     {
+        $this->importCSVDataSet(__DIR__ . '/../../Fixtures/pages.csv');
+        $this->writeSiteConfiguration(
+            'test',
+            $this->buildSiteConfiguration(1, '/'),
+            [
+                $this->buildDefaultLanguageConfiguration('EN', '/en/'),
+            ]
+        );
         (new ConnectionPool())->getConnectionForTable('sys_template')->insert('sys_template', [
             'pid' => 1,
             'root' => 1,
@@ -135,4 +154,42 @@ EOT
         );
         self::assertStringContainsString($expected, (string)$response->getBody());
     }
+
+    public static function renderWithAssignedParametersDataProvider(): array
+    {
+        return [
+            'parameter' => [
+                '<f:uri.typolink parameter="{parameter}" />',
+                [
+                    'parameter' => 'http://typo3.org/',
+                ],
+                'http://typo3.org/',
+            ],
+            'typolinkParameter object' => [
+                '<f:uri.typolink parameter="{parameter}" />',
+                [
+                    'parameter' => new TypolinkParameter('http://typo3.org/'),
+                ],
+                'http://typo3.org/',
+            ],
+            'invalid parameter' => [
+                '<f:uri.typolink parameter="{parameter}" />',
+                [
+                    'parameter' => new \stdClass(),
+                ],
+                '',
+            ],
+        ];
+    }
+
+    #[DataProvider('renderWithAssignedParametersDataProvider')]
+    #[Test]
+    public function renderWithAssignedParameters(string $template, array $assigns, string $expected): void
+    {
+        $context = $this->get(RenderingContextFactory::class)->create();
+        $context->getTemplatePaths()->setTemplateSource($template);
+        $view = new TemplateView($context);
+        $view->assignMultiple($assigns);
+        self::assertSame($expected, trim($view->render()));
+    }
 }
diff --git a/typo3/sysext/frontend/Classes/Typolink/TypolinkParameter.php b/typo3/sysext/frontend/Classes/Typolink/TypolinkParameter.php
new file mode 100644
index 000000000000..cbb69e4d1fe4
--- /dev/null
+++ b/typo3/sysext/frontend/Classes/Typolink/TypolinkParameter.php
@@ -0,0 +1,68 @@
+<?php
+
+declare(strict_types=1);
+
+/*
+ * This file is part of the TYPO3 CMS project.
+ *
+ * It is free software; you can redistribute it and/or modify it under
+ * the terms of the GNU General Public License, either version 2
+ * of the License, or any later version.
+ *
+ * For the full copyright and license information, please read the
+ * LICENSE.txt file that was distributed with this source code.
+ *
+ * The TYPO3 project - inspiring people to share!
+ */
+
+namespace TYPO3\CMS\Frontend\Typolink;
+
+/**
+ * This class represents an object containing the resolved parameters of a typolink
+ */
+readonly class TypolinkParameter implements \JsonSerializable
+{
+    public function __construct(
+        public string $url = '',
+        public string $target = '',
+        public string $class = '',
+        public string $title = '',
+        public string $additionalParams = '',
+        public array $customParams = [],
+    ) {}
+
+    public static function createFromTypolinkParts(array $typoLinkParts): TypolinkParameter
+    {
+        $url = $typoLinkParts['url'] ?? '';
+        $target = $typoLinkParts['target'] ?? '';
+        $class = $typoLinkParts['class'] ?? '';
+        $title = $typoLinkParts['title'] ?? '';
+        $additionalParams = $typoLinkParts['additionalParams'] ?? '';
+        unset($typoLinkParts['url'], $typoLinkParts['target'], $typoLinkParts['class'], $typoLinkParts['title'], $typoLinkParts['additionalParams']);
+
+        return new self(
+            $url,
+            $target,
+            $class,
+            $title,
+            $additionalParams,
+            $typoLinkParts
+        );
+    }
+
+    public function toArray(): array
+    {
+        return array_merge([
+            'url' => $this->url,
+            'target' => $this->target,
+            'class' => $this->class,
+            'title' => $this->title,
+            'additionalParams' => $this->additionalParams,
+        ], $this->customParams);
+    }
+
+    public function jsonSerialize(): array
+    {
+        return $this->toArray();
+    }
+}
-- 
GitLab