mirror of
https://github.com/makeplane/plane
synced 2024-06-14 14:31:34 +00:00
e1ae0d3b56
* remirror instances commented out to avoid prosemirror conflicts * styles migrated for remirror to tiptap transition * added bubblemenu support with extensions * fixed css for task lists and code with syntax highlighting * added support for slash command * fixed bubble menu to match styles and added better seperation in UI * saving with debounce logic added and it's stored in backend * added migration support by updating to html * Image uploads done * improved file structure and delete image function implemented * Integrated tiptap with Issue Modal * added additional props and Tiptap Integration with Comments * added tiptap integration with user activity feeds * added ref control support and bubble menu support for readonly editor * added tiptap support for plane pages * added tiptap support to gpt assistant modal (yet to be tested) * removed remirror instances and cleaned up code * improved code structure for extracting props in Tiptap * fixing ts errors for next build * fixing node ts error for Horizontal Rule * added ts fix for node types * temp fix * temp fix * added min height for issue description in modal * added resolutions to prosemirror-model version * trying pnpm overrides * explicitly added prosemirror deps * bugfixes * removed extra gap at the top and moved saved indicator to the bottom * fix: slash command scroll position * chore: update custom css variables * matched theme colours * fixed gpt-assistant modal * updated yarn lock * added debounced updates for the title and removed saved state after timeout * added css animations for saved state * build fixes and remove remirror instances * minor commenting fixes --------- Co-authored-by: Palanikannan1437 <73993394+Palanikannan1437@users.noreply.github.com> Co-authored-by: Aaryan Khandelwal <aaryankhandu123@gmail.com>
139 lines
4.0 KiB
TypeScript
139 lines
4.0 KiB
TypeScript
// @ts-nocheck
|
|
import { useEditor, EditorContent, Editor } from "@tiptap/react";
|
|
import { useDebouncedCallback } from "use-debounce";
|
|
import { EditorBubbleMenu } from "./bubble-menu";
|
|
import { TiptapExtensions } from "./extensions";
|
|
import { TiptapEditorProps } from "./props";
|
|
import { Node } from "@tiptap/pm/model";
|
|
import { Editor as CoreEditor } from "@tiptap/core";
|
|
import { useCallback, useImperativeHandle, useRef } from "react";
|
|
import { EditorState } from "@tiptap/pm/state";
|
|
import fileService from "services/file.service";
|
|
|
|
export interface ITiptapRichTextEditor {
|
|
value: string;
|
|
noBorder?: boolean;
|
|
borderOnFocus?: boolean;
|
|
customClassName?: string;
|
|
editorContentCustomClassNames?: string;
|
|
onChange?: (json: any, html: string) => void;
|
|
setIsSubmitting?: (isSubmitting: "submitting" | "submitted" | "saved") => void;
|
|
editable?: boolean;
|
|
forwardedRef?: any;
|
|
debouncedUpdatesEnabled?: boolean;
|
|
}
|
|
|
|
const Tiptap = (props: ITiptapRichTextEditor) => {
|
|
const {
|
|
onChange,
|
|
debouncedUpdatesEnabled,
|
|
forwardedRef,
|
|
editable,
|
|
setIsSubmitting,
|
|
editorContentCustomClassNames,
|
|
value,
|
|
noBorder,
|
|
borderOnFocus,
|
|
customClassName,
|
|
} = props;
|
|
|
|
const editor = useEditor({
|
|
editable: editable ?? true,
|
|
editorProps: TiptapEditorProps,
|
|
extensions: TiptapExtensions,
|
|
content: value,
|
|
onUpdate: async ({ editor }) => {
|
|
// for instant feedback loop
|
|
setIsSubmitting?.("submitting");
|
|
checkForNodeDeletions(editor);
|
|
if (debouncedUpdatesEnabled) {
|
|
debouncedUpdates({ onChange, editor });
|
|
} else {
|
|
onChange?.(editor.getJSON(), editor.getHTML());
|
|
}
|
|
},
|
|
});
|
|
|
|
const editorRef: React.MutableRefObject<Editor | null> = useRef(null);
|
|
|
|
useImperativeHandle(forwardedRef, () => ({
|
|
clearEditor: () => {
|
|
editorRef.current?.commands.clearContent();
|
|
},
|
|
setEditorValue: (content: string) => {
|
|
editorRef.current?.commands.setContent(content);
|
|
},
|
|
}));
|
|
|
|
const previousState = useRef<EditorState>();
|
|
|
|
const onNodeDeleted = useCallback(async (node: Node) => {
|
|
if (node.type.name === "image") {
|
|
const assetUrlWithWorkspaceId = new URL(node.attrs.src).pathname.substring(1);
|
|
const resStatus = await fileService.deleteImage(assetUrlWithWorkspaceId);
|
|
if (resStatus === 204) {
|
|
console.log("file deleted successfully");
|
|
}
|
|
}
|
|
}, []);
|
|
|
|
const checkForNodeDeletions = useCallback(
|
|
(editor: CoreEditor) => {
|
|
const prevNodesById: Record<string, Node> = {};
|
|
previousState.current?.doc.forEach((node) => {
|
|
if (node.attrs.id) {
|
|
prevNodesById[node.attrs.id] = node;
|
|
}
|
|
});
|
|
|
|
const nodesById: Record<string, Node> = {};
|
|
editor.state?.doc.forEach((node) => {
|
|
if (node.attrs.id) {
|
|
nodesById[node.attrs.id] = node;
|
|
}
|
|
});
|
|
|
|
previousState.current = editor.state;
|
|
|
|
for (const [id, node] of Object.entries(prevNodesById)) {
|
|
if (nodesById[id] === undefined) {
|
|
onNodeDeleted(node);
|
|
}
|
|
}
|
|
},
|
|
[onNodeDeleted]
|
|
);
|
|
|
|
const debouncedUpdates = useDebouncedCallback(async ({ onChange, editor }) => {
|
|
setTimeout(async () => {
|
|
if (onChange) {
|
|
onChange(editor.getJSON(), editor.getHTML());
|
|
}
|
|
}, 500);
|
|
}, 1000);
|
|
|
|
const editorClassNames = `relative w-full max-w-screen-lg sm:rounded-lg sm:shadow-lg mt-2 p-3 relative focus:outline-none rounded-md
|
|
${noBorder ? '' : 'border border-custom-border-200'
|
|
} ${borderOnFocus ? 'focus:border border-custom-border-300' : 'focus:border-0'
|
|
} ${customClassName}`;
|
|
|
|
if (!editor) return null;
|
|
editorRef.current = editor;
|
|
|
|
return (
|
|
<div
|
|
onClick={() => {
|
|
editor?.chain().focus().run();
|
|
}}
|
|
className={`tiptap-editor-container ${editorClassNames}`}
|
|
>
|
|
{editor && <EditorBubbleMenu editor={editor} />}
|
|
<div className={`${editorContentCustomClassNames}`}>
|
|
<EditorContent editor={editor} />
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default Tiptap;
|