Created
August 23, 2024 08:03
-
-
Save regexyl/958aa74f8c77e6ae0176a311b407abd4 to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import { Editor, Range } from '@tiptap/core' | |
import { EditorState, Plugin, PluginKey } from '@tiptap/pm/state' | |
import { Decoration, DecorationSet, EditorView } from '@tiptap/pm/view' | |
import { findSuggestionMatch as defaultFindSuggestionMatch } from './findSuggestionMatch.js' | |
export interface SuggestionOptions<I = any, TSelected = any> { | |
/** | |
* The plugin key for the suggestion plugin. | |
* @default 'suggestion' | |
* @example 'mention' | |
*/ | |
pluginKey?: PluginKey | |
/** | |
* The editor instance. | |
* @default null | |
*/ | |
editor: Editor | |
/** | |
* The character that triggers the suggestion. | |
* @default '@' | |
* @example '#' | |
*/ | |
char?: string | |
/** | |
* Allow spaces in the suggestion query. | |
* @default false | |
* @example true | |
*/ | |
allowSpaces?: boolean | |
/** | |
* Allow prefixes in the suggestion query. | |
* @default [' '] | |
* @example [' ', '@'] | |
*/ | |
allowedPrefixes?: string[] | null | |
/** | |
* Only match suggestions at the start of the line. | |
* @default false | |
* @example true | |
*/ | |
startOfLine?: boolean | |
/** | |
* The tag name of the decoration node. | |
* @default 'span' | |
* @example 'div' | |
*/ | |
decorationTag?: string | |
/** | |
* The class name of the decoration node. | |
* @default 'suggestion' | |
* @example 'mention' | |
*/ | |
decorationClass?: string | |
/** | |
* A function that is called when a suggestion is selected. | |
* @param props The props object. | |
* @param props.editor The editor instance. | |
* @param props.range The range of the suggestion. | |
* @param props.props The props of the selected suggestion. | |
* @returns void | |
* @example ({ editor, range, props }) => { props.command(props.props) } | |
*/ | |
command?: (props: { editor: Editor; range: Range; props: TSelected }) => void | |
/** | |
* A function that returns the suggestion items in form of an array. | |
* @param props The props object. | |
* @param props.editor The editor instance. | |
* @param props.query The current suggestion query. | |
* @returns An array of suggestion items. | |
* @example ({ editor, query }) => [{ id: 1, label: 'John Doe' }] | |
*/ | |
items?: (props: { query: string; editor: Editor }) => I[] | Promise<I[]> | |
/** | |
* The render function for the suggestion. | |
* @returns An object with render functions. | |
*/ | |
render?: () => { | |
onBeforeStart?: (props: SuggestionProps<I, TSelected>) => void; | |
onStart?: (props: SuggestionProps<I, TSelected>) => void; | |
onBeforeUpdate?: (props: SuggestionProps<I, TSelected>) => void; | |
onUpdate?: (props: SuggestionProps<I, TSelected>) => void; | |
onExit?: (props: SuggestionProps<I, TSelected>) => void; | |
onKeyDown?: (props: SuggestionKeyDownProps) => boolean; | |
} | |
/** | |
* A function that returns a boolean to indicate if the suggestion should be active. | |
* @param props The props object. | |
* @returns {boolean} | |
*/ | |
allow?: (props: { editor: Editor; state: EditorState; range: Range, isActive?: boolean }) => boolean | |
findSuggestionMatch?: typeof defaultFindSuggestionMatch | |
} | |
export interface SuggestionProps<I = any, TSelected = any> { | |
/** | |
* The editor instance. | |
*/ | |
editor: Editor | |
/** | |
* The range of the suggestion. | |
*/ | |
range: Range | |
/** | |
* The current suggestion query. | |
*/ | |
query: string | |
/** | |
* The current suggestion text. | |
*/ | |
text: string | |
/** | |
* The suggestion items array. | |
*/ | |
items: I[] | |
/** | |
* The state of loading items. | |
*/ | |
isLoading: boolean | |
/** | |
* A function that is called when a suggestion is selected. | |
* @param props The props object. | |
* @returns void | |
*/ | |
command: (props: TSelected) => void | |
/** | |
* The decoration node HTML element | |
* @default null | |
*/ | |
decorationNode: Element | null | |
/** | |
* The function that returns the client rect | |
* @default null | |
* @example () => new DOMRect(0, 0, 0, 0) | |
*/ | |
clientRect?: (() => DOMRect | null) | null | |
} | |
export interface SuggestionKeyDownProps { | |
view: EditorView | |
event: KeyboardEvent | |
range: Range | |
} | |
export const SuggestionPluginKey = new PluginKey('suggestion') | |
/** | |
* This utility allows you to create suggestions. | |
* @see https://tiptap.dev/api/utilities/suggestion | |
*/ | |
export function Suggestion<I = any, TSelected = any>({ | |
pluginKey = SuggestionPluginKey, | |
editor, | |
char = '@', | |
allowSpaces = false, | |
allowedPrefixes = [' '], | |
startOfLine = false, | |
decorationTag = 'span', | |
decorationClass = 'suggestion', | |
command = () => null, | |
items = () => [], | |
render = () => ({}), | |
allow = () => true, | |
findSuggestionMatch = defaultFindSuggestionMatch, | |
}: SuggestionOptions<I, TSelected>) { | |
let props: SuggestionProps<I, TSelected> | undefined | |
const renderer = render?.() | |
const plugin: Plugin<any> = new Plugin({ | |
key: pluginKey, | |
view() { | |
return { | |
update: async (view, prevState) => { | |
const prev = this.key?.getState(prevState) | |
const next = this.key?.getState(view.state) | |
// See how the state changed | |
const moved = prev.active && next.active && prev.range.from !== next.range.from | |
const started = !prev.active && next.active | |
const stopped = prev.active && !next.active | |
const changed = !started && !stopped && prev.query !== next.query | |
const handleStart = started || (moved && changed) | |
const handleChange = changed || moved | |
const handleExit = stopped | |
// Cancel when suggestion isn't active | |
if (!handleStart && !handleChange && !handleExit) { | |
return | |
} | |
const state = handleExit && !handleStart ? prev : next | |
const decorationNode = view.dom.querySelector( | |
`[data-decoration-id="${state.decorationId}"]`, | |
) | |
props = { | |
editor, | |
range: state.range, | |
query: state.query, | |
text: state.text, | |
items: [], | |
isLoading: false, | |
command: commandProps => { | |
return command({ | |
editor, | |
range: state.range, | |
props: commandProps, | |
}) | |
}, | |
decorationNode, | |
// virtual node for popper.js or tippy.js | |
// this can be used for building popups without a DOM node | |
clientRect: decorationNode | |
? () => { | |
// because of `items` can be asynchrounous we’ll search for the current decoration node | |
const { decorationId } = this.key?.getState(editor.state) // eslint-disable-line | |
const currentDecorationNode = view.dom.querySelector( | |
`[data-decoration-id="${decorationId}"]`, | |
) | |
return currentDecorationNode?.getBoundingClientRect() || null | |
} | |
: null, | |
} | |
if (handleStart) { | |
renderer?.onBeforeStart?.(props) | |
} | |
if (handleChange) { | |
renderer?.onBeforeUpdate?.(props) | |
} | |
if (handleChange || handleStart) { | |
props.isLoading = true | |
Promise.resolve(items({ editor, query: state.query })).then(result => { | |
if (props == null) { | |
return | |
} | |
props.isLoading = false | |
props.items = result | |
_postItemResolution(handleStart, handleChange, handleExit, props) | |
}) | |
} | |
_postItemResolution(handleStart, handleChange, handleExit, props) | |
}, | |
_postItemResolution(handleStart, handleChange, handleExit, props) { | |
if (handleExit) { | |
renderer?.onExit?.(props) | |
} | |
if (handleChange) { | |
renderer?.onUpdate?.(props) | |
} | |
if (handleStart) { | |
renderer?.onStart?.(props) | |
} | |
}, | |
destroy: () => { | |
if (!props) { | |
return | |
} | |
renderer?.onExit?.(props) | |
}, | |
} | |
}, | |
state: { | |
// Initialize the plugin's internal state. | |
init() { | |
const state: { | |
active: boolean | |
range: Range | |
query: null | string | |
text: null | string | |
composing: boolean | |
decorationId?: string | null | |
} = { | |
active: false, | |
range: { | |
from: 0, | |
to: 0, | |
}, | |
query: null, | |
text: null, | |
composing: false, | |
} | |
return state | |
}, | |
// Apply changes to the plugin state from a view transaction. | |
apply(transaction, prev, _oldState, state) { | |
const { isEditable } = editor | |
const { composing } = editor.view | |
const { selection } = transaction | |
const { empty, from } = selection | |
const next = { ...prev } | |
next.composing = composing | |
// We can only be suggesting if the view is editable, and: | |
// * there is no selection, or | |
// * a composition is active (see: https://github.com/ueberdosis/tiptap/issues/1449) | |
if (isEditable && (empty || editor.view.composing)) { | |
// Reset active state if we just left the previous suggestion range | |
if ((from < prev.range.from || from > prev.range.to) && !composing && !prev.composing) { | |
next.active = false | |
} | |
// Try to match against where our cursor currently is | |
const match = findSuggestionMatch({ | |
char, | |
allowSpaces, | |
allowedPrefixes, | |
startOfLine, | |
$position: selection.$from, | |
}) | |
const decorationId = `id_${Math.floor(Math.random() * 0xffffffff)}` | |
// If we found a match, update the current state to show it | |
if (match && allow({ | |
editor, state, range: match.range, isActive: prev.active, | |
})) { | |
next.active = true | |
next.decorationId = prev.decorationId ? prev.decorationId : decorationId | |
next.range = match.range | |
next.query = match.query | |
next.text = match.text | |
} else { | |
next.active = false | |
} | |
} else { | |
next.active = false | |
} | |
// Make sure to empty the range if suggestion is inactive | |
if (!next.active) { | |
next.decorationId = null | |
next.range = { from: 0, to: 0 } | |
next.query = null | |
next.text = null | |
} | |
return next | |
}, | |
}, | |
props: { | |
// Call the keydown hook if suggestion is active. | |
handleKeyDown(view, event) { | |
const { active, range } = plugin.getState(view.state) | |
if (!active) { | |
return false | |
} | |
return renderer?.onKeyDown?.({ view, event, range }) || false | |
}, | |
// Setup decorator on the currently active suggestion. | |
decorations(state) { | |
const { active, range, decorationId } = plugin.getState(state) | |
if (!active) { | |
return null | |
} | |
return DecorationSet.create(state.doc, [ | |
Decoration.inline(range.from, range.to, { | |
nodeName: decorationTag, | |
class: decorationClass, | |
'data-decoration-id': decorationId, | |
}), | |
]) | |
}, | |
}, | |
}) | |
return plugin | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment