import { MarkType } from '@tiptap/pm/model' import { getMarksBetween } from '../helpers/getMarksBetween.js' import { PasteRule, PasteRuleFinder } from '../PasteRule.js' import { ExtendedRegExpMatchArray } from '../types.js' import { callOrReturn } from '../utilities/callOrReturn.js' /** * Build an paste rule that adds a mark when the * matched text is pasted into it. */ export function markPasteRule(config: { find: PasteRuleFinder type: MarkType getAttributes?: | Record | ((match: ExtendedRegExpMatchArray, event: ClipboardEvent) => Record) | false | null }) { return new PasteRule({ find: config.find, handler: ({ state, range, match, pasteEvent, }) => { const attributes = callOrReturn(config.getAttributes, undefined, match, pasteEvent) if (attributes === false || attributes === null) { return null } const { tr } = state const captureGroup = match[match.length - 1] const fullMatch = match[0] let markEnd = range.to if (captureGroup) { const startSpaces = fullMatch.search(/\S/) const textStart = range.from + fullMatch.indexOf(captureGroup) const textEnd = textStart + captureGroup.length const excludedMarks = getMarksBetween(range.from, range.to, state.doc) .filter(item => { // @ts-ignore const excluded = item.mark.type.excluded as MarkType[] return excluded.find(type => type === config.type && type !== item.mark.type) }) .filter(item => item.to > textStart) if (excludedMarks.length) { return null } if (textEnd < range.to) { tr.delete(textEnd, range.to) } if (textStart > range.from) { tr.delete(range.from + startSpaces, textStart) } markEnd = range.from + startSpaces + captureGroup.length tr.addMark(range.from + startSpaces, markEnd, config.type.create(attributes || {})) tr.removeStoredMark(config.type) } }, }) }