docmost/apps/client/src/features/editor/title-editor.tsx

147 lines
4.1 KiB
TypeScript
Raw Normal View History

2024-04-13 17:16:31 +01:00
import "@/features/editor/styles/index.css";
import React, { useEffect, useState } from "react";
import { EditorContent, useEditor } from "@tiptap/react";
import { Document } from "@tiptap/extension-document";
import { Heading } from "@tiptap/extension-heading";
import { Text } from "@tiptap/extension-text";
import { Placeholder } from "@tiptap/extension-placeholder";
import { useAtomValue } from "jotai";
import {
pageEditorAtom,
titleEditorAtom,
} from "@/features/editor/atoms/editor-atoms";
2024-11-28 18:53:29 +00:00
import { useUpdatePageMutation } from "@/features/page/queries/page-query";
2024-04-13 17:16:31 +01:00
import { useDebouncedValue } from "@mantine/hooks";
import { useAtom } from "jotai";
import { treeDataAtom } from "@/features/page/tree/atoms/tree-data-atom";
import { updateTreeNodeName } from "@/features/page/tree/utils";
import { useQueryEmit } from "@/features/websocket/use-query-emit.ts";
import { History } from "@tiptap/extension-history";
import { buildPageUrl } from "@/features/page/page.utils.ts";
import { useNavigate } from "react-router-dom";
import { useTranslation } from "react-i18next";
2023-11-22 20:42:34 +00:00
export interface TitleEditorProps {
pageId: string;
slugId: string;
title: string;
spaceSlug: string;
editable: boolean;
2023-11-22 20:42:34 +00:00
}
export function TitleEditor({
pageId,
slugId,
title,
spaceSlug,
editable,
}: TitleEditorProps) {
const { t } = useTranslation();
const [debouncedTitleState, setDebouncedTitleState] = useState(null);
const [debouncedTitle] = useDebouncedValue(debouncedTitleState, 500);
2024-11-28 18:53:29 +00:00
const {
data: updatedPageData,
mutate: updatePageMutation,
status,
} = useUpdatePageMutation();
2023-11-24 18:06:32 +00:00
const pageEditor = useAtomValue(pageEditorAtom);
2023-11-22 20:42:34 +00:00
const [, setTitleEditor] = useAtom(titleEditorAtom);
const [treeData, setTreeData] = useAtom(treeDataAtom);
const emit = useQueryEmit();
const navigate = useNavigate();
const [activePageId, setActivePageId] = useState(pageId);
2023-11-22 20:42:34 +00:00
const titleEditor = useEditor({
extensions: [
Document.extend({
2024-04-13 17:16:31 +01:00
content: "heading",
2023-11-22 20:42:34 +00:00
}),
Heading.configure({
levels: [1],
}),
Text,
Placeholder.configure({
placeholder: t("Untitled"),
showOnlyWhenEditable: false,
2023-11-22 20:42:34 +00:00
}),
History.configure({
depth: 20,
}),
2023-11-22 20:42:34 +00:00
],
onCreate({ editor }) {
if (editor) {
// @ts-ignore
setTitleEditor(editor);
}
},
onUpdate({ editor }) {
const currentTitle = editor.getText();
setDebouncedTitleState(currentTitle);
setActivePageId(pageId);
2023-11-22 20:42:34 +00:00
},
editable: editable,
2023-11-22 20:42:34 +00:00
content: title,
});
useEffect(() => {
const pageSlug = buildPageUrl(spaceSlug, slugId, title);
navigate(pageSlug, { replace: true });
}, [title]);
useEffect(() => {
if (debouncedTitle !== null && activePageId === pageId) {
2024-11-28 18:53:29 +00:00
updatePageMutation({
pageId: pageId,
title: debouncedTitle,
});
2024-11-28 18:53:29 +00:00
}
}, [debouncedTitle]);
useEffect(() => {
if (status === "success" && updatedPageData) {
const newTreeData = updateTreeNodeName(treeData, pageId, debouncedTitle);
setTreeData(newTreeData);
setTimeout(() => {
emit({
operation: "updateOne",
2024-11-28 18:53:29 +00:00
spaceId: updatedPageData.spaceId,
entity: ["pages"],
id: pageId,
payload: { title: debouncedTitle, slugId: slugId },
});
}, 50);
2023-11-22 20:42:34 +00:00
}
2024-11-28 18:53:29 +00:00
}, [updatedPageData, status]);
2023-11-22 20:42:34 +00:00
useEffect(() => {
if (titleEditor && title !== titleEditor.getText()) {
titleEditor.commands.setContent(title);
}
2023-11-24 18:03:53 +00:00
}, [pageId, title, titleEditor]);
useEffect(() => {
setTimeout(() => {
2024-04-13 17:16:31 +01:00
titleEditor?.commands.focus("end");
2023-11-24 18:03:53 +00:00
}, 500);
}, [titleEditor]);
2023-11-22 20:42:34 +00:00
function handleTitleKeyDown(event) {
2023-11-24 18:03:53 +00:00
if (!titleEditor || !pageEditor || event.shiftKey) return;
2023-11-22 20:42:34 +00:00
const { key } = event;
const { $head } = titleEditor.state.selection;
2024-04-13 17:16:31 +01:00
const shouldFocusEditor =
key === "Enter" ||
key === "ArrowDown" ||
(key === "ArrowRight" && !$head.nodeAfter);
2023-11-22 20:42:34 +00:00
if (shouldFocusEditor) {
2024-04-13 17:16:31 +01:00
pageEditor.commands.focus("start");
2023-11-22 20:42:34 +00:00
}
}
2024-04-13 17:16:31 +01:00
return <EditorContent editor={titleEditor} onKeyDown={handleTitleKeyDown} />;
2023-11-22 20:42:34 +00:00
}