ckeditor/ckeditor5-engine

View on GitHub
src/model/liveposition.js

Summary

Maintainability
A
3 hrs
Test Coverage
/**
 * @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 engine/model/liveposition
 */

import Position from './position';
import EmitterMixin from '@ckeditor/ckeditor5-utils/src/emittermixin';
import mix from '@ckeditor/ckeditor5-utils/src/mix';
import CKEditorError from '@ckeditor/ckeditor5-utils/src/ckeditorerror';

/**
 * `LivePosition` is a type of {@link module:engine/model/position~Position Position}
 * that updates itself as {@link module:engine/model/document~Document document}
 * is changed through operations. It may be used as a bookmark.
 *
 * **Note:** Contrary to {@link module:engine/model/position~Position}, `LivePosition` works only in roots that are
 * {@link module:engine/model/rootelement~RootElement}.
 * If {@link module:engine/model/documentfragment~DocumentFragment} is passed, error will be thrown.
 *
 * **Note:** Be very careful when dealing with `LivePosition`. Each `LivePosition` instance bind events that might
 * have to be unbound.
 * Use {@link module:engine/model/liveposition~LivePosition#detach} whenever you don't need `LivePosition` anymore.
 *
 * @extends module:engine/model/position~Position
 */
export default class LivePosition extends Position {
    /**
     * Creates a live position.
     *
     * @see module:engine/model/position~Position
     * @param {module:engine/model/rootelement~RootElement} root
     * @param {Array.<Number>} path
     * @param {module:engine/model/position~PositionStickiness} [stickiness]
     */
    constructor( root, path, stickiness = 'toNone' ) {
        super( root, path, stickiness );

        if ( !this.root.is( 'rootElement' ) ) {
            /**
             * LivePosition's root has to be an instance of RootElement.
             *
             * @error liveposition-root-not-rootelement
             */
            throw new CKEditorError(
                'model-liveposition-root-not-rootelement: LivePosition\'s root has to be an instance of RootElement.',
                root
            );
        }

        bindWithDocument.call( this );
    }

    /**
     * Unbinds all events previously bound by `LivePosition`. Use it whenever you don't need `LivePosition` instance
     * anymore (i.e. when leaving scope in which it was declared or before re-assigning variable that was
     * referring to it).
     */
    detach() {
        this.stopListening();
    }

    /**
     * Checks whether this object is of the given.
     *
     *        livePosition.is( 'position' ); // -> true
     *        livePosition.is( 'model:position' ); // -> true
     *        livePosition.is( 'liveposition' ); // -> true
     *        livePosition.is( 'model:livePosition' ); // -> true
     *
     *        livePosition.is( 'view:position' ); // -> false
     *        livePosition.is( 'documentSelection' ); // -> false
     *
     * {@link module:engine/model/node~Node#is Check the entire list of model objects} which implement the `is()` method.
     *
     * @param {String} type
     * @returns {Boolean}
     */
    is( type ) {
        return type === 'livePosition' || type === 'model:livePosition' ||
            // From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.
            type == 'position' || type === 'model:position';
    }

    /**
     * Creates a {@link module:engine/model/position~Position position instance}, which is equal to this live position.
     *
     * @returns {module:engine/model/position~Position}
     */
    toPosition() {
        return new Position( this.root, this.path.slice(), this.stickiness );
    }

    /**
     * Creates a `LivePosition` instance that is equal to position.
     *
     * @param {module:engine/model/position~Position} position
     * @param {module:engine/model/position~PositionStickiness} [stickiness]
     * @returns {module:engine/model/position~Position}
     */
    static fromPosition( position, stickiness ) {
        return new this( position.root, position.path.slice(), stickiness ? stickiness : position.stickiness );
    }

    /**
     * @static
     * @protected
     * @method module:engine/model/liveposition~LivePosition._createAfter
     * @see module:engine/model/position~Position._createAfter
     * @param {module:engine/model/node~Node} node
     * @param {module:engine/model/position~PositionStickiness} [stickiness='toNone']
     * @returns {module:engine/model/liveposition~LivePosition}
     */

    /**
     * @static
     * @protected
     * @method module:engine/model/liveposition~LivePosition._createBefore
     * @see module:engine/model/position~Position._createBefore
     * @param {module:engine/model/node~Node} node
     * @param {module:engine/model/position~PositionStickiness} [stickiness='toNone']
     * @returns {module:engine/model/liveposition~LivePosition}
     */

    /**
     * @static
     * @protected
     * @method module:engine/model/liveposition~LivePosition._createAt
     * @see module:engine/model/position~Position._createAt
     * @param {module:engine/model/item~Item|module:engine/model/position~Position} itemOrPosition
     * @param {Number|'end'|'before'|'after'} [offset]
     * @param {module:engine/model/position~PositionStickiness} [stickiness='toNone']
     * @returns {module:engine/model/liveposition~LivePosition}
     */

    /**
     * Fired when `LivePosition` instance is changed due to changes on {@link module:engine/model/document~Document}.
     *
     * @event module:engine/model/liveposition~LivePosition#change
     * @param {module:engine/model/position~Position} oldPosition Position equal to this live position before it got changed.
     */
}

// Binds this `LivePosition` to the {@link module:engine/model/document~Document document} that owns
// this position's {@link module:engine/model/position~Position#root root}.
//
// @private
function bindWithDocument() {
    this.listenTo(
        this.root.document.model,
        'applyOperation',
        ( event, args ) => {
            const operation = args[ 0 ];

            if ( !operation.isDocumentOperation ) {
                return;
            }

            transform.call( this, operation );
        },
        { priority: 'low' }
    );
}

// Updates this position accordingly to the updates applied to the model. Bases on change events.
//
// @private
// @param {module:engine/model/operation/operation~Operation} operation Executed operation.
function transform( operation ) {
    const result = this.getTransformedByOperation( operation );

    if ( !this.isEqual( result ) ) {
        const oldPosition = this.toPosition();

        this.path = result.path;
        this.root = result.root;

        this.fire( 'change', oldPosition );
    }
}

mix( LivePosition, EmitterMixin );