From bdd32acb3e19fc7bada8e5f04dc0544f0efdade0 Mon Sep 17 00:00:00 2001
From: Mathias Brodala <mbrodala@pagemachine.de>
Date: Thu, 21 Nov 2019 20:08:25 +0100
Subject: [PATCH] [BUGFIX] Convert date objects to string in finisher options
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Resolves: #88899
Releases: master
Change-Id: I775b73eed5b257a65e48f6cc824e0ce855a67f6a
Reviewed-on: https://review.typo3.org/c/Packages/TYPO3.CMS/+/62361
Tested-by: TYPO3com <noreply@typo3.com>
Tested-by: Björn Jacob <bjoern.jacob@tritum.de>
Tested-by: Ralf Zimmermann <ralf.zimmermann@tritum.de>
Reviewed-by: Björn Jacob <bjoern.jacob@tritum.de>
Reviewed-by: Ralf Zimmermann <ralf.zimmermann@tritum.de>
---
 .../Domain/Finishers/AbstractFinisher.php     | 18 +++++
 .../Domain/Model/FormElements/Date.php        | 47 +++++++++++++
 .../Domain/Model/FormElements/DatePicker.php  | 20 +++++-
 .../StringableFormElementInterface.php        | 28 ++++++++
 .../form/Classes/Hooks/FormElementHooks.php   | 16 -----
 .../RenderAllFormValuesViewHelper.php         | 26 ++-----
 .../Configuration/Yaml/FormElements/Date.yaml |  2 +-
 .../Yaml/FormElements/DatePicker.yaml         |  2 +-
 .../Domain/Finishers/AbstractFinisherTest.php | 70 +++++++++++++++++++
 typo3/sysext/form/ext_localconf.php           |  3 -
 10 files changed, 191 insertions(+), 41 deletions(-)
 create mode 100644 typo3/sysext/form/Classes/Domain/Model/FormElements/Date.php
 create mode 100644 typo3/sysext/form/Classes/Domain/Model/FormElements/StringableFormElementInterface.php

diff --git a/typo3/sysext/form/Classes/Domain/Finishers/AbstractFinisher.php b/typo3/sysext/form/Classes/Domain/Finishers/AbstractFinisher.php
index 2c14a976c68c..9ee1ac728d81 100644
--- a/typo3/sysext/form/Classes/Domain/Finishers/AbstractFinisher.php
+++ b/typo3/sysext/form/Classes/Domain/Finishers/AbstractFinisher.php
@@ -21,6 +21,7 @@ use TYPO3\CMS\Core\Utility\ArrayUtility;
 use TYPO3\CMS\Core\Utility\Exception\MissingArrayPathException;
 use TYPO3\CMS\Extbase\Reflection\ObjectAccess;
 use TYPO3\CMS\Form\Domain\Finishers\Exception\FinisherException;
+use TYPO3\CMS\Form\Domain\Model\FormElements\StringableFormElementInterface;
 use TYPO3\CMS\Form\Domain\Runtime\FormRuntime;
 use TYPO3\CMS\Form\Service\TranslationService;
 use TYPO3\CMS\Frontend\Controller\TypoScriptFrontendController;
@@ -342,8 +343,23 @@ abstract class AbstractFinisher implements FinisherInterface
         if ($property === '__currentTimestamp') {
             return time();
         }
+
         // try to resolve the path '{...}' within the FormRuntime
         $value = ObjectAccess::getPropertyPath($formRuntime, $property);
+
+        if (is_object($value)) {
+            $element = $formRuntime->getFormDefinition()->getElementByIdentifier($property);
+
+            if (!$element instanceof StringableFormElementInterface) {
+                throw new FinisherException(
+                    sprintf('Cannot convert object value of "%s" to string', $property),
+                    1574362327
+                );
+            }
+
+            $value = $element->valueToString($value);
+        }
+
         if ($value === null) {
             // try to resolve the path '{...}' within the FinisherVariableProvider
             $value = ObjectAccess::getPropertyPath(
@@ -351,9 +367,11 @@ abstract class AbstractFinisher implements FinisherInterface
                 $property
             );
         }
+
         if ($value !== null) {
             return $value;
         }
+
         // in case no value could be resolved
         return '{' . $property . '}';
     }
