UNPKG

@lexical/react

Version:

This package provides Lexical components and hooks for React applications.

87 lines (82 loc) 2.7 kB
/** * Copyright (c) Meta Platforms, Inc. and affiliates. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. * */ import { LinkNode, TOGGLE_LINK_COMMAND, $toggleLink } from '@lexical/link'; import { useLexicalComposerContext } from '@lexical/react/LexicalComposerContext'; import { mergeRegister, objectKlassEquals } from '@lexical/utils'; import { COMMAND_PRIORITY_LOW, PASTE_COMMAND, $getSelection, $isRangeSelection, $isElementNode } from 'lexical'; import { useEffect } from 'react'; /** * Copyright (c) Meta Platforms, Inc. and affiliates. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. * */ function LinkPlugin({ validateUrl, attributes }) { const [editor] = useLexicalComposerContext(); useEffect(() => { if (!editor.hasNodes([LinkNode])) { throw new Error('LinkPlugin: LinkNode not registered on editor'); } return mergeRegister(editor.registerCommand(TOGGLE_LINK_COMMAND, payload => { if (payload === null) { $toggleLink(payload); return true; } else if (typeof payload === 'string') { if (validateUrl === undefined || validateUrl(payload)) { $toggleLink(payload, attributes); return true; } return false; } else { const { url, target, rel, title } = payload; $toggleLink(url, { ...attributes, rel, target, title }); return true; } }, COMMAND_PRIORITY_LOW), validateUrl !== undefined ? editor.registerCommand(PASTE_COMMAND, event => { const selection = $getSelection(); if (!$isRangeSelection(selection) || selection.isCollapsed() || !objectKlassEquals(event, ClipboardEvent)) { return false; } if (event.clipboardData === null) { return false; } const clipboardText = event.clipboardData.getData('text'); if (!validateUrl(clipboardText)) { return false; } // If we select nodes that are elements then avoid applying the link. if (!selection.getNodes().some(node => $isElementNode(node))) { editor.dispatchCommand(TOGGLE_LINK_COMMAND, { ...attributes, url: clipboardText }); event.preventDefault(); return true; } return false; }, COMMAND_PRIORITY_LOW) : () => { // Don't paste arbitrary text as a link when there's no validate function }); }, [editor, validateUrl, attributes]); return null; } export { LinkPlugin };