2008-10-14 00:20:41 +02:00
|
|
|
<?php
|
|
|
|
|
2016-10-14 03:30:05 +02:00
|
|
|
namespace SilverStripe\Forms\Tests;
|
|
|
|
|
|
|
|
use SilverStripe\Forms\HTMLEditor\HTMLEditorField;
|
2016-08-19 00:51:35 +02:00
|
|
|
use SilverStripe\Dev\SapphireTest;
|
|
|
|
use SilverStripe\Control\Controller;
|
2024-06-07 07:06:01 +02:00
|
|
|
use SilverStripe\Forms\CurrencyField;
|
|
|
|
use SilverStripe\Forms\DateField;
|
2016-08-19 00:51:35 +02:00
|
|
|
use SilverStripe\Forms\FieldList;
|
|
|
|
use SilverStripe\Forms\Form;
|
2016-10-14 03:30:05 +02:00
|
|
|
use SilverStripe\Forms\Tests\FormScaffolderTest\Article;
|
|
|
|
use SilverStripe\Forms\Tests\FormScaffolderTest\ArticleExtension;
|
|
|
|
use SilverStripe\Forms\Tests\FormScaffolderTest\Author;
|
2024-06-07 07:06:01 +02:00
|
|
|
use SilverStripe\Forms\Tests\FormScaffolderTest\Child;
|
|
|
|
use SilverStripe\Forms\Tests\FormScaffolderTest\ParentModel;
|
|
|
|
use SilverStripe\Forms\Tests\FormScaffolderTest\ParentChildJoin;
|
2016-10-14 03:30:05 +02:00
|
|
|
use SilverStripe\Forms\Tests\FormScaffolderTest\Tag;
|
2024-06-07 07:06:01 +02:00
|
|
|
use SilverStripe\Forms\TimeField;
|
2024-09-18 03:53:44 +02:00
|
|
|
use PHPUnit\Framework\Attributes\DataProvider;
|
2016-06-15 06:03:16 +02:00
|
|
|
|
2008-10-14 00:20:41 +02:00
|
|
|
/**
|
|
|
|
* Tests for DataObject FormField scaffolding
|
|
|
|
*/
|
2016-12-16 05:34:21 +01:00
|
|
|
class FormScaffolderTest extends SapphireTest
|
|
|
|
{
|
|
|
|
|
|
|
|
protected static $fixture_file = 'FormScaffolderTest.yml';
|
|
|
|
|
2017-03-24 12:17:26 +01:00
|
|
|
protected static $required_extensions = [
|
2016-12-16 05:34:21 +01:00
|
|
|
Article::class => [
|
|
|
|
ArticleExtension::class
|
|
|
|
]
|
|
|
|
];
|
|
|
|
|
2020-04-20 19:58:09 +02:00
|
|
|
protected static $extra_dataobjects = [
|
2016-12-16 05:34:21 +01:00
|
|
|
Article::class,
|
|
|
|
Tag::class,
|
|
|
|
Author::class,
|
2024-06-07 07:06:01 +02:00
|
|
|
ParentModel::class,
|
|
|
|
Child::class,
|
|
|
|
ParentChildJoin::class,
|
2020-04-20 19:58:09 +02:00
|
|
|
];
|
2016-12-16 05:34:21 +01:00
|
|
|
|
|
|
|
public function testGetCMSFieldsSingleton()
|
|
|
|
{
|
|
|
|
$article = new Article;
|
|
|
|
$fields = $article->getCMSFields();
|
2017-03-02 03:24:38 +01:00
|
|
|
$form = new Form(null, 'TestForm', $fields, new FieldList());
|
2016-12-16 05:34:21 +01:00
|
|
|
$form->loadDataFrom($article);
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
$fields->hasTabSet(),
|
|
|
|
'getCMSFields() produces a TabSet'
|
|
|
|
);
|
|
|
|
$this->assertNotNull(
|
|
|
|
$fields->dataFieldByName('Title'),
|
|
|
|
'getCMSFields() includes db fields'
|
|
|
|
);
|
|
|
|
$this->assertNotNull(
|
|
|
|
$fields->dataFieldByName('Content'),
|
|
|
|
'getCMSFields() includes db fields'
|
|
|
|
);
|
|
|
|
$this->assertNotNull(
|
|
|
|
$fields->dataFieldByName('AuthorID'),
|
|
|
|
'getCMSFields() includes has_one fields on singletons'
|
|
|
|
);
|
|
|
|
$this->assertNull(
|
|
|
|
$fields->dataFieldByName('Tags'),
|
|
|
|
"getCMSFields() doesn't include many_many fields if no ID is present"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetCMSFieldsInstance()
|
|
|
|
{
|
|
|
|
$article1 = $this->objFromFixture(Article::class, 'article1');
|
|
|
|
|
|
|
|
$fields = $article1->getCMSFields();
|
2017-03-02 03:24:38 +01:00
|
|
|
$form = new Form(null, 'TestForm', $fields, new FieldList());
|
2016-12-16 05:34:21 +01:00
|
|
|
$form->loadDataFrom($article1);
|
|
|
|
|
|
|
|
$this->assertNotNull(
|
|
|
|
$fields->dataFieldByName('AuthorID'),
|
|
|
|
'getCMSFields() includes has_one fields on instances'
|
|
|
|
);
|
|
|
|
$this->assertNotNull(
|
|
|
|
$fields->dataFieldByName('Tags'),
|
|
|
|
'getCMSFields() includes many_many fields if ID is present on instances'
|
|
|
|
);
|
|
|
|
$this->assertNotNull(
|
|
|
|
$fields->dataFieldByName('SubjectOfArticles'),
|
|
|
|
'getCMSFields() includes polymorphic has_many fields if ID is present on instances'
|
|
|
|
);
|
|
|
|
$this->assertNull(
|
|
|
|
$fields->dataFieldByName('Subject'),
|
|
|
|
"getCMSFields() doesn't include polymorphic has_one field"
|
|
|
|
);
|
|
|
|
$this->assertNull(
|
|
|
|
$fields->dataFieldByName('SubjectID'),
|
|
|
|
"getCMSFields() doesn't include polymorphic has_one id field"
|
|
|
|
);
|
|
|
|
$this->assertNull(
|
|
|
|
$fields->dataFieldByName('SubjectClass'),
|
|
|
|
"getCMSFields() doesn't include polymorphic has_one class field"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testUpdateCMSFields()
|
|
|
|
{
|
|
|
|
$article1 = $this->objFromFixture(Article::class, 'article1');
|
|
|
|
|
|
|
|
$fields = $article1->getCMSFields();
|
2017-03-02 03:24:38 +01:00
|
|
|
$form = new Form(null, 'TestForm', $fields, new FieldList());
|
2016-12-16 05:34:21 +01:00
|
|
|
$form->loadDataFrom($article1);
|
|
|
|
|
|
|
|
$this->assertNotNull(
|
|
|
|
$fields->dataFieldByName('AddedExtensionField'),
|
|
|
|
'getCMSFields() includes extended fields'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testRestrictCMSFields()
|
|
|
|
{
|
|
|
|
$article1 = $this->objFromFixture(Article::class, 'article1');
|
|
|
|
|
|
|
|
$fields = $article1->scaffoldFormFields(
|
2020-04-20 19:58:09 +02:00
|
|
|
[
|
|
|
|
'restrictFields' => ['Title']
|
|
|
|
]
|
2016-12-16 05:34:21 +01:00
|
|
|
);
|
2017-03-02 03:24:38 +01:00
|
|
|
$form = new Form(null, 'TestForm', $fields, new FieldList());
|
2016-12-16 05:34:21 +01:00
|
|
|
$form->loadDataFrom($article1);
|
|
|
|
|
|
|
|
$this->assertNotNull(
|
|
|
|
$fields->dataFieldByName('Title'),
|
2021-12-13 09:05:33 +01:00
|
|
|
'scaffoldCMSFields() includes explicitly defined "restrictFields"'
|
2016-12-16 05:34:21 +01:00
|
|
|
);
|
|
|
|
$this->assertNull(
|
|
|
|
$fields->dataFieldByName('Content'),
|
|
|
|
'getCMSFields() doesnt include fields left out in a "restrictFields" definition'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testFieldClassesOnGetCMSFields()
|
|
|
|
{
|
|
|
|
$article1 = $this->objFromFixture(Article::class, 'article1');
|
|
|
|
|
|
|
|
$fields = $article1->scaffoldFormFields(
|
2020-04-20 19:58:09 +02:00
|
|
|
[
|
|
|
|
'fieldClasses' => ['Title' => 'SilverStripe\\Forms\\HTMLEditor\\HTMLEditorField']
|
|
|
|
]
|
2016-12-16 05:34:21 +01:00
|
|
|
);
|
2017-03-02 03:24:38 +01:00
|
|
|
$form = new Form(null, 'TestForm', $fields, new FieldList());
|
2016-12-16 05:34:21 +01:00
|
|
|
$form->loadDataFrom($article1);
|
|
|
|
|
|
|
|
$this->assertNotNull(
|
|
|
|
$fields->dataFieldByName('Title')
|
|
|
|
);
|
|
|
|
$this->assertInstanceOf(
|
|
|
|
HTMLEditorField::class,
|
|
|
|
$fields->dataFieldByName('Title'),
|
|
|
|
'getCMSFields() doesnt include fields left out in a "restrictFields" definition'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetFormFields()
|
|
|
|
{
|
|
|
|
$fields = Article::singleton()->getFrontEndFields();
|
2017-03-02 03:24:38 +01:00
|
|
|
$form = new Form(null, 'TestForm', $fields, new FieldList());
|
2016-12-16 05:34:21 +01:00
|
|
|
$form->loadDataFrom(singleton(Article::class));
|
|
|
|
|
|
|
|
$this->assertFalse($fields->hasTabSet(), 'getFrontEndFields() doesnt produce a TabSet by default');
|
|
|
|
}
|
2024-06-07 07:06:01 +02:00
|
|
|
|
2024-09-18 03:53:44 +02:00
|
|
|
public static function provideScaffoldRelationFormFields()
|
2024-06-07 07:06:01 +02:00
|
|
|
{
|
2024-08-12 02:52:57 +02:00
|
|
|
$scenarios = [
|
|
|
|
'ignore no relations' => [
|
|
|
|
'includeInOwnTab' => true,
|
|
|
|
'ignoreRelations' => [],
|
|
|
|
],
|
|
|
|
'ignore some relations' => [
|
|
|
|
'includeInOwnTab' => true,
|
|
|
|
'ignoreRelations' => [
|
|
|
|
'ChildrenHasMany',
|
|
|
|
'ChildrenManyManyThrough',
|
|
|
|
],
|
|
|
|
],
|
2024-06-07 07:06:01 +02:00
|
|
|
];
|
2024-08-12 02:52:57 +02:00
|
|
|
foreach ($scenarios as $name => $scenario) {
|
|
|
|
$scenario['includeInOwnTab'] = false;
|
|
|
|
$scenarios[$name . ' - not in own tab'] = $scenario;
|
|
|
|
}
|
|
|
|
return $scenarios;
|
2024-06-07 07:06:01 +02:00
|
|
|
}
|
|
|
|
|
2024-09-18 03:53:44 +02:00
|
|
|
#[DataProvider('provideScaffoldRelationFormFields')]
|
2024-08-12 02:52:57 +02:00
|
|
|
public function testScaffoldRelationFormFields(bool $includeInOwnTab, array $ignoreRelations)
|
2024-06-07 07:06:01 +02:00
|
|
|
{
|
|
|
|
$parent = $this->objFromFixture(ParentModel::class, 'parent1');
|
|
|
|
Child::$includeInOwnTab = $includeInOwnTab;
|
2024-08-12 02:52:57 +02:00
|
|
|
$fields = $parent->scaffoldFormFields([
|
|
|
|
'includeRelations' => true,
|
|
|
|
'tabbed' => true,
|
|
|
|
'ignoreRelations' => $ignoreRelations,
|
|
|
|
]);
|
2024-06-07 07:06:01 +02:00
|
|
|
|
2024-08-12 02:52:57 +02:00
|
|
|
// has_one
|
2024-06-07 07:06:01 +02:00
|
|
|
foreach (array_keys(ParentModel::config()->uninherited('has_one')) as $hasOneName) {
|
|
|
|
$scaffoldedFormField = $fields->dataFieldByName($hasOneName . 'ID');
|
|
|
|
if ($hasOneName === 'ChildPolymorphic') {
|
|
|
|
$this->assertNull($scaffoldedFormField, "$hasOneName should be null");
|
|
|
|
} else {
|
|
|
|
$this->assertInstanceOf(DateField::class, $scaffoldedFormField, "$hasOneName should be a DateField");
|
|
|
|
}
|
|
|
|
}
|
2024-08-12 02:52:57 +02:00
|
|
|
// has_many
|
2024-06-07 07:06:01 +02:00
|
|
|
foreach (array_keys(ParentModel::config()->uninherited('has_many')) as $hasManyName) {
|
2024-08-12 02:52:57 +02:00
|
|
|
if (in_array($hasManyName, $ignoreRelations)) {
|
|
|
|
$this->assertNull($fields->dataFieldByName($hasManyName));
|
2024-06-07 07:06:01 +02:00
|
|
|
} else {
|
2024-08-12 02:52:57 +02:00
|
|
|
$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"));
|
|
|
|
}
|
2024-06-07 07:06:01 +02:00
|
|
|
}
|
|
|
|
}
|
2024-08-12 02:52:57 +02:00
|
|
|
// many_many
|
2024-06-07 07:06:01 +02:00
|
|
|
foreach (array_keys(ParentModel::config()->uninherited('many_many')) as $manyManyName) {
|
2024-08-12 02:52:57 +02:00
|
|
|
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'));
|
|
|
|
}
|
|
|
|
|
2024-09-18 03:53:44 +02:00
|
|
|
public static function provideTabs(): array
|
2024-08-12 02:52:57 +02:00
|
|
|
{
|
|
|
|
return [
|
|
|
|
'only main tab' => [
|
2024-09-18 03:53:44 +02:00
|
|
|
'tabbed' => true,
|
2024-08-12 02:52:57 +02:00
|
|
|
'mainTabOnly' => true,
|
|
|
|
],
|
|
|
|
'all tabs, all fields' => [
|
2024-09-18 03:53:44 +02:00
|
|
|
'tabbed' => true,
|
2024-08-12 02:52:57 +02:00
|
|
|
'mainTabOnly' => false,
|
|
|
|
],
|
|
|
|
'no tabs, no fields' => [
|
2024-09-18 03:53:44 +02:00
|
|
|
'tabbed' => false,
|
2024-08-12 02:52:57 +02:00
|
|
|
'mainTabOnly' => true,
|
|
|
|
],
|
|
|
|
'no tabs, all fields' => [
|
2024-09-18 03:53:44 +02:00
|
|
|
'tabbed' => false,
|
2024-08-12 02:52:57 +02:00
|
|
|
'mainTabOnly' => false,
|
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2024-09-18 03:53:44 +02:00
|
|
|
#[DataProvider('provideTabs')]
|
2024-08-12 02:52:57 +02:00
|
|
|
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);
|
2024-06-07 07:06:01 +02:00
|
|
|
} else {
|
2024-08-12 02:52:57 +02:00
|
|
|
// All fields, no tabs
|
|
|
|
$this->assertNull($fields->findTab('Root.Main'));
|
|
|
|
foreach ($relationTabs as $tabName) {
|
|
|
|
$this->assertNull($fields->findTab($tabName));
|
|
|
|
}
|
|
|
|
$this->assertListContains($fieldsToExpect, $fields->flattenFields());
|
2024-06-07 07:06:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-10-14 00:20:41 +02:00
|
|
|
}
|