UNPKG

@lexical/utils

Version:

This package contains misc utilities for Lexical.

346 lines (345 loc) 18.6 kB
/** * Copyright (c) Meta Platforms, Inc. and affiliates. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. * */ import { type CaretDirection, type EditorState, ElementNode, type Klass, type LexicalEditor, type LexicalNode, type NodeCaret, PointCaret, RootMode, type SiblingCaret, SplitAtPointCaretNextOptions, StateConfig, ValueOrUpdater } from 'lexical'; export { default as markSelection } from './markSelection'; export { default as mergeRegister } from './mergeRegister'; export { default as positionNodeOnRange } from './positionNodeOnRange'; export { default as selectionAlwaysOnDisplay } from './selectionAlwaysOnDisplay'; export { $splitNode, isBlockDomNode, isHTMLAnchorElement, isHTMLElement, isInlineDomNode, } from 'lexical'; export declare const CAN_USE_BEFORE_INPUT: boolean; export declare const CAN_USE_DOM: boolean; export declare const IS_ANDROID: boolean; export declare const IS_ANDROID_CHROME: boolean; export declare const IS_APPLE: boolean; export declare const IS_APPLE_WEBKIT: boolean; export declare const IS_CHROME: boolean; export declare const IS_FIREFOX: boolean; export declare const IS_IOS: boolean; export declare const IS_SAFARI: boolean; /** * Takes an HTML element and adds the classNames passed within an array, * ignoring any non-string types. A space can be used to add multiple classes * eg. addClassNamesToElement(element, ['element-inner active', true, null]) * will add both 'element-inner' and 'active' as classes to that element. * @param element - The element in which the classes are added * @param classNames - An array defining the class names to add to the element */ export declare function addClassNamesToElement(element: HTMLElement, ...classNames: Array<typeof undefined | boolean | null | string>): void; /** * Takes an HTML element and removes the classNames passed within an array, * ignoring any non-string types. A space can be used to remove multiple classes * eg. removeClassNamesFromElement(element, ['active small', true, null]) * will remove both the 'active' and 'small' classes from that element. * @param element - The element in which the classes are removed * @param classNames - An array defining the class names to remove from the element */ export declare function removeClassNamesFromElement(element: HTMLElement, ...classNames: Array<typeof undefined | boolean | null | string>): void; /** * Returns true if the file type matches the types passed within the acceptableMimeTypes array, false otherwise. * The types passed must be strings and are CASE-SENSITIVE. * eg. if file is of type 'text' and acceptableMimeTypes = ['TEXT', 'IMAGE'] the function will return false. * @param file - The file you want to type check. * @param acceptableMimeTypes - An array of strings of types which the file is checked against. * @returns true if the file is an acceptable mime type, false otherwise. */ export declare function isMimeType(file: File, acceptableMimeTypes: Array<string>): boolean; /** * Lexical File Reader with: * 1. MIME type support * 2. batched results (HistoryPlugin compatibility) * 3. Order aware (respects the order when multiple Files are passed) * * const filesResult = await mediaFileReader(files, ['image/']); * filesResult.forEach(file => editor.dispatchCommand('INSERT_IMAGE', \\{ * src: file.result, * \\})); */ export declare function mediaFileReader(files: Array<File>, acceptableMimeTypes: Array<string>): Promise<Array<{ file: File; result: string; }>>; export interface DFSNode { readonly depth: number; readonly node: LexicalNode; } /** * "Depth-First Search" starts at the root/top node of a tree and goes as far as it can down a branch end * before backtracking and finding a new path. Consider solving a maze by hugging either wall, moving down a * branch until you hit a dead-end (leaf) and backtracking to find the nearest branching path and repeat. * It will then return all the nodes found in the search in an array of objects. * @param startNode - The node to start the search, if omitted, it will start at the root node. * @param endNode - The node to end the search, if omitted, it will find all descendants of the startingNode. * @returns An array of objects of all the nodes found by the search, including their depth into the tree. * \\{depth: number, node: LexicalNode\\} It will always return at least 1 node (the start node). */ export declare function $dfs(startNode?: LexicalNode, endNode?: LexicalNode): Array<DFSNode>; /** * Get the adjacent caret in the same direction * * @param caret A caret or null * @returns `caret.getAdjacentCaret()` or `null` */ export declare function $getAdjacentCaret<D extends CaretDirection>(caret: null | NodeCaret<D>): null | SiblingCaret<LexicalNode, D>; /** * $dfs iterator (right to left). Tree traversal is done on the fly as new values are requested with O(1) memory. * @param startNode - The node to start the search, if omitted, it will start at the root node. * @param endNode - The node to end the search, if omitted, it will find all descendants of the startingNode. * @returns An iterator, each yielded value is a DFSNode. It will always return at least 1 node (the start node). */ export declare function $reverseDfs(startNode?: LexicalNode, endNode?: LexicalNode): Array<DFSNode>; /** * $dfs iterator (left to right). Tree traversal is done on the fly as new values are requested with O(1) memory. * @param startNode - The node to start the search, if omitted, it will start at the root node. * @param endNode - The node to end the search, if omitted, it will find all descendants of the startingNode. * @returns An iterator, each yielded value is a DFSNode. It will always return at least 1 node (the start node). */ export declare function $dfsIterator(startNode?: LexicalNode, endNode?: LexicalNode): IterableIterator<DFSNode>; /** * Returns the Node sibling when this exists, otherwise the closest parent sibling. For example * R -> P -> T1, T2 * -> P2 * returns T2 for node T1, P2 for node T2, and null for node P2. * @param node LexicalNode. * @returns An array (tuple) containing the found Lexical node and the depth difference, or null, if this node doesn't exist. */ export declare function $getNextSiblingOrParentSibling(node: LexicalNode): null | [LexicalNode, number]; export declare function $getDepth(node: null | LexicalNode): number; /** * Performs a right-to-left preorder tree traversal. * From the starting node it goes to the rightmost child, than backtracks to parent and finds new rightmost path. * It will return the next node in traversal sequence after the startingNode. * The traversal is similar to $dfs functions above, but the nodes are visited right-to-left, not left-to-right. * @param startingNode - The node to start the search. * @returns The next node in pre-order right to left traversal sequence or `null`, if the node does not exist */ export declare function $getNextRightPreorderNode(startingNode: LexicalNode): LexicalNode | null; /** * $dfs iterator (right to left). Tree traversal is done on the fly as new values are requested with O(1) memory. * @param startNode - The node to start the search, if omitted, it will start at the root node. * @param endNode - The node to end the search, if omitted, it will find all descendants of the startingNode. * @returns An iterator, each yielded value is a DFSNode. It will always return at least 1 node (the start node). */ export declare function $reverseDfsIterator(startNode?: LexicalNode, endNode?: LexicalNode): IterableIterator<DFSNode>; /** * Takes a node and traverses up its ancestors (toward the root node) * in order to find a specific type of node. * @param node - the node to begin searching. * @param klass - an instance of the type of node to look for. * @returns the node of type klass that was passed, or null if none exist. */ export declare function $getNearestNodeOfType<T extends ElementNode>(node: LexicalNode, klass: Klass<T>): T | null; /** * Returns the element node of the nearest ancestor, otherwise throws an error. * @param startNode - The starting node of the search * @returns The ancestor node found */ export declare function $getNearestBlockElementAncestorOrThrow(startNode: LexicalNode): ElementNode; export type DOMNodeToLexicalConversion = (element: Node) => LexicalNode; export type DOMNodeToLexicalConversionMap = Record<string, DOMNodeToLexicalConversion>; /** * Starts with a node and moves up the tree (toward the root node) to find a matching node based on * the search parameters of the findFn. (Consider JavaScripts' .find() function where a testing function must be * passed as an argument. eg. if( (node) => node.__type === 'div') ) return true; otherwise return false * @param startingNode - The node where the search starts. * @param findFn - A testing function that returns true if the current node satisfies the testing parameters. * @returns A parent node that matches the findFn parameters, or null if one wasn't found. */ export declare const $findMatchingParent: { <T extends LexicalNode>(startingNode: LexicalNode, findFn: (node: LexicalNode) => node is T): T | null; (startingNode: LexicalNode, findFn: (node: LexicalNode) => boolean): LexicalNode | null; }; /** * Attempts to resolve nested element nodes of the same type into a single node of that type. * It is generally used for marks/commenting * @param editor - The lexical editor * @param targetNode - The target for the nested element to be extracted from. * @param cloneNode - See {@link $createMarkNode} * @param handleOverlap - Handles any overlap between the node to extract and the targetNode * @returns The lexical editor */ export declare function registerNestedElementResolver<N extends ElementNode>(editor: LexicalEditor, targetNode: Klass<N>, cloneNode: (from: N) => N, handleOverlap: (from: N, to: N) => void): () => void; /** * Clones the editor and marks it as dirty to be reconciled. If there was a selection, * it would be set back to its previous state, or null otherwise. * @param editor - The lexical editor * @param editorState - The editor's state */ export declare function $restoreEditorState(editor: LexicalEditor, editorState: EditorState): void; /** * If the selected insertion area is the root/shadow root node (see {@link lexical!$isRootOrShadowRoot}), * the node will be appended there, otherwise, it will be inserted before the insertion area. * If there is no selection where the node is to be inserted, it will be appended after any current nodes * within the tree, as a child of the root node. A paragraph will then be added after the inserted node and selected. * @param node - The node to be inserted * @returns The node after its insertion */ export declare function $insertNodeToNearestRoot<T extends LexicalNode>(node: T): T; /** * If the insertion caret is the root/shadow root node (see {@link lexical!$isRootOrShadowRoot}), * the node will be inserted there, otherwise the parent nodes will be split according to the * given options. * @param node - The node to be inserted * @param caret - The location to insert or split from * @returns The node after its insertion */ export declare function $insertNodeToNearestRootAtCaret<T extends LexicalNode, D extends CaretDirection>(node: T, caret: PointCaret<D>, options?: SplitAtPointCaretNextOptions): NodeCaret<D>; /** * Wraps the node into another node created from a createElementNode function, eg. $createParagraphNode * @param node - Node to be wrapped. * @param createElementNode - Creates a new lexical element to wrap the to-be-wrapped node and returns it. * @returns A new lexical element with the previous node appended within (as a child, including its children). */ export declare function $wrapNodeInElement(node: LexicalNode, createElementNode: () => ElementNode): ElementNode; export type ObjectKlass<T> = new (...args: any[]) => T; /** * @param object = The instance of the type * @param objectClass = The class of the type * @returns Whether the object is has the same Klass of the objectClass, ignoring the difference across window (e.g. different iframes) */ export declare function objectKlassEquals<T>(object: unknown, objectClass: ObjectKlass<T>): object is T; /** * Filter the nodes * @param nodes Array of nodes that needs to be filtered * @param filterFn A filter function that returns node if the current node satisfies the condition otherwise null * @returns Array of filtered nodes */ export declare function $filter<T>(nodes: Array<LexicalNode>, filterFn: (node: LexicalNode) => null | T): Array<T>; /** * Appends the node before the first child of the parent node * @param parent A parent node * @param node Node that needs to be appended */ export declare function $insertFirst(parent: ElementNode, node: LexicalNode): void; /** * Calculates the zoom level of an element as a result of using * css zoom property. For browsers that implement standardized CSS * zoom (Firefox, Chrome >= 128), this will always return 1. * @param element */ export declare function calculateZoomLevel(element: Element | null): number; /** * Checks if the editor is a nested editor created by LexicalNestedComposer */ export declare function $isEditorIsNestedEditor(editor: LexicalEditor): boolean; /** * A depth first last-to-first traversal of root that stops at each node that matches * $predicate and ensures that its parent is root. This is typically used to discard * invalid or unsupported wrapping nodes. For example, a TableNode must only have * TableRowNode as children, but an importer might add invalid nodes based on * caption, tbody, thead, etc. and this will unwrap and discard those. * * @param root The root to start the traversal * @param $predicate Should return true for nodes that are permitted to be children of root * @returns true if this unwrapped or removed any nodes */ export declare function $unwrapAndFilterDescendants(root: ElementNode, $predicate: (node: LexicalNode) => boolean): boolean; /** * A depth first traversal of the children array that stops at and collects * each node that `$predicate` matches. This is typically used to discard * invalid or unsupported wrapping nodes on a children array in the `after` * of an {@link lexical!DOMConversionOutput}. For example, a TableNode must only have * TableRowNode as children, but an importer might add invalid nodes based on * caption, tbody, thead, etc. and this will unwrap and discard those. * * This function is read-only and performs no mutation operations, which makes * it suitable for import and export purposes but likely not for any in-place * mutation. You should use {@link $unwrapAndFilterDescendants} for in-place * mutations such as node transforms. * * @param children The children to traverse * @param $predicate Should return true for nodes that are permitted to be children of root * @returns The children or their descendants that match $predicate */ export declare function $descendantsMatching<T extends LexicalNode>(children: LexicalNode[], $predicate: (node: LexicalNode) => node is T): T[]; /** * Return an iterator that yields each child of node from first to last, taking * care to preserve the next sibling before yielding the value in case the caller * removes the yielded node. * * @param node The node whose children to iterate * @returns An iterator of the node's children */ export declare function $firstToLastIterator(node: ElementNode): Iterable<LexicalNode>; /** * Return an iterator that yields each child of node from last to first, taking * care to preserve the previous sibling before yielding the value in case the caller * removes the yielded node. * * @param node The node whose children to iterate * @returns An iterator of the node's children */ export declare function $lastToFirstIterator(node: ElementNode): Iterable<LexicalNode>; /** * Replace this node with its children * * @param node The ElementNode to unwrap and remove */ export declare function $unwrapNode(node: ElementNode): void; /** * Returns the Node sibling when this exists, otherwise the closest parent sibling. For example * R -> P -> T1, T2 * -> P2 * returns T2 for node T1, P2 for node T2, and null for node P2. * @param node LexicalNode. * @returns An array (tuple) containing the found Lexical node and the depth difference, or null, if this node doesn't exist. */ export declare function $getAdjacentSiblingOrParentSiblingCaret<D extends CaretDirection>(startCaret: NodeCaret<D>, rootMode?: RootMode): null | [NodeCaret<D>, number]; /** * A wrapper that creates bound functions and methods for the * StateConfig to save some boilerplate when defining methods * or exporting only the accessors from your modules rather * than exposing the StateConfig directly. */ export interface StateConfigWrapper<K extends string, V> { /** A reference to the stateConfig */ readonly stateConfig: StateConfig<K, V>; /** `(node) => $getState(node, stateConfig)` */ readonly $get: <T extends LexicalNode>(node: T) => V; /** `(node, valueOrUpdater) => $setState(node, stateConfig, valueOrUpdater)` */ readonly $set: <T extends LexicalNode>(node: T, valueOrUpdater: ValueOrUpdater<V>) => T; /** `[$get, $set]` */ readonly accessors: readonly [$get: this['$get'], $set: this['$set']]; /** * `() => function () { return $get(this) }` * * Should be called with an explicit `this` type parameter. * * @example * ```ts * class MyNode { * // … * myGetter = myWrapper.makeGetterMethod<this>(); * } * ``` */ makeGetterMethod<T extends LexicalNode>(): (this: T) => V; /** * `() => function (valueOrUpdater) { return $set(this, valueOrUpdater) }` * * Must be called with an explicit `this` type parameter. * * @example * ```ts * class MyNode { * // … * mySetter = myWrapper.makeSetterMethod<this>(); * } * ``` */ makeSetterMethod<T extends LexicalNode>(): (this: T, valueOrUpdater: ValueOrUpdater<V>) => T; } /** * EXPERIMENTAL * * A convenience interface for working with {@link $getState} and * {@link $setState}. * * @param stateConfig The stateConfig to wrap with convenience functionality * @returns a StateWrapper */ export declare function makeStateWrapper<K extends string, V>(stateConfig: StateConfig<K, V>): StateConfigWrapper<K, V>;