-
Notifications
You must be signed in to change notification settings - Fork 40
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
b60334d
commit e50607b
Showing
6 changed files
with
223 additions
and
3 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
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
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
178 changes: 178 additions & 0 deletions
178
src/components/chat-view/chat-input/plugins/mention/AutoLinkMentionPlugin.tsx
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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,178 @@ | ||
import { useLexicalComposerContext } from '@lexical/react/LexicalComposerContext' | ||
import { | ||
$createTextNode, | ||
$getSelection, | ||
$isRangeSelection, | ||
COMMAND_PRIORITY_LOW, | ||
PASTE_COMMAND, | ||
TextNode, | ||
} from 'lexical' | ||
import { useEffect } from 'react' | ||
|
||
import { MentionableUrl } from '../../../../../types/mentionable' | ||
import { | ||
getMentionableName, | ||
serializeMentionable, | ||
} from '../../../../../utils/mentionable' | ||
|
||
import { $createMentionNode } from './MentionNode' | ||
|
||
const URL_MATCHER = | ||
/^((https?:\/\/(www\.)?)|(www\.))[-a-zA-Z0-9@:%._+~#=]{1,256}\.[a-zA-Z0-9()]{1,6}\b([-a-zA-Z0-9()@:%_+.~#?&//=]*)$/ | ||
|
||
type URLMatch = { | ||
index: number | ||
length: number | ||
text: string | ||
url: string | ||
} | ||
|
||
function findURLs(text: string): URLMatch[] { | ||
const urls: URLMatch[] = [] | ||
|
||
let lastIndex = 0 | ||
for (const word of text.split(' ')) { | ||
if (URL_MATCHER.test(word)) { | ||
urls.push({ | ||
index: lastIndex, | ||
length: word.length, | ||
text: word, | ||
url: word.startsWith('http') ? word : `https://${word}`, | ||
// attributes: { rel: 'noreferrer', target: '_blank' }, // Optional link attributes | ||
}) | ||
} | ||
|
||
lastIndex += word.length + 1 // +1 for space | ||
} | ||
|
||
return urls | ||
} | ||
|
||
function $textNodeTransform(node: TextNode) { | ||
if (!node.isSimpleText()) { | ||
return | ||
} | ||
|
||
const text = node.getTextContent() | ||
|
||
// Find only 1st occurrence as transform will be re-run anyway for the rest | ||
// because newly inserted nodes are considered to be dirty | ||
const urlMatches = findURLs(text) | ||
if (urlMatches.length === 0) { | ||
return | ||
} | ||
const urlMatch = urlMatches[0] | ||
|
||
// Get the current selection | ||
const selection = $getSelection() | ||
|
||
// Check if the selection is a RangeSelection and the cursor is at the end of the URL | ||
if ( | ||
$isRangeSelection(selection) && | ||
selection.anchor.key === node.getKey() && | ||
selection.focus.key === node.getKey() && | ||
selection.anchor.offset === urlMatch.index + urlMatch.length && | ||
selection.focus.offset === urlMatch.index + urlMatch.length | ||
) { | ||
// If the cursor is at the end of the URL, don't transform | ||
return | ||
} | ||
|
||
let targetNode | ||
if (urlMatch.index === 0) { | ||
// First text chunk within string, splitting into 2 parts | ||
;[targetNode] = node.splitText(urlMatch.index + urlMatch.length) | ||
} else { | ||
// In the middle of a string | ||
;[, targetNode] = node.splitText( | ||
urlMatch.index, | ||
urlMatch.index + urlMatch.length, | ||
) | ||
} | ||
|
||
const mentionable: MentionableUrl = { | ||
type: 'url', | ||
url: urlMatch.url, | ||
} | ||
|
||
const mentionNode = $createMentionNode( | ||
getMentionableName(mentionable), | ||
serializeMentionable(mentionable), | ||
) | ||
|
||
targetNode.replace(mentionNode) | ||
|
||
const spaceNode = $createTextNode(' ') | ||
mentionNode.insertAfter(spaceNode) | ||
|
||
spaceNode.select() | ||
} | ||
|
||
export default function AutoLinkMentionPlugin() { | ||
const [editor] = useLexicalComposerContext() | ||
|
||
useEffect(() => { | ||
editor.registerCommand( | ||
PASTE_COMMAND, | ||
(event) => { | ||
const clipboardData = | ||
event instanceof ClipboardEvent ? event.clipboardData : null | ||
|
||
if (!clipboardData) return false | ||
|
||
const text = clipboardData.getData('text/plain') | ||
|
||
const urlMatches = findURLs(text) | ||
if (urlMatches.length === 0) { | ||
return false | ||
} | ||
|
||
const selection = $getSelection() | ||
if (!$isRangeSelection(selection)) { | ||
return false | ||
} | ||
|
||
const nodes = [] | ||
let lastIndex = 0 | ||
|
||
urlMatches.forEach((urlMatch) => { | ||
// Add text node for unmatched part | ||
if (urlMatch.index > lastIndex) { | ||
nodes.push($createTextNode(text.slice(lastIndex, urlMatch.index))) | ||
} | ||
|
||
// Add mention node | ||
nodes.push( | ||
$createMentionNode( | ||
urlMatch.text, | ||
serializeMentionable({ | ||
type: 'url', | ||
url: urlMatch.url, | ||
}), | ||
), | ||
) | ||
|
||
lastIndex = urlMatch.index + urlMatch.length | ||
|
||
// Add space node after mention if next character is not space or end of string | ||
if (lastIndex >= text.length || text[lastIndex] !== ' ') { | ||
nodes.push($createTextNode(' ')) | ||
} | ||
}) | ||
|
||
// Add remaining text if any | ||
if (lastIndex < text.length) { | ||
nodes.push($createTextNode(text.slice(lastIndex))) | ||
} | ||
|
||
selection.insertNodes(nodes) | ||
return true | ||
}, | ||
COMMAND_PRIORITY_LOW, | ||
) | ||
|
||
editor.registerNodeTransform(TextNode, $textNodeTransform) | ||
}, [editor]) | ||
|
||
return null | ||
} |
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
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