2012-03-06 22:48:09 +01:00
|
|
|
<?php
|
|
|
|
|
2016-10-14 03:30:05 +02:00
|
|
|
namespace SilverStripe\Forms\Tests\GridField;
|
|
|
|
|
2023-11-23 05:24:52 +01:00
|
|
|
use LogicException;
|
|
|
|
use ReflectionMethod;
|
2018-06-01 07:47:03 +02:00
|
|
|
use SilverStripe\Control\Controller;
|
2023-11-23 05:24:52 +01:00
|
|
|
use SilverStripe\Control\HTTPRequest;
|
2016-08-19 00:51:35 +02:00
|
|
|
use SilverStripe\Dev\CSSContentParser;
|
|
|
|
use SilverStripe\Dev\FunctionalTest;
|
2023-11-23 05:24:52 +01:00
|
|
|
use SilverStripe\Forms\FieldList;
|
2022-05-02 08:25:20 +02:00
|
|
|
use SilverStripe\Forms\Form;
|
2016-08-19 00:51:35 +02:00
|
|
|
use SilverStripe\Forms\GridField\GridField;
|
2023-11-23 05:24:52 +01:00
|
|
|
use SilverStripe\Forms\GridField\GridFieldConfig_RecordEditor;
|
2018-06-01 07:47:03 +02:00
|
|
|
use SilverStripe\Forms\GridField\GridFieldDetailForm;
|
2016-08-19 00:51:35 +02:00
|
|
|
use SilverStripe\Forms\GridField\GridFieldDetailForm_ItemRequest;
|
2018-06-01 07:47:03 +02:00
|
|
|
use SilverStripe\Forms\HiddenField;
|
2023-11-23 05:24:52 +01:00
|
|
|
use SilverStripe\Forms\Tests\GridField\GridFieldDetailFormTest\ArrayDataWithID;
|
2016-10-14 03:30:05 +02:00
|
|
|
use SilverStripe\Forms\Tests\GridField\GridFieldDetailFormTest\Category;
|
|
|
|
use SilverStripe\Forms\Tests\GridField\GridFieldDetailFormTest\CategoryController;
|
|
|
|
use SilverStripe\Forms\Tests\GridField\GridFieldDetailFormTest\GroupController;
|
2023-12-11 22:18:25 +01:00
|
|
|
use SilverStripe\Forms\Tests\GridField\GridFieldDetailFormTest\MultiRelationalPeopleGroup;
|
2016-10-14 03:30:05 +02:00
|
|
|
use SilverStripe\Forms\Tests\GridField\GridFieldDetailFormTest\PeopleGroup;
|
|
|
|
use SilverStripe\Forms\Tests\GridField\GridFieldDetailFormTest\Person;
|
2022-05-02 08:25:20 +02:00
|
|
|
use SilverStripe\Forms\Tests\GridField\GridFieldDetailFormTest\PolymorphicPeopleGroup;
|
2018-06-01 07:47:03 +02:00
|
|
|
use SilverStripe\Forms\Tests\GridField\GridFieldDetailFormTest\TestController;
|
2024-09-23 04:31:50 +02:00
|
|
|
use SilverStripe\Model\List\ArrayList;
|
|
|
|
use SilverStripe\Model\ArrayData;
|
2024-09-18 03:53:44 +02:00
|
|
|
use PHPUnit\Framework\Attributes\DataProvider;
|
2016-08-19 00:51:35 +02:00
|
|
|
|
2016-12-16 05:34:21 +01:00
|
|
|
class GridFieldDetailFormTest extends FunctionalTest
|
|
|
|
{
|
|
|
|
protected static $fixture_file = 'GridFieldDetailFormTest.yml';
|
|
|
|
|
2020-04-20 19:58:09 +02:00
|
|
|
protected static $extra_dataobjects = [
|
2016-12-16 05:34:21 +01:00
|
|
|
Person::class,
|
|
|
|
PeopleGroup::class,
|
2022-05-02 08:25:20 +02:00
|
|
|
PolymorphicPeopleGroup::class,
|
2023-12-11 22:18:25 +01:00
|
|
|
MultiRelationalPeopleGroup::class,
|
2016-12-16 05:34:21 +01:00
|
|
|
Category::class,
|
2020-04-20 19:58:09 +02:00
|
|
|
];
|
2016-12-16 05:34:21 +01:00
|
|
|
|
2017-03-24 12:17:26 +01:00
|
|
|
protected static $extra_controllers = [
|
2016-12-16 05:34:21 +01:00
|
|
|
CategoryController::class,
|
|
|
|
TestController::class,
|
|
|
|
GroupController::class,
|
|
|
|
];
|
|
|
|
|
2018-06-01 07:47:03 +02:00
|
|
|
protected static $disable_themes = true;
|
|
|
|
|
2016-12-16 05:34:21 +01:00
|
|
|
public function testValidator()
|
|
|
|
{
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
|
|
|
|
$response = $this->get('GridFieldDetailFormTest_Controller');
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$addlinkitem = $parser->getBySelector('.grid-field .new-link');
|
|
|
|
$addlink = (string) $addlinkitem[0]['href'];
|
|
|
|
|
|
|
|
$response = $this->get($addlink);
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$addform = $parser->getBySelector('#Form_ItemEditForm');
|
|
|
|
$addformurl = (string) $addform[0]['action'];
|
|
|
|
|
|
|
|
$response = $this->post(
|
|
|
|
$addformurl,
|
2020-04-20 19:58:09 +02:00
|
|
|
[
|
2016-12-16 05:34:21 +01:00
|
|
|
'FirstName' => 'Jeremiah',
|
|
|
|
'ajax' => 1,
|
|
|
|
'action_doSave' => 1
|
2020-04-20 19:58:09 +02:00
|
|
|
]
|
2016-12-16 05:34:21 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$errors = $parser->getBySelector('span.required');
|
2022-04-14 03:12:59 +02:00
|
|
|
$this->assertEquals(1, count($errors ?? []));
|
2016-12-16 05:34:21 +01:00
|
|
|
|
|
|
|
$response = $this->post(
|
|
|
|
$addformurl,
|
2020-04-20 19:58:09 +02:00
|
|
|
[
|
2016-12-16 05:34:21 +01:00
|
|
|
'ajax' => 1,
|
|
|
|
'action_doSave' => 1
|
2020-04-20 19:58:09 +02:00
|
|
|
]
|
2016-12-16 05:34:21 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$errors = $parser->getBySelector('span.required');
|
2022-04-14 03:12:59 +02:00
|
|
|
$this->assertEquals(2, count($errors ?? []));
|
2016-12-16 05:34:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testAddForm()
|
|
|
|
{
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
$group = PeopleGroup::get()
|
|
|
|
->filter('Name', 'My Group')
|
|
|
|
->sort('Name')
|
|
|
|
->First();
|
|
|
|
$count = $group->People()->Count();
|
|
|
|
|
|
|
|
$response = $this->get('GridFieldDetailFormTest_Controller');
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$addlinkitem = $parser->getBySelector('.grid-field .new-link');
|
|
|
|
$addlink = (string) $addlinkitem[0]['href'];
|
|
|
|
|
|
|
|
$response = $this->get($addlink);
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$addform = $parser->getBySelector('#Form_ItemEditForm');
|
|
|
|
$addformurl = (string) $addform[0]['action'];
|
|
|
|
|
|
|
|
$response = $this->post(
|
|
|
|
$addformurl,
|
2020-04-20 19:58:09 +02:00
|
|
|
[
|
2016-12-16 05:34:21 +01:00
|
|
|
'FirstName' => 'Jeremiah',
|
|
|
|
'Surname' => 'BullFrog',
|
|
|
|
'action_doSave' => 1
|
2020-04-20 19:58:09 +02:00
|
|
|
]
|
2016-12-16 05:34:21 +01:00
|
|
|
);
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
|
|
|
|
$group = PeopleGroup::get()
|
|
|
|
->filter('Name', 'My Group')
|
|
|
|
->sort('Name')
|
|
|
|
->First();
|
|
|
|
$this->assertEquals($count + 1, $group->People()->Count());
|
|
|
|
}
|
|
|
|
|
2022-05-02 08:25:20 +02:00
|
|
|
public function testAddFormWithPolymorphicHasOne()
|
|
|
|
{
|
|
|
|
// Log in for permissions check
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
// Prepare gridfield and other objects
|
|
|
|
$group = new PolymorphicPeopleGroup();
|
|
|
|
$group->write();
|
|
|
|
$gridField = $group->getCMSFields()->dataFieldByName('People');
|
|
|
|
$gridField->setForm(new Form());
|
|
|
|
$detailForm = $gridField->getConfig()->getComponentByType(GridFieldDetailForm::class);
|
|
|
|
$record = new Person();
|
|
|
|
|
|
|
|
// Trigger creation of the item edit form
|
|
|
|
$reflectionDetailForm = new \ReflectionClass($detailForm);
|
|
|
|
$reflectionMethod = $reflectionDetailForm->getMethod('getItemRequestHandler');
|
|
|
|
$reflectionMethod->setAccessible(true);
|
|
|
|
$itemrequest = $reflectionMethod->invoke($detailForm, $gridField, $record, new Controller());
|
|
|
|
$itemrequest->ItemEditForm();
|
|
|
|
|
|
|
|
// The polymorphic values should be pre-loaded
|
|
|
|
$this->assertEquals(PolymorphicPeopleGroup::class, $record->PolymorphicGroupClass);
|
|
|
|
$this->assertEquals($group->ID, $record->PolymorphicGroupID);
|
|
|
|
}
|
|
|
|
|
2023-12-11 22:18:25 +01:00
|
|
|
public function testAddFormWithMultiRelationalHasOne(): void
|
|
|
|
{
|
|
|
|
// Log in for permissions check
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
// Prepare gridfield and other objects
|
|
|
|
$group = new MultiRelationalPeopleGroup();
|
|
|
|
$group->write();
|
|
|
|
$gridField = $group->getCMSFields()->dataFieldByName('People');
|
|
|
|
$gridField->setForm(new Form());
|
|
|
|
$detailForm = $gridField->getConfig()->getComponentByType(GridFieldDetailForm::class);
|
|
|
|
$record = new Person();
|
|
|
|
|
|
|
|
// Trigger creation of the item edit form
|
|
|
|
$reflectionDetailForm = new \ReflectionClass($detailForm);
|
|
|
|
$reflectionMethod = $reflectionDetailForm->getMethod('getItemRequestHandler');
|
|
|
|
$reflectionMethod->setAccessible(true);
|
|
|
|
$itemrequest = $reflectionMethod->invoke($detailForm, $gridField, $record, new Controller());
|
|
|
|
$itemrequest->ItemEditForm();
|
|
|
|
|
|
|
|
// The polymorphic and multi-relational values should be pre-loaded
|
|
|
|
$this->assertEquals(MultiRelationalPeopleGroup::class, $record->MultiRelationalGroupClass);
|
|
|
|
$this->assertEquals('People', $record->MultiRelationalGroupRelation);
|
|
|
|
$this->assertEquals($group->ID, $record->MultiRelationalGroupID);
|
|
|
|
}
|
|
|
|
|
2016-12-16 05:34:21 +01:00
|
|
|
public function testViewForm()
|
|
|
|
{
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
|
|
|
|
$response = $this->get('GridFieldDetailFormTest_Controller');
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
|
|
|
|
$viewLink = $parser->getBySelector('.ss-gridfield-items .first .view-link');
|
|
|
|
$viewLink = (string) $viewLink[0]['href'];
|
|
|
|
|
|
|
|
$response = $this->get($viewLink);
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
|
|
|
|
$firstName = $parser->getBySelector('#Form_ItemEditForm_FirstName');
|
|
|
|
$surname = $parser->getBySelector('#Form_ItemEditForm_Surname');
|
|
|
|
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
$this->assertEquals('Jane', (string) $firstName[0]);
|
|
|
|
$this->assertEquals('Doe', (string) $surname[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testEditForm()
|
|
|
|
{
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
$group = PeopleGroup::get()
|
|
|
|
->filter('Name', 'My Group')
|
|
|
|
->sort('Name')
|
|
|
|
->First();
|
|
|
|
$firstperson = $group->People()->First();
|
|
|
|
$this->assertTrue($firstperson->Surname != 'Baggins');
|
|
|
|
|
|
|
|
$response = $this->get('GridFieldDetailFormTest_Controller');
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$editlinkitem = $parser->getBySelector('.ss-gridfield-items .first .edit-link');
|
|
|
|
$editlink = (string) $editlinkitem[0]['href'];
|
|
|
|
|
|
|
|
$response = $this->get($editlink);
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$editform = $parser->getBySelector('#Form_ItemEditForm');
|
|
|
|
$editformurl = (string) $editform[0]['action'];
|
|
|
|
|
|
|
|
$response = $this->post(
|
|
|
|
$editformurl,
|
2020-04-20 19:58:09 +02:00
|
|
|
[
|
2016-12-16 05:34:21 +01:00
|
|
|
'FirstName' => 'Bilbo',
|
|
|
|
'Surname' => 'Baggins',
|
|
|
|
'action_doSave' => 1
|
2020-04-20 19:58:09 +02:00
|
|
|
]
|
2016-12-16 05:34:21 +01:00
|
|
|
);
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
|
|
|
|
$group = PeopleGroup::get()
|
|
|
|
->filter('Name', 'My Group')
|
|
|
|
->sort('Name')
|
|
|
|
->First();
|
2020-04-20 19:58:09 +02:00
|
|
|
$this->assertListContains([['Surname' => 'Baggins']], $group->People());
|
2016-12-16 05:34:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testEditFormWithManyMany()
|
|
|
|
{
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
|
|
|
|
// Edit the first person
|
|
|
|
$response = $this->get('GridFieldDetailFormTest_CategoryController');
|
|
|
|
// Find the link to add a new favourite group
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$addLink = $parser->getBySelector('#Form_Form_testgroupsfield .new-link');
|
|
|
|
$addLink = (string) $addLink[0]['href'];
|
|
|
|
|
|
|
|
// Add a new favourite group
|
|
|
|
$response = $this->get($addLink);
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$addform = $parser->getBySelector('#Form_ItemEditForm');
|
|
|
|
$addformurl = (string) $addform[0]['action'];
|
|
|
|
|
|
|
|
$response = $this->post(
|
|
|
|
$addformurl,
|
2020-04-20 19:58:09 +02:00
|
|
|
[
|
2016-12-16 05:34:21 +01:00
|
|
|
'Name' => 'My Favourite Group',
|
|
|
|
'ajax' => 1,
|
|
|
|
'action_doSave' => 1
|
2020-04-20 19:58:09 +02:00
|
|
|
]
|
2016-12-16 05:34:21 +01:00
|
|
|
);
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
|
2020-08-27 01:18:54 +02:00
|
|
|
$person = $this->objFromFixture(Person::class, 'jane');
|
2016-12-16 05:34:21 +01:00
|
|
|
$favouriteGroup = $person->FavouriteGroups()->first();
|
|
|
|
|
|
|
|
$this->assertInstanceOf(PeopleGroup::class, $favouriteGroup);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testEditFormWithManyManyExtraData()
|
|
|
|
{
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
|
|
|
|
// Lists all categories for a person
|
|
|
|
$response = $this->get('GridFieldDetailFormTest_CategoryController');
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$editlinkitem = $parser->getBySelector('.ss-gridfield-items .first .edit-link');
|
|
|
|
$editlink = (string) $editlinkitem[0]['href'];
|
|
|
|
|
|
|
|
// Edit a single category, incl. manymany extrafields added manually
|
|
|
|
// through GridFieldDetailFormTest_CategoryController
|
|
|
|
$response = $this->get($editlink);
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$editform = $parser->getBySelector('#Form_ItemEditForm');
|
|
|
|
$editformurl = (string) $editform[0]['action'];
|
|
|
|
|
|
|
|
$manyManyField = $parser->getByXpath('//*[@id="Form_ItemEditForm"]//input[@name="ManyMany[IsPublished]"]');
|
|
|
|
$this->assertTrue((bool)$manyManyField);
|
|
|
|
|
|
|
|
// Test save of IsPublished field
|
|
|
|
$response = $this->post(
|
|
|
|
$editformurl,
|
2020-04-20 19:58:09 +02:00
|
|
|
[
|
2016-12-16 05:34:21 +01:00
|
|
|
'Name' => 'Updated Category',
|
2020-04-20 19:58:09 +02:00
|
|
|
'ManyMany' => [
|
2016-12-16 05:34:21 +01:00
|
|
|
'IsPublished' => 1,
|
|
|
|
'PublishedBy' => 'Richard'
|
2020-04-20 19:58:09 +02:00
|
|
|
],
|
2016-12-16 05:34:21 +01:00
|
|
|
'action_doSave' => 1
|
2020-04-20 19:58:09 +02:00
|
|
|
]
|
2016-12-16 05:34:21 +01:00
|
|
|
);
|
|
|
|
$this->assertFalse($response->isError());
|
2020-08-27 01:18:54 +02:00
|
|
|
$person = $this->objFromFixture(Person::class, 'jane');
|
2020-04-20 19:58:09 +02:00
|
|
|
$category = $person->Categories()->filter(['Name' => 'Updated Category'])->First();
|
2016-12-16 05:34:21 +01:00
|
|
|
$this->assertEquals(
|
2020-04-20 19:58:09 +02:00
|
|
|
[
|
2016-12-16 05:34:21 +01:00
|
|
|
'IsPublished' => 1,
|
|
|
|
'PublishedBy' => 'Richard'
|
2020-04-20 19:58:09 +02:00
|
|
|
],
|
2016-12-16 05:34:21 +01:00
|
|
|
$person->Categories()->getExtraData('', $category->ID)
|
|
|
|
);
|
|
|
|
|
|
|
|
// Test update of value with falsey value
|
|
|
|
$response = $this->post(
|
|
|
|
$editformurl,
|
2020-04-20 19:58:09 +02:00
|
|
|
[
|
2016-12-16 05:34:21 +01:00
|
|
|
'Name' => 'Updated Category',
|
2020-04-20 19:58:09 +02:00
|
|
|
'ManyMany' => [
|
2016-12-16 05:34:21 +01:00
|
|
|
'PublishedBy' => ''
|
2020-04-20 19:58:09 +02:00
|
|
|
],
|
2016-12-16 05:34:21 +01:00
|
|
|
'action_doSave' => 1
|
2020-04-20 19:58:09 +02:00
|
|
|
]
|
2016-12-16 05:34:21 +01:00
|
|
|
);
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
|
2020-08-27 01:18:54 +02:00
|
|
|
$person = $this->objFromFixture(Person::class, 'jane');
|
2020-04-20 19:58:09 +02:00
|
|
|
$category = $person->Categories()->filter(['Name' => 'Updated Category'])->First();
|
2016-12-16 05:34:21 +01:00
|
|
|
$this->assertEquals(
|
2020-04-20 19:58:09 +02:00
|
|
|
[
|
2016-12-16 05:34:21 +01:00
|
|
|
'IsPublished' => 0,
|
|
|
|
'PublishedBy' => ''
|
2020-04-20 19:58:09 +02:00
|
|
|
],
|
2016-12-16 05:34:21 +01:00
|
|
|
$person->Categories()->getExtraData('', $category->ID)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testNestedEditForm()
|
|
|
|
{
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
|
|
|
|
$group = $this->objFromFixture(PeopleGroup::class, 'group');
|
|
|
|
$person = $group->People()->First();
|
|
|
|
$category = $person->Categories()->First();
|
|
|
|
|
|
|
|
// Get first form (GridField managing PeopleGroup)
|
|
|
|
$response = $this->get('GridFieldDetailFormTest_GroupController');
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
|
|
|
|
$groupEditLink = $parser->getByXpath(
|
|
|
|
'//tr[contains(@class, "ss-gridfield-item") and contains(@data-id, "'
|
|
|
|
. $group->ID . '")]//a'
|
|
|
|
);
|
|
|
|
$this->assertEquals(
|
|
|
|
'GridFieldDetailFormTest_GroupController/Form/field/testfield/item/' . $group->ID . '/edit',
|
|
|
|
(string)$groupEditLink[0]['href']
|
|
|
|
);
|
|
|
|
|
|
|
|
// Get second level form (GridField managing Person)
|
|
|
|
$response = $this->get((string)$groupEditLink[0]['href']);
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$personEditLink = $parser->getByXpath(
|
|
|
|
'//fieldset[@id="Form_ItemEditForm_People"]' .
|
|
|
|
'//tr[contains(@class, "ss-gridfield-item") and contains(@data-id, "' . $person->ID . '")]//a'
|
|
|
|
);
|
|
|
|
$this->assertEquals(
|
|
|
|
sprintf(
|
2022-07-17 22:49:08 +02:00
|
|
|
'/GridFieldDetailFormTest_GroupController/Form/field/testfield/item/%d/ItemEditForm/field/People'
|
|
|
|
. '/item/%d/edit%s',
|
2016-12-16 05:34:21 +01:00
|
|
|
$group->ID,
|
2022-07-17 22:49:08 +02:00
|
|
|
$person->ID,
|
|
|
|
'?gridState-People-1=%7B%22GridFieldAddRelation%22%3Anull%7D'
|
2016-12-16 05:34:21 +01:00
|
|
|
),
|
|
|
|
(string)$personEditLink[0]['href']
|
|
|
|
);
|
|
|
|
|
|
|
|
// Get third level form (GridField managing Category)
|
|
|
|
$response = $this->get((string)$personEditLink[0]['href']);
|
|
|
|
$this->assertFalse($response->isError());
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$categoryEditLink = $parser->getByXpath(
|
|
|
|
'//fieldset[@id="Form_ItemEditForm_Categories"]'
|
|
|
|
. '//tr[contains(@class, "ss-gridfield-item") and contains(@data-id, "' . $category->ID . '")]//a'
|
|
|
|
);
|
|
|
|
$this->assertEquals(
|
|
|
|
sprintf(
|
2022-07-17 22:49:08 +02:00
|
|
|
'/GridFieldDetailFormTest_GroupController/Form/field/testfield/item/%d/ItemEditForm/field/People'
|
|
|
|
. '/item/%d/ItemEditForm/field/Categories/item/%d/edit%s',
|
2016-12-16 05:34:21 +01:00
|
|
|
$group->ID,
|
|
|
|
$person->ID,
|
2022-07-17 22:49:08 +02:00
|
|
|
$category->ID,
|
|
|
|
'?gridState-Categories-2=%7B%22GridFieldAddRelation%22%3Anull%7D'
|
2016-12-16 05:34:21 +01:00
|
|
|
),
|
|
|
|
(string)$categoryEditLink[0]['href']
|
|
|
|
);
|
|
|
|
|
|
|
|
// Fourth level form would be a Category detail view
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testCustomItemRequestClass()
|
|
|
|
{
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
|
|
|
|
$component = new GridFieldDetailForm();
|
|
|
|
$this->assertEquals('SilverStripe\\Forms\\GridField\\GridFieldDetailForm_ItemRequest', $component->getItemRequestClass());
|
|
|
|
$component->setItemRequestClass('GridFieldDetailFormTest_ItemRequest');
|
|
|
|
$this->assertEquals('GridFieldDetailFormTest_ItemRequest', $component->getItemRequestClass());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testItemEditFormCallback()
|
|
|
|
{
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
|
|
|
|
$category = new Category();
|
|
|
|
$component = new GridFieldDetailForm();
|
|
|
|
$component->setItemEditFormCallback(
|
|
|
|
function ($form, $component) {
|
|
|
|
$form->Fields()->push(new HiddenField('Callback'));
|
|
|
|
}
|
|
|
|
);
|
|
|
|
// Note: A lot of scaffolding to execute the tested logic,
|
2017-06-22 12:50:45 +02:00
|
|
|
// due to the coupling of form creation with itemRequest handling (and its context)
|
|
|
|
$itemRequest = new GridFieldDetailForm_ItemRequest(
|
2016-12-16 05:34:21 +01:00
|
|
|
GridField::create('Categories', 'Categories'),
|
|
|
|
$component,
|
|
|
|
$category,
|
2017-06-22 12:50:45 +02:00
|
|
|
Controller::curr(),
|
2016-12-16 05:34:21 +01:00
|
|
|
'Form'
|
|
|
|
);
|
2017-06-22 12:50:45 +02:00
|
|
|
$itemRequest->setRequest(Controller::curr()->getRequest());
|
|
|
|
$form = $itemRequest->ItemEditForm();
|
2016-12-16 05:34:21 +01:00
|
|
|
$this->assertNotNull($form->Fields()->fieldByName('Callback'));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tests that a has-many detail form is pre-populated with the parent ID.
|
|
|
|
*/
|
|
|
|
public function testHasManyFormPrePopulated()
|
|
|
|
{
|
|
|
|
$group = $this->objFromFixture(
|
|
|
|
PeopleGroup::class,
|
|
|
|
'group'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
|
|
|
|
$response = $this->get('GridFieldDetailFormTest_Controller');
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$addLink = $parser->getBySelector('.grid-field .new-link');
|
|
|
|
$addLink = (string) $addLink[0]['href'];
|
|
|
|
|
|
|
|
$response = $this->get($addLink);
|
|
|
|
$parser = new CSSContentParser($response->getBody());
|
|
|
|
$title = $parser->getBySelector('#Form_ItemEditForm_GroupID_Holder span');
|
|
|
|
$id = $parser->getBySelector('#Form_ItemEditForm_GroupID_Holder input');
|
|
|
|
|
|
|
|
$this->assertEquals($group->Name, (string) $title[0]);
|
|
|
|
$this->assertEquals($group->ID, (string) $id[0]['value']);
|
|
|
|
}
|
2020-08-27 01:18:54 +02:00
|
|
|
|
|
|
|
public function testRedirectMissingRecords()
|
|
|
|
{
|
|
|
|
$origAutoFollow = $this->autoFollowRedirection;
|
|
|
|
$this->autoFollowRedirection = false;
|
|
|
|
|
2021-12-13 09:05:33 +01:00
|
|
|
// GridField is filtered people in "My Group", which doesn't include "jack"
|
2020-08-27 01:18:54 +02:00
|
|
|
$included = $this->objFromFixture(Person::class, 'joe');
|
|
|
|
$excluded = $this->objFromFixture(Person::class, 'jack');
|
|
|
|
|
|
|
|
$response = $this->get(sprintf(
|
|
|
|
'GridFieldDetailFormTest_Controller/Form/field/testfield/item/%d/edit',
|
|
|
|
$included->ID
|
|
|
|
));
|
|
|
|
$this->assertFalse(
|
|
|
|
$response->isRedirect(),
|
|
|
|
'Existing records are not redirected'
|
|
|
|
);
|
|
|
|
|
|
|
|
$response = $this->get(sprintf(
|
|
|
|
'GridFieldDetailFormTest_Controller/Form/field/testfield/item/%d/edit',
|
|
|
|
$excluded->ID
|
|
|
|
));
|
|
|
|
$this->assertTrue(
|
|
|
|
$response->isRedirect(),
|
|
|
|
'Non-existing records are redirected'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->autoFollowRedirection = $origAutoFollow;
|
|
|
|
}
|
2023-11-23 05:24:52 +01:00
|
|
|
|
2024-09-18 03:53:44 +02:00
|
|
|
public static function provideGetRecordFromRequestFindExisting()
|
2023-11-23 05:24:52 +01:00
|
|
|
{
|
|
|
|
return [
|
|
|
|
'No records' => [
|
|
|
|
'data' => [],
|
|
|
|
'hasRecord' => false,
|
|
|
|
],
|
|
|
|
'Records exist but without ID field' => [
|
|
|
|
'data' => [new ArrayDataWithID()],
|
|
|
|
'hasRecord' => false,
|
|
|
|
],
|
|
|
|
'Record exists with matching ID' => [
|
|
|
|
'data' => [new ArrayDataWithID(['ID' => 32])],
|
|
|
|
'hasRecord' => true,
|
|
|
|
],
|
|
|
|
'Record exists, no matching ID' => [
|
|
|
|
'data' => [new ArrayDataWithID(['ID' => 1])],
|
|
|
|
'hasRecord' => false,
|
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2024-09-18 03:53:44 +02:00
|
|
|
#[DataProvider('provideGetRecordFromRequestFindExisting')]
|
2023-11-23 05:24:52 +01:00
|
|
|
public function testGetRecordFromRequestFindExisting(array $data, bool $hasRecord)
|
|
|
|
{
|
|
|
|
$controller = new TestController();
|
|
|
|
$form = $controller->Form(null, new ArrayList($data));
|
|
|
|
$gridField = $form->Fields()->dataFieldByName('testfield');
|
|
|
|
if (empty($data)) {
|
|
|
|
$gridField->setModelClass(ArrayDataWithID::class);
|
|
|
|
}
|
|
|
|
$component = $gridField->getConfig()->getComponentByType(GridFieldDetailForm::class);
|
|
|
|
$request = new HTTPRequest('GET', $gridField->Link('item/32'));
|
|
|
|
$request->match(Controller::join_links($gridField->Link(), 'item/$ID'));
|
|
|
|
|
|
|
|
$reflectionMethod = new ReflectionMethod($component, 'getRecordFromRequest');
|
|
|
|
$reflectionMethod->setAccessible(true);
|
|
|
|
$this->assertSame($hasRecord, (bool) $reflectionMethod->invoke($component, $gridField, $request));
|
|
|
|
}
|
|
|
|
|
2024-09-18 03:53:44 +02:00
|
|
|
public static function provideGetRecordFromRequestCreateNew()
|
2023-11-23 05:24:52 +01:00
|
|
|
{
|
|
|
|
// Note that in all of these scenarios a new record gets created, so it *shouldn't* matter what's already in there.
|
|
|
|
return [
|
|
|
|
'No records' => [
|
|
|
|
'data' => [],
|
|
|
|
],
|
|
|
|
'Records exist but without ID field' => [
|
|
|
|
'data' => [new ArrayDataWithID()],
|
|
|
|
],
|
|
|
|
'Record exists with ID field' => [
|
|
|
|
'data' => [new ArrayDataWithID(['ID' => 32])],
|
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2024-09-18 03:53:44 +02:00
|
|
|
#[DataProvider('provideGetRecordFromRequestCreateNew')]
|
2023-11-23 05:24:52 +01:00
|
|
|
public function testGetRecordFromRequestCreateNew(array $data)
|
|
|
|
{
|
|
|
|
$controller = new TestController();
|
|
|
|
$form = $controller->Form(null, new ArrayList($data));
|
|
|
|
$gridField = $form->Fields()->dataFieldByName('testfield');
|
|
|
|
if (empty($data)) {
|
|
|
|
$gridField->setModelClass(ArrayDataWithID::class);
|
|
|
|
}
|
|
|
|
$component = $gridField->getConfig()->getComponentByType(GridFieldDetailForm::class);
|
|
|
|
$request = new HTTPRequest('GET', $gridField->Link('item/new'));
|
|
|
|
$request->match(Controller::join_links($gridField->Link(), 'item/$ID'));
|
|
|
|
|
|
|
|
$reflectionMethod = new ReflectionMethod($component, 'getRecordFromRequest');
|
|
|
|
$reflectionMethod->setAccessible(true);
|
|
|
|
$this->assertEquals(new ArrayDataWithID(['ID' => 0]), $reflectionMethod->invoke($component, $gridField, $request));
|
|
|
|
}
|
|
|
|
|
2024-09-18 03:53:44 +02:00
|
|
|
public static function provideGetRecordFromRequestWithoutData()
|
2023-11-23 05:24:52 +01:00
|
|
|
{
|
|
|
|
// Note that in all of these scenarios a new record gets created, so it *shouldn't* matter what's already in there.
|
|
|
|
return [
|
|
|
|
'No records' => [
|
|
|
|
'data' => [],
|
|
|
|
],
|
|
|
|
'Records exist but without ID field' => [
|
|
|
|
'data' => [new ArrayData()],
|
|
|
|
],
|
|
|
|
'Record exists with ID field' => [
|
|
|
|
'data' => [new ArrayData(['ID' => 32])],
|
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2024-09-18 03:53:44 +02:00
|
|
|
#[DataProvider('provideGetRecordFromRequestWithoutData')]
|
2023-11-23 05:24:52 +01:00
|
|
|
public function testGetRecordFromRequestWithoutData(array $data)
|
|
|
|
{
|
|
|
|
$controller = new TestController();
|
|
|
|
$form = $controller->Form(null, new ArrayList($data));
|
|
|
|
$gridField = $form->Fields()->dataFieldByName('testfield');
|
|
|
|
if (empty($data)) {
|
|
|
|
$gridField->setModelClass(ArrayData::class);
|
|
|
|
}
|
|
|
|
$component = $gridField->getConfig()->getComponentByType(GridFieldDetailForm::class);
|
|
|
|
$request = new HTTPRequest('GET', $gridField->Link('item/new'));
|
|
|
|
$request->match(Controller::join_links($gridField->Link(), 'item/$ID'));
|
|
|
|
|
|
|
|
$this->expectException(LogicException::class);
|
|
|
|
$this->expectExceptionMessage(ArrayData::class . ' must have an ID field.');
|
|
|
|
|
|
|
|
$reflectionMethod = new ReflectionMethod($component, 'getRecordFromRequest');
|
|
|
|
$reflectionMethod->setAccessible(true);
|
|
|
|
$reflectionMethod->invoke($component, $gridField, $request);
|
|
|
|
}
|
2012-03-06 22:48:09 +01:00
|
|
|
}
|