import { Editor, posToDOMRect } from '@tiptap/core' import { EditorState, Plugin, PluginKey } from 'prosemirror-state' import { EditorView } from 'prosemirror-view' import tippy, { Instance, Props } from 'tippy.js' export interface FloatingMenuPluginProps { pluginKey: PluginKey | string, editor: Editor, element: HTMLElement, tippyOptions?: Partial, shouldShow?: ((props: { editor: Editor, view: EditorView, state: EditorState, oldState?: EditorState, }) => boolean) | null, } export type FloatingMenuViewProps = FloatingMenuPluginProps & { view: EditorView, } export class FloatingMenuView { public editor: Editor public element: HTMLElement public view: EditorView public preventHide = false public tippy: Instance | undefined public tippyOptions?: Partial public shouldShow: Exclude = ({ view, state }) => { const { selection } = state const { $anchor, empty } = selection const isRootDepth = $anchor.depth === 1 const isEmptyTextBlock = $anchor.parent.isTextblock && !$anchor.parent.type.spec.code && !$anchor.parent.textContent if ( !view.hasFocus() || !empty || !isRootDepth || !isEmptyTextBlock ) { return false } return true } constructor({ editor, element, view, tippyOptions = {}, shouldShow, }: FloatingMenuViewProps) { this.editor = editor this.element = element this.view = view if (shouldShow) { this.shouldShow = shouldShow } this.element.addEventListener('mousedown', this.mousedownHandler, { capture: true }) this.editor.on('focus', this.focusHandler) this.editor.on('blur', this.blurHandler) this.tippyOptions = tippyOptions // Detaches menu content from its current parent this.element.remove() this.element.style.visibility = 'visible' } mousedownHandler = () => { this.preventHide = true } focusHandler = () => { // we use `setTimeout` to make sure `selection` is already updated setTimeout(() => this.update(this.editor.view)) } blurHandler = ({ event }: { event: FocusEvent }) => { if (this.preventHide) { this.preventHide = false return } if ( event?.relatedTarget && this.element.parentNode?.contains(event.relatedTarget as Node) ) { return } this.hide() } createTooltip() { const { element: editorElement } = this.editor.options const editorIsAttached = !!editorElement.parentElement if (this.tippy || !editorIsAttached) { return } this.tippy = tippy(editorElement, { duration: 0, getReferenceClientRect: null, content: this.element, interactive: true, trigger: 'manual', placement: 'right', hideOnClick: 'toggle', ...this.tippyOptions, }) // maybe we have to hide tippy on its own blur event as well if (this.tippy.popper.firstChild) { (this.tippy.popper.firstChild as HTMLElement).addEventListener('blur', event => { this.blurHandler({ event }) }) } } update(view: EditorView, oldState?: EditorState) { const { state } = view const { doc, selection } = state const { from, to } = selection const isSame = oldState && oldState.doc.eq(doc) && oldState.selection.eq(selection) if (isSame) { return } this.createTooltip() const shouldShow = this.shouldShow?.({ editor: this.editor, view, state, oldState, }) if (!shouldShow) { this.hide() return } this.tippy?.setProps({ getReferenceClientRect: () => posToDOMRect(view, from, to), }) this.show() } show() { this.tippy?.show() } hide() { this.tippy?.hide() } destroy() { this.tippy?.destroy() this.element.removeEventListener('mousedown', this.mousedownHandler, { capture: true }) this.editor.off('focus', this.focusHandler) this.editor.off('blur', this.blurHandler) } } export const FloatingMenuPlugin = (options: FloatingMenuPluginProps) => { return new Plugin({ key: typeof options.pluginKey === 'string' ? new PluginKey(options.pluginKey) : options.pluginKey, view: view => new FloatingMenuView({ view, ...options }), }) }