From b86a48f5c14ab1747b9aa1581d7713d0aad011fe Mon Sep 17 00:00:00 2001 From: Guy Sartorelli <36352093+GuySartorelli@users.noreply.github.com> Date: Mon, 25 Nov 2024 10:28:34 +1300 Subject: [PATCH] FIX Ensure schema data includes attributes (#11469) --- src/Forms/FormField.php | 22 +++++++++++++++ src/Forms/SearchableDropdownTrait.php | 1 - src/Forms/TreeDropdownField.php | 2 -- .../SilverStripe/Forms/TreeDropdownField.ss | 2 +- tests/php/Forms/FormFieldTest.php | 10 +++++++ .../FormSchemaTest/testGetNestedSchema.json | 25 ++++++++++++----- .../Forms/FormSchemaTest/testGetSchema.json | 4 ++- .../FormSchemaTest/testSchemaValidation.json | 27 +++++++++++++++---- 8 files changed, 76 insertions(+), 17 deletions(-) diff --git a/src/Forms/FormField.php b/src/Forms/FormField.php index d5a08e39a95..fe953afba8d 100644 --- a/src/Forms/FormField.php +++ b/src/Forms/FormField.php @@ -1493,6 +1493,16 @@ public function getSchemaDataDefaults() if ($titleTip instanceof Tip) { $data['titleTip'] = $titleTip->getTipSchema(); } + $attributes = $this->getAttributes(); + // Remove value from attributes because otherwise it breaks react fields + unset($attributes['value']); + // Remove above attributes from attributes list to avoid double-ups + foreach (array_keys($data) as $key) { + // HTML attributes are always lowercase so we need to make sure to transform our js key names + // to lowercase before unsetting them. + unset($attributes[strtolower($key)]); + } + $data['attributes'] = $attributes; return $data; } @@ -1562,6 +1572,18 @@ public function getSchemaValidation() return $validationList; } + /** + * Gets the data-schema and data-state attributes for the input element. + * Can't be included in getAttributesHtml because that would result in + * an infinite loop. + */ + public function getSchemaAttributesHtml(): DBHTMLText + { + $content = 'data-schema="' . htmlspecialchars(json_encode($this->getSchemaData())) + . '" data-state="' . htmlspecialchars(json_encode($this->getSchemaState())) . '"'; + return DBHTMLText::create()->setValue($content); + } + /** * @return Tip */ diff --git a/src/Forms/SearchableDropdownTrait.php b/src/Forms/SearchableDropdownTrait.php index fa751c49a2f..56e80821c62 100644 --- a/src/Forms/SearchableDropdownTrait.php +++ b/src/Forms/SearchableDropdownTrait.php @@ -312,7 +312,6 @@ public function getAttributes(): array parent::getAttributes(), [ 'name' => $name, - 'data-schema' => json_encode($this->getSchemaData()), ] ); } diff --git a/src/Forms/TreeDropdownField.php b/src/Forms/TreeDropdownField.php index 5bf454f5513..7a8889dafca 100644 --- a/src/Forms/TreeDropdownField.php +++ b/src/Forms/TreeDropdownField.php @@ -620,8 +620,6 @@ public function getAttributes() $attributes = [ 'class' => $this->extraClass(), 'id' => $this->ID(), - 'data-schema' => json_encode($this->getSchemaData()), - 'data-state' => json_encode($this->getSchemaState()), ]; $attributes = array_merge($attributes, $this->attributes); diff --git a/templates/SilverStripe/Forms/TreeDropdownField.ss b/templates/SilverStripe/Forms/TreeDropdownField.ss index fad2a4e5f3c..c17e1582d2b 100644 --- a/templates/SilverStripe/Forms/TreeDropdownField.ss +++ b/templates/SilverStripe/Forms/TreeDropdownField.ss @@ -1,6 +1,6 @@
data-metadata="$Metadata.ATT"<% end_if %> > diff --git a/tests/php/Forms/FormFieldTest.php b/tests/php/Forms/FormFieldTest.php index 7c75ddb3162..5c54010b005 100644 --- a/tests/php/Forms/FormFieldTest.php +++ b/tests/php/Forms/FormFieldTest.php @@ -414,6 +414,16 @@ public function testGetSchemaDataDefaultsTitleTip() $this->assertSame('Test tip', $schema['titleTip']['content']); } + public function testGetSchemaDataDefaultsAttributes() + { + $field = new FormField('MyField'); + $field->setAttribute('foo', 'bar'); + $this->assertEquals('bar', $field->getAttribute('foo')); + $schema = $field->getSchemaDataDefaults(); + $this->assertArrayHasKey('foo', $schema['attributes']); + $this->assertEquals('bar', $schema['attributes']['foo']); + } + public function testGetSchemaData() { $field = new FormField('MyField'); diff --git a/tests/php/Forms/FormSchemaTest/testGetNestedSchema.json b/tests/php/Forms/FormSchemaTest/testGetNestedSchema.json index 8386f140c0b..016419da2e6 100644 --- a/tests/php/Forms/FormSchemaTest/testGetNestedSchema.json +++ b/tests/php/Forms/FormSchemaTest/testGetNestedSchema.json @@ -30,7 +30,9 @@ "disabled": false, "autoFocus": false, "customValidationMessage": "", - "attributes": [], + "attributes": { + "class": "text" + }, "data": { "maxlength": null }, @@ -53,7 +55,9 @@ "disabled": false, "autoFocus": false, "customValidationMessage": "", - "attributes": [], + "attributes": { + "class": "hidden" + }, "data": [], "validation": [] } @@ -77,7 +81,8 @@ "autoFocus": false, "customValidationMessage": "", "attributes": { - "type": "submit" + "type": "submit", + "class": "action" }, "data": { "icon": "save" @@ -102,7 +107,8 @@ "autoFocus": false, "customValidationMessage": "", "attributes": { - "type": "button" + "type": "button", + "class": "action" }, "data": { "icon": null @@ -126,7 +132,10 @@ "disabled": false, "autoFocus": false, "customValidationMessage": "", - "attributes": [], + "attributes": { + "class": "field CompositeField popover", + "tabindex": null + }, "data": { "popoverTitle": null, "placement": "bottom", @@ -159,7 +168,8 @@ "autoFocus": false, "customValidationMessage": "", "attributes": { - "type": "submit" + "type": "submit", + "class": "action" }, "data": { "icon": null @@ -184,7 +194,8 @@ "autoFocus": false, "customValidationMessage": "", "attributes": { - "type": "submit" + "type": "submit", + "class": "action" }, "data": { "icon": null diff --git a/tests/php/Forms/FormSchemaTest/testGetSchema.json b/tests/php/Forms/FormSchemaTest/testGetSchema.json index 57a2ef642bf..39a807e57b6 100644 --- a/tests/php/Forms/FormSchemaTest/testGetSchema.json +++ b/tests/php/Forms/FormSchemaTest/testGetSchema.json @@ -30,7 +30,9 @@ "disabled": false, "autoFocus": false, "customValidationMessage": "", - "attributes": [], + "attributes": { + "class": "hidden" + }, "data": [], "validation": [] } diff --git a/tests/php/Forms/FormSchemaTest/testSchemaValidation.json b/tests/php/Forms/FormSchemaTest/testSchemaValidation.json index e7f3bc3d0b0..1e551f61c41 100644 --- a/tests/php/Forms/FormSchemaTest/testSchemaValidation.json +++ b/tests/php/Forms/FormSchemaTest/testSchemaValidation.json @@ -36,7 +36,13 @@ "length": 40 } }, - "attributes": [], + "attributes": { + "class": "text", + "required": "required", + "aria-required": "true", + "maxLength": 40, + "size": 30 + }, "data": { "maxlength": 40 } @@ -62,7 +68,12 @@ "validation": { "date": true }, - "attributes": [], + "attributes": { + "class": "date text", + "lang": "en-US", + "min": null, + "max": null + }, "data": { "html5": true, "min": null, @@ -90,7 +101,9 @@ "validation": { "numeric": true }, - "attributes": [], + "attributes": { + "class": "numeric text" + }, "data": { "maxlength": null } @@ -115,7 +128,9 @@ "validation": { "currency": true }, - "attributes": [], + "attributes": { + "class": "currency text" + }, "data": { "maxlength": null } @@ -138,7 +153,9 @@ "autoFocus": false, "customValidationMessage": "", "validation": [], - "attributes": [], + "attributes": { + "class": "hidden" + }, "data": [] } ],