CKEditor5.php in Drupal 10
Namespace
Drupal\ckeditor5\Plugin\EditorFile
core/modules/ckeditor5/src/Plugin/Editor/CKEditor5.phpView source
<?php
declare (strict_types=1);
namespace Drupal\ckeditor5\Plugin\Editor;
use Drupal\ckeditor5\CKEditor5StylesheetsMessage;
use Drupal\ckeditor5\HTMLRestrictions;
use Drupal\ckeditor5\Plugin\CKEditor5Plugin\Heading;
use Drupal\ckeditor5\Plugin\CKEditor5PluginDefinition;
use Drupal\ckeditor5\Plugin\CKEditor5PluginManagerInterface;
use Drupal\Component\Serialization\Json;
use Drupal\Component\Utility\NestedArray;
use Drupal\Core\Cache\CacheBackendInterface;
use Drupal\Core\Config\Schema\SchemaCheckTrait;
use Drupal\Core\Extension\ModuleHandlerInterface;
use Drupal\Core\Form\FormStateInterface;
use Drupal\Core\Form\SubformState;
use Drupal\Core\Form\SubformStateInterface;
use Drupal\Core\Language\LanguageManagerInterface;
use Drupal\Core\Messenger\MessengerInterface;
use Drupal\Core\Plugin\ContainerFactoryPluginInterface;
use Drupal\ckeditor5\SmartDefaultSettings;
use Drupal\Core\Validation\Plugin\Validation\Constraint\PrimitiveTypeConstraint;
use Drupal\editor\EditorInterface;
use Drupal\editor\Entity\Editor;
use Drupal\editor\Plugin\EditorBase;
use Drupal\filter\FilterFormatInterface;
use Psr\Log\LoggerInterface;
use Symfony\Component\DependencyInjection\ContainerInterface;
use Symfony\Component\Validator\ConstraintViolation;
use Symfony\Component\Validator\ConstraintViolationListInterface;
/**
* Defines a CKEditor 5-based text editor for Drupal.
*
* @Editor(
* id = "ckeditor5",
* label = @Translation("CKEditor 5"),
* supports_content_filtering = TRUE,
* supports_inline_editing = TRUE,
* is_xss_safe = FALSE,
* supported_element_types = {
* "textarea"
* }
* )
*
* @internal
* Plugin classes are internal.
*/
class CKEditor5 extends EditorBase implements ContainerFactoryPluginInterface {
use SchemaCheckTrait;
/**
* The CKEditor plugin manager.
*
* @var \Drupal\ckeditor5\Plugin\CKEditor5PluginManagerInterface
*/
protected $ckeditor5PluginManager;
/**
* The language manager.
*
* @var \Drupal\Core\Language\LanguageManagerInterface
*/
protected $languageManager;
/**
* The module handler.
*
* @var \Drupal\Core\Extension\ModuleHandlerInterface
*/
protected $moduleHandler;
/**
* Smart default settings utility.
*
* @var \Drupal\ckeditor5\SmartDefaultSettings
*/
protected $smartDefaultSettings;
/**
* The set of configured CKEditor 5 plugins.
*
* @var \Drupal\ckeditor5\Plugin\CKEditor5PluginInterface[]
*/
private $plugins = [];
/**
* The submitted editor.
*
* @var \Drupal\editor\EditorInterface
*/
private $submittedEditor;
/**
* The cache.
*
* @var \Drupal\Core\Cache\CacheBackendInterface
*/
protected $cache;
/**
* The ckeditor_stylesheets message utility.
*
* @var \Drupal\ckeditor5\CKEditor5StylesheetsMessage
*/
private $stylesheetsMessage;
/**
* A logger instance.
*
* @var \Psr\Log\LoggerInterface
*/
protected $logger;
/**
* Constructs a CKEditor5 editor plugin.
*
* @param array $configuration
* A configuration array containing information about the plugin instance.
* @param string $plugin_id
* The plugin_id for the plugin instance.
* @param mixed $plugin_definition
* The plugin implementation definition.
* @param \Drupal\ckeditor5\Plugin\CKEditor5PluginManagerInterface $ckeditor5_plugin_manager
* The CKEditor5 plugin manager.
* @param \Drupal\Core\Language\LanguageManagerInterface $language_manager
* The language manager
* @param \Drupal\Core\Extension\ModuleHandlerInterface $module_handler
* The module handler.
* @param \Drupal\ckeditor5\SmartDefaultSettings $smart_default_settings
* The smart default settings utility.
* @param \Drupal\Core\Cache\CacheBackendInterface $cache
* The cache.
* @param \Drupal\ckeditor5\CKEditor5StylesheetsMessage $stylesheets_message
* The ckeditor_stylesheets message utility.
* @param \Psr\Log\LoggerInterface $logger
* A logger instance.
*/
public function __construct(array $configuration, $plugin_id, $plugin_definition, CKEditor5PluginManagerInterface $ckeditor5_plugin_manager, LanguageManagerInterface $language_manager, ModuleHandlerInterface $module_handler, SmartDefaultSettings $smart_default_settings, CacheBackendInterface $cache, CKEditor5StylesheetsMessage $stylesheets_message, LoggerInterface $logger) {
parent::__construct($configuration, $plugin_id, $plugin_definition);
$this->ckeditor5PluginManager = $ckeditor5_plugin_manager;
$this->languageManager = $language_manager;
$this->moduleHandler = $module_handler;
$this->smartDefaultSettings = $smart_default_settings;
$this->cache = $cache;
$this->stylesheetsMessage = $stylesheets_message;
$this->logger = $logger;
}
/**
* {@inheritdoc}
*/
public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) {
return new static($configuration, $plugin_id, $plugin_definition, $container
->get('plugin.manager.ckeditor5.plugin'), $container
->get('language_manager'), $container
->get('module_handler'), $container
->get('ckeditor5.smart_default_settings'), $container
->get('cache.default'), $container
->get('ckeditor5.stylesheets.message'), $container
->get('logger.channel.ckeditor5'));
}
/**
* {@inheritdoc}
*/
public function getDefaultSettings() {
return [
'toolbar' => [
'items' => [
'heading',
'bold',
'italic',
],
],
'plugins' => [
'ckeditor5_heading' => Heading::DEFAULT_CONFIGURATION,
],
];
}
/**
* Validates a Text Editor + Text Format pair.
*
* Drupal is designed to only verify schema conformity (and validation) of
* individual config entities. The Text Editor module layers a tightly coupled
* Editor entity on top of the Filter module's FilterFormat config entity.
* This inextricable coupling is clearly visible in EditorInterface:
* \Drupal\editor\EditorInterface::getFilterFormat(). They are always paired.
* Because not every text editor is guaranteed to be compatible with every
* text format, the pair must be validated.
*
* @param \Drupal\editor\EditorInterface $text_editor
* The paired text editor to validate.
* @param \Drupal\filter\FilterFormatInterface $text_format
* The paired text format to validate.
* @param bool $all_compatibility_problems
* Get all compatibility problems (default) or only fundamental ones.
*
* @return \Symfony\Component\Validator\ConstraintViolationListInterface
* The validation constraint violations.
*
* @throws \InvalidArgumentException
* Thrown when the text editor is not configured to use CKEditor 5.
*
* @see \Drupal\editor\EditorInterface::getFilterFormat()
* @see ckeditor5.pair.schema.yml
*/
public static function validatePair(EditorInterface $text_editor, FilterFormatInterface $text_format, bool $all_compatibility_problems = TRUE) : ConstraintViolationListInterface {
if ($text_editor
->getEditor() !== 'ckeditor5') {
throw new \InvalidArgumentException('This text editor is not configured to use CKEditor 5.');
}
$typed_config_manager = \Drupal::getContainer()
->get('config.typed');
$typed_config = $typed_config_manager
->createFromNameAndData('ckeditor5_valid_pair__format_and_editor', [
// A mix of:
// - editor.editor.*.settings — note that "settings" is top-level in
// editor.editor.*, and so it is here, so all validation constraints
// will continue to work fine.
'settings' => $text_editor
->toArray()['settings'],
// - filter.format.*.filters — note that "filters" is top-level in
// filter.format.*, and so it is here, so all validation constraints
// will continue to work fine.
'filters' => $text_format
->toArray()['filters'],
// - editor.editor.*.image_upload — note that "image_upload" is
// top-level in editor.editor.*, and so it is here, so all validation
// constraints will continue to work fine.
'image_upload' => $text_editor
->toArray()['image_upload'],
]);
$violations = $typed_config
->validate();
// Only consider validation constraint violations covering the pair, so not
// irrelevant details such as a PrimitiveTypeConstraint in filter settings,
// which do not affect CKEditor 5 anyway.
foreach ($violations as $i => $violation) {
assert($violation instanceof ConstraintViolation);
if (explode('.', $violation
->getPropertyPath())[0] === 'filters' && is_a($violation
->getConstraint(), PrimitiveTypeConstraint::class)) {
$violations
->remove($i);
}
}
if (!$all_compatibility_problems) {
foreach ($violations as $i => $violation) {
// Remove all violations that are not fundamental — these are at the
// root (property path '').
if ($violation
->getPropertyPath() !== '') {
$violations
->remove($i);
}
}
}
return $violations;
}
/**
* {@inheritdoc}
*/
public function buildConfigurationForm(array $form, FormStateInterface $form_state) {
$editor = $form_state
->get('editor');
assert($editor instanceof Editor);
$language = $this->languageManager
->getCurrentLanguage();
// When enabling CKEditor 5, generate sensible settings from the
// pre-existing text editor/format rather than the hardcoded defaults
// whenever possible.
// @todo Remove after https://www.drupal.org/project/drupal/issues/3226673.
$format = $form_state
->getFormObject()
->getEntity();
assert($format instanceof FilterFormatInterface);
if ($editor
->isNew() && !$form_state
->get('ckeditor5_is_active') && $form_state
->get('ckeditor5_is_selected')) {
assert($editor
->getSettings() === $this
->getDefaultSettings());
if (!$format
->isNew()) {
[
$editor,
$messages,
] = $this->smartDefaultSettings
->computeSmartDefaultSettings($editor, $format);
$form_state
->set('used_smart_default_settings', TRUE);
foreach ($messages as $type => $messages_per_type) {
foreach ($messages_per_type as $message) {
$this
->messenger()
->addMessage($message, $type);
}
}
if (isset($messages[MessengerInterface::TYPE_WARNING]) || isset($messages[MessengerInterface::TYPE_ERROR])) {
$this
->messenger()
->addMessage($this
->t('Check <a href=":handbook">this handbook page</a> for details about compatibility issues of contrib modules.', [
':handbook' => 'https://www.drupal.org/node/3273985',
]), MessengerInterface::TYPE_WARNING);
}
}
$eventual_editor_and_format = $this
->getEventualEditorWithPrimedFilterFormat($form_state, $editor);
// Provide the validated eventual pair in form state to
// ::getGeneratedAllowedHtmlValue(), to update filter_html's
// "allowed_html".
$form_state
->set('ckeditor5_validated_pair', $eventual_editor_and_format);
// Ensure that CKEditor 5 plugins that need to interact with the Editor
// config entity are able to access the computed Editor, which was cloned
// from $form_state->get('editor').
// @see \Drupal\ckeditor5\Plugin\CKEditor5Plugin\ImageUpload::buildConfigurationForm
$form_state
->set('editor', $editor);
}
if ($css_warning = $this->stylesheetsMessage
->getWarning()) {
// Explicitly render this single warning message visually close to the
// text editor since this is a very long form. Otherwise, it may be
// interpreted as a text format problem, or ignored entirely.
// All other messages will be rendered in the default location.
// @see \Drupal\Core\Render\Element\StatusMessages
$form['css_warning'] = [
'#theme' => 'status_messages',
'#message_list' => [
'warning' => [
$css_warning,
],
],
'#status_headings' => [
'warning' => t('Warning message'),
],
];
}
// AJAX validation errors should appear visually close to the text editor
// since this is a very long form: otherwise they would not be noticed.
$form['real_time_validation_errors_location'] = [
'#type' => 'container',
'#id' => 'ckeditor5-realtime-validation-messages-container',
];
$form['toolbar'] = [
'#type' => 'container',
'#title' => $this
->t('CKEditor 5 toolbar configuration'),
'#theme' => 'ckeditor5_settings_toolbar',
'#attached' => [
'library' => $this->ckeditor5PluginManager
->getAdminLibraries(),
'drupalSettings' => [
'ckeditor5' => [
'language' => [
'dir' => $language
->getDirection(),
'langcode' => $language
->getId(),
],
],
],
],
];
$form['available_items_description'] = [
'#type' => 'container',
'#markup' => $this
->t('Press the down arrow key to add to the toolbar.'),
'#id' => 'available-button-description',
'#attributes' => [
'class' => [
'visually-hidden',
],
],
];
$form['active_items_description'] = [
'#type' => 'container',
'#markup' => $this
->t('Move this button in the toolbar by pressing the left or right arrow keys. Press the up arrow key to remove from the toolbar.'),
'#id' => 'active-button-description',
'#attributes' => [
'class' => [
'visually-hidden',
],
],
];
// The items are encoded in markup to provide a no-JS fallback.
// Although CKEditor 5 is useless without JS it would still be possible
// to see all the available toolbar items provided by plugins in the format
// that needs to be entered in the textarea. The UI app parses this list.
$form['toolbar']['available'] = [
'#type' => 'container',
'#title' => 'Available items',
'#id' => 'ckeditor5-toolbar-buttons-available',
'available_items' => [
'#markup' => Json::encode($this->ckeditor5PluginManager
->getToolbarItems()),
],
];
$editor_settings = $editor
->getSettings();
// This form field requires a JSON-style array of valid toolbar items.
// e.g. ["bold","italic","|","uploadImage"].
// CKEditor 5 config for toolbar items takes an array of strings which
// correspond to the keys under toolbar_items in a plugin yml or annotation.
// @see https://ckeditor.com/docs/ckeditor5/latest/features/toolbar/toolbar.html
$form['toolbar']['items'] = [
'#type' => 'textarea',
'#title' => $this
->t('Toolbar items'),
'#rows' => 1,
'#default_value' => Json::encode($editor_settings['toolbar']['items']),
'#id' => 'ckeditor5-toolbar-buttons-selected',
'#attributes' => [
'tabindex' => '-1',
'aria-hidden' => 'true',
],
];
$form['plugin_settings'] = [
'#type' => 'vertical_tabs',
'#title' => $this
->t('CKEditor5 plugin settings'),
'#id' => 'ckeditor5-plugin-settings',
];
$this
->injectPluginSettingsForm($form, $form_state, $editor);
// Allow reliable detection of switching to CKEditor 5 from another text
// editor (or none at all).
$form['is_already_using_ckeditor5'] = [
'#type' => 'hidden',
'#default_value' => TRUE,
];
return $form;
}
/**
* Determines whether the plugin settings form should be visible.
*
* @param \Drupal\ckeditor5\Plugin\CKEditor5PluginDefinition $definition
* The configurable CKEditor 5 plugin to assess the visibility for.
* @param \Drupal\editor\EditorInterface $editor
* A configured text editor object.
*
* @return bool
* Whether this configurable plugin's settings form should be visible.
*/
private function shouldHaveVisiblePluginSettingsForm(CKEditor5PluginDefinition $definition, EditorInterface $editor) : bool {
assert($definition
->isConfigurable());
$enabled_plugins = $this->ckeditor5PluginManager
->getEnabledDefinitions($editor);
$plugin_id = $definition
->id();
// Enabled plugins should be configurable.
if (isset($enabled_plugins[$plugin_id])) {
return TRUE;
}
// There are two circumstances where a plugin not listed in $enabled_plugins
// due to isEnabled() returning false, that should still have its config
// form provided:
// 1 - A conditionally enabled plugin that does not depend on a toolbar item
// to be active AND the plugins it depends on are enabled (if any) AND the
// filter it depends on is enabled (if any).
// 2 - A conditionally enabled plugin that does depend on a toolbar item,
// and that toolbar item is active.
if ($definition
->hasConditions()) {
$conditions = $definition
->getConditions();
if (!array_key_exists('toolbarItem', $conditions)) {
$conclusion = TRUE;
// The filter this plugin depends on must be enabled.
if (array_key_exists('filter', $conditions)) {
$required_filter = $conditions['filter'];
$format_filters = $editor
->getFilterFormat()
->filters();
$conclusion = $conclusion && $format_filters
->has($required_filter) && $format_filters
->get($required_filter)->status;
}
// The CKEditor 5 plugins this plugin depends on must be enabled.
if (array_key_exists('plugins', $conditions)) {
$all_plugins = $this->ckeditor5PluginManager
->getDefinitions();
$dependencies = array_intersect_key($all_plugins, array_flip($conditions['plugins']));
$unmet_dependencies = array_diff_key($dependencies, $enabled_plugins);
$conclusion = $conclusion && empty($unmet_dependencies);
}
return $conclusion;
}
elseif (in_array($conditions['toolbarItem'], $editor
->getSettings()['toolbar']['items'], TRUE)) {
return TRUE;
}
}
return FALSE;
}
/**
* Injects the CKEditor plugins settings forms as a vertical tabs subform.
*
* @param array &$form
* A reference to an associative array containing the structure of the form.
* @param \Drupal\Core\Form\FormStateInterface $form_state
* The current state of the form.
* @param \Drupal\editor\EditorInterface $editor
* A configured text editor object.
*/
private function injectPluginSettingsForm(array &$form, FormStateInterface $form_state, EditorInterface $editor) : void {
$definitions = $this->ckeditor5PluginManager
->getDefinitions();
$eventual_editor_and_format = $this
->getEventualEditorWithPrimedFilterFormat($form_state, $editor);
foreach ($definitions as $plugin_id => $definition) {
if ($definition
->isConfigurable() && $this
->shouldHaveVisiblePluginSettingsForm($definition, $eventual_editor_and_format)) {
$plugin = $this->ckeditor5PluginManager
->getPlugin($plugin_id, $editor);
$plugin_settings_form = [];
$form['plugins'][$plugin_id] = [
'#type' => 'details',
'#title' => $definition
->label(),
'#open' => TRUE,
'#group' => 'editor][settings][plugin_settings',
'#attributes' => [
'data-ckeditor5-plugin-id' => $plugin_id,
],
];
$form['plugins'][$plugin_id] += $plugin
->buildConfigurationForm($plugin_settings_form, $form_state);
}
}
}
/**
* Form #after_build callback: provides text editor state changes.
*
* Updates the internal $this->entity object with submitted values when the
* form is being rebuilt (e.g. submitted via AJAX), so that subsequent
* processing (e.g. AJAX callbacks) can rely on it.
*
* @see \Drupal\Core\Entity\EntityForm::afterBuild()
*/
public static function assessActiveTextEditorAfterBuild(array $element, FormStateInterface $form_state) : array {
// The case of the form being built initially, and the text editor plugin in
// use is already CKEditor 5.
if (!$form_state
->isProcessingInput()) {
$editor = $form_state
->get('editor');
$already_using_ckeditor5 = $editor && $editor
->getEditor() === 'ckeditor5';
}
else {
// Whenever there is user input, this cannot be the initial build of the
// form and hence we need to inspect user input.
$already_using_ckeditor5 = FALSE;
NestedArray::getValue($form_state
->getUserInput(), [
'editor',
'settings',
'is_already_using_ckeditor5',
], $already_using_ckeditor5);
}
$form_state
->set('ckeditor5_is_active', $already_using_ckeditor5);
$form_state
->set('ckeditor5_is_selected', $form_state
->getValue([
'editor',
'editor',
]) === 'ckeditor5');
// Disable inline form errors when using CKEditor 5 because it prevents
// useful error messages from vertical tabs from being visible to the user.
// @todo Remove this workaround in
// https://www.drupal.org/project/drupal/issues/3263668
if ($form_state
->get('ckeditor5_is_selected')) {
$element['#disable_inline_form_errors'] = TRUE;
}
return $element;
}
/**
* Validate callback to inform the user of CKEditor 5 compatibility problems.
*/
public static function validateSwitchingToCKEditor5(array $form, FormStateInterface $form_state) : void {
if (!$form_state
->get('ckeditor5_is_active') && $form_state
->get('ckeditor5_is_selected')) {
$minimal_ckeditor5_editor = Editor::create([
'format' => NULL,
'editor' => 'ckeditor5',
]);
$submitted_filter_format = CKEditor5::getSubmittedFilterFormat($form_state);
$fundamental_incompatibilities = CKEditor5::validatePair($minimal_ckeditor5_editor, $submitted_filter_format, FALSE);
foreach ($fundamental_incompatibilities as $violation) {
// If the violation uses the nonAllowedElementsMessage template, it can
// be skipped because this is a violation that automatically fixed
// within SmartDefaultSettings, but SmartDefaultSettings does not
// execute until this validator passes.
if ($violation
->getMessageTemplate() === $violation
->getConstraint()->nonAllowedElementsMessage) {
continue;
}
// @codingStandardsIgnoreLine
$form_state
->setErrorByName('editor][editor', t($violation
->getMessageTemplate(), $violation
->getParameters()));
}
}
}
/**
* Value callback to set the CKEditor 5-generated "allowed_html" value.
*
* Used to set the value of filter_html's "allowed_html" form item if the form
* has been validated and hence `ckeditor5_validated_pair` is available
* in form state. This allows setting a guaranteed to be valid value.
*
* `ckeditor5_validated_pair` can be set from two places:
* - When switching to CKEditor 5, this is populated by
* CKEditor5::buildConfigurationForm().
* - When making filter or editor settings changes, it is populated by
* CKEditor5::validateConfigurationForm().
*
* @param array $element
* An associative array containing the properties of the element.
* @param mixed $input
* The incoming input to populate the form element. If this is FALSE,
* the element's default value should be returned.
* @param \Drupal\Core\Form\FormStateInterface $form_state
* The current state of the form.
*
* @return string
* The value to assign to the element.
*/
public static function getGeneratedAllowedHtmlValue(array &$element, $input, FormStateInterface $form_state) : string {
if ($form_state
->isValidationComplete()) {
$validated_format = $form_state
->get('ckeditor5_validated_pair')
->getFilterFormat();
$configuration = $validated_format
->filters()
->get('filter_html')
->getConfiguration();
return $configuration['settings']['allowed_html'];
}
else {
if ($input !== FALSE) {
return $input;
}
return $element['#default_value'];
}
}
/**
* {@inheritdoc}
*/
public function validateConfigurationForm(array &$form, FormStateInterface $form_state) {
$json = $form_state
->getValue([
'toolbar',
'items',
]);
$toolbar_items = Json::decode($json);
// This basic validation must live in the form logic because it can only
// occur in a form context.
if (!$toolbar_items) {
$form_state
->setErrorByName('toolbar][items', $this
->t('Invalid toolbar value.'));
return;
}
// Construct a Text Editor config entity with the submitted values for
// validation. Do this on a clone: do not manipulate form state.
$submitted_editor = clone $form_state
->get('editor');
$settings = $submitted_editor
->getSettings();
// Update settings first to match the submitted toolbar items. This is
// necessary for ::shouldHaveVisiblePluginSettingsForm() to work.
$settings['toolbar']['items'] = $toolbar_items;
$submitted_editor
->setSettings($settings);
$eventual_editor_and_format_for_plugin_settings_visibility = $this
->getEventualEditorWithPrimedFilterFormat($form_state, $submitted_editor);
$settings['plugins'] = [];
$default_configurations = [];
foreach ($this->ckeditor5PluginManager
->getDefinitions() as $plugin_id => $definition) {
if (!$definition
->isConfigurable()) {
continue;
}
// Create a fresh instance of this CKEditor 5 plugin, not tied to a text
// editor configuration entity.
$plugin = $this->ckeditor5PluginManager
->getPlugin($plugin_id, NULL);
// If this plugin is configurable but it has empty default configuration,
// that means the configuration must be stored out of band.
// @see \Drupal\ckeditor5\Plugin\CKEditor5Plugin\ImageUpload
// @see editor_image_upload_settings_form()
$default_configuration = $plugin
->defaultConfiguration();
$configuration_stored_out_of_band = empty($default_configuration);
// If this plugin is configurable but has not yet had user interaction,
// the default configuration will still be active and may trigger
// validation errors. Do not trigger those validation errors until the
// form is actually saved, to allow the user to first configure other
// CKEditor 5 functionality.
$default_configurations[$plugin_id] = $default_configuration;
if ($form_state
->hasValue([
'plugins',
$plugin_id,
])) {
$subform = $form['plugins'][$plugin_id];
$subform_state = SubformState::createForSubform($subform, $form, $form_state);
$plugin
->validateConfigurationForm($subform, $subform_state);
$plugin
->submitConfigurationForm($subform, $subform_state);
// If the configuration is stored out of band, ::submitConfigurationForm
// will already have stored it. If it is not stored out of band,
// populate $settings, to populate $submitted_editor.
if (!$configuration_stored_out_of_band) {
$settings['plugins'][$plugin_id] = $plugin
->getConfiguration();
}
}
elseif ($this
->shouldHaveVisiblePluginSettingsForm($definition, $eventual_editor_and_format_for_plugin_settings_visibility)) {
if (!$configuration_stored_out_of_band) {
$settings['plugins'][$plugin_id] = $default_configuration;
}
}
}
// All plugin settings have been collected, including defaults that depend
// on visibility. Store the collected settings, throw away the interim state
// that allowed determining which defaults to add.
unset($eventual_editor_and_format_for_plugin_settings_visibility);
$submitted_editor
->setSettings($settings);
// Validate the text editor + text format pair.
// Note that the eventual pair is computed and validated, not the received
// pair: if the filter_html filter is in use, the CKEditor 5 configuration
// dictates the filter_html's filter plugin's "allowed_html" setting.
// @see ckeditor5_form_filter_format_form_alter()
// @see ::getGeneratedAllowedHtmlValue()
$eventual_editor_and_format = $this
->getEventualEditorWithPrimedFilterFormat($form_state, $submitted_editor);
$violations = CKEditor5::validatePair($eventual_editor_and_format, $eventual_editor_and_format
->getFilterFormat());
foreach ($violations as $violation) {
$property_path_parts = explode('.', $violation
->getPropertyPath());
// Special case: AJAX updates that do not submit the form (that cannot
// result in configuration being saved).
if ($form_state
->getSubmitHandlers() === [
'editor_form_filter_admin_format_editor_configure',
]) {
// Ensure that plugins' validation constraints do not immediately
// trigger a validation error: the user may choose to configure other
// CKEditor 5 aspects first.
if ($property_path_parts[0] === 'settings' && $property_path_parts[1] === 'plugins') {
$plugin_id = $property_path_parts[2];
// This CKEditor 5 plugin settings form was just added: the user has
// not yet had a chance to configure it.
if (!$form_state
->hasValue([
'plugins',
$plugin_id,
])) {
continue;
}
// This CKEditor 5 plugin settings form was added recently, the user
// is triggering AJAX rebuilds of the configuration UI because they're
// configuring other functionality first. Only require these to be
// valid at form submission time.
if ($form_state
->getValue([
'plugins',
$plugin_id,
]) === $default_configurations[$plugin_id]) {
continue;
}
}
}
$form_item_name = static::mapPairViolationPropertyPathsToFormNames($violation
->getPropertyPath(), $form);
// When adding a toolbar item, it is possible that not all conditions for
// using it have been met yet. FormBuilder refuses to rebuild forms when a
// validation error is present. But to meet the condition for the toolbar
// item, configuration must be set in a vertical tab that must still
// appear. Work-around: reduce the validation error to a warning message.
// @see \Drupal\ckeditor5\Plugin\Validation\Constraint\ToolbarItemConditionsMetConstraintValidator
if ($form_state
->isRedirectDisabled() && $form_item_name === 'editor][settings][toolbar][items') {
$this
->messenger()
->addWarning($violation
->getMessage());
continue;
}
$form_state
->getCompleteFormState()
->setErrorByName($form_item_name, $violation
->getMessage());
}
// Pass it on to ::submitConfigurationForm().
$form_state
->get('editor')
->setSettings($settings);
// Provide the validated eventual pair in form state to
// ::getGeneratedAllowedHtmlValue(), to update filter_html's
// "allowed_html".
$form_state
->set('ckeditor5_validated_pair', $eventual_editor_and_format);
assert(TRUE === $this
->checkConfigSchema(\Drupal::getContainer()
->get('config.typed'), 'editor.editor.id_does_not_matter', $submitted_editor
->toArray()), 'Schema errors: ' . print_r($this
->checkConfigSchema(\Drupal::getContainer()
->get('config.typed'), 'editor.editor.id_does_not_matter', $submitted_editor
->toArray()), TRUE));
}
/**
* Gets the submitted text format config entity from form state.
*
* Needed for validation.
*
* @param \Drupal\Core\Form\FormStateInterface $filter_format_form_state
* The text format configuration form's form state.
*
* @return \Drupal\filter\FilterFormatInterface
* A FilterFormat config entity representing the current filter form state.
*/
protected static function getSubmittedFilterFormat(FormStateInterface $filter_format_form_state) : FilterFormatInterface {
$submitted_filter_format = clone $filter_format_form_state
->getFormObject()
->getEntity();
assert($submitted_filter_format instanceof FilterFormatInterface);
// Get only the values of the filter_format form state that are relevant for
// checking compatibility. This logic is copied from FilterFormatFormBase.
// @see \Drupal\ckeditor5\Plugin\Validation\Constraint\FundamentalCompatibilityConstraintValidator
// @see \Drupal\filter\FilterFormatFormBase::submitForm()
$filter_format_form_values = array_intersect_key($filter_format_form_state
->getValues(), array_flip([
'filters',
'filter_settings',
]));
foreach ($filter_format_form_values as $key => $value) {
if ($key !== 'filters') {
$submitted_filter_format
->set($key, $value);
}
else {
foreach ($value as $instance_id => $config) {
$submitted_filter_format
->setFilterConfig($instance_id, $config);
}
}
}
return $submitted_filter_format;
}
/**
* Gets the eventual text format config entity: from form state + editor.
*
* Needed for validation.
*
* @param \Drupal\Core\Form\SubformStateInterface $editor_form_state
* The text editor configuration form's form state.
* @param \Drupal\editor\EditorInterface $submitted_editor
* The current text editor config entity.
*
* @return \Drupal\editor\EditorInterface
* A clone of the received Editor config entity , with a primed associated
* FilterFormat that corresponds to the current form state, to avoid the
* stored FilterFormat config entity being loaded.
*/
protected function getEventualEditorWithPrimedFilterFormat(SubformStateInterface $editor_form_state, EditorInterface $submitted_editor) : EditorInterface {
$submitted_filter_format = static::getSubmittedFilterFormat($editor_form_state
->getCompleteFormState());
$pair = static::createEphemeralPairedEditor($submitted_editor, $submitted_filter_format);
// When CKEditor 5 plugins are disabled in the form-based admin UI, the
// associated settings (if any) should be omitted too, except for plugins
// that are enabled using `requiresConfiguration` (because whether they are
// enabled or not depends on the associated settings).
$original_settings = $pair
->getSettings();
$enabled_plugins = $this->ckeditor5PluginManager
->getEnabledDefinitions($pair);
$config_enabled_plugins = [];
foreach ($this->ckeditor5PluginManager
->getDefinitions() as $id => $definition) {
if ($definition
->hasConditions() && isset($definition
->getConditions()['requiresConfiguration'])) {
$config_enabled_plugins[$id] = TRUE;
}
}
$updated_settings = [
'plugins' => array_intersect_key($original_settings['plugins'], $enabled_plugins + $config_enabled_plugins),
] + $original_settings;
$pair
->setSettings($updated_settings);
if ($pair
->getFilterFormat()
->filters('filter_html')->status) {
// Compute elements provided by the current CKEditor 5 settings.
$restrictions = new HTMLRestrictions($this->ckeditor5PluginManager
->getProvidedElements(array_keys($enabled_plugins), $pair));
// Compute eventual filter_html setting. Eventual as in: this is the list
// of eventually allowed HTML tags.
// @see \Drupal\filter\FilterFormatFormBase::submitForm()
// @see ckeditor5_form_filter_format_form_alter()
$filter_html_config = $pair
->getFilterFormat()
->filters('filter_html')
->getConfiguration();
$filter_html_config['settings']['allowed_html'] = $restrictions
->toFilterHtmlAllowedTagsString();
$pair
->getFilterFormat()
->setFilterConfig('filter_html', $filter_html_config);
}
return $pair;
}
/**
* Creates an ephemeral pair of text editor + text format config entity.
*
* Clones the given text editor config entity object and then overwrites its
* $filterFormat property, to prevent loading the text format config entity
* from entity storage in calls to Editor::hasAssociatedFilterFormat() and
* Editor::getFilterFormat().
* This is necessary to be able to evaluate unsaved text editor and format
* config entities:
* - for assessing which CKEditor 5 plugins are enabled and whose settings
* forms to show
* - for validating them.
*
* @param \Drupal\editor\EditorInterface $editor
* The submitted text editor config entity, constructed from form values.
* @param \Drupal\filter\FilterFormatInterface $filter_format
* The submitted text format config entity, constructed from form values.
*
* @return \Drupal\editor\EditorInterface
* A clone of the given text editor config entity, with its $filterFormat
* property set to a clone of the given text format config entity.
*
* @throws \ReflectionException
*
* @see \Drupal\ckeditor5\Plugin\CKEditor5PluginManager::isPluginDisabled()
* @todo Remove this in https://www.drupal.org/project/drupal/issues/3231347
*/
protected static function createEphemeralPairedEditor(EditorInterface $editor, FilterFormatInterface $filter_format) : EditorInterface {
$paired_editor = clone $editor;
// If the editor is still being configured, the configuration may not yet be
// valid. Explicitly mark the ephemeral paired editor as new to allow other
// code to treat this accordingly.
// @see \Drupal\ckeditor5\Plugin\CKEditor5PluginManager::getProvidedElements()
$paired_editor
->enforceIsNew(TRUE);
$reflector = new \ReflectionObject($paired_editor);
$property = $reflector
->getProperty('filterFormat');
$property
->setAccessible(TRUE);
$property
->setValue($paired_editor, clone $filter_format);
return $paired_editor;
}
/**
* Maps Text Editor config object property paths to form names.
*
* @param string $property_path
* A config object property path.
* @param array $subform
* The subform being checked.
*
* @return string
* The corresponding form name in the subform.
*/
protected static function mapViolationPropertyPathsToFormNames(string $property_path, array $subform) : string {
$parts = explode('.', $property_path);
// The "settings" form element does exist, but one level above the Text
// Editor-specific form. This is operating on a subform.
$shifted = array_shift($parts);
assert($shifted === 'settings');
// It is not required (nor sensible) for the form structure to match the
// config schema structure 1:1. Automatically identify the relevant form
// name. Try to be specific. Worst case, an entire plugin settings vertical
// tab is targeted. (Hence the minimum of 2 parts: the property path gets at
// minimum mapped to 'toolbar.items' or 'plugins.<plugin ID>'.)
while (count($parts) > 2 && !NestedArray::keyExists($subform, $parts)) {
array_pop($parts);
}
assert(NestedArray::keyExists($subform, $parts));
return implode('][', array_merge([
'settings',
], $parts));
}
/**
* Maps Text Editor + Text Format pair property paths to form names.
*
* @param string $property_path
* A config object property path.
* @param array $form
* The form being checked.
*
* @return string
* The corresponding form name in the complete form.
*/
protected static function mapPairViolationPropertyPathsToFormNames(string $property_path, array $form) : string {
// Fundamental compatibility errors are at the root. Map these to the text
// editor plugin dropdown.
if ($property_path === '') {
return 'editor][editor';
}
// Filters are top-level.
if (preg_match('/^filters\\..*/', $property_path)) {
return implode('][', array_merge(explode('.', $property_path), [
'settings',
]));
}
// Everything else is in the subform.
return 'editor][' . static::mapViolationPropertyPathsToFormNames($property_path, $form);
}
/**
* {@inheritdoc}
*/
public function submitConfigurationForm(array &$form, FormStateInterface $form_state) {
// @see ::validateConfigurationForm()
$editor = $form_state
->get('editor');
// Prepare the editor settings for editor_form_filter_admin_format_submit().
// This strips away unwanted form values too, because those never can exist
// in the already validated Editor config entity.
$form_state
->setValues($editor
->getSettings());
parent::submitConfigurationForm($form, $form_state);
if ($form_state
->get('used_smart_default_settings')) {
$format_name = $editor
->getFilterFormat()
->get('name');
$this->logger
->info($this
->t('The migration of %text_format to CKEditor 5 has been saved.', [
'%text_format' => $format_name,
]));
}
}
/**
* {@inheritdoc}
*/
public function getJSSettings(Editor $editor) {
$toolbar_items = $editor
->getSettings()['toolbar']['items'];
$plugin_config = $this->ckeditor5PluginManager
->getCKEditor5PluginConfig($editor);
$settings = [
'toolbar' => [
'items' => $toolbar_items,
'shouldNotGroupWhenFull' => in_array('-', $toolbar_items, TRUE),
],
] + $plugin_config;
if ($this->moduleHandler
->moduleExists('locale')) {
$language_interface = $this->languageManager
->getCurrentLanguage();
$settings['language']['ui'] = _ckeditor5_get_langcode_mapping($language_interface
->getId());
}
return $settings;
}
/**
* {@inheritdoc}
*/
public function getLibraries(Editor $editor) {
$plugin_libraries = $this->ckeditor5PluginManager
->getEnabledLibraries($editor);
if ($this->moduleHandler
->moduleExists('locale')) {
$language_interface = $this->languageManager
->getCurrentLanguage();
$plugin_libraries[] = 'core/ckeditor5.translations.' . _ckeditor5_get_langcode_mapping($language_interface
->getId());
}
return $plugin_libraries;
}
}