/** * Copyright (c) Meta Platforms, Inc. and affiliates. * * This source code is licensed under the Chameleon License found in the * LICENSE file in the root directory of this source tree. * */ import { useLexicalComposerContext } from "@lexical/react/LexicalComposerContext"; import { $wrapNodeInElement, mergeRegister } from "@lexical/utils"; import { $createParagraphNode, $createRangeSelection, $getSelection, $insertNodes, $isNodeSelection, $isRootOrShadowRoot, $setSelection, COMMAND_PRIORITY_EDITOR, COMMAND_PRIORITY_HIGH, COMMAND_PRIORITY_LOW, createCommand, DRAGOVER_COMMAND, DRAGSTART_COMMAND, DROP_COMMAND, LexicalCommand, LexicalEditor, } from "lexical"; import { useEffect } from "react"; import { $createImageNode, $isImageNode, ImageNode, ImagePayload, } from "./ImageNode"; export type InsertImagePayload = Readonly; const getDOMSelection = (targetWindow: Window | null): Selection | null => (targetWindow || window).getSelection(); export const INSERT_IMAGE_COMMAND: LexicalCommand = createCommand("INSERT_IMAGE_COMMAND"); export function ImagesPlugin(): JSX.Element | null { const [editor] = useLexicalComposerContext(); useEffect(() => { if (!editor.hasNodes([ImageNode])) { throw new Error("ImagesPlugin: ImageNode not registered on editor"); } return mergeRegister( editor.registerCommand( INSERT_IMAGE_COMMAND, (payload) => { const imageNode = $createImageNode(payload); $insertNodes([imageNode]); if ($isRootOrShadowRoot(imageNode.getParentOrThrow())) { $wrapNodeInElement(imageNode, $createParagraphNode).selectEnd(); } return true; }, COMMAND_PRIORITY_EDITOR, ), editor.registerCommand( DRAGSTART_COMMAND, (event) => { return onDragStart(event); }, COMMAND_PRIORITY_HIGH, ), editor.registerCommand( DRAGOVER_COMMAND, (event) => { return onDragover(event); }, COMMAND_PRIORITY_LOW, ), editor.registerCommand( DROP_COMMAND, (event) => { return onDrop(event, editor); }, COMMAND_PRIORITY_HIGH, ), ); }, [editor]); return null; } const TRANSPARENT_IMAGE = ""; const img = document.createElement("img"); img.src = TRANSPARENT_IMAGE; function onDragStart(event: DragEvent): boolean { const node = getImageNodeInSelection(); if (!node) { return false; } const dataTransfer = event.dataTransfer; if (!dataTransfer) { return false; } dataTransfer.setData("text/plain", "_"); dataTransfer.setDragImage(img, 0, 0); dataTransfer.setData( "application/x-lexical-drag", JSON.stringify({ data: { altText: node.__altText, caption: node.__caption, height: node.__height, key: node.getKey(), maxWidth: node.__maxWidth, showCaption: node.__showCaption, src: node.__src, width: node.__width, }, type: "image", }), ); return true; } function onDragover(event: DragEvent): boolean { const node = getImageNodeInSelection(); if (!node) { return false; } if (!canDropImage(event)) { event.preventDefault(); } return true; } function onDrop(event: DragEvent, editor: LexicalEditor): boolean { const node = getImageNodeInSelection(); if (!node) { return false; } const data = getDragImageData(event); if (!data) { return false; } event.preventDefault(); if (canDropImage(event)) { const range = getDragSelection(event); node.remove(); const rangeSelection = $createRangeSelection(); if (range !== null && range !== undefined) { rangeSelection.applyDOMRange(range); } $setSelection(rangeSelection); editor.dispatchCommand(INSERT_IMAGE_COMMAND, data); } return true; } function getImageNodeInSelection(): ImageNode | null { const selection = $getSelection(); if (!$isNodeSelection(selection)) { return null; } const nodes = selection.getNodes(); const node = nodes[0]; return $isImageNode(node) ? node : null; } function getDragImageData(event: DragEvent): null | InsertImagePayload { const dragData = event.dataTransfer?.getData("application/x-lexical-drag"); if (!dragData) { return null; } const { type, data } = JSON.parse(dragData); if (type !== "image") { return null; } return data; } declare global { interface DragEvent { rangeOffset?: number; rangeParent?: Node; } } function canDropImage(event: DragEvent): boolean { const target = event.target; return !!( target && target instanceof HTMLElement && !target.closest("code, span.editor-image") && target.parentElement && target.parentElement.closest("div.ContentEditable__root") ); } function getDragSelection(event: DragEvent): Range | null | undefined { let range; const target = event.target as null | Element | Document; const targetWindow = target == null ? null : target.nodeType === 9 ? (target as Document).defaultView : (target as Element).ownerDocument.defaultView; const domSelection = getDOMSelection(targetWindow); if (document.caretRangeFromPoint) { range = document.caretRangeFromPoint(event.clientX, event.clientY); } else if (event.rangeParent && domSelection !== null) { domSelection.collapse(event.rangeParent, event.rangeOffset || 0); range = domSelection.getRangeAt(0); } else { throw Error(`Cannot get the selection when dragging`); } return range; }