UNPKG

@ckeditor/ckeditor5-engine

Version:

The editing engine of CKEditor 5 – the best browser-based rich text editor.

137 lines (136 loc) 5.81 kB
/** * @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 engine/view/textproxy */ import { ViewTypeCheckable } from './typecheckable.js'; import { CKEditorError } from '@ckeditor/ckeditor5-utils'; /** * ViewTextProxy is a wrapper for substring of {@link module:engine/view/text~ViewText}. Instance of this class is created by * {@link module:engine/view/treewalker~ViewTreeWalker} when only a part of {@link module:engine/view/text~ViewText} needs to be returned. * * `ViewTextProxy` has an API similar to {@link module:engine/view/text~ViewText Text} and allows to do most of the common tasks performed * on view nodes. * * **Note:** Some `ViewTextProxy` instances may represent whole text node, not just a part of it. * See {@link module:engine/view/textproxy~ViewTextProxy#isPartial}. * * **Note:** `ViewTextProxy` is a readonly interface. * * **Note:** `ViewTextProxy` instances are created on the fly basing * on the current state of parent {@link module:engine/view/text~ViewText}. * Because of this it is highly unrecommended to store references to `TextProxy instances because they might get * invalidated due to operations on Document. Also ViewTextProxy is not a {@link module:engine/view/node~ViewNode} so it cannot be * inserted as a child of {@link module:engine/view/element~ViewElement}. * * `ViewTextProxy` instances are created by {@link module:engine/view/treewalker~ViewTreeWalker view tree walker}. * You should not need to create an instance of this class by your own. */ export class ViewTextProxy extends ViewTypeCheckable { /** * Reference to the {@link module:engine/view/text~ViewText} element which ViewTextProxy is a substring. */ textNode; /** * Text data represented by this text proxy. */ data; /** * Offset in the `textNode` where this `ViewTextProxy` instance starts. */ offsetInText; /** * Creates a text proxy. * * @internal * @param textNode Text node which part is represented by this text proxy. * @param offsetInText Offset in {@link module:engine/view/textproxy~ViewTextProxy#textNode text node} * from which the text proxy starts. * @param length Text proxy length, that is how many text node's characters, starting from `offsetInText` it represents. */ constructor(textNode, offsetInText, length) { super(); this.textNode = textNode; if (offsetInText < 0 || offsetInText > textNode.data.length) { /** * Given offsetInText value is incorrect. * * @error view-textproxy-wrong-offsetintext */ throw new CKEditorError('view-textproxy-wrong-offsetintext', this); } if (length < 0 || offsetInText + length > textNode.data.length) { /** * Given length value is incorrect. * * @error view-textproxy-wrong-length */ throw new CKEditorError('view-textproxy-wrong-length', this); } this.data = textNode.data.substring(offsetInText, offsetInText + length); this.offsetInText = offsetInText; } /** * Offset size of this node. */ get offsetSize() { return this.data.length; } /** * Flag indicating whether `ViewTextProxy` instance covers only part of the original {@link module:engine/view/text~ViewText text node} * (`true`) or the whole text node (`false`). * * This is `false` when text proxy starts at the very beginning of {@link module:engine/view/textproxy~ViewTextProxy#textNode textNode} * ({@link module:engine/view/textproxy~ViewTextProxy#offsetInText offsetInText} equals `0`) and text proxy sizes is equal to * text node size. */ get isPartial() { return this.data.length !== this.textNode.data.length; } /** * Parent of this text proxy, which is same as parent of text node represented by this text proxy. */ get parent() { return this.textNode.parent; } /** * Root of this text proxy, which is same as root of text node represented by this text proxy. */ get root() { return this.textNode.root; } /** * {@link module:engine/view/document~ViewDocument View document} that owns this text proxy, or `null` if the text proxy is inside * {@link module:engine/view/documentfragment~ViewDocumentFragment document fragment}. */ get document() { return this.textNode.document; } /** * Returns ancestors array of this text proxy. * * @param options Options object. * @param options.includeSelf When set to `true`, textNode will be also included in parent's array. * @param options.parentFirst When set to `true`, array will be sorted from text proxy parent to * root element, otherwise root element will be the first item in the array. * @returns Array with ancestors. */ getAncestors(options = {}) { const ancestors = []; let parent = options.includeSelf ? this.textNode : this.parent; while (parent !== null) { ancestors[options.parentFirst ? 'push' : 'unshift'](parent); parent = parent.parent; } return ancestors; } } // The magic of type inference using `is` method is centralized in `TypeCheckable` class. // Proper overload would interfere with that. ViewTextProxy.prototype.is = function (type) { return type === '$textProxy' || type === 'view:$textProxy' || // This are legacy values kept for backward compatibility. type === 'textProxy' || type === 'view:textProxy'; };