2009-05-26 00:21:02 +02:00
|
|
|
<?php
|
2015-08-30 07:02:55 +02:00
|
|
|
|
2016-08-19 00:51:35 +02:00
|
|
|
namespace SilverStripe\Forms;
|
2016-06-15 06:03:16 +02:00
|
|
|
|
2017-01-26 05:20:08 +01:00
|
|
|
use InvalidArgumentException;
|
2016-08-19 00:51:35 +02:00
|
|
|
use SilverStripe\ORM\ArrayLib;
|
2016-06-15 06:03:16 +02:00
|
|
|
use SilverStripe\ORM\FieldType\DBMoney;
|
|
|
|
use SilverStripe\ORM\DataObjectInterface;
|
|
|
|
|
2009-05-26 00:21:02 +02:00
|
|
|
/**
|
2010-11-13 03:05:34 +01:00
|
|
|
* A form field that can save into a {@link Money} database field.
|
2014-08-15 08:53:05 +02:00
|
|
|
* See {@link CurrencyField} for a similiar implementation
|
2010-11-13 03:05:34 +01:00
|
|
|
* that can save into a single float database field without indicating the currency.
|
2014-08-15 08:53:05 +02:00
|
|
|
*
|
2009-05-26 00:21:02 +02:00
|
|
|
* @author Ingo Schommer, SilverStripe Ltd. (<firstname>@silverstripe.com)
|
|
|
|
*/
|
2016-11-29 00:31:16 +01:00
|
|
|
class MoneyField extends FormField
|
|
|
|
{
|
|
|
|
|
|
|
|
// TODO replace with `FormField::SCHEMA_DATA_TYPE_TEXT` when MoneyField is implemented
|
|
|
|
protected $schemaDataType = 'MoneyField';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Limit the currencies
|
2017-01-26 05:20:08 +01:00
|
|
|
*
|
|
|
|
* @var array
|
2016-11-29 00:31:16 +01:00
|
|
|
*/
|
2017-01-26 05:20:08 +01:00
|
|
|
protected $allowedCurrencies = [];
|
2016-11-29 00:31:16 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var NumericField
|
|
|
|
*/
|
|
|
|
protected $fieldAmount = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var FormField
|
|
|
|
*/
|
|
|
|
protected $fieldCurrency = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets field for the currency selector
|
|
|
|
*
|
|
|
|
* @return FormField
|
|
|
|
*/
|
|
|
|
public function getCurrencyField()
|
|
|
|
{
|
|
|
|
return $this->fieldCurrency;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets field for the amount input
|
|
|
|
*
|
|
|
|
* @return NumericField
|
|
|
|
*/
|
|
|
|
public function getAmountField()
|
|
|
|
{
|
|
|
|
return $this->fieldAmount;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function __construct($name, $title = null, $value = "")
|
|
|
|
{
|
|
|
|
$this->setName($name);
|
2017-01-26 05:20:08 +01:00
|
|
|
$this->fieldAmount = NumericField::create(
|
|
|
|
"{$name}[Amount]",
|
|
|
|
_t('MoneyField.FIELDLABELAMOUNT', 'Amount')
|
|
|
|
)
|
|
|
|
->setScale(2);
|
|
|
|
$this->buildCurrencyField();
|
2016-11-29 00:31:16 +01:00
|
|
|
|
|
|
|
parent::__construct($name, $title, $value);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function __clone()
|
|
|
|
{
|
|
|
|
$this->fieldAmount = clone $this->fieldAmount;
|
|
|
|
$this->fieldCurrency = clone $this->fieldCurrency;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Builds a new currency field based on the allowed currencies configured
|
|
|
|
*
|
|
|
|
* @return FormField
|
|
|
|
*/
|
|
|
|
protected function buildCurrencyField()
|
|
|
|
{
|
|
|
|
$name = $this->getName();
|
2017-01-26 05:20:08 +01:00
|
|
|
|
|
|
|
// Validate allowed currencies
|
|
|
|
$currencyValue = $this->fieldCurrency ? $this->fieldCurrency->dataValue() : null;
|
2016-11-29 00:31:16 +01:00
|
|
|
$allowedCurrencies = $this->getAllowedCurrencies();
|
2017-01-26 05:20:08 +01:00
|
|
|
if (count($allowedCurrencies) === 1) {
|
|
|
|
// Dropdown field for multiple currencies
|
|
|
|
$field = HiddenField::create("{$name}[Currency]");
|
|
|
|
reset($allowedCurrencies);
|
|
|
|
$currencyValue = key($allowedCurrencies);
|
|
|
|
} elseif ($allowedCurrencies) {
|
|
|
|
// Dropdown field for multiple currencies
|
|
|
|
$field = DropdownField::create(
|
2016-11-29 00:31:16 +01:00
|
|
|
"{$name}[Currency]",
|
|
|
|
_t('MoneyField.FIELDLABELCURRENCY', 'Currency'),
|
2017-01-26 05:20:08 +01:00
|
|
|
$allowedCurrencies
|
2016-11-29 00:31:16 +01:00
|
|
|
);
|
|
|
|
} else {
|
2017-01-26 05:20:08 +01:00
|
|
|
// Free-text entry for currency value
|
|
|
|
$field = TextField::create(
|
2016-11-29 00:31:16 +01:00
|
|
|
"{$name}[Currency]",
|
|
|
|
_t('MoneyField.FIELDLABELCURRENCY', 'Currency')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$field->setReadonly($this->isReadonly());
|
|
|
|
$field->setDisabled($this->isDisabled());
|
2017-01-26 05:20:08 +01:00
|
|
|
if ($currencyValue) {
|
|
|
|
$field->setValue($currencyValue);
|
|
|
|
}
|
|
|
|
$this->fieldCurrency = $field;
|
2016-11-29 00:31:16 +01:00
|
|
|
return $field;
|
|
|
|
}
|
|
|
|
|
2017-01-26 05:20:08 +01:00
|
|
|
public function setSubmittedValue($value, $data = null)
|
2016-11-29 00:31:16 +01:00
|
|
|
{
|
2017-01-26 05:20:08 +01:00
|
|
|
if (empty($value)) {
|
|
|
|
$this->value = null;
|
|
|
|
$this->fieldCurrency->setValue(null);
|
|
|
|
$this->fieldAmount->setValue(null);
|
|
|
|
return $this;
|
|
|
|
}
|
2016-11-29 00:31:16 +01:00
|
|
|
|
2017-01-26 05:20:08 +01:00
|
|
|
// Handle submitted array value
|
|
|
|
if (!is_array($value)) {
|
|
|
|
throw new InvalidArgumentException("Value is not submitted array");
|
2016-11-29 00:31:16 +01:00
|
|
|
}
|
|
|
|
|
2017-01-26 05:20:08 +01:00
|
|
|
// Update each field
|
|
|
|
$this->fieldCurrency->setSubmittedValue($value['Currency'], $value);
|
|
|
|
$this->fieldAmount->setSubmittedValue($value['Amount'], $value);
|
2016-11-29 00:31:16 +01:00
|
|
|
|
2017-01-26 05:20:08 +01:00
|
|
|
// Get data value
|
|
|
|
$this->value = $this->dataValue();
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setValue($value, $data = null)
|
|
|
|
{
|
|
|
|
if (empty($value)) {
|
|
|
|
$this->value = null;
|
|
|
|
$this->fieldCurrency->setValue(null);
|
|
|
|
$this->fieldAmount->setValue(null);
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert string to array
|
|
|
|
// E.g. `44.00 NZD`
|
|
|
|
if (is_string($value) &&
|
|
|
|
preg_match('/^(?<amount>[\\d\\.]+)( (?<currency>\w{3}))?$/i', $value, $matches)
|
|
|
|
) {
|
|
|
|
$currency = isset($matches['currency']) ? strtoupper($matches['currency']) : null;
|
|
|
|
$value = [
|
|
|
|
'Currency' => $currency,
|
|
|
|
'Amount' => (float)$matches['amount'],
|
|
|
|
];
|
|
|
|
} elseif ($value instanceof DBMoney) {
|
|
|
|
$value = [
|
|
|
|
'Currency' => $value->getCurrency(),
|
|
|
|
'Amount' => $value->getAmount(),
|
|
|
|
];
|
|
|
|
} else {
|
|
|
|
throw new InvalidArgumentException("Invalid currency format");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save value
|
|
|
|
$this->fieldCurrency->setValue($value['Currency'], $value);
|
|
|
|
$this->fieldAmount->setValue($value['Amount'], $value);
|
|
|
|
$this->value = $this->dataValue();
|
2016-11-29 00:31:16 +01:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2017-01-26 05:20:08 +01:00
|
|
|
/**
|
|
|
|
* Get value as DBMoney object useful for formatting the number
|
|
|
|
*
|
|
|
|
* @return DBMoney
|
|
|
|
*/
|
|
|
|
protected function getDBMoney()
|
|
|
|
{
|
|
|
|
return DBMoney::create_field('Money', [
|
|
|
|
'Currency' => $this->fieldCurrency->dataValue(),
|
|
|
|
'Amount' => $this->fieldAmount->dataValue()
|
|
|
|
])
|
|
|
|
->setLocale($this->getLocale());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function dataValue()
|
|
|
|
{
|
|
|
|
// Non-localised money
|
|
|
|
return $this->getDBMoney()->getValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function Value()
|
|
|
|
{
|
|
|
|
// Localised money
|
|
|
|
return $this->getDBMoney()->Nice();
|
|
|
|
}
|
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* 30/06/2009 - Enhancement:
|
|
|
|
* SaveInto checks if set-methods are available and use them
|
|
|
|
* instead of setting the values in the money class directly. saveInto
|
|
|
|
* initiates a new Money class object to pass through the values to the setter
|
|
|
|
* method.
|
|
|
|
*
|
|
|
|
* (see @link MoneyFieldTest_CustomSetter_Object for more information)
|
|
|
|
*
|
|
|
|
* @param DataObjectInterface|Object $dataObject
|
|
|
|
*/
|
|
|
|
public function saveInto(DataObjectInterface $dataObject)
|
|
|
|
{
|
|
|
|
$fieldName = $this->getName();
|
|
|
|
if ($dataObject->hasMethod("set$fieldName")) {
|
2017-01-26 05:20:08 +01:00
|
|
|
$dataObject->$fieldName = $this->getDBMoney();
|
2016-11-29 00:31:16 +01:00
|
|
|
} else {
|
|
|
|
$currencyField = "{$fieldName}Currency";
|
|
|
|
$amountField = "{$fieldName}Amount";
|
|
|
|
|
|
|
|
$dataObject->$currencyField = $this->fieldCurrency->dataValue();
|
|
|
|
$dataObject->$amountField = $this->fieldAmount->dataValue();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a readonly version of this field.
|
|
|
|
*/
|
|
|
|
public function performReadonlyTransformation()
|
|
|
|
{
|
|
|
|
$clone = clone $this;
|
|
|
|
$clone->fieldAmount = $clone->fieldAmount->performReadonlyTransformation();
|
|
|
|
$clone->fieldCurrency = $clone->fieldCurrency->performReadonlyTransformation();
|
|
|
|
$clone->setReadonly(true);
|
|
|
|
return $clone;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setReadonly($bool)
|
|
|
|
{
|
|
|
|
parent::setReadonly($bool);
|
|
|
|
|
|
|
|
$this->fieldAmount->setReadonly($bool);
|
|
|
|
$this->fieldCurrency->setReadonly($bool);
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setDisabled($bool)
|
|
|
|
{
|
|
|
|
parent::setDisabled($bool);
|
|
|
|
|
|
|
|
$this->fieldAmount->setDisabled($bool);
|
|
|
|
$this->fieldCurrency->setDisabled($bool);
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-01-26 05:20:08 +01:00
|
|
|
* Set list of currencies. Currencies should be in the 3-letter ISO 4217 currency code.
|
|
|
|
*
|
|
|
|
* @param array $currencies
|
2016-11-29 00:31:16 +01:00
|
|
|
* @return $this
|
|
|
|
*/
|
2017-01-26 05:20:08 +01:00
|
|
|
public function setAllowedCurrencies($currencies)
|
2016-11-29 00:31:16 +01:00
|
|
|
{
|
2017-01-26 05:20:08 +01:00
|
|
|
if (empty($currencies)) {
|
|
|
|
$currencies = [];
|
|
|
|
} elseif (is_string($currencies)) {
|
|
|
|
$currencies = [
|
|
|
|
$currencies => $currencies
|
|
|
|
];
|
|
|
|
} elseif (!is_array($currencies)) {
|
|
|
|
throw new InvalidArgumentException("Invalid currency list");
|
|
|
|
} elseif (!ArrayLib::is_associative($currencies)) {
|
|
|
|
$currencies = array_combine($currencies, $currencies);
|
|
|
|
}
|
2016-11-29 00:31:16 +01:00
|
|
|
|
2017-01-26 05:20:08 +01:00
|
|
|
$this->allowedCurrencies = $currencies;
|
2016-11-29 00:31:16 +01:00
|
|
|
|
2017-01-26 05:20:08 +01:00
|
|
|
// Rebuild currency field
|
|
|
|
$this->buildCurrencyField();
|
2016-11-29 00:31:16 +01:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getAllowedCurrencies()
|
|
|
|
{
|
|
|
|
return $this->allowedCurrencies;
|
|
|
|
}
|
|
|
|
|
2017-01-26 05:20:08 +01:00
|
|
|
/**
|
|
|
|
* Assign locale to format this currency in
|
|
|
|
*
|
|
|
|
* @param string $locale
|
|
|
|
* @return $this
|
|
|
|
*/
|
2016-11-29 00:31:16 +01:00
|
|
|
public function setLocale($locale)
|
|
|
|
{
|
2017-01-26 05:20:08 +01:00
|
|
|
$this->fieldAmount->setLocale($locale);
|
2016-11-29 00:31:16 +01:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2017-01-26 05:20:08 +01:00
|
|
|
/**
|
|
|
|
* Get locale to format this currency in.
|
|
|
|
* Defaults to current locale.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
2016-11-29 00:31:16 +01:00
|
|
|
public function getLocale()
|
|
|
|
{
|
2017-01-26 05:20:08 +01:00
|
|
|
return $this->fieldAmount->getLocale();
|
2016-11-29 00:31:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Validate this field
|
|
|
|
*
|
|
|
|
* @param Validator $validator
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function validate($validator)
|
|
|
|
{
|
2017-01-26 05:20:08 +01:00
|
|
|
// Validate currency
|
|
|
|
$currencies = $this->getAllowedCurrencies();
|
|
|
|
$currency = $this->fieldCurrency->dataValue();
|
|
|
|
if ($currency && $currencies && !in_array($currency, $currencies)) {
|
|
|
|
$validator->validationError(
|
|
|
|
$this->getName(),
|
|
|
|
_t(
|
|
|
|
__CLASS__.'.INVALID_CURRENCY',
|
|
|
|
'Currency {currency} is not in the list of allowed currencies',
|
|
|
|
['currency' => $currency]
|
|
|
|
)
|
|
|
|
);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Field-specific validation
|
|
|
|
return $this->fieldAmount->validate($validator) && $this->fieldCurrency->validate($validator);
|
2016-11-29 00:31:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public function setForm($form)
|
|
|
|
{
|
|
|
|
$this->fieldCurrency->setForm($form);
|
|
|
|
$this->fieldAmount->setForm($form);
|
|
|
|
return parent::setForm($form);
|
|
|
|
}
|
2009-05-26 00:21:02 +02:00
|
|
|
}
|