import { Plugin, Transaction } from 'prosemirror-state' import { Command as ProseMirrorCommand } from 'prosemirror-commands' import { InputRule } from 'prosemirror-inputrules' import { Editor } from './Editor' import { Node } from './Node' import mergeDeep from './utilities/mergeDeep' import { GlobalAttributes, RawCommands, ParentConfig } from './types' import { ExtensionConfig } from '.' declare module '@tiptap/core' { interface ExtensionConfig { [key: string]: any; /** * Name */ name: string, /** * Priority */ priority?: number, /** * Default options */ defaultOptions?: Options, /** * Global attributes */ addGlobalAttributes?: (this: { options: Options, parentConfig: ParentConfig>, }) => GlobalAttributes | {}, /** * Raw */ addCommands?: (this: { options: Options, editor: Editor, parentConfig: ParentConfig>, }) => Partial, /** * Keyboard shortcuts */ addKeyboardShortcuts?: (this: { options: Options, editor: Editor, parentConfig: ParentConfig>, }) => { [key: string]: ProseMirrorCommand, }, /** * Input rules */ addInputRules?: (this: { options: Options, editor: Editor, parentConfig: ParentConfig>, }) => InputRule[], /** * Paste rules */ addPasteRules?: (this: { options: Options, editor: Editor, parentConfig: ParentConfig>, }) => Plugin[], /** * ProseMirror plugins */ addProseMirrorPlugins?: (this: { options: Options, editor: Editor, parentConfig: ParentConfig>, }) => Plugin[], /** * Extend Node Schema */ extendNodeSchema?: (( this: { options: Options, parentConfig: ParentConfig>, }, extension: Node, ) => { [key: string]: any, }) | null, /** * Extend Mark Schema */ extendMarkSchema?: (( this: { options: Options, parentConfig: ParentConfig>, }, extension: Node, ) => { [key: string]: any, }) | null, /** * The editor is not ready yet. */ onBeforeCreate?: ((this: { options: Options, editor: Editor, parentConfig: ParentConfig>, }) => void) | null, /** * The editor is ready. */ onCreate?: ((this: { options: Options, editor: Editor, parentConfig: ParentConfig>, }) => void) | null, /** * The content has changed. */ onUpdate?: ((this: { options: Options, editor: Editor, parentConfig: ParentConfig>, }) => void) | null, /** * The selection has changed. */ onSelectionUpdate?: ((this: { options: Options, editor: Editor, parentConfig: ParentConfig>, }) => void) | null, /** * The editor state has changed. */ onTransaction?: (( this: { options: Options, editor: Editor, parentConfig: ParentConfig>, }, props: { transaction: Transaction, }, ) => void) | null, /** * The editor is focused. */ onFocus?: (( this: { options: Options, editor: Editor, parentConfig: ParentConfig>, }, props: { event: FocusEvent, }, ) => void) | null, /** * The editor isn’t focused anymore. */ onBlur?: (( this: { options: Options, editor: Editor, parentConfig: ParentConfig>, }, props: { event: FocusEvent, }, ) => void) | null, /** * The editor is destroyed. */ onDestroy?: ((this: { options: Options, editor: Editor, parentConfig: ParentConfig>, }) => void) | null, } } export class Extension { type = 'extension' config: ExtensionConfig = { name: 'extension', priority: 100, defaultOptions: {}, } // parentConfig: Partial = {} parent: any options!: Options constructor(config: ExtensionConfig) { this.config = { ...this.config, ...config, } this.options = this.config.defaultOptions } static create(config: ExtensionConfig) { return new Extension(config) } configure(options: Partial = {}) { return Extension .create(this.config as ExtensionConfig) .#configure(options) } #configure = (options: Partial) => { this.options = mergeDeep(this.config.defaultOptions, options) as Options return this } extend(extendedConfig: Partial> = {}) { const extension = new Extension({ // ...this.config, ...extendedConfig, } as ExtensionConfig) // extension.parentConfig = this.config extension.parent = this extension.options = { ...(this.config.defaultOptions ?? {}), ...(extendedConfig.defaultOptions ?? {}), } return extension } }