import { DOMOutputSpec, Node as ProseMirrorNode, NodeSpec, NodeType, } from 'prosemirror-model' import { Plugin, Transaction } from 'prosemirror-state' import { NodeConfig } from '.' import { Editor } from './Editor' import { getExtensionField } from './helpers/getExtensionField' import { InputRule } from './InputRule' import { PasteRule } from './PasteRule' import { AnyConfig, Attributes, Extensions, GlobalAttributes, KeyboardShortcutCommand, NodeViewRenderer, ParentConfig, RawCommands, } from './types' import { callOrReturn } from './utilities/callOrReturn' import { mergeDeep } from './utilities/mergeDeep' declare module '@tiptap/core' { interface NodeConfig { [key: string]: any; /** * Name */ name: string, /** * Priority */ priority?: number, /** * Default options */ defaultOptions?: Options, /** * Default Options */ addOptions?: (this: { name: string, parent: Exclude>['addOptions'], undefined>, }) => Options, /** * Default Storage */ addStorage?: (this: { name: string, options: Options, parent: Exclude>['addStorage'], undefined>, }) => Storage, /** * Global attributes */ addGlobalAttributes?: (this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['addGlobalAttributes'], }) => GlobalAttributes | {}, /** * Raw */ addCommands?: (this: { name: string, options: Options, storage: Storage, editor: Editor, type: NodeType, parent: ParentConfig>['addCommands'], }) => Partial, /** * Keyboard shortcuts */ addKeyboardShortcuts?: (this: { name: string, options: Options, storage: Storage, editor: Editor, type: NodeType, parent: ParentConfig>['addKeyboardShortcuts'], }) => { [key: string]: KeyboardShortcutCommand, }, /** * Input rules */ addInputRules?: (this: { name: string, options: Options, storage: Storage, editor: Editor, type: NodeType, parent: ParentConfig>['addInputRules'], }) => InputRule[], /** * Paste rules */ addPasteRules?: (this: { name: string, options: Options, storage: Storage, editor: Editor, type: NodeType, parent: ParentConfig>['addPasteRules'], }) => PasteRule[], /** * ProseMirror plugins */ addProseMirrorPlugins?: (this: { name: string, options: Options, storage: Storage, editor: Editor, type: NodeType, parent: ParentConfig>['addProseMirrorPlugins'], }) => Plugin[], /** * Extensions */ addExtensions?: (this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['addExtensions'], }) => Extensions, /** * Extend Node Schema */ extendNodeSchema?: (( this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['extendNodeSchema'], }, extension: Node, ) => Record) | null, /** * Extend Mark Schema */ extendMarkSchema?: (( this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['extendMarkSchema'], }, extension: Node, ) => Record) | null, /** * The editor is not ready yet. */ onBeforeCreate?: ((this: { name: string, options: Options, storage: Storage, editor: Editor, type: NodeType, parent: ParentConfig>['onBeforeCreate'], }) => void) | null, /** * The editor is ready. */ onCreate?: ((this: { name: string, options: Options, storage: Storage, editor: Editor, type: NodeType, parent: ParentConfig>['onCreate'], }) => void) | null, /** * The content has changed. */ onUpdate?: ((this: { name: string, options: Options, storage: Storage, editor: Editor, type: NodeType, parent: ParentConfig>['onUpdate'], }) => void) | null, /** * The selection has changed. */ onSelectionUpdate?: ((this: { name: string, options: Options, storage: Storage, editor: Editor, type: NodeType, parent: ParentConfig>['onSelectionUpdate'], }) => void) | null, /** * The editor state has changed. */ onTransaction?: (( this: { name: string, options: Options, storage: Storage, editor: Editor, type: NodeType, parent: ParentConfig>['onTransaction'], }, props: { transaction: Transaction, }, ) => void) | null, /** * The editor is focused. */ onFocus?: (( this: { name: string, options: Options, storage: Storage, editor: Editor, type: NodeType, parent: ParentConfig>['onFocus'], }, props: { event: FocusEvent, }, ) => void) | null, /** * The editor isn’t focused anymore. */ onBlur?: (( this: { name: string, options: Options, storage: Storage, editor: Editor, type: NodeType, parent: ParentConfig>['onBlur'], }, props: { event: FocusEvent, }, ) => void) | null, /** * The editor is destroyed. */ onDestroy?: ((this: { name: string, options: Options, storage: Storage, editor: Editor, type: NodeType, parent: ParentConfig>['onDestroy'], }) => void) | null, /** * Node View */ addNodeView?: ((this: { name: string, options: Options, storage: Storage, editor: Editor, type: NodeType, parent: ParentConfig>['addNodeView'], }) => NodeViewRenderer) | null, /** * TopNode */ topNode?: boolean, /** * Content */ content?: NodeSpec['content'] | ((this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['content'], }) => NodeSpec['content']), /** * Marks */ marks?: NodeSpec['marks'] | ((this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['marks'], }) => NodeSpec['marks']), /** * Group */ group?: NodeSpec['group'] | ((this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['group'], }) => NodeSpec['group']), /** * Inline */ inline?: NodeSpec['inline'] | ((this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['inline'], }) => NodeSpec['inline']), /** * Atom */ atom?: NodeSpec['atom'] | ((this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['atom'], }) => NodeSpec['atom']), /** * Selectable */ selectable?: NodeSpec['selectable'] | ((this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['selectable'], }) => NodeSpec['selectable']), /** * Draggable */ draggable?: NodeSpec['draggable'] | ((this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['draggable'], }) => NodeSpec['draggable']), /** * Code */ code?: NodeSpec['code'] | ((this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['code'], }) => NodeSpec['code']), /** * Whitespace */ whitespace?: NodeSpec['whitespace'] | ((this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['whitespace'], }) => NodeSpec['whitespace']), /** * Defining */ defining?: NodeSpec['defining'] | ((this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['defining'], }) => NodeSpec['defining']), /** * Isolating */ isolating?: NodeSpec['isolating'] | ((this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['isolating'], }) => NodeSpec['isolating']), /** * Parse HTML */ parseHTML?: ( this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['parseHTML'], }, ) => NodeSpec['parseDOM'], /** * Render HTML */ renderHTML?: (( this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['renderHTML'], }, props: { node: ProseMirrorNode, HTMLAttributes: Record, } ) => DOMOutputSpec) | null, /** * Render Text */ renderText?: (( this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['renderText'], }, props: { node: ProseMirrorNode, pos: number, parent: ProseMirrorNode, index: number, } ) => string) | null, /** * Add Attributes */ addAttributes?: ( this: { name: string, options: Options, storage: Storage, parent: ParentConfig>['addAttributes'], }, ) => Attributes | {}, } } export class Node { type = 'node' name = 'node' parent: Node | null = null child: Node | null = null options: Options storage: Storage config: NodeConfig = { name: this.name, defaultOptions: {}, } constructor(config: Partial> = {}) { this.config = { ...this.config, ...config, } this.name = this.config.name if (config.defaultOptions) { console.warn(`[tiptap warn]: BREAKING CHANGE: "defaultOptions" is deprecated. Please use "addOptions" instead. Found in extension: "${this.name}".`) } // TODO: remove `addOptions` fallback this.options = this.config.defaultOptions if (this.config.addOptions) { this.options = callOrReturn(getExtensionField( this, 'addOptions', { name: this.name, }, )) } this.storage = callOrReturn(getExtensionField( this, 'addStorage', { name: this.name, options: this.options, }, )) || {} } static create(config: Partial> = {}) { return new Node(config) } configure(options: Partial = {}) { // return a new instance so we can use the same extension // with different calls of `configure` const extension = this.extend() extension.options = mergeDeep(this.options as Record, options) as Options extension.storage = callOrReturn(getExtensionField( extension, 'addStorage', { name: extension.name, options: extension.options, }, )) return extension } extend(extendedConfig: Partial> = {}) { const extension = new Node(extendedConfig) extension.parent = this this.child = extension extension.name = extendedConfig.name ? extendedConfig.name : extension.parent.name if (extendedConfig.defaultOptions) { console.warn(`[tiptap warn]: BREAKING CHANGE: "defaultOptions" is deprecated. Please use "addOptions" instead. Found in extension: "${extension.name}".`) } extension.options = callOrReturn(getExtensionField( extension, 'addOptions', { name: extension.name, }, )) extension.storage = callOrReturn(getExtensionField( extension, 'addStorage', { name: extension.name, options: extension.options, }, )) return extension } }