2016-02-12 16:00:15 +13:00
|
|
|
<?php
|
|
|
|
|
2016-08-19 10:51:35 +12:00
|
|
|
namespace SilverStripe\Forms\HTMLEditor;
|
|
|
|
|
|
|
|
use SilverStripe\Core\Convert;
|
|
|
|
use SilverStripe\Control\Controller;
|
|
|
|
use SilverStripe\Control\Director;
|
|
|
|
use SilverStripe\View\Requirements;
|
|
|
|
use SilverStripe\View\SSViewer;
|
2016-07-19 14:09:15 +12:00
|
|
|
use SilverStripe\View\ThemeResourceLoader;
|
2016-08-19 10:51:35 +12:00
|
|
|
use TinyMCE_Compressor;
|
2016-07-14 00:36:52 +12:00
|
|
|
|
2016-02-12 16:00:15 +13:00
|
|
|
/**
|
|
|
|
* Default configuration for HtmlEditor specific to tinymce
|
|
|
|
*/
|
2016-05-06 11:43:47 +12:00
|
|
|
class TinyMCEConfig extends HTMLEditorConfig {
|
2016-02-12 16:00:15 +13:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Location of module relative to BASE_DIR. This must contain the following dirs
|
|
|
|
* - plugins
|
|
|
|
* - themes
|
|
|
|
* - skins
|
|
|
|
*
|
2016-11-01 17:40:59 +13:00
|
|
|
* If left blank defaults to ADMIN_THIRDPARTY_DIR . '/tinymce'
|
|
|
|
*
|
2016-02-12 16:00:15 +13:00
|
|
|
* @config
|
|
|
|
* @var string
|
|
|
|
*/
|
2016-11-01 17:40:59 +13:00
|
|
|
private static $base_dir = null;
|
2016-02-12 16:00:15 +13:00
|
|
|
|
|
|
|
/**
|
|
|
|
* TinyMCE JS settings
|
|
|
|
*
|
|
|
|
* @link https://www.tinymce.com/docs/configure/
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $settings = array(
|
|
|
|
'fix_list_elements' => true, // https://www.tinymce.com/docs/configure/content-filtering/#fix_list_elements
|
|
|
|
'friendly_name' => '(Please set a friendly name for this config)',
|
|
|
|
'priority' => 0, // used for Per-member config override
|
|
|
|
'browser_spellcheck' => true,
|
|
|
|
'body_class' => 'typography',
|
|
|
|
'elementpath' => false, // https://www.tinymce.com/docs/configure/editor-appearance/#elementpath
|
|
|
|
'relative_urls' => true,
|
|
|
|
'remove_script_host' => true,
|
|
|
|
'convert_urls' => false, // Prevent site-root images being rewritten to base relative
|
|
|
|
'menubar' => false,
|
2016-04-04 09:59:31 +12:00
|
|
|
'language' => 'en',
|
2016-02-12 16:00:15 +13:00
|
|
|
);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Holder list of enabled plugins
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $plugins = array(
|
|
|
|
'table' => null,
|
|
|
|
'emoticons' => null,
|
|
|
|
'paste' => null,
|
|
|
|
'code' => null,
|
|
|
|
'link' => null,
|
|
|
|
'importcss' => null,
|
|
|
|
);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Theme name
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $theme = 'modern';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the theme
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getTheme() {
|
|
|
|
return $this->theme;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the theme name
|
|
|
|
*
|
|
|
|
* @param string $theme
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setTheme($theme) {
|
|
|
|
$this->theme = $theme;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Holder list of buttons, organised by line. This array is 1-based indexed array
|
|
|
|
*
|
|
|
|
* {@link https://www.tinymce.com/docs/advanced/editor-control-identifiers/#toolbarcontrols}
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $buttons = array(
|
|
|
|
1 => array(
|
|
|
|
'bold', 'italic', 'underline', 'removeformat', '|',
|
|
|
|
'alignleft', 'aligncenter', 'alignright', 'alignjustify', '|',
|
|
|
|
'bullist', 'numlist', 'outdent', 'indent',
|
|
|
|
),
|
|
|
|
2 => array(
|
|
|
|
'formatselect', '|',
|
|
|
|
'paste', 'pastetext', '|',
|
|
|
|
'table', 'ssmedia', 'sslink', 'unlink', '|',
|
|
|
|
'code'
|
|
|
|
),
|
|
|
|
3 => array()
|
|
|
|
);
|
|
|
|
|
|
|
|
public function getOption($key) {
|
|
|
|
if(isset($this->settings[$key])) {
|
|
|
|
return $this->settings[$key];
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setOption($key,$value) {
|
|
|
|
$this->settings[$key] = $value;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setOptions($options) {
|
|
|
|
foreach ($options as $key => $value) {
|
|
|
|
$this->settings[$key] = $value;
|
|
|
|
}
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get all settings
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
protected function getSettings() {
|
|
|
|
return $this->settings;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getAttributes() {
|
|
|
|
return [
|
|
|
|
'data-editor' => 'tinyMCE', // Register ss.editorWrappers.tinyMCE
|
|
|
|
'data-config' => Convert::array2json($this->getConfig())
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enable one or several plugins. Will maintain unique list if already
|
|
|
|
* enabled plugin is re-passed. If passed in as a map of plugin-name to path,
|
|
|
|
* the plugin will be loaded by tinymce.PluginManager.load() instead of through tinyMCE.init().
|
|
|
|
* Keep in mind that these externals plugins require a dash-prefix in their name.
|
|
|
|
*
|
|
|
|
* @see http://wiki.moxiecode.com/index.php/TinyMCE:API/tinymce.PluginManager/load
|
|
|
|
*
|
|
|
|
* If passing in a non-associative array, the plugin name should be located in the standard tinymce
|
|
|
|
* plugins folder.
|
|
|
|
*
|
|
|
|
* If passing in an associative array, the key of each item should be the plugin name.
|
|
|
|
* The value of each item is one of:
|
|
|
|
* - null - Will be treated as a stardard plugin in the standard location
|
|
|
|
* - relative path - Will be treated as a relative url
|
|
|
|
* - absolute url - Some url to an external plugin
|
|
|
|
*
|
|
|
|
* @param string $plugin,... a string, or several strings, or a single array of strings - The plugins to enable
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function enablePlugins($plugin) {
|
|
|
|
$plugins = func_get_args();
|
|
|
|
if (is_array(current($plugins))) {
|
|
|
|
$plugins = current($plugins);
|
|
|
|
}
|
|
|
|
foreach ($plugins as $name => $path) {
|
|
|
|
// if plugins are passed without a path
|
|
|
|
if(is_numeric($name)) {
|
|
|
|
$name = $path;
|
|
|
|
$path = null;
|
|
|
|
}
|
|
|
|
if (!array_key_exists($name, $this->plugins)) {
|
|
|
|
$this->plugins[$name] = $path;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enable one or several plugins. Will properly handle being passed a plugin that is already disabled
|
|
|
|
* @param string $plugin,... a string, or several strings, or a single array of strings - The plugins to enable
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function disablePlugins($plugin) {
|
|
|
|
$plugins = func_get_args();
|
|
|
|
if (is_array(current($plugins))) {
|
|
|
|
$plugins = current($plugins);
|
|
|
|
}
|
|
|
|
foreach ($plugins as $name) {
|
|
|
|
unset($this->plugins[$name]);
|
|
|
|
}
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the list of all enabled plugins as an associative array.
|
|
|
|
* Array keys are the plugin names, and values are potentially the plugin location
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getPlugins() {
|
|
|
|
return $this->plugins;
|
|
|
|
}
|
|
|
|
|
2016-04-04 09:59:31 +12:00
|
|
|
/**
|
|
|
|
* Get list of plugins without custom locations, which is the set of
|
|
|
|
* plugins which can be loaded via the standard plugin path, and could
|
|
|
|
* potentially be minified
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getInternalPlugins() {
|
|
|
|
// Return only plugins with no custom url
|
|
|
|
$plugins = [];
|
|
|
|
foreach($this->getPlugins() as $name => $url) {
|
|
|
|
if(empty($url)) {
|
|
|
|
$plugins[] = $name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $plugins;
|
|
|
|
}
|
|
|
|
|
2016-02-12 16:00:15 +13:00
|
|
|
/**
|
|
|
|
* Get all button rows, skipping empty rows
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getButtons() {
|
|
|
|
return array_filter($this->buttons);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Totally re-set the buttons on a given line
|
|
|
|
*
|
|
|
|
* @param int $line The line number to redefine, from 1 to 3
|
|
|
|
* @param string $buttons,... A string or several strings, or a single array of strings.
|
|
|
|
* The button names to assign to this line.
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setButtonsForLine($line, $buttons) {
|
|
|
|
if (func_num_args() > 2) {
|
|
|
|
$buttons = func_get_args();
|
|
|
|
array_shift($buttons);
|
|
|
|
}
|
|
|
|
$this->buttons[$line] = is_array($buttons) ? $buttons : array($buttons);
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add buttons to the end of a line
|
|
|
|
* @param int $line The line number to redefine, from 1 to 3
|
|
|
|
* @param string $buttons,... A string or several strings, or a single array of strings.
|
|
|
|
* The button names to add to this line
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function addButtonsToLine($line, $buttons) {
|
|
|
|
if(func_num_args() > 2) {
|
|
|
|
$buttons = func_get_args();
|
|
|
|
array_shift($buttons);
|
|
|
|
}
|
|
|
|
if(!is_array($buttons)) {
|
|
|
|
$buttons = [$buttons];
|
|
|
|
}
|
|
|
|
foreach ($buttons as $button) {
|
|
|
|
$this->buttons[$line][] = $button;
|
|
|
|
}
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Internal function for adding and removing buttons related to another button
|
|
|
|
* @param string $name The name of the button to modify
|
|
|
|
* @param int $offset The offset relative to that button to perform an array_splice at.
|
|
|
|
* 0 for before $name, 1 for after.
|
|
|
|
* @param int $del The number of buttons to remove at the position given by index(string) + offset
|
|
|
|
* @param mixed $add An array or single item to insert at the position given by index(string) + offset,
|
|
|
|
* or null for no insertion
|
|
|
|
* @return bool True if $name matched a button, false otherwise
|
|
|
|
*/
|
|
|
|
protected function modifyButtons($name, $offset, $del=0, $add=null) {
|
|
|
|
foreach ($this->buttons as &$buttons) {
|
|
|
|
if (($idx = array_search($name, $buttons)) !== false) {
|
|
|
|
if ($add) {
|
|
|
|
array_splice($buttons, $idx + $offset, $del, $add);
|
|
|
|
} else {
|
|
|
|
array_splice($buttons, $idx + $offset, $del);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Insert buttons before the first occurance of another button
|
|
|
|
* @param string $before the name of the button to insert other buttons before
|
|
|
|
* @param string $buttons,... a string, or several strings, or a single array of strings.
|
|
|
|
* The button names to insert before that button
|
|
|
|
* @return bool True if insertion occured, false if it did not (because the given button name was not found)
|
|
|
|
*/
|
|
|
|
public function insertButtonsBefore($before, $buttons) {
|
|
|
|
if(func_num_args() > 2) {
|
|
|
|
$buttons = func_get_args();
|
|
|
|
array_shift($buttons);
|
|
|
|
}
|
|
|
|
if(!is_array($buttons)) {
|
|
|
|
$buttons = [$buttons];
|
|
|
|
}
|
|
|
|
return $this->modifyButtons($before, 0, 0, $buttons);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Insert buttons after the first occurance of another button
|
|
|
|
* @param string $after the name of the button to insert other buttons before
|
|
|
|
* @param string $buttons,... a string, or several strings, or a single array of strings.
|
|
|
|
* The button names to insert after that button
|
|
|
|
* @return bool True if insertion occured, false if it did not (because the given button name was not found)
|
|
|
|
*/
|
|
|
|
public function insertButtonsAfter($after, $buttons) {
|
|
|
|
if(func_num_args() > 2) {
|
|
|
|
$buttons = func_get_args();
|
|
|
|
array_shift($buttons);
|
|
|
|
}
|
|
|
|
if(!is_array($buttons)) {
|
|
|
|
$buttons = [$buttons];
|
|
|
|
}
|
|
|
|
return $this->modifyButtons($after, 1, 0, $buttons);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove the first occurance of buttons
|
|
|
|
* @param string $buttons,... one or more strings - the name of the buttons to remove
|
|
|
|
*/
|
|
|
|
public function removeButtons($buttons) {
|
|
|
|
if(func_num_args() > 1) {
|
|
|
|
$buttons = func_get_args();
|
|
|
|
}
|
|
|
|
if(!is_array($buttons)) {
|
|
|
|
$buttons = [$buttons];
|
|
|
|
}
|
|
|
|
foreach ($buttons as $button) {
|
|
|
|
$this->modifyButtons($button, 0, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate the JavaScript that will set TinyMCE's configuration:
|
|
|
|
* - Parse all configurations into JSON objects to be used in JavaScript
|
|
|
|
* - Includes TinyMCE and configurations using the {@link Requirements} system
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
protected function getConfig() {
|
|
|
|
$settings = $this->getSettings();
|
|
|
|
|
|
|
|
// https://www.tinymce.com/docs/configure/url-handling/#document_base_url
|
|
|
|
$settings['document_base_url'] = Director::absoluteBaseURL();
|
|
|
|
|
|
|
|
// https://www.tinymce.com/docs/api/class/tinymce.editormanager/#baseURL
|
|
|
|
$tinyMCEBaseURL = Controller::join_links(
|
|
|
|
Director::absoluteBaseURL(),
|
2016-11-01 17:40:59 +13:00
|
|
|
$this->config()->get('base_dir') ?: ADMIN_THIRDPARTY_DIR . '/tinymce'
|
2016-02-12 16:00:15 +13:00
|
|
|
);
|
|
|
|
$settings['baseURL'] = $tinyMCEBaseURL;
|
|
|
|
|
|
|
|
// map all plugins to absolute urls for loading
|
|
|
|
$plugins = array();
|
|
|
|
foreach($this->getPlugins() as $plugin => $path) {
|
|
|
|
if(!$path) {
|
|
|
|
// Empty paths: Convert to urls in standard base url
|
|
|
|
$path = Controller::join_links(
|
|
|
|
$tinyMCEBaseURL,
|
|
|
|
"plugins/{$plugin}/plugin.min.js"
|
|
|
|
);
|
|
|
|
} elseif(!Director::is_absolute_url($path)) {
|
|
|
|
// Non-absolute urls are made absolute
|
|
|
|
$path = Director::absoluteURL($path);
|
|
|
|
}
|
|
|
|
$plugins[$plugin] = $path;
|
|
|
|
}
|
|
|
|
|
|
|
|
// https://www.tinymce.com/docs/configure/integration-and-setup/#external_plugins
|
|
|
|
if($plugins) {
|
|
|
|
$settings['external_plugins'] = $plugins;
|
|
|
|
}
|
|
|
|
|
|
|
|
// https://www.tinymce.com/docs/configure/editor-appearance/#groupingtoolbarcontrols
|
|
|
|
$buttons = $this->getButtons();
|
|
|
|
$settings['toolbar'] = [];
|
|
|
|
foreach($buttons as $rowButtons) {
|
|
|
|
$row = implode(' ', $rowButtons);
|
|
|
|
if(count($buttons) > 1) {
|
|
|
|
$settings['toolbar'][] = $row;
|
|
|
|
} else {
|
|
|
|
$settings['toolbar'] = $row;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// https://www.tinymce.com/docs/configure/content-appearance/#content_css
|
|
|
|
$settings['content_css'] = $this->getEditorCSS();
|
|
|
|
|
|
|
|
// https://www.tinymce.com/docs/configure/editor-appearance/#theme_url
|
|
|
|
$theme = $this->getTheme();
|
|
|
|
if(!Director::is_absolute_url($theme)) {
|
|
|
|
$theme = Controller::join_links($tinyMCEBaseURL, "themes/{$theme}/theme.min.js");
|
|
|
|
}
|
|
|
|
$settings['theme_url'] = $theme;
|
|
|
|
|
|
|
|
// Send back
|
|
|
|
return $settings;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get location of all editor.css files
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
protected function getEditorCSS() {
|
|
|
|
$editor = array();
|
2016-07-14 00:36:52 +12:00
|
|
|
|
2016-07-19 14:09:15 +12:00
|
|
|
// Add standard editor.css
|
|
|
|
$editor[] = Director::absoluteURL(FRAMEWORK_ADMIN_DIR . '/client/dist/styles/editor.css');
|
2016-07-14 00:36:52 +12:00
|
|
|
|
2016-07-19 14:09:15 +12:00
|
|
|
// Themed editor.css
|
|
|
|
$themedEditor = ThemeResourceLoader::instance()->findThemedCSS('editor', SSViewer::get_themes());
|
|
|
|
if($themedEditor) {
|
|
|
|
$editor[] = Director::absoluteURL($themedEditor, Director::BASE);
|
2016-02-12 16:00:15 +13:00
|
|
|
}
|
2016-07-19 14:09:15 +12:00
|
|
|
|
2016-04-01 17:09:24 +13:00
|
|
|
return $editor;
|
|
|
|
}
|
2016-02-12 16:00:15 +13:00
|
|
|
|
2016-04-01 17:09:24 +13:00
|
|
|
/**
|
|
|
|
* Generate gzipped TinyMCE configuration including plugins and languages.
|
|
|
|
* This ends up "pre-loading" TinyMCE bundled with the required plugins
|
|
|
|
* so that multiple HTTP requests on the client don't need to be made.
|
2016-04-04 09:59:31 +12:00
|
|
|
*
|
|
|
|
* @return string
|
2016-04-01 17:09:24 +13:00
|
|
|
*/
|
2016-04-04 09:59:31 +12:00
|
|
|
public function getScriptURL() {
|
|
|
|
// If gzip is disabled just return core script url
|
2016-08-19 10:51:35 +12:00
|
|
|
$useGzip = HTMLEditorField::config()->get('use_gzip');
|
2016-04-04 09:59:31 +12:00
|
|
|
if(!$useGzip) {
|
2016-09-15 14:48:38 +12:00
|
|
|
return ADMIN_THIRDPARTY_DIR . '/tinymce/tinymce.min.js';
|
2016-04-01 17:09:24 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
// tinyMCE JS requirement
|
2016-09-15 14:48:38 +12:00
|
|
|
require_once ADMIN_THIRDPARTY_PATH . '/tinymce/tiny_mce_gzip.php';
|
2016-04-04 09:59:31 +12:00
|
|
|
$tag = TinyMCE_Compressor::renderTag(array(
|
2016-09-15 14:48:38 +12:00
|
|
|
'url' => ADMIN_THIRDPARTY_DIR . '/tinymce/tiny_mce_gzip.php',
|
2016-04-04 09:59:31 +12:00
|
|
|
'plugins' => implode(',', $this->getInternalPlugins()),
|
|
|
|
'themes' => $this->getTheme(),
|
|
|
|
'languages' => $this->getOption('language')
|
|
|
|
), true);
|
|
|
|
preg_match('/src="([^"]*)"/', $tag, $matches);
|
|
|
|
return html_entity_decode($matches[1]);
|
2016-04-01 17:09:24 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
public function init() {
|
|
|
|
// include TinyMCE Javascript
|
2016-04-04 09:59:31 +12:00
|
|
|
Requirements::javascript($this->getScriptURL());
|
2016-04-01 17:09:24 +13:00
|
|
|
}
|
2016-02-12 16:00:15 +13:00
|
|
|
}
|