2016-08-19 10:51:35 +12:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace SilverStripe\Forms\HTMLEditor;
|
|
|
|
|
2017-06-29 18:45:17 +12:00
|
|
|
use SilverStripe\Assets\Shortcodes\ImageShortcodeProvider;
|
2016-08-19 10:51:35 +12:00
|
|
|
use SilverStripe\Forms\TextareaField;
|
|
|
|
use SilverStripe\ORM\DataObject;
|
|
|
|
use SilverStripe\ORM\DataObjectInterface;
|
|
|
|
use Exception;
|
2017-04-18 10:34:18 +12:00
|
|
|
use SilverStripe\View\Parsers\HTMLValue;
|
2016-08-19 10:51:35 +12:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A TinyMCE-powered WYSIWYG HTML editor field with image and link insertion and tracking capabilities. Editor fields
|
|
|
|
* are created from <textarea> tags, which are then converted with JavaScript.
|
2016-09-15 14:48:38 +12:00
|
|
|
*
|
|
|
|
* Caution: The form field does not include any JavaScript or CSS when used outside of the CMS context,
|
|
|
|
* since the required frontend dependencies are included through CMS bundling.
|
2016-08-19 10:51:35 +12:00
|
|
|
*/
|
2016-11-29 12:31:16 +13:00
|
|
|
class HTMLEditorField extends TextareaField
|
|
|
|
{
|
|
|
|
|
|
|
|
private static $casting = [
|
|
|
|
'Value' => 'HTMLText',
|
|
|
|
];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Use TinyMCE's GZIP compressor
|
|
|
|
*
|
|
|
|
* @config
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
private static $use_gzip = true;
|
|
|
|
|
2016-12-15 12:20:29 +00:00
|
|
|
/**
|
|
|
|
* @config
|
|
|
|
* @var string Default alignment for Images and Media. Options: leftAlone|center|left|right
|
|
|
|
*/
|
|
|
|
private static $media_alignment = 'leftAlone';
|
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
/**
|
|
|
|
* Should we check the valid_elements (& extended_valid_elements) rules from HTMLEditorConfig server side?
|
|
|
|
*
|
|
|
|
* @config
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
private static $sanitise_server_side = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Number of rows
|
|
|
|
*
|
|
|
|
* @config
|
|
|
|
* @var int
|
|
|
|
*/
|
2017-08-01 13:59:35 +12:00
|
|
|
private static $default_rows = 20;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Extra height per row
|
|
|
|
*
|
|
|
|
* @var int
|
|
|
|
*/
|
|
|
|
private static $fixed_row_height = 20;
|
2016-11-29 12:31:16 +13:00
|
|
|
|
|
|
|
/**
|
|
|
|
* ID or instance of editorconfig
|
|
|
|
*
|
|
|
|
* @var string|HTMLEditorConfig
|
|
|
|
*/
|
|
|
|
protected $editorConfig = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the HTMLEditorConfig instance
|
|
|
|
*
|
|
|
|
* @return HTMLEditorConfig
|
|
|
|
*/
|
|
|
|
public function getEditorConfig()
|
|
|
|
{
|
|
|
|
// Instance override
|
|
|
|
if ($this->editorConfig instanceof HTMLEditorConfig) {
|
|
|
|
return $this->editorConfig;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get named / active config
|
|
|
|
return HTMLEditorConfig::get($this->editorConfig);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Assign a new configuration instance or identifier
|
|
|
|
*
|
|
|
|
* @param string|HTMLEditorConfig $config
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setEditorConfig($config)
|
|
|
|
{
|
|
|
|
$this->editorConfig = $config;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a new HTMLEditorField.
|
|
|
|
* @see TextareaField::__construct()
|
|
|
|
*
|
|
|
|
* @param string $name The internal field name, passed to forms.
|
|
|
|
* @param string $title The human-readable field label.
|
|
|
|
* @param mixed $value The value of the field.
|
|
|
|
* @param string $config HTMLEditorConfig identifier to be used. Default to the active one.
|
|
|
|
*/
|
|
|
|
public function __construct($name, $title = null, $value = '', $config = null)
|
|
|
|
{
|
|
|
|
parent::__construct($name, $title, $value);
|
|
|
|
|
|
|
|
if ($config) {
|
|
|
|
$this->setEditorConfig($config);
|
|
|
|
}
|
|
|
|
|
2017-02-22 16:14:53 +13:00
|
|
|
$this->setRows(HTMLEditorField::config()->default_rows);
|
2016-11-29 12:31:16 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
public function getAttributes()
|
|
|
|
{
|
2017-08-01 13:59:35 +12:00
|
|
|
// Fix CSS height based on rows
|
|
|
|
$rowHeight = $this->config()->get('fixed_row_height');
|
|
|
|
$attributes = [];
|
|
|
|
if ($rowHeight) {
|
|
|
|
$height = $this->getRows() * $rowHeight;
|
|
|
|
$attributes['style'] = sprintf('height: %dpx;', $height);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Merge attributes
|
2016-11-29 12:31:16 +13:00
|
|
|
return array_merge(
|
2017-08-01 13:59:35 +12:00
|
|
|
$attributes,
|
2016-11-29 12:31:16 +13:00
|
|
|
parent::getAttributes(),
|
|
|
|
$this->getEditorConfig()->getAttributes()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param DataObject|DataObjectInterface $record
|
|
|
|
* @throws Exception
|
|
|
|
*/
|
|
|
|
public function saveInto(DataObjectInterface $record)
|
|
|
|
{
|
|
|
|
if ($record->hasField($this->name) && $record->escapeTypeForField($this->name) != 'xml') {
|
|
|
|
throw new Exception(
|
|
|
|
'HTMLEditorField->saveInto(): This field should save into a HTMLText or HTMLVarchar field.'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sanitise if requested
|
2017-04-18 10:34:18 +12:00
|
|
|
$htmlValue = HTMLValue::create($this->Value());
|
2017-02-22 16:14:53 +13:00
|
|
|
if (HTMLEditorField::config()->sanitise_server_side) {
|
2016-11-29 12:31:16 +13:00
|
|
|
$santiser = HTMLEditorSanitiser::create(HTMLEditorConfig::get_active());
|
|
|
|
$santiser->sanitise($htmlValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
// optionally manipulate the HTML after a TinyMCE edit and prior to a save
|
|
|
|
$this->extend('processHTML', $htmlValue);
|
|
|
|
|
|
|
|
// Store into record
|
|
|
|
$record->{$this->name} = $htmlValue->getContent();
|
|
|
|
}
|
|
|
|
|
2017-01-26 17:20:08 +13:00
|
|
|
public function setValue($value, $data = null)
|
2016-11-29 12:31:16 +13:00
|
|
|
{
|
|
|
|
// Regenerate links prior to preview, so that the editor can see them.
|
2017-06-29 18:45:17 +12:00
|
|
|
$value = ImageShortcodeProvider::regenerate_html_links($value);
|
2016-11-29 12:31:16 +13:00
|
|
|
return parent::setValue($value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return HTMLEditorField_Readonly
|
|
|
|
*/
|
|
|
|
public function performReadonlyTransformation()
|
|
|
|
{
|
2017-01-18 16:58:48 +13:00
|
|
|
return $this->castedCopy(HTMLEditorField_Readonly::class);
|
2016-11-29 12:31:16 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
public function performDisabledTransformation()
|
|
|
|
{
|
|
|
|
return $this->performReadonlyTransformation();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function Field($properties = array())
|
|
|
|
{
|
|
|
|
// Include requirements
|
|
|
|
$this->getEditorConfig()->init();
|
|
|
|
return parent::Field($properties);
|
|
|
|
}
|
2016-08-19 10:51:35 +12:00
|
|
|
}
|