ckeditor5-image-upload-base64
Version:
The development environment of CKEditor 5 – the best browser-based rich text editor.
116 lines (94 loc) • 3.8 kB
JavaScript
/**
* @license Copyright (c) 2003-2020, CKSource - Frederico Knabben. All rights reserved.
* For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
*/
/**
* @module page-break/pagebreakcommand
*/
import Command from '@ckeditor/ckeditor5-core/src/command';
import { findOptimalInsertionPosition } from '@ckeditor/ckeditor5-widget/src/utils';
/**
* The page break command.
*
* The command is registered by {@link module:page-break/pagebreakediting~PageBreakEditing} as `'pageBreak'`.
*
* To insert a page break at the current selection, execute the command:
*
* editor.execute( 'pageBreak' );
*
* @extends module:core/command~Command
*/
export default class PageBreakCommand extends Command {
/**
* @inheritDoc
*/
refresh() {
this.isEnabled = isPageBreakAllowed( this.editor.model );
}
/**
* Executes the command.
*
* @fires execute
*/
execute() {
const model = this.editor.model;
model.change( writer => {
const pageBreakElement = writer.createElement( 'pageBreak' );
model.insertContent( pageBreakElement );
let nextElement = pageBreakElement.nextSibling;
// Check whether an element next to the inserted page break is defined and can contain a text.
const canSetSelection = nextElement && model.schema.checkChild( nextElement, '$text' );
// If the element is missing, but a paragraph could be inserted next to the page break, let's add it.
if ( !canSetSelection && model.schema.checkChild( pageBreakElement.parent, 'paragraph' ) ) {
nextElement = writer.createElement( 'paragraph' );
model.insertContent( nextElement, writer.createPositionAfter( pageBreakElement ) );
}
// Put the selection inside the element, at the beginning.
if ( nextElement ) {
writer.setSelection( nextElement, 0 );
}
} );
}
}
// Checks if the `pageBreak` element can be inserted at the current model selection.
//
// @param {module:engine/model/model~Model} model
// @returns {Boolean}
function isPageBreakAllowed( model ) {
const schema = model.schema;
const selection = model.document.selection;
return isPageBreakAllowedInParent( selection, schema, model ) &&
!checkSelectionOnObject( selection, schema );
}
// Checks if a page break is allowed by the schema in the optimal insertion parent.
//
// @param {module:engine/model/selection~Selection|module:engine/model/documentselection~DocumentSelection} selection
// @param {module:engine/model/schema~Schema} schema
// @param {module:engine/model/model~Model} model Model instance.
// @returns {Boolean}
function isPageBreakAllowedInParent( selection, schema, model ) {
const parent = getInsertPageBreakParent( selection, model );
return schema.checkChild( parent, 'pageBreak' );
}
// Checks if the selection is on object.
//
// @param {module:engine/model/selection~Selection|module:engine/model/documentselection~DocumentSelection} selection
// @param {module:engine/model/schema~Schema} schema
// @returns {Boolean}
function checkSelectionOnObject( selection, schema ) {
const selectedElement = selection.getSelectedElement();
return selectedElement && schema.isObject( selectedElement );
}
// Returns a node that will be used to insert a page break with `model.insertContent` to check if the page break can be placed there.
//
// @param {module:engine/model/selection~Selection|module:engine/model/documentselection~DocumentSelection} selection
// @param {module:engine/model/model~Model} model Model instance.
// @returns {module:engine/model/element~Element}
function getInsertPageBreakParent( selection, model ) {
const insertAt = findOptimalInsertionPosition( selection, model );
const parent = insertAt.parent;
if ( parent.isEmpty && !parent.is( 'element', '$root' ) ) {
return parent.parent;
}
return parent;
}