2015-02-13 17:35:39 +13:00
|
|
|
<?php
|
|
|
|
|
2016-08-19 10:51:35 +12:00
|
|
|
namespace SilverStripe\Forms;
|
2016-06-15 16:03:16 +12:00
|
|
|
|
|
|
|
use SilverStripe\ORM\DataObject;
|
|
|
|
use SilverStripe\ORM\DataObjectInterface;
|
2018-10-18 22:52:14 +13:00
|
|
|
use SilverStripe\ORM\FieldType\DBMultiEnum;
|
2016-06-15 16:03:16 +12:00
|
|
|
use SilverStripe\ORM\Relation;
|
|
|
|
|
2015-02-13 17:35:39 +13:00
|
|
|
/**
|
|
|
|
* Represents a SelectField that may potentially have multiple selections, and may have
|
|
|
|
* a {@link ManyManyList} as a data source.
|
|
|
|
*/
|
2016-11-29 12:31:16 +13:00
|
|
|
abstract class MultiSelectField extends SelectField
|
|
|
|
{
|
|
|
|
|
|
|
|
/**
|
|
|
|
* List of items to mark as checked, and may not be unchecked
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
2020-04-20 18:58:09 +01:00
|
|
|
protected $defaultItems = [];
|
2016-11-29 12:31:16 +13:00
|
|
|
|
|
|
|
protected $schemaDataType = FormField::SCHEMA_DATA_TYPE_MULTISELECT;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Extracts the value of this field, normalised as an array.
|
|
|
|
* Scalar values will return a single length array, even if empty
|
|
|
|
*
|
|
|
|
* @return array List of values as an array
|
|
|
|
*/
|
|
|
|
public function getValueArray()
|
|
|
|
{
|
|
|
|
return $this->getListValues($this->Value());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Default selections, regardless of the {@link setValue()} settings.
|
|
|
|
* Note: Items marked as disabled through {@link setDisabledItems()} can still be
|
|
|
|
* selected by default through this method.
|
|
|
|
*
|
|
|
|
* @param array $items Collection of array keys, as defined in the $source array
|
|
|
|
* @return $this Self reference
|
|
|
|
*/
|
|
|
|
public function setDefaultItems($items)
|
|
|
|
{
|
|
|
|
$this->defaultItems = $this->getListValues($items);
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Default selections, regardless of the {@link setValue()} settings.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getDefaultItems()
|
|
|
|
{
|
|
|
|
return $this->defaultItems;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Load a value into this MultiSelectField
|
|
|
|
*
|
2017-01-26 17:20:08 +13:00
|
|
|
* @param mixed $value
|
2016-11-29 12:31:16 +13:00
|
|
|
* @param null|array|DataObject $obj {@see Form::loadDataFrom}
|
|
|
|
* @return $this
|
|
|
|
*/
|
2017-01-26 17:20:08 +13:00
|
|
|
public function setValue($value, $obj = null)
|
2016-11-29 12:31:16 +13:00
|
|
|
{
|
|
|
|
// If we're not passed a value directly,
|
|
|
|
// we can look for it in a relation method on the object passed as a second arg
|
|
|
|
if ($obj instanceof DataObject) {
|
|
|
|
$this->loadFrom($obj);
|
|
|
|
} else {
|
2017-01-26 17:20:08 +13:00
|
|
|
parent::setValue($value);
|
2016-11-29 12:31:16 +13:00
|
|
|
}
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Load the value from the dataobject into this field
|
|
|
|
*
|
|
|
|
* @param DataObject|DataObjectInterface $record
|
|
|
|
*/
|
|
|
|
public function loadFrom(DataObjectInterface $record)
|
|
|
|
{
|
|
|
|
$fieldName = $this->getName();
|
|
|
|
if (empty($fieldName) || empty($record)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$relation = $record->hasMethod($fieldName)
|
|
|
|
? $record->$fieldName()
|
|
|
|
: null;
|
|
|
|
|
|
|
|
// Detect DB relation or field
|
|
|
|
if ($relation instanceof Relation) {
|
|
|
|
// Load ids from relation
|
2022-04-14 13:12:59 +12:00
|
|
|
$value = array_values($relation->getIDList() ?? []);
|
2016-11-29 12:31:16 +13:00
|
|
|
parent::setValue($value);
|
|
|
|
} elseif ($record->hasField($fieldName)) {
|
2018-10-18 22:52:14 +13:00
|
|
|
// 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);
|
|
|
|
}
|
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
parent::setValue($value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Save the current value of this MultiSelectField into a DataObject.
|
|
|
|
* If the field it is saving to is a has_many or many_many relationship,
|
|
|
|
* it is saved by setByIDList(), otherwise it creates a comma separated
|
|
|
|
* list for a standard DB text/varchar field.
|
|
|
|
*
|
|
|
|
* @param DataObject|DataObjectInterface $record The record to save into
|
|
|
|
*/
|
|
|
|
public function saveInto(DataObjectInterface $record)
|
|
|
|
{
|
|
|
|
$fieldName = $this->getName();
|
|
|
|
if (empty($fieldName) || empty($record)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$relation = $record->hasMethod($fieldName)
|
|
|
|
? $record->$fieldName()
|
|
|
|
: null;
|
|
|
|
|
|
|
|
// Detect DB relation or field
|
|
|
|
$items = $this->getValueArray();
|
|
|
|
if ($relation instanceof Relation) {
|
|
|
|
// Save ids into relation
|
|
|
|
$relation->setByIDList($items);
|
|
|
|
} elseif ($record->hasField($fieldName)) {
|
2018-10-18 22:52:14 +13:00
|
|
|
// 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);
|
|
|
|
}
|
2016-11-29 12:31:16 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Encode a list of values into a string, or null if empty (to simplify empty checks)
|
|
|
|
*
|
|
|
|
* @param array $value
|
|
|
|
* @return string|null
|
|
|
|
*/
|
|
|
|
public function stringEncode($value)
|
|
|
|
{
|
|
|
|
return $value
|
|
|
|
? json_encode(array_values($value))
|
|
|
|
: null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Extract a string value into an array of values
|
|
|
|
*
|
|
|
|
* @param string $value
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
protected function stringDecode($value)
|
|
|
|
{
|
|
|
|
// Handle empty case
|
|
|
|
if (empty($value)) {
|
2020-04-20 18:58:09 +01:00
|
|
|
return [];
|
2016-11-29 12:31:16 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
// If json deserialisation fails, then fallover to legacy format
|
2022-04-14 13:12:59 +12:00
|
|
|
$result = json_decode($value ?? '', true);
|
2016-11-29 12:31:16 +13:00
|
|
|
if ($result !== false) {
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new \InvalidArgumentException("Invalid string encoded value for multi select field");
|
|
|
|
}
|
|
|
|
|
2018-10-18 22:52:14 +13:00
|
|
|
/**
|
|
|
|
* 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) {
|
2022-04-14 13:12:59 +12:00
|
|
|
return str_replace(',', '', $x ?? '');
|
2018-10-18 22:52:14 +13:00
|
|
|
},
|
2022-04-14 13:12:59 +12:00
|
|
|
array_values($value ?? [])
|
2018-10-18 22:52:14 +13:00
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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 [];
|
|
|
|
}
|
|
|
|
|
2022-04-14 13:12:59 +12:00
|
|
|
return preg_split('/\s*,\s*/', trim($value ?? ''));
|
2018-10-18 22:52:14 +13:00
|
|
|
}
|
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
/**
|
|
|
|
* Validate this field
|
|
|
|
*
|
|
|
|
* @param Validator $validator
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function validate($validator)
|
|
|
|
{
|
|
|
|
$values = $this->getValueArray();
|
|
|
|
$validValues = $this->getValidValues();
|
|
|
|
|
|
|
|
// Filter out selected values not in the data source
|
|
|
|
$self = $this;
|
|
|
|
$invalidValues = array_filter(
|
2022-04-14 13:12:59 +12:00
|
|
|
$values ?? [],
|
2016-11-29 12:31:16 +13:00
|
|
|
function ($userValue) use ($self, $validValues) {
|
|
|
|
foreach ($validValues as $formValue) {
|
|
|
|
if ($self->isSelectedValue($formValue, $userValue)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
);
|
|
|
|
if (empty($invalidValues)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// List invalid items
|
|
|
|
$validator->validationError(
|
|
|
|
$this->getName(),
|
|
|
|
_t(
|
2017-04-20 13:15:24 +12:00
|
|
|
'SilverStripe\\Forms\\MultiSelectField.SOURCE_VALIDATION',
|
2016-11-29 12:31:16 +13:00
|
|
|
"Please select values within the list provided. Invalid option(s) {value} given",
|
2020-04-20 18:58:09 +01:00
|
|
|
['value' => implode(',', $invalidValues)]
|
2016-11-29 12:31:16 +13:00
|
|
|
),
|
|
|
|
"validation"
|
|
|
|
);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Transforms the source data for this CheckboxSetField
|
|
|
|
* into a comma separated list of values.
|
|
|
|
*
|
|
|
|
* @return ReadonlyField
|
|
|
|
*/
|
|
|
|
public function performReadonlyTransformation()
|
|
|
|
{
|
2016-10-05 14:30:19 +13:00
|
|
|
$field = $this->castedCopy('SilverStripe\\Forms\\LookupField');
|
|
|
|
$field->setSource($this->getSource());
|
|
|
|
$field->setReadonly(true);
|
2015-02-13 17:35:39 +13:00
|
|
|
|
2019-05-27 12:35:00 +12:00
|
|
|
// Pass through default items
|
|
|
|
if (!$this->getValueArray() && $this->getDefaultItems()) {
|
|
|
|
$field->setValue($this->getDefaultItems());
|
|
|
|
}
|
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
return $field;
|
|
|
|
}
|
2015-02-13 17:35:39 +13:00
|
|
|
}
|