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";
|
2024-04-27 15:40:22 +01:00
|
|
|
import { useQueryEmit } from "@/features/websocket/use-query-emit.ts";
|
|
|
|
import { History } from "@tiptap/extension-history";
|
2024-05-31 21:51:44 +01:00
|
|
|
import { buildPageUrl } from "@/features/page/page.utils.ts";
|
2024-10-26 15:48:40 +01:00
|
|
|
import { useNavigate } from "react-router-dom";
|
2025-01-04 21:17:17 +08:00
|
|
|
import { useTranslation } from "react-i18next";
|
2023-11-22 20:42:34 +00:00
|
|
|
|
|
|
|
export interface TitleEditorProps {
|
|
|
|
pageId: string;
|
2024-05-18 03:19:42 +01:00
|
|
|
slugId: string;
|
2024-04-21 16:38:59 +01:00
|
|
|
title: string;
|
2024-05-31 21:51:44 +01:00
|
|
|
spaceSlug: string;
|
2024-06-03 02:54:12 +01:00
|
|
|
editable: boolean;
|
2023-11-22 20:42:34 +00:00
|
|
|
}
|
|
|
|
|
2024-05-31 21:51:44 +01:00
|
|
|
export function TitleEditor({
|
|
|
|
pageId,
|
|
|
|
slugId,
|
|
|
|
title,
|
|
|
|
spaceSlug,
|
2024-06-03 02:54:12 +01:00
|
|
|
editable,
|
2024-05-31 21:51:44 +01:00
|
|
|
}: TitleEditorProps) {
|
2025-01-04 21:17:17 +08:00
|
|
|
const { t } = useTranslation();
|
2024-05-18 03:19:42 +01:00
|
|
|
const [debouncedTitleState, setDebouncedTitleState] = useState(null);
|
2024-10-26 15:48:40 +01:00
|
|
|
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);
|
2023-11-23 02:11:28 +00:00
|
|
|
const [treeData, setTreeData] = useAtom(treeDataAtom);
|
2024-04-27 15:40:22 +01:00
|
|
|
const emit = useQueryEmit();
|
2024-05-18 03:19:42 +01:00
|
|
|
const navigate = useNavigate();
|
2024-10-26 15:48:40 +01:00
|
|
|
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({
|
2025-01-04 21:17:17 +08:00
|
|
|
placeholder: t("Untitled"),
|
2024-06-03 02:54:12 +01:00
|
|
|
showOnlyWhenEditable: false,
|
2023-11-22 20:42:34 +00:00
|
|
|
}),
|
2024-04-27 15:40:22 +01: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);
|
2024-10-26 15:48:40 +01:00
|
|
|
setActivePageId(pageId);
|
2023-11-22 20:42:34 +00:00
|
|
|
},
|
2024-06-03 02:54:12 +01:00
|
|
|
editable: editable,
|
2023-11-22 20:42:34 +00:00
|
|
|
content: title,
|
|
|
|
});
|
|
|
|
|
|
|
|
useEffect(() => {
|
2024-05-31 21:51:44 +01:00
|
|
|
const pageSlug = buildPageUrl(spaceSlug, slugId, title);
|
2024-05-18 03:19:42 +01:00
|
|
|
navigate(pageSlug, { replace: true });
|
|
|
|
}, [title]);
|
|
|
|
|
|
|
|
useEffect(() => {
|
2024-10-26 15:48:40 +01:00
|
|
|
if (debouncedTitle !== null && activePageId === pageId) {
|
2024-11-28 18:53:29 +00:00
|
|
|
updatePageMutation({
|
2024-05-18 03:19:42 +01:00
|
|
|
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);
|
2023-11-23 02:11:28 +00:00
|
|
|
|
2024-04-27 15:40:22 +01:00
|
|
|
setTimeout(() => {
|
|
|
|
emit({
|
|
|
|
operation: "updateOne",
|
2024-11-28 18:53:29 +00:00
|
|
|
spaceId: updatedPageData.spaceId,
|
2024-04-27 15:40:22 +01:00
|
|
|
entity: ["pages"],
|
|
|
|
id: pageId,
|
2024-05-18 03:19:42 +01:00
|
|
|
payload: { title: debouncedTitle, slugId: slugId },
|
2024-04-27 15:40:22 +01:00
|
|
|
});
|
|
|
|
}, 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
|
|
|
|
2023-11-23 02:11:28 +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-23 02:11:28 +00:00
|
|
|
|
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
|
|
|
}
|