From 046babfd3a94222128f08cef653859ab66eaf3cd Mon Sep 17 00:00:00 2001 From: Priestch Date: Thu, 17 Dec 2020 08:35:04 +0800 Subject: [PATCH] fix: can not write second mention bug, see #874 --- .../tiptap-extensions/src/nodes/Mention.js | 2 +- .../src/plugins/Suggestions.js | 33 ++++++++++++++++++- 2 files changed, 33 insertions(+), 2 deletions(-) diff --git a/packages/tiptap-extensions/src/nodes/Mention.js b/packages/tiptap-extensions/src/nodes/Mention.js index 1514fdb6b..59aca1935 100644 --- a/packages/tiptap-extensions/src/nodes/Mention.js +++ b/packages/tiptap-extensions/src/nodes/Mention.js @@ -43,7 +43,7 @@ export default class Mention extends Node { }, group: 'inline', inline: true, - content: 'inline*', + content: 'text*', selectable: false, atom: true, toDOM: node => [ diff --git a/packages/tiptap-extensions/src/plugins/Suggestions.js b/packages/tiptap-extensions/src/plugins/Suggestions.js index 0f11f3ee6..087293f9e 100644 --- a/packages/tiptap-extensions/src/plugins/Suggestions.js +++ b/packages/tiptap-extensions/src/plugins/Suggestions.js @@ -2,6 +2,37 @@ import { Plugin, PluginKey } from 'prosemirror-state' import { Decoration, DecorationSet } from 'prosemirror-view' import { insertText } from 'tiptap-commands' +function getTextBetween(node, from, to, blockSeparator, inlineSeparator, leafText = '\0') { + let text = '' + let blockSeparated = true + let inlineNode = null + node.content.nodesBetween(from, to, (innerNode, pos) => { + if (innerNode.isText) { + if (inlineNode) { + inlineNode = null + return + } + text += innerNode.text.slice(Math.max(from, pos) - pos, to - pos) + blockSeparated = !blockSeparator + } else if (innerNode.isLeaf && leafText) { + text += leafText + blockSeparated = !blockSeparator + } else if (innerNode.isInline && !innerNode.isLeaf) { + text += inlineSeparator + if (innerNode.textContent) { + text += innerNode.textContent + inlineNode = innerNode + } + text += inlineSeparator + blockSeparated = !blockSeparated + } else if (!blockSeparated && innerNode.isBlock) { + text += blockSeparator + blockSeparated = true + } + }, 0) + return text +} + // Create a matcher that matches when a specific character is typed. Useful for @mentions and #tags. function triggerCharacter({ char = '@', @@ -26,7 +57,7 @@ function triggerCharacter({ // Lookup the boundaries of the current node const textFrom = $position.before() const textTo = $position.end() - const text = $position.doc.textBetween(textFrom, textTo, '\0', '\0') + const text = getTextBetween($position.doc, textFrom, textTo, '\0', '\0') let match = regexp.exec(text) let position