diff --git a/typo3/sysext/form/Classes/Domain/Model/FormElements/Date.php b/typo3/sysext/form/Classes/Domain/Model/FormElements/Date.php
new file mode 100644
index 000000000000..c4d9dcf99bf1
--- /dev/null
+++ b/typo3/sysext/form/Classes/Domain/Model/FormElements/Date.php
@@ -0,0 +1,47 @@
+<?php
+declare(strict_types = 1);
+namespace TYPO3\CMS\Form\Domain\Model\FormElements;
+
+/*
+ * This file is part of the TYPO3 CMS project.
+ *
+ * It originated from the Neos.Form package (www.neos.io)
+ *
+ * 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!
+ */
+
+/**
+ * A date form element
+ *
+ * Scope: frontend
+ */
+class Date extends AbstractFormElement implements StringableFormElementInterface
+{
+
+    /**
+     * Initializes the Form Element by setting the data type to "DateTime"
+     * @internal
+     */
+    public function initializeFormElement()
+    {
+        $this->setDataType('DateTime');
+        parent::initializeFormElement();
+    }
+
+    /**
+     * @param \DateTime $value
+     */
+    public function valueToString($value): string
+    {
+        $dateFormat = $this->properties['displayFormat'] ?? 'Y-m-d';
+
+        return $value->format($dateFormat);
+    }
+}
diff --git a/typo3/sysext/form/Classes/Domain/Model/FormElements/DatePicker.php b/typo3/sysext/form/Classes/Domain/Model/FormElements/DatePicker.php
index 7b30a887246f..14bc841b1da6 100644
--- a/typo3/sysext/form/Classes/Domain/Model/FormElements/DatePicker.php
+++ b/typo3/sysext/form/Classes/Domain/Model/FormElements/DatePicker.php
@@ -22,7 +22,7 @@ namespace TYPO3\CMS\Form\Domain\Model\FormElements;
  *
  * Scope: frontend
  */
-class DatePicker extends AbstractFormElement
+class DatePicker extends AbstractFormElement implements StringableFormElementInterface
 {
 
     /**
@@ -34,4 +34,22 @@ class DatePicker extends AbstractFormElement
         $this->setDataType('DateTime');
         parent::initializeFormElement();
     }
+
+    /**
+     * @param \DateTime $value
+     */
+    public function valueToString($value): string
+    {
+        $dateFormat = \DateTime::W3C;
+
+        if (isset($this->properties['dateFormat'])) {
+            $dateFormat = $this->properties['dateFormat'];
+
+            if ($this->properties['displayTimeSelector'] ?? false) {
+                $dateFormat .= ' H:i';
+            }
+        }
+
+        return $value->format($dateFormat);
+    }
 }
diff --git a/typo3/sysext/form/Classes/Domain/Model/FormElements/StringableFormElementInterface.php b/typo3/sysext/form/Classes/Domain/Model/FormElements/StringableFormElementInterface.php
new file mode 100644
index 000000000000..84aecc4d39c2
--- /dev/null
+++ b/typo3/sysext/form/Classes/Domain/Model/FormElements/StringableFormElementInterface.php
@@ -0,0 +1,28 @@
+<?php
+declare(strict_types = 1);
+namespace TYPO3\CMS\Form\Domain\Model\FormElements;
+
+/*
+ * This file is part of the TYPO3 CMS project.
+ *
+ * It originated from the Neos.Form package (www.neos.io)
+ *
+ * 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!
+ */
+
+/**
+ * Interface for form elements capable of converting their complex values to string
+ *
+ * @internal
+ */
+interface StringableFormElementInterface
+{
+    public function valueToString($value): string;
+}
diff --git a/typo3/sysext/form/Classes/Hooks/FormElementHooks.php b/typo3/sysext/form/Classes/Hooks/FormElementHooks.php
index 536b9bf831f6..052953aed21e 100644
--- a/typo3/sysext/form/Classes/Hooks/FormElementHooks.php
+++ b/typo3/sysext/form/Classes/Hooks/FormElementHooks.php
@@ -80,20 +80,4 @@ class FormElementHooks
             }
         }
     }
-
-    /**
-     * This hook is invoked whenever a form element is created.
-     * Note that this hook will be called **after** all properties from the
-     * prototype configuration are set in the form element but **before**
-     * the properties from the form definition are set in the form element.
-     *
-     * @param RenderableInterface $renderable
-     */
-    public function initializeFormElement(RenderableInterface $renderable)
-    {
-        if ($renderable->getType() === 'Date' || $renderable->getType() === 'DatePicker') {
-            // Set the property mapping type for the `Date` and `DatePicker` element.
-            $renderable->setDataType('DateTime');
-        }
-    }
 }
