mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 14:05:37 +02:00
Pulls/4/remove reliance on admin dir (#6876)
* Stop relying on external constants * Revise getTinyMCEPath method to throw exception when no path can be computed * Throw exception on no gzip, better admin module check
This commit is contained in:
parent
ff35baaeab
commit
4af71b9ed7
@ -5,11 +5,13 @@ namespace SilverStripe\Forms\HTMLEditor;
|
||||
use SilverStripe\Core\Convert;
|
||||
use SilverStripe\Control\Controller;
|
||||
use SilverStripe\Control\Director;
|
||||
use SilverStripe\Core\Manifest\ModuleLoader;
|
||||
use SilverStripe\i18n\i18n;
|
||||
use SilverStripe\View\Requirements;
|
||||
use SilverStripe\View\SSViewer;
|
||||
use SilverStripe\View\ThemeResourceLoader;
|
||||
use TinyMCE_Compressor;
|
||||
use Exception;
|
||||
|
||||
/**
|
||||
* Default configuration for HtmlEditor specific to tinymce
|
||||
@ -189,7 +191,7 @@ class TinyMCEConfig extends HTMLEditorConfig
|
||||
* - themes
|
||||
* - skins
|
||||
*
|
||||
* If left blank defaults to ADMIN_THIRDPARTY_DIR . '/tinymce'
|
||||
* If left blank defaults to [admin dir]/tinyme
|
||||
*
|
||||
* @config
|
||||
* @var string
|
||||
@ -555,8 +557,7 @@ class TinyMCEConfig extends HTMLEditorConfig
|
||||
// https://www.tinymce.com/docs/api/class/tinymce.editormanager/#baseURL
|
||||
$tinyMCEBaseURL = Controller::join_links(
|
||||
Director::absoluteBaseURL(),
|
||||
TinyMCEConfig::config()->get('base_dir')
|
||||
?: ADMIN_THIRDPARTY_DIR . '/tinymce'
|
||||
$this->getTinyMCEPath()
|
||||
);
|
||||
$settings['baseURL'] = $tinyMCEBaseURL;
|
||||
|
||||
@ -617,7 +618,7 @@ class TinyMCEConfig extends HTMLEditorConfig
|
||||
$editor = array();
|
||||
|
||||
// Add standard editor.css
|
||||
$editor[] = Director::absoluteURL(FRAMEWORK_ADMIN_DIR . '/client/dist/styles/editor.css');
|
||||
$editor[] = Director::absoluteURL(ltrim($this->getAdminPath() . '/client/dist/styles/editor.css', '/'));
|
||||
|
||||
// Themed editor.css
|
||||
$themedEditor = ThemeResourceLoader::instance()->findThemedCSS('editor', SSViewer::get_themes());
|
||||
@ -640,18 +641,25 @@ class TinyMCEConfig extends HTMLEditorConfig
|
||||
// If gzip is disabled just return core script url
|
||||
$useGzip = HTMLEditorField::config()->get('use_gzip');
|
||||
if (!$useGzip) {
|
||||
return ADMIN_THIRDPARTY_DIR . '/tinymce/tinymce.min.js';
|
||||
return $this->getTinyMCEPath() . '/tinymce.min.js';
|
||||
}
|
||||
|
||||
// tinyMCE JS requirement
|
||||
require_once ADMIN_THIRDPARTY_PATH . '/tinymce/tiny_mce_gzip.php';
|
||||
$gzipPath = BASE_PATH . '/' . $this->getTinyMCEPath() . '/tiny_mce_gzip.php';
|
||||
if (!file_exists($gzipPath)) {
|
||||
throw new Exception("HTMLEditorField.use_gzip enabled, but file $gzipPath does not exist!");
|
||||
}
|
||||
|
||||
require_once $gzipPath;
|
||||
|
||||
$tag = TinyMCE_Compressor::renderTag(array(
|
||||
'url' => ADMIN_THIRDPARTY_DIR . '/tinymce/tiny_mce_gzip.php',
|
||||
'url' => $this->getTinyMCEPath() . '/tiny_mce_gzip.php',
|
||||
'plugins' => implode(',', $this->getInternalPlugins()),
|
||||
'themes' => $this->getTheme(),
|
||||
'languages' => $this->getOption('language')
|
||||
), true);
|
||||
preg_match('/src="([^"]*)"/', $tag, $matches);
|
||||
|
||||
return html_entity_decode($matches[1]);
|
||||
}
|
||||
|
||||
@ -676,4 +684,45 @@ class TinyMCEConfig extends HTMLEditorConfig
|
||||
}
|
||||
return 'en';
|
||||
}
|
||||
|
||||
/**
|
||||
* @return string|false
|
||||
*/
|
||||
public function getAdminPath()
|
||||
{
|
||||
$module = $this->getAdminModule();
|
||||
if ($module) {
|
||||
return $module->getRelativePath();
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return string|false
|
||||
*/
|
||||
public function getTinyMCEPath()
|
||||
{
|
||||
$configDir = static::config()->get('base_dir');
|
||||
if ($configDir) {
|
||||
return $configDir;
|
||||
}
|
||||
|
||||
if ($admin = $this->getAdminModule()) {
|
||||
return $admin->getResourcePath('thirdparty/tinymce');
|
||||
}
|
||||
|
||||
throw new Exception(sprintf(
|
||||
'If the silverstripe/admin module is not installed,
|
||||
you must set the TinyMCE path in %s.base_dir',
|
||||
__CLASS__
|
||||
));
|
||||
}
|
||||
|
||||
/**
|
||||
* @return \SilverStripe\Core\Manifest\Module
|
||||
*/
|
||||
protected function getAdminModule()
|
||||
{
|
||||
return ModuleLoader::instance()->getManifest()->getModule('silverstripe/admin');
|
||||
}
|
||||
}
|
||||
|
@ -2,16 +2,27 @@
|
||||
|
||||
namespace SilverStripe\Forms\Tests\HTMLEditor;
|
||||
|
||||
use SilverStripe\Control\Director;
|
||||
use SilverStripe\Core\Config\Config;
|
||||
use SilverStripe\Core\Convert;
|
||||
use SilverStripe\Core\Manifest\ModuleLoader;
|
||||
use SilverStripe\Core\Manifest\ModuleManifest;
|
||||
use SilverStripe\Dev\SapphireTest;
|
||||
use SilverStripe\Forms\HTMLEditor\HTMLEditorField;
|
||||
use SilverStripe\Forms\HTMLEditor\TinyMCEConfig;
|
||||
use SilverStripe\Forms\HTMLEditor\HTMLEditorConfig;
|
||||
use \Exception;
|
||||
|
||||
class HTMLEditorConfigTest extends SapphireTest
|
||||
{
|
||||
|
||||
protected function setUp()
|
||||
{
|
||||
parent::setUp();
|
||||
|
||||
TinyMCEConfig::config()->set('base_dir', 'test/thirdparty/tinymce');
|
||||
}
|
||||
|
||||
public function testEnablePluginsByString()
|
||||
{
|
||||
$c = new TinyMCEConfig();
|
||||
@ -37,18 +48,18 @@ class HTMLEditorConfigTest extends SapphireTest
|
||||
|
||||
public function testEnablePluginsByArrayWithPaths()
|
||||
{
|
||||
Config::inst()->update('SilverStripe\\Control\\Director', 'alternate_base_url', 'http://mysite.com/subdir');
|
||||
Config::inst()->update(Director::class, 'alternate_base_url', 'http://mysite.com/subdir');
|
||||
$c = new TinyMCEConfig();
|
||||
$c->setTheme('modern');
|
||||
$c->setOption('language', 'es');
|
||||
$c->disablePlugins('table', 'emoticons', 'paste', 'code', 'link', 'importcss');
|
||||
$c->enablePlugins(
|
||||
array(
|
||||
'plugin1' => 'mypath/plugin1.js',
|
||||
'plugin2' => '/anotherbase/mypath/plugin2.js',
|
||||
'plugin3' => 'https://www.google.com/plugin.js',
|
||||
'plugin4' => null,
|
||||
'plugin5' => null,
|
||||
'plugin1' => 'mypath/plugin1.js',
|
||||
'plugin2' => '/anotherbase/mypath/plugin2.js',
|
||||
'plugin3' => 'https://www.google.com/plugin.js',
|
||||
'plugin4' => null,
|
||||
'plugin5' => null,
|
||||
)
|
||||
);
|
||||
$attributes = $c->getAttributes();
|
||||
@ -80,24 +91,51 @@ class HTMLEditorConfigTest extends SapphireTest
|
||||
// Plugin specified with standard location
|
||||
$this->assertContains('plugin4', array_keys($plugins));
|
||||
$this->assertEquals(
|
||||
'http://mysite.com/subdir/'.ADMIN_THIRDPARTY_DIR.'/tinymce/plugins/plugin4/plugin.min.js',
|
||||
'http://mysite.com/subdir/test/thirdparty/tinymce/plugins/plugin4/plugin.min.js',
|
||||
$plugins['plugin4']
|
||||
);
|
||||
|
||||
// Check that internal plugins are extractable separately
|
||||
$this->assertEquals(['plugin4', 'plugin5'], $c->getInternalPlugins());
|
||||
}
|
||||
|
||||
public function testPluginCompression()
|
||||
{
|
||||
$module = ModuleLoader::instance()->getManifest()->getModule('silverstripe/admin');
|
||||
if (!$module) {
|
||||
$this->markTestSkipped('No silverstripe/admin module loaded');
|
||||
}
|
||||
TinyMCEConfig::config()->remove('base_dir');
|
||||
Config::inst()->update(Director::class, 'alternate_base_url', 'http://mysite.com/subdir');
|
||||
$c = new TinyMCEConfig();
|
||||
$c->setTheme('modern');
|
||||
$c->setOption('language', 'es');
|
||||
$c->disablePlugins('table', 'emoticons', 'paste', 'code', 'link', 'importcss');
|
||||
$c->enablePlugins(
|
||||
array(
|
||||
'plugin1' => 'mypath/plugin1.js',
|
||||
'plugin2' => '/anotherbase/mypath/plugin2.js',
|
||||
'plugin3' => 'https://www.google.com/plugin.js',
|
||||
'plugin4' => null,
|
||||
'plugin5' => null,
|
||||
)
|
||||
);
|
||||
$attributes = $c->getAttributes();
|
||||
$config = Convert::json2array($attributes['data-config']);
|
||||
$plugins = $config['external_plugins'];
|
||||
$this->assertNotEmpty($plugins);
|
||||
|
||||
// Test plugins included via gzip compresser
|
||||
HTMLEditorField::config()->update('use_gzip', true);
|
||||
$this->assertEquals(
|
||||
ADMIN_THIRDPARTY_DIR . '/tinymce/tiny_mce_gzip.php?js=1&plugins=plugin4,plugin5&themes=modern&languages=es&diskcache=true&src=true',
|
||||
'silverstripe-admin/thirdparty/tinymce/tiny_mce_gzip.php?js=1&plugins=plugin4,plugin5&themes=modern&languages=es&diskcache=true&src=true',
|
||||
$c->getScriptURL()
|
||||
);
|
||||
|
||||
// If gzip is disabled only the core plugin is loaded
|
||||
HTMLEditorField::config()->remove('use_gzip');
|
||||
$this->assertEquals(
|
||||
ADMIN_THIRDPARTY_DIR . '/tinymce/tinymce.min.js',
|
||||
'silverstripe-admin/thirdparty/tinymce/tinymce.min.js',
|
||||
$c->getScriptURL()
|
||||
);
|
||||
}
|
||||
@ -149,4 +187,40 @@ class HTMLEditorConfigTest extends SapphireTest
|
||||
$this->assertNotEmpty($aAttributes['data-config']);
|
||||
$this->assertNotEmpty($cAttributes['data-config']);
|
||||
}
|
||||
|
||||
public function testExceptionThrownWhenTinyMCEPathCannotBeComputed()
|
||||
{
|
||||
TinyMCEConfig::config()->remove('base_dir');
|
||||
ModuleLoader::instance()->pushManifest(new ModuleManifest(
|
||||
dirname(__FILE__),
|
||||
false
|
||||
));
|
||||
$c = new TinyMCEConfig();
|
||||
|
||||
$this->setExpectedExceptionRegExp(
|
||||
Exception::class,
|
||||
'/module is not installed/'
|
||||
);
|
||||
|
||||
$c->getScriptURL();
|
||||
|
||||
ModuleLoader::instance()->popManifest();
|
||||
}
|
||||
|
||||
public function testExceptionThrownWhenTinyMCEGZipPathDoesntExist()
|
||||
{
|
||||
HTMLEditorField::config()->set('use_gzip', true);
|
||||
$stub = $this->getMockBuilder(TinyMCEConfig::class)
|
||||
->setMethods(['getTinyMCEPath'])
|
||||
->getMock();
|
||||
$stub->method('getTinyMCEPath')
|
||||
->willReturn('fail');
|
||||
|
||||
$this->setExpectedExceptionRegExp(
|
||||
Exception::class,
|
||||
'/does not exist/'
|
||||
);
|
||||
|
||||
$stub->getScriptURL();
|
||||
}
|
||||
}
|
||||
|
@ -21,6 +21,7 @@ use SilverStripe\Forms\HTMLReadonlyField;
|
||||
use SilverStripe\Forms\Tests\HTMLEditor\HTMLEditorFieldTest\DummyMediaFormFieldExtension;
|
||||
use SilverStripe\Forms\Tests\HTMLEditor\HTMLEditorFieldTest\TestObject;
|
||||
use SilverStripe\ORM\FieldType\DBHTMLText;
|
||||
use SilverStripe\Forms\HTMLEditor\TinyMCEConfig;
|
||||
|
||||
class HTMLEditorFieldTest extends FunctionalTest
|
||||
{
|
||||
@ -77,6 +78,10 @@ class HTMLEditorFieldTest extends FunctionalTest
|
||||
|
||||
public function testCasting()
|
||||
{
|
||||
// Shim TinyMCE so silverstripe/admin doesn't have to be installed
|
||||
TinyMCEConfig::config()->set('base_dir', 'test');
|
||||
HtmlEditorField::config()->set('use_gzip', false);
|
||||
|
||||
// Test special characters
|
||||
$inputText = "These are some unicodes: ä, ö, & ü";
|
||||
$field = new HTMLEditorField("Test", "Test");
|
||||
|
Loading…
Reference in New Issue
Block a user