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:
Ingo Schommer 2009-07-17 02:09:27 +00:00
parent 99d6f2241f
commit 946025adb9
3 changed files with 240 additions and 111 deletions

View File

@ -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";
@ -575,14 +559,16 @@ JS;
function php($data) { function php($data) {
$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,10 +576,13 @@ 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();
$valid = $field->validate($validator)&&$valid; if($items) foreach($items as $item) {
foreach($item->Fields() as $field) {
$valid = $field->validate($validator) && $valid;
}
} }
//debug::show($this->form->Message()); //debug::show($this->form->Message());
@ -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) {

View File

@ -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'
);
}
} }
/** /**

View File

@ -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