From 7b91207c12405cb42f2f7658fbf1db72be1cc2d6 Mon Sep 17 00:00:00 2001 From: Guy Sartorelli <36352093+GuySartorelli@users.noreply.github.com> Date: Fri, 9 Aug 2024 09:27:38 +1200 Subject: [PATCH 1/3] FIX Don't error if template global is null (#11331) --- src/View/SSViewer_Scope.php | 3 +++ tests/php/View/SSViewerTest.php | 6 ++++++ tests/php/View/SSViewerTest/TestGlobalProvider.php | 9 +++++++-- 3 files changed, 16 insertions(+), 2 deletions(-) diff --git a/src/View/SSViewer_Scope.php b/src/View/SSViewer_Scope.php index f82b84062d8..3733dd3af5e 100644 --- a/src/View/SSViewer_Scope.php +++ b/src/View/SSViewer_Scope.php @@ -177,6 +177,9 @@ public function resetLocalScope() public function getObj($name, $arguments = [], $cache = false, $cacheName = null) { $on = $this->itemIterator ? $this->itemIterator->current() : $this->item; + if ($on === null) { + return null; + } return $on->obj($name, $arguments, $cache, $cacheName); } diff --git a/tests/php/View/SSViewerTest.php b/tests/php/View/SSViewerTest.php index ce91527f3d7..2ce97e2b97e 100644 --- a/tests/php/View/SSViewerTest.php +++ b/tests/php/View/SSViewerTest.php @@ -364,6 +364,12 @@ public function testGlobalVariablesAreEscaped() ); } + public function testGlobalVariablesReturnNull() + { + $this->assertEquals('

', $this->render('

$SSViewerTest_GlobalReturnsNull

')); + $this->assertEquals('

', $this->render('

$SSViewerTest_GlobalReturnsNull.Chained.Properties

