FIX: CheckboxSetField can now save into DBMultiEnum

Fixes https://github.com/silverstripe/silverstripe-framework/issues/1489
This commit is contained in:
Sam Minnee 2018-10-18 22:52:14 +13:00
parent 6bd2281b83
commit 76255c9fb5
3 changed files with 171 additions and 3 deletions

View File

@ -4,6 +4,7 @@ namespace SilverStripe\Forms;
use SilverStripe\ORM\DataObject; use SilverStripe\ORM\DataObject;
use SilverStripe\ORM\DataObjectInterface; use SilverStripe\ORM\DataObjectInterface;
use SilverStripe\ORM\FieldType\DBMultiEnum;
use SilverStripe\ORM\Relation; use SilverStripe\ORM\Relation;
/** /**
@ -98,7 +99,15 @@ abstract class MultiSelectField extends SelectField
$value = array_values($relation->getIDList()); $value = array_values($relation->getIDList());
parent::setValue($value); parent::setValue($value);
} elseif ($record->hasField($fieldName)) { } elseif ($record->hasField($fieldName)) {
// Load dataValue from field... a CSV for DBMultiEnum
if ($record->obj($fieldName) instanceof DBMultiEnum) {
$value = $this->csvDecode($record->$fieldName);
// ... JSON-encoded string for other fields
} else {
$value = $this->stringDecode($record->$fieldName); $value = $this->stringDecode($record->$fieldName);
}
parent::setValue($value); parent::setValue($value);
} }
} }
@ -129,10 +138,16 @@ abstract class MultiSelectField extends SelectField
// Save ids into relation // Save ids into relation
$relation->setByIDList($items); $relation->setByIDList($items);
} elseif ($record->hasField($fieldName)) { } elseif ($record->hasField($fieldName)) {
// Save dataValue into field // Save dataValue into field... a CSV for DBMultiEnum
if ($record->obj($fieldName) instanceof DBMultiEnum) {
$record->$fieldName = $this->csvEncode($items);
// ... JSON-encoded string for other fields
} else {
$record->$fieldName = $this->stringEncode($items); $record->$fieldName = $this->stringEncode($items);
} }
} }
}
/** /**
* Encode a list of values into a string, or null if empty (to simplify empty checks) * Encode a list of values into a string, or null if empty (to simplify empty checks)
@ -169,6 +184,45 @@ abstract class MultiSelectField extends SelectField
throw new \InvalidArgumentException("Invalid string encoded value for multi select field"); throw new \InvalidArgumentException("Invalid string encoded value for multi select field");
} }
/**
* Encode a list of values into a string as a comma separated list.
* Commas will be stripped from the items passed in
*
* @param array $value
* @return string|null
*/
protected function csvEncode($value)
{
if (!$value) {
return null;
}
return implode(
',',
array_map(
function ($x) {
return str_replace(',', '', $x);
},
array_values($value)
)
);
}
/**
* Decode a list of values from a comma separated string.
* Spaces are trimmed
*
* @param string $value
* @return array
*/
protected function csvDecode($value)
{
if (!$value) {
return [];
}
return preg_split('/\s*,\s*/', trim($value));
}
/** /**
* Validate this field * Validate this field
* *

View File

@ -0,0 +1,16 @@
<?php
namespace SilverStripe\Forms\Tests\CheckboxSetFieldTest;
use SilverStripe\Dev\TestOnly;
use SilverStripe\ORM\DataObject;
class MultiEnumArticle extends DataObject implements TestOnly
{
private static $table_name = 'CheckboxSetFieldTest_MultiEnumArticle';
private static $db = array(
"Content" => "Text",
"Colours" => "MultiEnum('Red,Blue,Green')",
);
}

View File

@ -0,0 +1,98 @@
<?php
namespace SilverStripe\Forms\Tests;
use SilverStripe\Forms\Tests\CheckboxSetFieldTest\MultiEnumArticle;
use SilverStripe\ORM\DB;
use SilverStripe\Dev\SapphireTest;
use SilverStripe\Control\Controller;
use SilverStripe\Forms\CheckboxSetField;
use SilverStripe\Forms\FieldList;
use SilverStripe\Forms\Form;
use SilverStripe\ORM\Connect\MySQLDatabase;
class CheckboxSetFieldMulitEnumTest extends SapphireTest
{
protected $usesDatabase = true;
public static function getExtraDataObjects()
{
// Don't add this for other database
if (DB::get_conn() instanceof MySQLDatabase) {
return [
MultiEnumArticle::class,
];
}
}
public function setUp()
{
if (!(DB::get_conn() instanceof MySQLDatabase)) {
$this->markTestSkipped('DBMultiEnum only supported by MySQL');
return;
}
parent::setUp();
}
public function tearDown()
{
if (!(DB::get_conn() instanceof MySQLDatabase)) {
return;
}
parent::tearDown();
}
public function testLoadDataFromMultiEnum()
{
$article = new MultiEnumArticle();
$article->Colours = 'Red,Green';
$field = new CheckboxSetField(
'Colours',
'Colours',
[
'Red' => 'Red',
'Blue' => 'Blue',
'Green' => 'Green',
]
);
$form = new Form(
Controller::curr(),
'Form',
new FieldList($field),
new FieldList()
);
$form->loadDataFrom($article);
$value = $field->Value();
$this->assertEquals(['Red', 'Green'], $value);
}
public function testSavingIntoMultiEnum()
{
$field = new CheckboxSetField(
'Colours',
'Colours',
[
'Red' => 'Red',
'Blue' => 'Blue',
'Green' => 'Green',
]
);
$article = new MultiEnumArticle();
$field->setValue(array('Red' => 'Red', 'Blue' => 'Blue'));
$field->saveInto($article);
$article->write();
$dbValue = DB::query(
sprintf(
'SELECT "Colours" FROM "CheckboxSetFieldTest_MultiEnumArticle" WHERE "ID" = %s',
$article->ID
)
)->value();
// JSON encoded values
$this->assertEquals('Red,Blue', $dbValue);
}
}