@mdxeditor/editor
Version:
React component for rich text markdown editing
328 lines (327 loc) • 11.9 kB
JavaScript
import { $wrapNodeInElement, mergeRegister } from "@lexical/utils";
import { Signal, withLatestFrom, Cell, Action, mapTo, map } from "@mdxeditor/gurx";
import { $insertNodes, $isRootOrShadowRoot, $createParagraphNode, $getNodeByKey, COMMAND_PRIORITY_EDITOR, DRAGSTART_COMMAND, COMMAND_PRIORITY_HIGH, DRAGOVER_COMMAND, COMMAND_PRIORITY_LOW, DROP_COMMAND, PASTE_COMMAND, COMMAND_PRIORITY_CRITICAL, createCommand, $createRangeSelection, $setSelection, $getSelection, $isNodeSelection } from "lexical";
import { realmPlugin } from "../../RealmWithPlugins.js";
import { CAN_USE_DOM } from "../../utils/detectMac.js";
import { activeEditor$, createActiveEditorSubscription$, addImportVisitor$, addLexicalNode$, addExportVisitor$, addComposerChild$ } from "../core/index.js";
import { ImageDialog } from "./ImageDialog.js";
import { $createImageNode, ImageNode, $isImageNode } from "./ImageNode.js";
import { LexicalImageVisitor } from "./LexicalImageVisitor.js";
import { MdastImageVisitor, MdastHtmlImageVisitor, MdastJsxImageVisitor } from "./MdastImageVisitor.js";
const internalInsertImage$ = Signal((r) => {
r.sub(r.pipe(internalInsertImage$, withLatestFrom(activeEditor$)), ([values, theEditor]) => {
theEditor == null ? void 0 : theEditor.update(() => {
const imageNode = $createImageNode({ altText: values.altText ?? "", src: values.src, title: values.title ?? "" });
$insertNodes([imageNode]);
if ($isRootOrShadowRoot(imageNode.getParentOrThrow())) {
$wrapNodeInElement(imageNode, $createParagraphNode).selectEnd();
}
});
});
});
const insertImage$ = Signal((r) => {
r.sub(r.pipe(insertImage$, withLatestFrom(imageUploadHandler$)), ([values, imageUploadHandler]) => {
const handler = (src) => {
r.pub(internalInsertImage$, { ...values, src });
};
if ("file" in values) {
imageUploadHandler == null ? void 0 : imageUploadHandler(values.file).then(handler).catch((e) => {
throw e;
});
} else {
handler(values.src);
}
});
});
const imageAutocompleteSuggestions$ = Cell([]);
const disableImageResize$ = Cell(false);
const imageUploadHandler$ = Cell(null);
const imagePreviewHandler$ = Cell(null);
const imageDialogState$ = Cell(
{ type: "inactive" },
(r) => {
r.sub(
r.pipe(saveImage$, withLatestFrom(activeEditor$, imageUploadHandler$, imageDialogState$)),
([values, theEditor, imageUploadHandler, dialogState]) => {
const handler = dialogState.type === "editing" ? (src) => {
theEditor == null ? void 0 : theEditor.update(() => {
const { nodeKey } = dialogState;
const imageNode = $getNodeByKey(nodeKey);
imageNode.setTitle(values.title);
imageNode.setAltText(values.altText);
imageNode.setSrc(src);
});
r.pub(imageDialogState$, { type: "inactive" });
} : (src) => {
r.pub(internalInsertImage$, { ...values, src });
r.pub(imageDialogState$, { type: "inactive" });
};
if (values.file.length > 0) {
imageUploadHandler == null ? void 0 : imageUploadHandler(values.file.item(0)).then(handler).catch((e) => {
throw e;
});
} else if (values.src) {
handler(values.src);
}
}
);
r.pub(createActiveEditorSubscription$, (editor) => {
const theUploadHandler = r.getValue(imageUploadHandler$);
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, !!theUploadHandler);
},
COMMAND_PRIORITY_LOW
),
editor.registerCommand(
DROP_COMMAND,
(event) => {
return onDrop(event, editor, r.getValue(imageUploadHandler$));
},
COMMAND_PRIORITY_HIGH
),
editor.registerCommand(
PASTE_COMMAND,
(event) => {
var _a, _b;
if (!theUploadHandler) {
let fromWeb = Array.from(((_a = event.clipboardData) == null ? void 0 : _a.items) ?? []);
fromWeb = fromWeb.filter((i) => i.type.includes("text"));
if (!fromWeb.length || fromWeb.length === 0) {
return true;
}
return false;
}
let cbPayload = Array.from(((_b = event.clipboardData) == null ? void 0 : _b.items) ?? []);
cbPayload = cbPayload.filter((i) => i.type.includes("image"));
if (!cbPayload.length || cbPayload.length === 0) {
return false;
}
const imageUploadHandlerValue = r.getValue(imageUploadHandler$);
Promise.all(cbPayload.map((file) => imageUploadHandlerValue(file.getAsFile()))).then((urls) => {
urls.forEach((url) => {
editor.dispatchCommand(INSERT_IMAGE_COMMAND, {
src: url,
altText: ""
});
});
}).catch((e) => {
throw e;
});
return true;
},
COMMAND_PRIORITY_CRITICAL
)
);
});
}
);
const openNewImageDialog$ = Action((r) => {
r.link(r.pipe(openNewImageDialog$, mapTo({ type: "new" })), imageDialogState$);
});
const openEditImageDialog$ = Signal((r) => {
r.link(
r.pipe(
openEditImageDialog$,
map((payload) => ({ type: "editing", ...payload }))
),
imageDialogState$
);
});
const closeImageDialog$ = Action((r) => {
r.link(r.pipe(closeImageDialog$, mapTo({ type: "inactive" })), imageDialogState$);
});
const disableImageSettingsButton$ = Cell(false);
const saveImage$ = Signal();
const imagePlugin = realmPlugin({
init(realm, params) {
realm.pubIn({
[addImportVisitor$]: [MdastImageVisitor, MdastHtmlImageVisitor, MdastJsxImageVisitor],
[addLexicalNode$]: ImageNode,
[addExportVisitor$]: LexicalImageVisitor,
[addComposerChild$]: (params == null ? void 0 : params.ImageDialog) ?? ImageDialog,
[imageUploadHandler$]: (params == null ? void 0 : params.imageUploadHandler) ?? null,
[imageAutocompleteSuggestions$]: (params == null ? void 0 : params.imageAutocompleteSuggestions) ?? [],
[disableImageResize$]: Boolean(params == null ? void 0 : params.disableImageResize),
[disableImageSettingsButton$]: Boolean(params == null ? void 0 : params.disableImageSettingsButton),
[imagePreviewHandler$]: (params == null ? void 0 : params.imagePreviewHandler) ?? null
});
},
update(realm, params) {
realm.pubIn({
[imageUploadHandler$]: (params == null ? void 0 : params.imageUploadHandler) ?? null,
[imageAutocompleteSuggestions$]: (params == null ? void 0 : params.imageAutocompleteSuggestions) ?? [],
[disableImageResize$]: Boolean(params == null ? void 0 : params.disableImageResize),
[imagePreviewHandler$]: (params == null ? void 0 : params.imagePreviewHandler) ?? null
});
}
});
const getDOMSelection = (targetWindow) => CAN_USE_DOM ? (targetWindow ?? window).getSelection() : null;
const INSERT_IMAGE_COMMAND = createCommand("INSERT_IMAGE_COMMAND");
const TRANSPARENT_IMAGE = "";
function onDragStart(event) {
const node = getImageNodeInSelection();
if (!node) {
return false;
}
const dataTransfer = event.dataTransfer;
if (!dataTransfer) {
return false;
}
dataTransfer.setData("text/plain", "_");
const img = document.createElement("img");
img.src = TRANSPARENT_IMAGE;
dataTransfer.setDragImage(img, 0, 0);
dataTransfer.setData(
"application/x-lexical-drag",
JSON.stringify({
data: {
altText: node.__altText,
title: node.__title,
key: node.getKey(),
src: node.__src
},
type: "image"
})
);
return true;
}
function onDragover(event, hasUploadHandler) {
var _a;
if (hasUploadHandler) {
let cbPayload = Array.from(((_a = event.dataTransfer) == null ? void 0 : _a.items) ?? []);
cbPayload = cbPayload.filter((i) => i.type.includes("image"));
if (cbPayload.length > 0) {
event.preventDefault();
return true;
}
}
const node = getImageNodeInSelection();
if (!node) {
return false;
}
if (!canDropImage(event)) {
event.preventDefault();
}
return true;
}
function onDrop(event, editor, imageUploadHandler) {
var _a;
let cbPayload = Array.from(((_a = event.dataTransfer) == null ? void 0 : _a.items) ?? []);
cbPayload = cbPayload.filter((i) => i.type.includes("image"));
if (cbPayload.length > 0) {
if (imageUploadHandler !== null) {
event.preventDefault();
Promise.all(cbPayload.map((image) => imageUploadHandler(image.getAsFile()))).then((urls) => {
urls.forEach((url) => {
editor.dispatchCommand(INSERT_IMAGE_COMMAND, {
src: url,
altText: ""
});
});
}).catch((e) => {
throw e;
});
return true;
}
}
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 !== void 0) {
rangeSelection.applyDOMRange(range);
}
$setSelection(rangeSelection);
editor.dispatchCommand(INSERT_IMAGE_COMMAND, data);
}
return true;
}
function getImageNodeInSelection() {
const selection = $getSelection();
if (!$isNodeSelection(selection)) {
return null;
}
const nodes = selection.getNodes();
const node = nodes[0];
return $isImageNode(node) ? node : null;
}
function getDragImageData(event) {
var _a;
const dragData = (_a = event.dataTransfer) == null ? void 0 : _a.getData("application/x-lexical-drag");
if (!dragData) {
return null;
}
const { type, data } = JSON.parse(dragData);
if (type !== "image") {
return null;
}
return data;
}
function canDropImage(event) {
const target = event.target;
return !!(target && target instanceof HTMLElement && target.parentElement);
}
function getDragSelection(event) {
let range;
const target = event.target;
const targetWindow = target == null ? null : target.nodeType === 9 ? target.defaultView : target.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;
}
export {
$createImageNode,
$isImageNode,
INSERT_IMAGE_COMMAND,
ImageNode,
closeImageDialog$,
disableImageResize$,
disableImageSettingsButton$,
imageAutocompleteSuggestions$,
imageDialogState$,
imagePlugin,
imagePreviewHandler$,
imageUploadHandler$,
insertImage$,
openEditImageDialog$,
openNewImageDialog$,
saveImage$
};