')); + } + public function testCoreGlobalVariableCalls() { $this->assertEquals( diff --git a/tests/php/View/SSViewerTest/TestGlobalProvider.php b/tests/php/View/SSViewerTest/TestGlobalProvider.php index 9fccbb0c064..f005f24ce1b 100644 --- a/tests/php/View/SSViewerTest/TestGlobalProvider.php +++ b/tests/php/View/SSViewerTest/TestGlobalProvider.php @@ -18,8 +18,8 @@ public static function get_template_global_variables() 'SSViewerTest_GlobalReferencedByString' => 'get_reference', 'SSViewerTest_GlobalReferencedInArray' => ['method' => 'get_reference'], - 'SSViewerTest_GlobalThatTakesArguments' => ['method' => 'get_argmix', 'casting' => 'HTMLFragment'] - + 'SSViewerTest_GlobalThatTakesArguments' => ['method' => 'get_argmix', 'casting' => 'HTMLFragment'], + 'SSViewerTest_GlobalReturnsNull' => 'getNull', ]; } @@ -43,4 +43,9 @@ public static function get_argmix() $args = func_get_args(); return 'z' . implode(':', $args) . 'z'; } + + public static function getNull() + { + return null; + } } From eee7a84a480b028a3d925bc7483cae9c45e9d252 Mon Sep 17 00:00:00 2001 From: Guy Sartorelli <36352093+GuySartorelli@users.noreply.github.com> Date: Mon, 12 Aug 2024 09:33:55 +1200 Subject: [PATCH 2/3] NEW Add new method TabSet::changeTabOrder(). (#11329) --- src/Forms/TabSet.php | 33 ++++++++++++++++++++++++++++++ tests/php/Forms/TabSetTest.php | 37 ++++++++++++++++++++++++++++++++++ 2 files changed, 70 insertions(+) create mode 100644 tests/php/Forms/TabSetTest.php diff --git a/src/Forms/TabSet.php b/src/Forms/TabSet.php index c397b62a25d..47037193e7a 100644 --- a/src/Forms/TabSet.php +++ b/src/Forms/TabSet.php @@ -249,6 +249,39 @@ public function insertAfter($insertAfter, $field, $appendIfMissing = true) return parent::insertAfter($insertAfter, $field, $appendIfMissing); } + /** + * Change the order of tabs which are direct children of this TabSet by specifying an ordered list of + * tab names. + * + * This works well in conjunction with SilverStripe's scaffolding functions: take the scaffold, and + * shuffle the tabs around to the order that you want. + * + * Tab names should exclude prefixes. For example if this TabSet is "Root", include "Main" not "Root.Main" + */ + public function changeTabOrder(array $tabNames): static + { + // Build a map of tabs indexed by their name. This will make the 2nd step much easier. + $existingTabs = []; + foreach ($this->children as $tab) { + $existingTabs[$tab->getName()] = $tab; + } + + // Iterate through the ordered list of names, building a new array. + // While we're doing this, empty out $existingTabs so that we can keep track of leftovers. + // Unrecognised field names are okay; just ignore them. + $orderedTabs = []; + foreach ($tabNames as $tabName) { + if (isset($existingTabs[$tabName])) { + $orderedTabs[] = $existingTabs[$tabName]; + unset($existingTabs[$tabName]); + } + } + + // Add the leftover fields to the end of the ordered list. + $this->setTabs(FieldList::create([...$orderedTabs, ...$existingTabs])); + return $this; + } + /** * Sets an additional default for $schemaData. * The existing keys are immutable. HideNav is added in this overriding method to ensure it is not ignored by diff --git a/tests/php/Forms/TabSetTest.php b/tests/php/Forms/TabSetTest.php new file mode 100644 index 00000000000..35e9cb1a90c --- /dev/null +++ b/tests/php/Forms/TabSetTest.php @@ -0,0 +1,37 @@ +findOrMakeTab('Root.Main'); + $fieldList->findOrMakeTab('Root.Next'); + $fieldList->findOrMakeTab('Root.More'); + $fieldList->findOrMakeTab('Root.Extra'); + $fieldList->addFieldToTab('Root', new TabSet('SubTabSet')); + $fieldList->findOrMakeTab('Root.SubTabSet.Another'); + + // Reorder tabs - intentionally leaving some alone, which will be added to the end. + $tabSet->changeTabOrder([ + 'SubTabSet', + 'More', + 'Main', + 'Non-Existent', // will be ignored + 'Another', // will be ignored + ]); + // Order is correct + $this->assertSame(['SubTabSet', 'More', 'Main', 'Next', 'Extra'], $tabSet->getChildren()->column('Name')); + // Sub-tab is still there + $this->assertNotNull($fieldList->findTab('Root.SubTabSet.Another')); + } +} From dca62c7380724fc44c8f390f4d129834c3ec1111 Mon Sep 17 00:00:00 2001 From: Guy Sartorelli <36352093+GuySartorelli@users.noreply.github.com> Date: Mon, 12 Aug 2024 12:52:57 +1200 Subject: [PATCH 3/3] NEW Make CMSFields scaffolding configurable, plus new options (#11328) Note that includeRelations was intentionally changed to not include has_one in 524d7a90117d447bc7705066500b1637aa75f0e0 --- src/Forms/FieldList.php | 3 + src/Forms/FormScaffolder.php | 54 ++++++++- src/ORM/DataObject.php | 29 ++++- tests/php/Forms/FieldListTest.php | 2 + tests/php/Forms/FormScaffolderTest.php | 161 +++++++++++++++++++++++-- 5 files changed, 228 insertions(+), 21 deletions(-) diff --git a/src/Forms/FieldList.php b/src/Forms/FieldList.php index 85143cf1390..39713759e43 100644 --- a/src/Forms/FieldList.php +++ b/src/Forms/FieldList.php @@ -407,6 +407,9 @@ public function findTab($tabName) $currentPointer = $this; foreach ($parts as $k => $part) { + if ($currentPointer === null) { + return null; + } $currentPointer = $currentPointer->fieldByName($part); } diff --git a/src/Forms/FormScaffolder.php b/src/Forms/FormScaffolder.php index 6ef58cf9021..87f59b85933 100644 --- a/src/Forms/FormScaffolder.php +++ b/src/Forms/FormScaffolder.php @@ -28,8 +28,15 @@ class FormScaffolder */ public $tabbed = false; + /** + * Only set up the "Root.Main" tab, but skip scaffolding actual FormFields. + * If $tabbed is false, an empty FieldList will be returned. + */ + public bool $mainTabOnly = false; + /** * @var boolean $ajaxSafe + * @deprecated 5.3.0 Will be removed without equivalent functionality. */ public $ajaxSafe = false; @@ -39,6 +46,11 @@ class FormScaffolder */ public $restrictFields; + /** + * Numeric array of field names and has_one relations to explicitly not scaffold. + */ + public array $ignoreFields = []; + /** * @var array $fieldClasses Optional mapping of fieldnames to subclasses of {@link FormField}. * By default the scaffolder will determine the field instance by {@link DBField::scaffoldFormField()}. @@ -46,10 +58,21 @@ class FormScaffolder public $fieldClasses; /** - * @var boolean $includeRelations Include has_one, has_many and many_many relations + * @var boolean $includeRelations Include has_many and many_many relations */ public $includeRelations = false; + /** + * Array of relation names to use as an allow list. + * If left blank, all has_many and many_many relations will be scaffolded unless explicitly ignored. + */ + public array $restrictRelations = []; + + /** + * Numeric array of has_many and many_many relations to explicitly not scaffold. + */ + public array $ignoreRelations = []; + /** * @param DataObject $obj */ @@ -76,12 +99,20 @@ public function getFieldList() $mainTab->setTitle(_t(__CLASS__ . '.TABMAIN', 'Main')); } + if ($this->mainTabOnly) { + return $fields; + } + // Add logical fields directly specified in db config foreach ($this->obj->config()->get('db') as $fieldName => $fieldType) { - // Skip restricted fields + // Skip fields that aren't in the allow list if ($this->restrictFields && !in_array($fieldName, $this->restrictFields ?? [])) { continue; } + // Skip ignored fields + if (in_array($fieldName, $this->ignoreFields)) { + continue; + } if ($this->fieldClasses && isset($this->fieldClasses[$fieldName])) { $fieldClass = $this->fieldClasses[$fieldName]; @@ -110,6 +141,9 @@ public function getFieldList() if ($this->restrictFields && !in_array($relationship, $this->restrictFields ?? [])) { continue; } + if (in_array($relationship, $this->ignoreFields)) { + continue; + } $fieldName = $component === 'SilverStripe\\ORM\\DataObject' ? $relationship // Polymorphic has_one field is composite, so don't refer to ID subfield : "{$relationship}ID"; @@ -138,6 +172,12 @@ public function getFieldList() && ($this->includeRelations === true || isset($this->includeRelations['has_many'])) ) { foreach ($this->obj->hasMany() as $relationship => $component) { + if (!empty($this->restrictRelations) && !in_array($relationship, $this->restrictRelations)) { + continue; + } + if (in_array($relationship, $this->ignoreRelations)) { + continue; + } $includeInOwnTab = true; $fieldLabel = $this->obj->fieldLabel($relationship); $fieldClass = (isset($this->fieldClasses[$relationship])) @@ -177,6 +217,12 @@ public function getFieldList() && ($this->includeRelations === true || isset($this->includeRelations['many_many'])) ) { foreach ($this->obj->manyMany() as $relationship => $component) { + if (!empty($this->restrictRelations) && !in_array($relationship, $this->restrictRelations)) { + continue; + } + if (in_array($relationship, $this->ignoreRelations)) { + continue; + } static::addManyManyRelationshipFields( $fields, $relationship, @@ -252,8 +298,12 @@ protected function getParamsArray() { return [ 'tabbed' => $this->tabbed, + 'mainTabOnly' => $this->mainTabOnly, 'includeRelations' => $this->includeRelations, + 'restrictRelations' => $this->restrictRelations, + 'ignoreRelations' => $this->ignoreRelations, 'restrictFields' => $this->restrictFields, + 'ignoreFields' => $this->ignoreFields, 'fieldClasses' => $this->fieldClasses, 'ajaxSafe' => $this->ajaxSafe ]; diff --git a/src/ORM/DataObject.php b/src/ORM/DataObject.php index aa687ae2482..c496f631d9f 100644 --- a/src/ORM/DataObject.php +++ b/src/ORM/DataObject.php @@ -289,6 +289,15 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity */ private static $table_name = null; + /** + * Settings used by the FormScaffolder that scaffolds fields for getCMSFields() + */ + private static array $scaffold_cms_fields_settings = [ + 'includeRelations' => true, + 'tabbed' => true, + 'ajaxSafe' => true, + ]; + /** * Non-static relationship cache, indexed by component name. * @@ -2469,8 +2478,12 @@ public function scaffoldFormFields($_params = null) $params = array_merge( [ 'tabbed' => false, + 'mainTabOnly' => false, 'includeRelations' => false, + 'restrictRelations' => [], + 'ignoreRelations' => [], 'restrictFields' => false, + 'ignoreFields' => [], 'fieldClasses' => false, 'ajaxSafe' => false ], @@ -2479,8 +2492,12 @@ public function scaffoldFormFields($_params = null) $fs = FormScaffolder::create($this); $fs->tabbed = $params['tabbed']; + $fs->mainTabOnly = $params['mainTabOnly']; $fs->includeRelations = $params['includeRelations']; + $fs->restrictRelations = $params['restrictRelations']; + $fs->ignoreRelations = $params['ignoreRelations']; $fs->restrictFields = $params['restrictFields']; + $fs->ignoreFields = $params['ignoreFields']; $fs->fieldClasses = $params['fieldClasses']; $fs->ajaxSafe = $params['ajaxSafe']; @@ -2600,12 +2617,12 @@ protected function afterUpdateCMSFields(callable $callback) */ public function getCMSFields() { - $tabbedFields = $this->scaffoldFormFields([ - // Don't allow has_many/many_many relationship editing before the record is first saved - 'includeRelations' => ($this->ID > 0), - 'tabbed' => true, - 'ajaxSafe' => true - ]); + $scaffoldOptions = static::config()->get('scaffold_cms_fields_settings'); + // Don't allow has_many/many_many relationship editing before the record is first saved + if (!$this->isInDB()) { + $scaffoldOptions['includeRelations'] = false; + } + $tabbedFields = $this->scaffoldFormFields($scaffoldOptions); $this->extend('updateCMSFields', $tabbedFields); diff --git a/tests/php/Forms/FieldListTest.php b/tests/php/Forms/FieldListTest.php index ec235dfeef3..2847151fa8d 100644 --- a/tests/php/Forms/FieldListTest.php +++ b/tests/php/Forms/FieldListTest.php @@ -259,6 +259,8 @@ public function testFindTab() $this->assertNull($fields->findTab('More')); $this->assertEquals($fields->findTab('Root.More'), $more); $this->assertEquals($fields->findTab('Root.More.Tab4'), $tab4); + + $this->assertNull($fields->findTab('This.Doesnt.Exist')); } /** diff --git a/tests/php/Forms/FormScaffolderTest.php b/tests/php/Forms/FormScaffolderTest.php index 48a02a1cd95..9116bc82de5 100644 --- a/tests/php/Forms/FormScaffolderTest.php +++ b/tests/php/Forms/FormScaffolderTest.php @@ -173,21 +173,40 @@ public function testGetFormFields() public function provideScaffoldRelationFormFields() { - return [ - [true], - [false], + $scenarios = [ + 'ignore no relations' => [ + 'includeInOwnTab' => true, + 'ignoreRelations' => [], + ], + 'ignore some relations' => [ + 'includeInOwnTab' => true, + 'ignoreRelations' => [ + 'ChildrenHasMany', + 'ChildrenManyManyThrough', + ], + ], ]; + foreach ($scenarios as $name => $scenario) { + $scenario['includeInOwnTab'] = false; + $scenarios[$name . ' - not in own tab'] = $scenario; + } + return $scenarios; } /** * @dataProvider provideScaffoldRelationFormFields */ - public function testScaffoldRelationFormFields(bool $includeInOwnTab) + public function testScaffoldRelationFormFields(bool $includeInOwnTab, array $ignoreRelations) { $parent = $this->objFromFixture(ParentModel::class, 'parent1'); Child::$includeInOwnTab = $includeInOwnTab; - $fields = $parent->scaffoldFormFields(['includeRelations' => true, 'tabbed' => true]); + $fields = $parent->scaffoldFormFields([ + 'includeRelations' => true, + 'tabbed' => true, + 'ignoreRelations' => $ignoreRelations, + ]); + // has_one foreach (array_keys(ParentModel::config()->uninherited('has_one')) as $hasOneName) { $scaffoldedFormField = $fields->dataFieldByName($hasOneName . 'ID'); if ($hasOneName === 'ChildPolymorphic') { @@ -196,20 +215,136 @@ public function testScaffoldRelationFormFields(bool $includeInOwnTab) $this->assertInstanceOf(DateField::class, $scaffoldedFormField, "$hasOneName should be a DateField"); } } + // has_many foreach (array_keys(ParentModel::config()->uninherited('has_many')) as $hasManyName) { - $this->assertInstanceOf(CurrencyField::class, $fields->dataFieldByName($hasManyName), "$hasManyName should be a CurrencyField"); - if ($includeInOwnTab) { - $this->assertNotNull($fields->findTab("Root.$hasManyName")); + if (in_array($hasManyName, $ignoreRelations)) { + $this->assertNull($fields->dataFieldByName($hasManyName)); } else { - $this->assertNull($fields->findTab("Root.$hasManyName")); + $this->assertInstanceOf(CurrencyField::class, $fields->dataFieldByName($hasManyName), "$hasManyName should be a CurrencyField"); + if ($includeInOwnTab) { + $this->assertNotNull($fields->findTab("Root.$hasManyName")); + } else { + $this->assertNull($fields->findTab("Root.$hasManyName")); + } } } + // many_many foreach (array_keys(ParentModel::config()->uninherited('many_many')) as $manyManyName) { - $this->assertInstanceOf(TimeField::class, $fields->dataFieldByName($manyManyName), "$manyManyName should be a TimeField"); - if ($includeInOwnTab) { - $this->assertNotNull($fields->findTab("Root.$manyManyName")); + if (in_array($hasManyName, $ignoreRelations)) { + $this->assertNull($fields->dataFieldByName($hasManyName)); + } else { + $this->assertInstanceOf(TimeField::class, $fields->dataFieldByName($manyManyName), "$manyManyName should be a TimeField"); + if ($includeInOwnTab) { + $this->assertNotNull($fields->findTab("Root.$manyManyName")); + } else { + $this->assertNull($fields->findTab("Root.$manyManyName")); + } + } + } + } + + public function testScaffoldIgnoreFields(): void + { + $article1 = $this->objFromFixture(Article::class, 'article1'); + $fields = $article1->scaffoldFormFields([ + 'ignoreFields' => [ + 'Content', + 'Author', + ], + ]); + $this->assertSame(['ExtendedField', 'Title'], $fields->column('Name')); + } + + public function testScaffoldRestrictRelations(): void + { + $article1 = $this->objFromFixture(Article::class, 'article1'); + $fields = $article1->scaffoldFormFields([ + 'includeRelations' => true, + 'restrictRelations' => [ + 'Tags', + ], + // Ensure no db or has_one fields get scaffolded + 'restrictFields' => [ + 'non-existent', + ], + ]); + $this->assertSame(['Tags'], $fields->column('Name')); + } + + public function provideTabs(): array + { + return [ + 'only main tab' => [ + 'tabs' => true, + 'mainTabOnly' => true, + ], + 'all tabs, all fields' => [ + 'tabs' => true, + 'mainTabOnly' => false, + ], + 'no tabs, no fields' => [ + 'tabs' => false, + 'mainTabOnly' => true, + ], + 'no tabs, all fields' => [ + 'tabs' => false, + 'mainTabOnly' => false, + ], + ]; + } + + /** + * @dataProvider provideTabs + */ + public function testTabs(bool $tabbed, bool $mainTabOnly): void + { + $parent = $this->objFromFixture(ParentModel::class, 'parent1'); + Child::$includeInOwnTab = true; + $fields = $parent->scaffoldFormFields([ + 'tabbed' => $tabbed, + 'mainTabOnly' => $mainTabOnly, + 'includeRelations' => true, + ]); + + $fieldsToExpect = [ + ['Name' => 'Title'], + ['Name' => 'ChildID'], + ['Name' => 'ChildrenHasMany'], + ['Name' => 'ChildrenManyMany'], + ['Name' => 'ChildrenManyManyThrough'], + ]; + $relationTabs = [ + 'Root.ChildrenHasMany', + 'Root.ChildrenManyMany', + 'Root.ChildrenManyManyThrough', + ]; + + if ($tabbed) { + $this->assertNotNull($fields->findTab('Root.Main')); + if ($mainTabOnly) { + // Only Root.Main with no fields + $this->assertListNotContains($fieldsToExpect, $fields->flattenFields()); + foreach ($relationTabs as $tabName) { + $this->assertNull($fields->findTab($tabName)); + } + } else { + // All fields in all tabs + $this->assertListContains($fieldsToExpect, $fields->flattenFields()); + foreach ($relationTabs as $tabName) { + $this->assertNotNull($fields->findTab($tabName)); + } + } + } else { + if ($mainTabOnly) { + // Empty list + $this->assertEmpty($fields); } else { - $this->assertNull($fields->findTab("Root.$manyManyName")); + // All fields, no tabs + $this->assertNull($fields->findTab('Root.Main')); + foreach ($relationTabs as $tabName) { + $this->assertNull($fields->findTab($tabName)); + } + $this->assertListContains($fieldsToExpect, $fields->flattenFields()); } } }