mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 14:05:37 +02:00
API CHANGE Removed TableField->FieldSet() and TableField->SubmittedFieldSet(), please use Items() and TableField_Item->Fields() instead (merged branches/2.3-nzct)
BUGFIX Fixed re-loading of unsaved TableField form data (e.g. after a validation error). The (now removed) method SubmittedFieldSet() was setting incremental temporary identifiers ("new1", "new2", etc), which wasn't picked up by Items() (merged from branches/2.3-nzct) BUGFIX Using $this->value instead of $_POST to process submitted data in TableField (merged from branches/2.3-nzct) BUGFIX TableField validation logic iterates over TableField_Item instances to get all formfields, rather than creating their own set in SubmittedFieldSet() MINOR Removed repitition of temporary Form generation in TableField by generateTableFieldItem() method (merged from branches/2.3-nzct) git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@82089 467b73ca-7a2a-4603-9d3b-597d59a354a9
This commit is contained in:
parent
99d6f2241f
commit
946025adb9
@ -181,88 +181,75 @@ class TableField extends TableListField {
|
|||||||
* @return DataObjectSet Collection of {@link TableField_Item}
|
* @return DataObjectSet Collection of {@link TableField_Item}
|
||||||
*/
|
*/
|
||||||
function Items() {
|
function Items() {
|
||||||
$output = new DataObjectSet();
|
// holds TableField_Item instances
|
||||||
if($items = $this->sourceItems()) {
|
$items = new DataObjectSet();
|
||||||
foreach ($items as $item) {
|
|
||||||
// Load the data in to a temporary form (for correct field types)
|
$sourceItems = $this->sourceItems();
|
||||||
$fieldset = $this->FieldSetForRow();
|
|
||||||
if($fieldset){
|
// either load all rows from the field value,
|
||||||
$form = new Form($this, null, $fieldset, new FieldSet());
|
// (e.g. when validation failed), or from sourceItems()
|
||||||
$form->loadDataFrom($item);
|
if($this->value) {
|
||||||
// Add the item to our new DataObjectSet, with a wrapper class.
|
if(!$sourceItems) $sourceItems = new DataObjectSet();
|
||||||
$output->push(new TableField_Item($item, $this, $form, $this->fieldTypes));
|
|
||||||
|
// get an array keyed by rows, rather than values
|
||||||
|
$rows = $this->sortData(ArrayLib::invert($this->value));
|
||||||
|
// ignore all rows which are already saved
|
||||||
|
if(isset($rows['new'])) {
|
||||||
|
$newRows = $this->sortData($rows['new']);
|
||||||
|
// iterate over each value (not each row)
|
||||||
|
$i = 0;
|
||||||
|
foreach($newRows as $idx => $newRow){
|
||||||
|
// set a pseudo-ID
|
||||||
|
$newRow['ID'] = "new";
|
||||||
|
|
||||||
|
// unset any extradata
|
||||||
|
foreach($newRow as $k => $v){
|
||||||
|
if($this->extraData && array_key_exists($k, $this->extraData)){
|
||||||
|
unset($newRow[$k]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// generate a temporary DataObject container (not saved in the database)
|
||||||
|
$sourceClass = $this->sourceClass;
|
||||||
|
$sourceItems->push(new $sourceClass($newRow));
|
||||||
|
|
||||||
|
$i++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Create a temporary DataObject
|
|
||||||
if($this->Can('add')) {
|
// generate a new TableField_Item instance from each collected item
|
||||||
if($this->showAddRow){
|
if($sourceItems) foreach($sourceItems as $sourceItem) {
|
||||||
$output->push(new TableField_Item(null, $this, null, $this->fieldTypes, true));
|
$items->push($this->generateTableFieldItem($sourceItem));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// add an empty TableField_Item for a single "add row"
|
||||||
|
if($this->showAddRow && $this->Can('add')) {
|
||||||
|
$items->push(new TableField_Item(null, $this, null, $this->fieldTypes, true));
|
||||||
}
|
}
|
||||||
return $output;
|
|
||||||
|
return $items;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get all fields for each row contained in the TableField.
|
* Generates a new {@link TableField} instance
|
||||||
* Does not include the empty row.
|
* by loading a fieldset for this row into a temporary form.
|
||||||
*
|
*
|
||||||
* @return array
|
* @param DataObject $dataObj
|
||||||
|
* @return TableField_Item
|
||||||
*/
|
*/
|
||||||
function FieldSet() {
|
protected function generateTableFieldItem($dataObj) {
|
||||||
$fields = array ();
|
|
||||||
if($items = $this->sourceItems()) {
|
|
||||||
foreach($items as $item) {
|
|
||||||
// Load the data in to a temporary form (for correct field types)
|
// Load the data in to a temporary form (for correct field types)
|
||||||
$fieldset = $this->FieldSetForRow();
|
$form = new Form(
|
||||||
if ($fieldset)
|
$this,
|
||||||
{
|
null,
|
||||||
// TODO Needs to be attached to a form existing in the DOM-tree
|
$this->FieldSetForRow(),
|
||||||
$form = new Form($this, 'EditForm', $fieldset, new FieldSet());
|
new FieldSet()
|
||||||
$form->loadDataFrom($item);
|
);
|
||||||
$row = new TableField_Item($item, $this, $form, $this->fieldTypes);
|
$form->loadDataFrom($dataObj);
|
||||||
$fields = array_merge($fields, $row->Fields()->toArray());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $fields;
|
// Add the item to our new DataObjectSet, with a wrapper class.
|
||||||
}
|
return new TableField_Item($dataObj, $this, $form, $this->fieldTypes);
|
||||||
|
|
||||||
function SubmittedFieldSet(&$sourceItems){
|
|
||||||
$fields = array ();
|
|
||||||
if(isset($_POST[$this->name])&&$rows = $_POST[$this->name]){
|
|
||||||
if(count($rows)){
|
|
||||||
foreach($rows as $idx => $row){
|
|
||||||
if($idx == 'new'){
|
|
||||||
$newitems = ArrayLib::invert($row);
|
|
||||||
if(count($newitems)){
|
|
||||||
$sourceItems = new DataObjectSet();
|
|
||||||
foreach($newitems as $k => $newitem){
|
|
||||||
$fieldset = $this->FieldSetForRow();
|
|
||||||
if($fieldset){
|
|
||||||
$newitem['ID'] = "new".$k;
|
|
||||||
foreach($newitem as $k => $v){
|
|
||||||
if($this->extraData && array_key_exists($k, $this->extraData)){
|
|
||||||
unset($newitem[$k]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$sourceItem = new DataObject($newitem);
|
|
||||||
if(!$sourceItem->isEmpty()){
|
|
||||||
$sourceItems->push($sourceItem);
|
|
||||||
$form = new Form($this, "EditForm", $fieldset, new FieldSet());
|
|
||||||
$form->loadDataFrom($sourceItem);
|
|
||||||
$item = new TableField_Item($sourceItem, $this, $form, $this->fieldTypes);
|
|
||||||
$fields = array_merge($fields, $item->Fields()->toArray());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return $fields;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -537,21 +524,18 @@ class TableField extends TableListField {
|
|||||||
function jsValidation() {
|
function jsValidation() {
|
||||||
$js = "";
|
$js = "";
|
||||||
|
|
||||||
$fields = $this->FieldSet();
|
$items = $this->Items();
|
||||||
$fields = new FieldSet($fields);
|
if($items) foreach($items as $item) {
|
||||||
// TODO doesn't automatically update validation when adding a row
|
foreach($item->Fields() as $field) {
|
||||||
foreach($fields as $field) {
|
|
||||||
//if the field type has some special specific specification for validation of itself
|
//if the field type has some special specific specification for validation of itself
|
||||||
$js .= $field->jsValidation($this->form->class."_".$this->form->Name());
|
$js .= $field->jsValidation($this->form->class."_".$this->form->Name());
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// TODO Implement custom requiredFields
|
// TODO Implement custom requiredFields
|
||||||
$items = $this->sourceItems();
|
$items = $this->sourceItems();
|
||||||
if($items && $this->requiredFields && $items->count()) {
|
if($items && $this->requiredFields && $items->count()) {
|
||||||
foreach ($this->requiredFields as $field) {
|
foreach ($this->requiredFields as $field) {
|
||||||
/*if($fields->dataFieldByName($field)) {
|
|
||||||
$js .= "\t\t\t\t\trequire('$field');\n";
|
|
||||||
}*/
|
|
||||||
foreach($items as $item){
|
foreach($items as $item){
|
||||||
$cellName = $this->Name().'['.$item->ID.']['.$field.']';
|
$cellName = $this->Name().'['.$item->ID.']['.$field.']';
|
||||||
$js .= "\n";
|
$js .= "\n";
|
||||||
@ -576,11 +560,13 @@ JS;
|
|||||||
$valid = true;
|
$valid = true;
|
||||||
|
|
||||||
if($data['methodName'] != 'delete') {
|
if($data['methodName'] != 'delete') {
|
||||||
$fields = $this->FieldSet();
|
$items = $this->Items();
|
||||||
$fields = new FieldSet($fields);
|
if($items) foreach($items as $item) {
|
||||||
foreach($fields as $field){
|
foreach($item->Fields() as $field) {
|
||||||
$valid = $field->validate($this) && $valid;
|
$valid = $field->validate($this) && $valid;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return $valid;
|
return $valid;
|
||||||
} else {
|
} else {
|
||||||
return $valid;
|
return $valid;
|
||||||
@ -590,11 +576,14 @@ JS;
|
|||||||
function validate($validator) {
|
function validate($validator) {
|
||||||
$errorMessage = '';
|
$errorMessage = '';
|
||||||
$valid = true;
|
$valid = true;
|
||||||
$fields = $this->SubmittedFieldSet($sourceItemsNew);
|
|
||||||
$fields = new FieldSet($fields);
|
// @todo should only contain new elements
|
||||||
foreach($fields as $field){
|
$items = $this->Items();
|
||||||
|
if($items) foreach($items as $item) {
|
||||||
|
foreach($item->Fields() as $field) {
|
||||||
$valid = $field->validate($validator) && $valid;
|
$valid = $field->validate($validator) && $valid;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
//debug::show($this->form->Message());
|
//debug::show($this->form->Message());
|
||||||
if($this->requiredFields&&$sourceItemsNew&&$sourceItemsNew->count()) {
|
if($this->requiredFields&&$sourceItemsNew&&$sourceItemsNew->count()) {
|
||||||
@ -702,8 +691,10 @@ class TableField_Item extends TableListField_Item {
|
|||||||
$origFieldName = $field->Name();
|
$origFieldName = $field->Name();
|
||||||
|
|
||||||
// set unique fieldname with id
|
// set unique fieldname with id
|
||||||
$combinedFieldName = $this->parent->Name() . "[" . $this->ID . "][" . $origFieldName . "]";
|
$combinedFieldName = $this->parent->Name() . "[" . $this->ID() . "][" . $origFieldName . "]";
|
||||||
if($this->isAddRow) $combinedFieldName .= '[]';
|
// ensure to set field to nested array notation
|
||||||
|
// if its an unsaved row, or the "add row" which is present by default
|
||||||
|
if($this->isAddRow || $this->ID() == 'new') $combinedFieldName .= '[]';
|
||||||
|
|
||||||
// get value
|
// get value
|
||||||
if(strpos($origFieldName,'.') === false) {
|
if(strpos($origFieldName,'.') === false) {
|
||||||
|
@ -3,27 +3,34 @@
|
|||||||
class TableFieldTest extends SapphireTest {
|
class TableFieldTest extends SapphireTest {
|
||||||
static $fixture_file = 'sapphire/tests/forms/TableFieldTest.yml';
|
static $fixture_file = 'sapphire/tests/forms/TableFieldTest.yml';
|
||||||
|
|
||||||
|
function testAdd() {
|
||||||
function testTableFieldSaving() {
|
$group = $this->objFromFixture('Group','group1_no_perms');
|
||||||
$group = $this->objFromFixture('Group','a');
|
|
||||||
|
|
||||||
$tableField = new TableField(
|
$tableField = new TableField(
|
||||||
"Permissions",
|
"Permissions",
|
||||||
"Permission",
|
"Permission",
|
||||||
array(
|
array(
|
||||||
"Code" => _t('SecurityAdmin.CODE', 'Code'),
|
"Code" => 'Code',
|
||||||
"Arg" => _t('SecurityAdmin.OPTIONALID', 'Optional ID'),
|
"Arg" => 'Arg',
|
||||||
),
|
),
|
||||||
array(
|
array(
|
||||||
"Code" => "PermissionDropdownField",
|
"Code" => "TextField",
|
||||||
"Arg" => "TextField",
|
"Arg" => "TextField",
|
||||||
),
|
),
|
||||||
"GroupID",
|
"GroupID",
|
||||||
$group->ID
|
$group->ID
|
||||||
);
|
);
|
||||||
$form = new Form(new TableFieldTest_Controller(), "Form", new FieldSet($tableField), new FieldSet());
|
$form = new Form(
|
||||||
|
new TableFieldTest_Controller(),
|
||||||
|
"Form",
|
||||||
|
new FieldSet($tableField),
|
||||||
|
new FieldSet()
|
||||||
|
);
|
||||||
|
|
||||||
/* The field starts emppty. Save some new data. We have replicated the array structure that the specific layout of the form generates. */
|
// Test Insert
|
||||||
|
|
||||||
|
// The field starts emppty. Save some new data.
|
||||||
|
// We have replicated the array structure that the specific layout of the form generates.
|
||||||
$tableField->setValue(array(
|
$tableField->setValue(array(
|
||||||
'new' => array(
|
'new' => array(
|
||||||
'Code' => array(
|
'Code' => array(
|
||||||
@ -38,7 +45,7 @@ class TableFieldTest extends SapphireTest {
|
|||||||
));
|
));
|
||||||
$tableField->saveInto($group);
|
$tableField->saveInto($group);
|
||||||
|
|
||||||
/* Let's check that the 2 permissions entries have been saved */
|
// Let's check that the 2 permissions entries have been saved
|
||||||
$permissions = $group->Permissions()->toDropdownMap('Arg', 'Code');
|
$permissions = $group->Permissions()->toDropdownMap('Arg', 'Code');
|
||||||
$this->assertEquals(array(
|
$this->assertEquals(array(
|
||||||
1 => 'CMS_ACCESS_CMSMain',
|
1 => 'CMS_ACCESS_CMSMain',
|
||||||
@ -46,7 +53,7 @@ class TableFieldTest extends SapphireTest {
|
|||||||
), $permissions);
|
), $permissions);
|
||||||
|
|
||||||
|
|
||||||
/* Now let's perform an update query */
|
// Test repeated insert
|
||||||
$value = array();
|
$value = array();
|
||||||
foreach($group->Permissions() as $permission) {
|
foreach($group->Permissions() as $permission) {
|
||||||
$value[$permission->ID] = array("Code" => $permission->Code, "Arg" => $permission->Arg);
|
$value[$permission->ID] = array("Code" => $permission->Code, "Arg" => $permission->Arg);
|
||||||
@ -62,7 +69,7 @@ class TableFieldTest extends SapphireTest {
|
|||||||
$tableField->setValue($value);
|
$tableField->setValue($value);
|
||||||
$tableField->saveInto($group);
|
$tableField->saveInto($group);
|
||||||
|
|
||||||
/* Let's check that the 2 existing permissions entries, and the 1 new one, have been saved */
|
// Let's check that the 2 existing permissions entries, and the 1 new one, have been saved
|
||||||
$permissions = $group->Permissions()->toDropdownMap('Arg', 'Code');
|
$permissions = $group->Permissions()->toDropdownMap('Arg', 'Code');
|
||||||
$this->assertEquals(array(
|
$this->assertEquals(array(
|
||||||
1 => 'CMS_ACCESS_CMSMain',
|
1 => 'CMS_ACCESS_CMSMain',
|
||||||
@ -72,6 +79,88 @@ class TableFieldTest extends SapphireTest {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function testEdit() {
|
||||||
|
$group = $this->objFromFixture('Group','group2_existing_perms');
|
||||||
|
$perm1 = $this->objFromFixture('Permission', 'perm1');
|
||||||
|
$perm2 = $this->objFromFixture('Permission', 'perm2');
|
||||||
|
|
||||||
|
$tableField = new TableField(
|
||||||
|
"Permissions",
|
||||||
|
"Permission",
|
||||||
|
array(
|
||||||
|
"Code" => 'Code',
|
||||||
|
"Arg" => 'Arg',
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
"Code" => "TextField",
|
||||||
|
"Arg" => "TextField",
|
||||||
|
),
|
||||||
|
"GroupID",
|
||||||
|
$group->ID
|
||||||
|
);
|
||||||
|
$form = new Form(
|
||||||
|
new TableFieldTest_Controller(),
|
||||||
|
"Form",
|
||||||
|
new FieldSet($tableField),
|
||||||
|
new FieldSet()
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->assertEquals($tableField->sourceItems()->Count(), 2);
|
||||||
|
|
||||||
|
// We have replicated the array structure that the specific layout of the form generates.
|
||||||
|
$tableField->setValue(array(
|
||||||
|
$perm1->ID => array(
|
||||||
|
'Code' => 'Perm1 Modified',
|
||||||
|
'Arg' => '101'
|
||||||
|
),
|
||||||
|
$perm2->ID => array(
|
||||||
|
'Code' => 'Perm2 Modified',
|
||||||
|
'Arg' => '102'
|
||||||
|
)
|
||||||
|
));
|
||||||
|
$tableField->saveInto($group);
|
||||||
|
|
||||||
|
// Let's check that the 2 permissions entries have been saved
|
||||||
|
$permissions = $group->Permissions()->toDropdownMap('Arg', 'Code');
|
||||||
|
$this->assertEquals(array(
|
||||||
|
101 => 'Perm1 Modified',
|
||||||
|
102 => 'Perm2 Modified',
|
||||||
|
), $permissions);
|
||||||
|
}
|
||||||
|
|
||||||
|
function testDelete() {
|
||||||
|
$group = $this->objFromFixture('Group','group2_existing_perms');
|
||||||
|
$perm1 = $this->objFromFixture('Permission', 'perm1');
|
||||||
|
$perm2 = $this->objFromFixture('Permission', 'perm2');
|
||||||
|
|
||||||
|
$tableField = new TableField(
|
||||||
|
"Permissions",
|
||||||
|
"Permission",
|
||||||
|
array(
|
||||||
|
"Code" => 'Code',
|
||||||
|
"Arg" => 'Arg',
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
"Code" => "TextField",
|
||||||
|
"Arg" => "TextField",
|
||||||
|
),
|
||||||
|
"GroupID",
|
||||||
|
$group->ID
|
||||||
|
);
|
||||||
|
$form = new Form(
|
||||||
|
new TableFieldTest_Controller(),
|
||||||
|
"Form",
|
||||||
|
new FieldSet($tableField),
|
||||||
|
new FieldSet()
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->assertContains($perm1->ID, $tableField->sourceItems()->column('ID'));
|
||||||
|
|
||||||
|
$response = $tableField->Items()->find('ID', $perm1->ID)->delete();
|
||||||
|
|
||||||
|
$this->assertNotContains($perm1->ID, $tableField->sourceItems()->column('ID'));
|
||||||
|
}
|
||||||
|
|
||||||
function testAutoRelationSettingOn() {
|
function testAutoRelationSettingOn() {
|
||||||
$tf = new TableField(
|
$tf = new TableField(
|
||||||
'HasManyRelations',
|
'HasManyRelations',
|
||||||
@ -150,6 +239,45 @@ class TableFieldTest extends SapphireTest {
|
|||||||
//$this->assertEquals($data['TestTableField']['new']['Currency'][0], 1234.56);
|
//$this->assertEquals($data['TestTableField']['new']['Currency'][0], 1234.56);
|
||||||
//$this->assertEquals($data['TestTableField']['new']['Currency'][1], 1234.57);
|
//$this->assertEquals($data['TestTableField']['new']['Currency'][1], 1234.57);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function testHasItemsWhenSetAsArray() {
|
||||||
|
$tf = new TableField(
|
||||||
|
'TestTableField',
|
||||||
|
'TableFieldTest_HasManyRelation',
|
||||||
|
array(
|
||||||
|
'Value' => 'Value'
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
'Value' => 'TextField'
|
||||||
|
)
|
||||||
|
);
|
||||||
|
$tf->setValue(array(
|
||||||
|
'new' => array(
|
||||||
|
'Value' => array(
|
||||||
|
'One',
|
||||||
|
'Two',
|
||||||
|
)
|
||||||
|
)
|
||||||
|
));
|
||||||
|
$items = $tf->Items();
|
||||||
|
$itemsArr = $items->toArray();
|
||||||
|
|
||||||
|
// includes the two values and an "add" row
|
||||||
|
$this->assertEquals($items->Count(), 3);
|
||||||
|
|
||||||
|
// first row
|
||||||
|
$this->assertEquals(
|
||||||
|
$itemsArr[0]->Fields()->fieldByName('TestTableField[new][Value][]')->Value(),
|
||||||
|
'One'
|
||||||
|
);
|
||||||
|
|
||||||
|
// second row
|
||||||
|
$this->assertEquals(
|
||||||
|
$itemsArr[1]->Fields()->fieldByName('TestTableField[new][Value][]')->Value(),
|
||||||
|
'Two'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1,3 +1,13 @@
|
|||||||
|
Permission:
|
||||||
|
perm1:
|
||||||
|
Code: Perm1
|
||||||
|
Arg: 1
|
||||||
|
perm2:
|
||||||
|
Code: Perm2
|
||||||
|
Arg: 2
|
||||||
Group:
|
Group:
|
||||||
a:
|
group1_no_perms:
|
||||||
Title: Group A
|
Title: Group A
|
||||||
|
group2_existing_perms:
|
||||||
|
Title: Group B
|
||||||
|
Permissions: =>Permission.perm1,=>Permission.perm2
|
Loading…
Reference in New Issue
Block a user