@ckeditor/ckeditor5-heading
Version:
Headings feature for CKEditor 5.
71 lines (70 loc) • 2.64 kB
JavaScript
/**
* @license Copyright (c) 2003-2025, CKSource Holding sp. z o.o. All rights reserved.
* For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-licensing-options
*/
/**
* @module heading/headingcommand
*/
import { Command } from 'ckeditor5/src/core.js';
import { first } from 'ckeditor5/src/utils.js';
/**
* The heading command. It is used by the {@link module:heading/heading~Heading heading feature} to apply headings.
*/
export default class HeadingCommand extends Command {
/**
* Set of defined model's elements names that this command support.
* See {@link module:heading/headingconfig~HeadingOption}.
*/
modelElements;
/**
* Creates an instance of the command.
*
* @param editor Editor instance.
* @param modelElements Names of the element which this command can apply in the model.
*/
constructor(editor, modelElements) {
super(editor);
this.modelElements = modelElements;
}
/**
* @inheritDoc
*/
refresh() {
const block = first(this.editor.model.document.selection.getSelectedBlocks());
this.value = !!block && this.modelElements.includes(block.name) && block.name;
this.isEnabled = !!block && this.modelElements.some(heading => checkCanBecomeHeading(block, heading, this.editor.model.schema));
}
/**
* Executes the command. Applies the heading to the selected blocks or, if the first selected
* block is a heading already, turns selected headings (of this level only) to paragraphs.
*
* @param options.value Name of the element which this command will apply in the model.
* @fires execute
*/
execute(options) {
const model = this.editor.model;
const document = model.document;
const modelElement = options.value;
model.change(writer => {
const blocks = Array.from(document.selection.getSelectedBlocks())
.filter(block => {
return checkCanBecomeHeading(block, modelElement, model.schema);
});
for (const block of blocks) {
if (!block.is('element', modelElement)) {
writer.rename(block, modelElement);
}
}
});
}
}
/**
* Checks whether the given block can be replaced by a specific heading.
*
* @param block A block to be tested.
* @param heading Command element name in the model.
* @param schema The schema of the document.
*/
function checkCanBecomeHeading(block, heading, schema) {
return schema.checkChild(block.parent, heading) && !schema.isObject(block);
}