2007-07-19 12:40:28 +02:00
|
|
|
<?php
|
2016-08-19 00:51:35 +02:00
|
|
|
|
|
|
|
namespace SilverStripe\Forms;
|
|
|
|
|
2016-07-12 08:51:08 +02:00
|
|
|
use SilverStripe\ORM\FieldType\DBHTMLText;
|
|
|
|
|
2007-07-19 12:40:28 +02:00
|
|
|
/**
|
2018-03-16 16:04:26 +01:00
|
|
|
* The action buttons are `<input type="submit">` as well as <button> tags.
|
2014-08-15 08:53:05 +02:00
|
|
|
*
|
2010-10-15 05:55:22 +02:00
|
|
|
* Upon clicking the button below will redirect the user to doAction under the current controller.
|
2014-08-15 08:53:05 +02:00
|
|
|
*
|
2010-10-15 05:55:22 +02:00
|
|
|
* <code>
|
|
|
|
* new FormAction (
|
|
|
|
* // doAction has to be a defined controller member
|
|
|
|
* $action = "doAction",
|
|
|
|
* $title = "Submit button"
|
|
|
|
* )
|
|
|
|
* </code>
|
2007-07-19 12:40:28 +02:00
|
|
|
*/
|
2016-11-29 00:31:16 +01:00
|
|
|
class FormAction extends FormField
|
|
|
|
{
|
2007-07-19 12:40:28 +02:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* @config
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private static $casting = [
|
|
|
|
'ButtonContent' => 'HTMLFragment',
|
|
|
|
];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Action name, normally prefixed with 'action_'
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $action;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Identifier of icon, if supported on the frontend
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $icon = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @skipUpgrade
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $schemaComponent = 'FormAction';
|
|
|
|
|
|
|
|
/**
|
2018-03-16 16:04:26 +01:00
|
|
|
* Enables the use of `<button>` instead of `<input>`
|
2016-11-29 00:31:16 +01:00
|
|
|
* in {@link Field()} - for more customisable styling.
|
|
|
|
*
|
|
|
|
* @var boolean
|
|
|
|
*/
|
|
|
|
public $useButtonTag = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Literal button content, used when useButtonTag is true.
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $buttonContent = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Should validation be skipped when performing this action?
|
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
protected $validationExempt = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new action button.
|
|
|
|
*
|
|
|
|
* @param string $action The method to call when the button is clicked
|
|
|
|
* @param string $title The label on the button. This should be plain text, not escaped as HTML.
|
|
|
|
* @param Form form The parent form, auto-set when the field is placed inside a form
|
|
|
|
*/
|
|
|
|
public function __construct($action, $title = "", $form = null)
|
|
|
|
{
|
|
|
|
$this->action = "action_$action";
|
|
|
|
$this->setForm($form);
|
|
|
|
|
|
|
|
parent::__construct($this->action, $title);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add extra options to data
|
|
|
|
*/
|
|
|
|
public function getSchemaDataDefaults()
|
|
|
|
{
|
|
|
|
$defaults = parent::getSchemaDataDefaults();
|
|
|
|
$defaults['attributes']['type'] = $this->getUseButtonTag() ? 'button' : 'submit';
|
|
|
|
$defaults['data']['icon'] = $this->getIcon();
|
|
|
|
return $defaults;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get button icon, if supported
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getIcon()
|
|
|
|
{
|
|
|
|
return $this->icon;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets button icon
|
|
|
|
*
|
|
|
|
* @param string $icon Icon identifier (not path)
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setIcon($icon)
|
|
|
|
{
|
|
|
|
$this->icon = $icon;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the action name
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function actionName()
|
|
|
|
{
|
|
|
|
return substr($this->name, 7);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the full action name, including action_
|
|
|
|
* This provides an opportunity to replace it with something else
|
|
|
|
*
|
|
|
|
* @param string $fullAction
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setFullAction($fullAction)
|
|
|
|
{
|
|
|
|
$this->action = $fullAction;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array $properties
|
|
|
|
* @return DBHTMLText
|
|
|
|
*/
|
|
|
|
public function Field($properties = array())
|
|
|
|
{
|
|
|
|
$properties = array_merge(
|
|
|
|
$properties,
|
|
|
|
array(
|
|
|
|
'Name' => $this->action,
|
|
|
|
'Title' => ($this->description && !$this->useButtonTag) ? $this->description : $this->Title(),
|
|
|
|
'UseButtonTag' => $this->useButtonTag
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
return parent::Field($properties);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array $properties
|
|
|
|
* @return DBHTMLText
|
|
|
|
*/
|
|
|
|
public function FieldHolder($properties = array())
|
|
|
|
{
|
|
|
|
return $this->Field($properties);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function Type()
|
|
|
|
{
|
|
|
|
return 'action';
|
|
|
|
}
|
|
|
|
|
2017-05-09 06:42:08 +02:00
|
|
|
public function getInputType()
|
2016-11-29 00:31:16 +01:00
|
|
|
{
|
2017-05-08 10:17:57 +02:00
|
|
|
if (isset($this->attributes['type'])) {
|
2017-05-09 06:42:08 +02:00
|
|
|
return $this->attributes['type'];
|
2017-05-08 10:17:57 +02:00
|
|
|
} else {
|
2017-05-09 06:42:08 +02:00
|
|
|
return (isset($this->attributes['src'])) ? 'image' : 'submit';
|
2017-05-08 10:17:57 +02:00
|
|
|
}
|
2017-05-09 06:42:08 +02:00
|
|
|
}
|
2016-11-29 00:31:16 +01:00
|
|
|
|
2017-05-09 06:42:08 +02:00
|
|
|
public function getAttributes()
|
|
|
|
{
|
2016-11-29 00:31:16 +01:00
|
|
|
return array_merge(
|
|
|
|
parent::getAttributes(),
|
|
|
|
array(
|
|
|
|
'disabled' => ($this->isReadonly() || $this->isDisabled()),
|
|
|
|
'value' => $this->Title(),
|
2017-05-09 06:42:08 +02:00
|
|
|
'type' => $this->getInputType(),
|
2016-11-29 00:31:16 +01:00
|
|
|
'title' => ($this->useButtonTag) ? $this->description : null,
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add content inside a button field. This should be pre-escaped raw HTML and should be used sparingly.
|
|
|
|
*
|
|
|
|
* @param string $content
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setButtonContent($content)
|
|
|
|
{
|
|
|
|
$this->buttonContent = (string) $content;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the content inside the button field. This is raw HTML, and should be used sparingly.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getButtonContent()
|
|
|
|
{
|
|
|
|
return $this->buttonContent;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enable or disable the rendering of this action as a <button />
|
|
|
|
*
|
|
|
|
* @param boolean
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setUseButtonTag($bool)
|
|
|
|
{
|
|
|
|
$this->useButtonTag = $bool;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine if this action is rendered as a <button />
|
|
|
|
*
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
public function getUseButtonTag()
|
|
|
|
{
|
|
|
|
return $this->useButtonTag;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set whether this action can be performed without validating the data
|
|
|
|
*
|
|
|
|
* @param bool $exempt
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setValidationExempt($exempt = true)
|
|
|
|
{
|
|
|
|
$this->validationExempt = $exempt;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-03-16 16:04:26 +01:00
|
|
|
* Get whether this action can be performed without validating the data
|
2016-11-29 00:31:16 +01:00
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function getValidationExempt()
|
|
|
|
{
|
|
|
|
return $this->validationExempt;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Does not transform to readonly by purpose.
|
|
|
|
* Globally disabled buttons would break the CMS.
|
|
|
|
*/
|
|
|
|
public function performReadonlyTransformation()
|
|
|
|
{
|
|
|
|
$clone = clone $this;
|
|
|
|
$clone->setReadonly(true);
|
|
|
|
return $clone;
|
|
|
|
}
|
2012-03-24 04:04:52 +01:00
|
|
|
}
|