diff --git a/typo3/sysext/form/Classes/ViewHelpers/RenderAllFormValuesViewHelper.php b/typo3/sysext/form/Classes/ViewHelpers/RenderAllFormValuesViewHelper.php
index abaac177dbc0..c5f0d3f74050 100644
--- a/typo3/sysext/form/Classes/ViewHelpers/RenderAllFormValuesViewHelper.php
+++ b/typo3/sysext/form/Classes/ViewHelpers/RenderAllFormValuesViewHelper.php
@@ -20,6 +20,7 @@ namespace TYPO3\CMS\Form\ViewHelpers;
 use TYPO3\CMS\Core\Resource\File;
 use TYPO3\CMS\Extbase\Domain\Model\FileReference;
 use TYPO3\CMS\Form\Domain\Model\FormElements\FormElementInterface;
+use TYPO3\CMS\Form\Domain\Model\FormElements\StringableFormElementInterface;
 use TYPO3\CMS\Form\Domain\Model\Renderable\CompositeRenderableInterface;
 use TYPO3\CMS\Form\Domain\Model\Renderable\RenderableInterface;
 use TYPO3\CMS\Form\Domain\Model\Renderable\RootRenderableInterface;
@@ -184,26 +185,13 @@ class RenderAllFormValuesViewHelper extends AbstractViewHelper
     public static function processObject(FormElementInterface $element, $object): string
     {
         $properties = $element->getProperties();
-        if ($object instanceof \DateTime) {
-            if (
-                $element->getType() === 'DatePicker'
-                && isset($properties['dateFormat'])
-            ) {
-                $dateFormat = $properties['dateFormat'];
-                if (isset($properties['displayTimeSelector']) && $properties['displayTimeSelector'] === true) {
-                    $dateFormat .= ' H:i';
-                }
-            } elseif ($element->getType() === 'Date') {
-                if (isset($properties['displayFormat'])) {
-                    $dateFormat = $properties['displayFormat'];
-                } else {
-                    $dateFormat = 'Y-m-d';
-                }
-            } else {
-                $dateFormat = \DateTime::W3C;
-            }
 
-            return $object->format($dateFormat);
+        if ($element instanceof StringableFormElementInterface) {
+            return $element->valueToString($object);
+        }
+
+        if ($object instanceof \DateTime) {
+            return $object->format(\DateTime::W3C);
         }
 
         if ($object instanceof File || $object instanceof FileReference) {
diff --git a/typo3/sysext/form/Configuration/Yaml/FormElements/Date.yaml b/typo3/sysext/form/Configuration/Yaml/FormElements/Date.yaml
index b1020ed44d00..78b52375ecd8 100644
--- a/typo3/sysext/form/Configuration/Yaml/FormElements/Date.yaml
+++ b/typo3/sysext/form/Configuration/Yaml/FormElements/Date.yaml
@@ -317,7 +317,7 @@ TYPO3:
                 group: html5
                 groupSorting: 500
                 iconIdentifier: form-date-picker
-              implementationClassName: TYPO3\CMS\Form\Domain\Model\FormElements\GenericFormElement
+              implementationClassName: TYPO3\CMS\Form\Domain\Model\FormElements\Date
               properties:
                 containerClassAttribute: input
                 elementClassAttribute: ''
diff --git a/typo3/sysext/form/Configuration/Yaml/FormElements/DatePicker.yaml b/typo3/sysext/form/Configuration/Yaml/FormElements/DatePicker.yaml
index 0ea3cba9fe49..7c552f21b70d 100644
--- a/typo3/sysext/form/Configuration/Yaml/FormElements/DatePicker.yaml
+++ b/typo3/sysext/form/Configuration/Yaml/FormElements/DatePicker.yaml
@@ -123,7 +123,7 @@ TYPO3:
                         9999:
                           identifier: removeButton
                           templateName: Inspector-RemoveElementEditor
-              implementationClassName: TYPO3\CMS\Form\Domain\Model\FormElements\GenericFormElement
+              implementationClassName: TYPO3\CMS\Form\Domain\Model\FormElements\DatePicker
               properties:
                 containerClassAttribute: input
                 elementClassAttribute: 'small form-control'
diff --git a/typo3/sysext/form/Tests/Unit/Domain/Finishers/AbstractFinisherTest.php b/typo3/sysext/form/Tests/Unit/Domain/Finishers/AbstractFinisherTest.php
index fabd9d9c37d7..4c2b7449e70a 100644
--- a/typo3/sysext/form/Tests/Unit/Domain/Finishers/AbstractFinisherTest.php
+++ b/typo3/sysext/form/Tests/Unit/Domain/Finishers/AbstractFinisherTest.php
@@ -21,6 +21,8 @@ use TYPO3\CMS\Form\Domain\Finishers\AbstractFinisher;
 use TYPO3\CMS\Form\Domain\Finishers\Exception\FinisherException;
 use TYPO3\CMS\Form\Domain\Finishers\FinisherContext;
 use TYPO3\CMS\Form\Domain\Finishers\FinisherVariableProvider;
+use TYPO3\CMS\Form\Domain\Model\FormDefinition;
+use TYPO3\CMS\Form\Domain\Model\FormElements\StringableFormElementInterface;
 use TYPO3\CMS\Form\Domain\Runtime\FormRuntime;
 use TYPO3\TestingFramework\Core\Unit\UnitTestCase;
 
@@ -480,6 +482,74 @@ class AbstractFinisherTest extends UnitTestCase
         self::assertSame($expected, $mockAbstractFinisher->_call('substituteRuntimeReferences', $input, $formRuntimeProphecy->reveal()));
     }
 
+    /**
+     * @test
+     */
+    public function substituteRuntimeReferencesConvertsObjectsToString(): void
+    {
+        $date = new \DateTime('@1574415600');
+        $formRuntimeProphecy = $this->createFormRuntimeProphecy([
+            'date-1' => $date,
+        ]);
+
+        $stringableElement = new class implements StringableFormElementInterface {
+            /**
+             * @param \DateTimeInterface $value
+             */
+            public function valueToString($value): string
+            {
+                return $value->format('Y-m-d');
+            }
+        };
+        $formDefinitionProphecy = $this->prophesize(FormDefinition::class);
+        $formDefinitionProphecy->getElementByIdentifier('date-1')->willReturn($stringableElement);
+        $formRuntimeProphecy->getFormDefinition()->willReturn($formDefinitionProphecy->reveal());
+
+        $mockAbstractFinisher = $this->getAccessibleMockForAbstractClass(
+            AbstractFinisher::class,
+            [],
+            '',
+            false
+        );
+        $result = $mockAbstractFinisher->_call(
+            'substituteRuntimeReferences',
+            'When: {date-1}',
+            $formRuntimeProphecy->reveal()
+        );
+
+        self::assertSame('When: 2019-11-22', $result);
+    }
+
+    /**
+     * @test
+     */
+    public function substituteRuntimeReferencesThrowsExceptionOnObjectWithoutStringableElement(): void
+    {
+        $formRuntimeProphecy = $this->createFormRuntimeProphecy([
+            'date-1' => new \DateTime(),
+        ]);
+
+        $formDefinitionProphecy = $this->prophesize(FormDefinition::class);
+        $formDefinitionProphecy->getElementByIdentifier('date-1')->willReturn($this->prophesize(FormElementInterface::class)->reveal());
+        $formRuntimeProphecy->getFormDefinition()->willReturn($formDefinitionProphecy->reveal());
+
+        $mockAbstractFinisher = $this->getAccessibleMockForAbstractClass(
+            AbstractFinisher::class,
+            [],
+            '',
+            false
+        );
+
+        $this->expectException(FinisherException::class);
+        $this->expectExceptionCode(1574362327);
+
+        $mockAbstractFinisher->_call(
+            'substituteRuntimeReferences',
+            'When: {date-1}',
+            $formRuntimeProphecy->reveal()
+        );
+    }
+
     /**
      * @test
      */
diff --git a/typo3/sysext/form/ext_localconf.php b/typo3/sysext/form/ext_localconf.php
index 43c8e5c5ebc3..b9cda2f07234 100644
--- a/typo3/sysext/form/ext_localconf.php
+++ b/typo3/sysext/form/ext_localconf.php
@@ -58,9 +58,6 @@ call_user_func(function () {
     $GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/form']['afterSubmit'][1489772699]
         = \TYPO3\CMS\Form\Hooks\FormElementHooks::class;
 
-    $GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/form']['initializeFormElement'][1489772699]
-        = \TYPO3\CMS\Form\Hooks\FormElementHooks::class;
-
     $GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/form']['beforeRendering'][1489772699]
         = \TYPO3\CMS\Form\Hooks\FormElementHooks::class;
 
-- 
GitLab