mirror of
https://github.com/docmost/docmost.git
synced 2026-05-08 15:23:07 +08:00
d2629afff2
* feat: add heading extension with unique ID support and scroll functionality * Added unique id for heading * remove baseUrl heading storage * move heading to extensions package * WIP * support anchors in mentions * enhance scrolling functionality * nodeId function * fix nanoid import * Bring unique-id extension local * fixes * fix internal link scroll in public pages * add unique id server side * rename mention anchor to anchorId * capture first anchorId on paste --------- Co-authored-by: Romik <40670677+RomikMakavana@users.noreply.github.com>
98 lines
2.5 KiB
TypeScript
98 lines
2.5 KiB
TypeScript
import "@/features/editor/styles/index.css";
|
|
import React, { useCallback, useEffect, useMemo, useRef } from "react";
|
|
import { EditorProvider } from "@tiptap/react";
|
|
import { mainExtensions } from "@/features/editor/extensions/extensions";
|
|
import { Document } from "@tiptap/extension-document";
|
|
import { Heading, generateNodeId, UniqueID } from "@docmost/editor-ext";
|
|
import { Text } from "@tiptap/extension-text";
|
|
import { Placeholder } from "@tiptap/extension-placeholder";
|
|
import { useAtom } from "jotai";
|
|
import { readOnlyEditorAtom } from "@/features/editor/atoms/editor-atoms.ts";
|
|
import { useEditorScroll } from "./hooks/use-editor-scroll";
|
|
|
|
interface PageEditorProps {
|
|
title: string;
|
|
content: any;
|
|
pageId?: string;
|
|
}
|
|
|
|
export default function ReadonlyPageEditor({
|
|
title,
|
|
content,
|
|
pageId,
|
|
}: PageEditorProps) {
|
|
const [, setReadOnlyEditor] = useAtom(readOnlyEditorAtom);
|
|
const isComponentMounted = useRef(false);
|
|
const editorCreated = useRef(false);
|
|
|
|
const canScroll = useCallback(
|
|
() => isComponentMounted.current && editorCreated.current,
|
|
[isComponentMounted, editorCreated],
|
|
);
|
|
const initialScrollTo = window.location.hash
|
|
? window.location.hash.slice(1)
|
|
: "";
|
|
const { handleScrollTo } = useEditorScroll({ canScroll, initialScrollTo });
|
|
|
|
useEffect(() => {
|
|
isComponentMounted.current = true;
|
|
}, []);
|
|
|
|
const extensions = useMemo(() => {
|
|
const filteredExtensions = mainExtensions.filter(
|
|
(ext) => ext.name !== "uniqueID",
|
|
);
|
|
|
|
return [
|
|
...filteredExtensions,
|
|
UniqueID.configure({
|
|
types: ["heading", "paragraph"],
|
|
updateDocument: false,
|
|
}),
|
|
];
|
|
}, []);
|
|
|
|
const titleExtensions = [
|
|
Document.extend({
|
|
content: "heading",
|
|
}),
|
|
Heading,
|
|
Text,
|
|
Placeholder.configure({
|
|
placeholder: "Untitled",
|
|
showOnlyWhenEditable: false,
|
|
}),
|
|
];
|
|
|
|
return (
|
|
<>
|
|
<EditorProvider
|
|
editable={false}
|
|
immediatelyRender={true}
|
|
extensions={titleExtensions}
|
|
content={title}
|
|
></EditorProvider>
|
|
|
|
<EditorProvider
|
|
editable={false}
|
|
immediatelyRender={true}
|
|
extensions={extensions}
|
|
content={content}
|
|
onCreate={({ editor }) => {
|
|
if (editor) {
|
|
if (pageId) {
|
|
editor.storage.pageId = pageId;
|
|
}
|
|
// @ts-ignore
|
|
setReadOnlyEditor(editor);
|
|
|
|
handleScrollTo(editor);
|
|
editorCreated.current = true;
|
|
}
|
|
}}
|
|
></EditorProvider>
|
|
<div style={{ paddingBottom: "20vh" }}></div>
|
|
</>
|
|
);
|
|
}
|