diff --git a/@noctaCrdt/Crdt.ts b/@noctaCrdt/Crdt.ts index 34812d0c..674c25a0 100644 --- a/@noctaCrdt/Crdt.ts +++ b/@noctaCrdt/Crdt.ts @@ -103,6 +103,17 @@ export class EditorCRDT extends CRDT { return operation; } + localUpdate(block: Block, pageId: string): RemoteBlockUpdateOperation { + const updatedBlock = this.LinkedList.nodeMap[JSON.stringify(block.id)]; + updatedBlock.animation = block.animation; + updatedBlock.icon = block.icon; + updatedBlock.indent = block.indent; + updatedBlock.style = block.style; + updatedBlock.type = block.type; + // this.LinkedList.nodeMap[JSON.stringify(block.id)] = block; + return { node: updatedBlock, pageId }; + } + remoteUpdate(block: Block, pageId: string): RemoteBlockUpdateOperation { const updatedBlock = this.LinkedList.nodeMap[JSON.stringify(block.id)]; updatedBlock.animation = block.animation; diff --git a/@noctaCrdt/LinkedList.ts b/@noctaCrdt/LinkedList.ts index 5adfcef6..74a1109e 100644 --- a/@noctaCrdt/LinkedList.ts +++ b/@noctaCrdt/LinkedList.ts @@ -27,8 +27,6 @@ export abstract class LinkedList> { deleteNode(id: T["id"]): void { const nodeToDelete = this.getNode(id); - console.log(this.nodeMap); - console.log("nodeToDelete", nodeToDelete, id); if (!nodeToDelete) return; if (this.head && id.equals(this.head)) { diff --git a/client/src/App.tsx b/client/src/App.tsx index 8878f7b0..98190cc3 100644 --- a/client/src/App.tsx +++ b/client/src/App.tsx @@ -1,9 +1,9 @@ -import { useEffect } from "react"; import { useRefreshQuery } from "@apis/auth"; -import { ErrorModal } from "@components/modal/ErrorModal"; -import { WorkSpace } from "@features/workSpace/WorkSpace"; import { useErrorStore } from "@stores/useErrorStore"; import { useUserInfo } from "@stores/useUserStore"; +import { useEffect } from "react"; +import { ErrorModal } from "@components/modal/ErrorModal"; +import { WorkSpace } from "@features/workSpace/WorkSpace"; import { useSocketStore } from "./stores/useSocketStore"; const App = () => { diff --git a/client/src/constants/option.ts b/client/src/constants/option.ts index a31fe835..75193a17 100644 --- a/client/src/constants/option.ts +++ b/client/src/constants/option.ts @@ -5,14 +5,14 @@ export const OPTION_CATEGORIES = { id: "type", label: "전환", options: [ - { id: "p", label: "p" }, - { id: "h1", label: "h1" }, - { id: "h2", label: "h2" }, - { id: "h3", label: "h3" }, - { id: "ul", label: "ul" }, - { id: "ol", label: "ol" }, - { id: "checkbox", label: "checkbox" }, - { id: "blockquote", label: "blockquote" }, + { id: "p", label: "기본" }, + { id: "h1", label: "제목 1" }, + { id: "h2", label: "제목 2" }, + { id: "h3", label: "제목 3" }, + { id: "ul", label: "리스트" }, + { id: "ol", label: "순서 리스트" }, + { id: "checkbox", label: "체크박스" }, + { id: "blockquote", label: "인용문" }, ] as { id: ElementType; label: string }[], }, ANIMATION: { diff --git a/client/src/features/editor/Editor.style.ts b/client/src/features/editor/Editor.style.ts index 7292d919..54f6861e 100644 --- a/client/src/features/editor/Editor.style.ts +++ b/client/src/features/editor/Editor.style.ts @@ -50,3 +50,16 @@ export const checkbox = css({ backgroundColor: "blue.500", }, }); + +export const addNewBlockButton = css({ + display: "flex", + gap: "spacing.sm", + borderRadius: "4px", + padding: "spacing.sm", + color: "gray.900", + opacity: 0.8, + cursor: "pointer", + "&:hover": { + opacity: 1, + }, +}); diff --git a/client/src/features/editor/Editor.tsx b/client/src/features/editor/Editor.tsx index 0af75b70..d3e22eba 100644 --- a/client/src/features/editor/Editor.tsx +++ b/client/src/features/editor/Editor.tsx @@ -8,9 +8,15 @@ import { RemoteCharInsertOperation, serializedEditorDataProps, } from "node_modules/@noctaCrdt/Interfaces.ts"; -import { useRef, useState, useCallback, useEffect, useMemo } from "react"; +import { useRef, useState, useCallback, useEffect, useMemo, useLayoutEffect } from "react"; import { useSocketStore } from "@src/stores/useSocketStore.ts"; -import { editorContainer, editorTitleContainer, editorTitle } from "./Editor.style"; +import { setCaretPosition } from "@src/utils/caretUtils.ts"; +import { + editorContainer, + editorTitleContainer, + editorTitle, + addNewBlockButton, +} from "./Editor.style"; import { Block } from "./components/block/Block.tsx"; import { useBlockDragAndDrop } from "./hooks/useBlockDragAndDrop"; import { useBlockOptionSelect } from "./hooks/useBlockOption.ts"; @@ -25,7 +31,6 @@ interface EditorProps { export interface EditorStateProps { clock: number; linkedList: BlockLinkedList; - currentBlock: BlockId | null; } // TODO: pageId, editorCRDT를 props로 받아와야함 export const Editor = ({ onTitleChange, pageId, serializedEditorData }: EditorProps) => { @@ -47,7 +52,6 @@ export const Editor = ({ onTitleChange, pageId, serializedEditorData }: EditorPr const [editorState, setEditorState] = useState({ clock: editorCRDT.current.clock, linkedList: editorCRDT.current.LinkedList, - currentBlock: null as BlockId | null, }); const { sensors, handleDragEnd } = useBlockDragAndDrop({ editorCRDT: editorCRDT.current, @@ -84,37 +88,9 @@ export const Editor = ({ onTitleChange, pageId, serializedEditorData }: EditorPr onTitleChange(e.target.value); }; - const handleBlockClick = (blockId: BlockId, e: React.MouseEvent) => { - try { - const block = editorState.linkedList.getNode(blockId); - if (!block) { - console.warn("Block not found:", blockId); - return; - } - - const selection = window.getSelection(); - const range = document.caretRangeFromPoint(e.clientX, e.clientY); - - if (!selection || !range) { - console.warn("Selection or range not available"); - return; - } - - // 새로운 Range로 Selection 설정 - selection.removeAllRanges(); - selection.addRange(range); - - // 현재 캐럿 위치를 저장 - const caretPosition = selection.focusOffset; - block.crdt.currentCaret = caretPosition; - - setEditorState((prev) => ({ - ...prev, - currentBlock: blockId, - })); - } catch (error) { - console.error("Error handling block click:", error); - } + const handleBlockClick = (blockId: BlockId) => { + editorCRDT.current.currentBlock = + editorCRDT.current.LinkedList.nodeMap[JSON.stringify(blockId)]; }; const handleBlockInput = useCallback( @@ -136,29 +112,55 @@ export const Editor = ({ onTitleChange, pageId, serializedEditorData }: EditorPr if (caretPosition === 0) { const [addedChar] = newContent; charNode = block.crdt.localInsert(0, addedChar, block.id, pageId); - block.crdt.currentCaret = 1; + editorCRDT.current.currentBlock!.crdt.currentCaret = caretPosition; + requestAnimationFrame(() => { + setCaretPosition({ + blockId: block.id, + linkedList: editorCRDT.current.LinkedList, + position: caretPosition, + }); + }); } else if (caretPosition > currentContent.length) { const addedChar = newContent[newContent.length - 1]; charNode = block.crdt.localInsert(currentContent.length, addedChar, block.id, pageId); - block.crdt.currentCaret = caretPosition; + editorCRDT.current.currentBlock!.crdt.currentCaret = caretPosition; + requestAnimationFrame(() => { + setCaretPosition({ + blockId: block.id, + linkedList: editorCRDT.current.LinkedList, + position: caretPosition, + }); + }); } else { const addedChar = newContent[caretPosition - 1]; charNode = block.crdt.localInsert(caretPosition - 1, addedChar, block.id, pageId); - block.crdt.currentCaret = caretPosition; + editorCRDT.current.currentBlock!.crdt.currentCaret = caretPosition; + requestAnimationFrame(() => { + setCaretPosition({ + blockId: block.id, + linkedList: editorCRDT.current.LinkedList, + position: caretPosition, + }); + }); } sendCharInsertOperation({ node: charNode.node, blockId: block.id, pageId }); } else if (newContent.length < currentContent.length) { // 문자가 삭제된 경우 operationNode = block.crdt.localDelete(caretPosition, block.id, pageId); - block.crdt.currentCaret = caretPosition; sendCharDeleteOperation(operationNode); + editorCRDT.current.currentBlock!.crdt.currentCaret = caretPosition; + requestAnimationFrame(() => { + setCaretPosition({ + blockId: block.id, + linkedList: editorCRDT.current.LinkedList, + position: caretPosition, + }); + }); } - - setEditorState((prev) => ({ + setEditorState({ clock: editorCRDT.current.clock, linkedList: editorCRDT.current.LinkedList, - currentBlock: prev.currentBlock, - })); + }); }, [sendCharInsertOperation, sendCharDeleteOperation], ); @@ -186,6 +188,15 @@ export const Editor = ({ onTitleChange, pageId, serializedEditorData }: EditorPr const subscriptionRef = useRef(false); + useLayoutEffect(() => { + if (!editorCRDT.current.currentBlock) return; + setCaretPosition({ + blockId: editorCRDT.current.currentBlock.id, + linkedList: editorCRDT.current.LinkedList, + position: editorCRDT.current.currentBlock?.crdt.currentCaret, + }); + }, [editorCRDT.current.currentBlock?.crdt.read().length]); + useEffect(() => { if (subscriptionRef.current) return; subscriptionRef.current = true; @@ -195,22 +206,20 @@ export const Editor = ({ onTitleChange, pageId, serializedEditorData }: EditorPr console.log(operation, "block : 입력 확인합니다이"); if (!editorCRDT.current) return; editorCRDT.current.remoteInsert(operation); - setEditorState((prev) => ({ + setEditorState({ clock: editorCRDT.current.clock, linkedList: editorCRDT.current.LinkedList, - currentBlock: prev.currentBlock, - })); + }); }, onRemoteBlockDelete: (operation) => { console.log(operation, "block : 삭제 확인합니다이"); if (!editorCRDT.current) return; editorCRDT.current.remoteDelete(operation); - setEditorState((prev) => ({ + setEditorState({ clock: editorCRDT.current.clock, linkedList: editorCRDT.current.LinkedList, - currentBlock: prev.currentBlock, - })); + }); }, onRemoteCharInsert: (operation) => { @@ -219,11 +228,10 @@ export const Editor = ({ onTitleChange, pageId, serializedEditorData }: EditorPr const targetBlock = editorCRDT.current.LinkedList.nodeMap[JSON.stringify(operation.blockId)]; targetBlock.crdt.remoteInsert(operation); - setEditorState((prev) => ({ + setEditorState({ clock: editorCRDT.current.clock, linkedList: editorCRDT.current.LinkedList, - currentBlock: prev.currentBlock, - })); + }); }, onRemoteCharDelete: (operation) => { @@ -232,11 +240,10 @@ export const Editor = ({ onTitleChange, pageId, serializedEditorData }: EditorPr const targetBlock = editorCRDT.current.LinkedList.nodeMap[JSON.stringify(operation.blockId)]; targetBlock.crdt.remoteDelete(operation); - setEditorState((prev) => ({ + setEditorState({ clock: editorCRDT.current.clock, linkedList: editorCRDT.current.LinkedList, - currentBlock: prev.currentBlock, - })); + }); }, onRemoteBlockUpdate: (operation) => { @@ -245,22 +252,20 @@ export const Editor = ({ onTitleChange, pageId, serializedEditorData }: EditorPr // ?? console.log("타입", operation.node); editorCRDT.current.remoteUpdate(operation.node, operation.pageId); - setEditorState((prev) => ({ + setEditorState({ clock: editorCRDT.current.clock, linkedList: editorCRDT.current.LinkedList, - currentBlock: prev.currentBlock, - })); + }); }, onRemoteBlockReorder: (operation) => { console.log(operation, "block : 재정렬 확인합니다이"); if (!editorCRDT.current) return; editorCRDT.current.remoteReorder(operation); - setEditorState((prev) => ({ + setEditorState({ clock: editorCRDT.current.clock, linkedList: editorCRDT.current.LinkedList, - currentBlock: prev.currentBlock, - })); + }); }, onRemoteCursor: (position) => { @@ -274,18 +279,16 @@ export const Editor = ({ onTitleChange, pageId, serializedEditorData }: EditorPr }; }, []); - const tempBlock = () => { + const addNewBlock = () => { const index = editorCRDT.current.LinkedList.spread().length; // 로컬 삽입을 수행하고 연산 객체를 반환받음 const operation = editorCRDT.current.localInsert(index, ""); sendBlockInsertOperation({ node: operation.node, pageId }); - console.log("operation clock", operation.node); - setEditorState(() => ({ + setEditorState({ clock: operation.node.id.clock, linkedList: editorCRDT.current.LinkedList, - currentBlock: operation.node.id, - })); + }); }; return ( @@ -309,7 +312,7 @@ export const Editor = ({ onTitleChange, pageId, serializedEditorData }: EditorPr key={`${block.id.client}-${block.id.clock}`} id={`${block.id.client}-${block.id.clock}`} block={block} - isActive={block.id === editorState.currentBlock} + isActive={block.id === editorCRDT.current.currentBlock?.id} onInput={handleBlockInput} onCompositionEnd={handleCompositionEnd} onKeyDown={handleKeyDown} @@ -322,7 +325,11 @@ export const Editor = ({ onTitleChange, pageId, serializedEditorData }: EditorPr ))} -
임시
+ {editorState.linkedList.spread().length === 0 && ( +
+ 클릭해서 새로운 블록을 추가하세요 +
+ )} ); diff --git a/client/src/features/editor/components/block/Block.tsx b/client/src/features/editor/components/block/Block.tsx index 897c8b57..68b01f4e 100644 --- a/client/src/features/editor/components/block/Block.tsx +++ b/client/src/features/editor/components/block/Block.tsx @@ -4,7 +4,7 @@ import { AnimationType, ElementType } from "@noctaCrdt/Interfaces"; import { Block as CRDTBlock } from "@noctaCrdt/Node"; import { BlockId } from "@noctaCrdt/NodeId"; import { motion } from "framer-motion"; -import { memo, useRef, useLayoutEffect } from "react"; +import { memo, useRef } from "react"; import { useBlockAnimation } from "../../hooks/useBlockAnimtaion"; import { IconBlock } from "../IconBlock/IconBlock"; import { MenuBlock } from "../MenuBlock/MenuBlock"; @@ -18,7 +18,7 @@ interface BlockProps { onInput: (e: React.FormEvent, block: CRDTBlock) => void; onCompositionEnd: (e: React.CompositionEvent, block: CRDTBlock) => void; onKeyDown: (e: React.KeyboardEvent) => void; - onClick: (blockId: BlockId, e: React.MouseEvent) => void; + onClick: (blockId: BlockId) => void; onAnimationSelect: (blockId: BlockId, animation: AnimationType) => void; onTypeSelect: (blockId: BlockId, type: ElementType) => void; onCopySelect: (blockId: BlockId) => void; @@ -56,31 +56,6 @@ export const Block: React.FC = memo( onInput(e, block); }; - const setFocusAndCursor = () => { - if (blockRef.current && isActive) { - const selection = window.getSelection(); - if (!selection) return; - const range = document.createRange(); - const content = - blockRef.current.firstChild || blockRef.current.appendChild(document.createTextNode("")); - // const position = Math.min(block.crdt.currentCaret, content.textContent?.length || 0); - const position = Math.min( - blockCRDTRef.current.crdt.currentCaret, - content.textContent?.length || 0, - ); - range.setStart(content, position); - range.collapse(true); - selection.removeAllRanges(); - selection.addRange(range); - } - }; - - useLayoutEffect(() => { - // ✅ 추가 - setFocusAndCursor(); - // block.crdt.currentCaret - }, [isActive, blockCRDTRef.current.crdt.currentCaret]); - const handleAnimationSelect = (animation: AnimationType) => { onAnimationSelect(block.id, animation); }; @@ -130,7 +105,7 @@ export const Block: React.FC = memo( onKeyDown={onKeyDown} onInput={handleInput} onCompositionEnd={(e) => onCompositionEnd(e, block)} - onClick={(e) => onClick(block.id, e)} + onClick={() => onClick(block.id)} contentEditable suppressContentEditableWarning className={textContainerStyle({ diff --git a/client/src/features/editor/hooks/useBlockDragAndDrop.ts b/client/src/features/editor/hooks/useBlockDragAndDrop.ts index 61c35eb9..ae880a5b 100644 --- a/client/src/features/editor/hooks/useBlockDragAndDrop.ts +++ b/client/src/features/editor/hooks/useBlockDragAndDrop.ts @@ -91,7 +91,6 @@ export const useBlockDragAndDrop = ({ setEditorState({ clock: editorCRDT.clock, linkedList: editorCRDT.LinkedList, - currentBlock: editorState.currentBlock, }); } catch (error) { console.error("Failed to reorder blocks:", error); diff --git a/client/src/features/editor/hooks/useBlockOption.ts b/client/src/features/editor/hooks/useBlockOption.ts index 595c65e6..6b023bd5 100644 --- a/client/src/features/editor/hooks/useBlockOption.ts +++ b/client/src/features/editor/hooks/useBlockOption.ts @@ -18,7 +18,6 @@ interface useBlockOptionSelectProps { React.SetStateAction<{ clock: number; linkedList: BlockLinkedList; - currentBlock: BlockId | null; }> >; pageId: string; @@ -43,6 +42,7 @@ export const useBlockOptionSelect = ({ if (!block) return; block.type = type; + editorCRDT.currentBlock = block; editorCRDT.remoteUpdate(block, pageId); sendBlockUpdateOperation({ @@ -50,11 +50,10 @@ export const useBlockOptionSelect = ({ pageId, }); - setEditorState((prev) => ({ + setEditorState({ clock: editorCRDT.clock, linkedList: editorCRDT.LinkedList, - currentBlock: blockId || prev.currentBlock, - })); + }); }; const handleAnimationSelect = (blockId: BlockId, animation: AnimationType) => { @@ -62,6 +61,7 @@ export const useBlockOptionSelect = ({ if (!block) return; block.animation = animation; + editorCRDT.currentBlock = block; editorCRDT.remoteUpdate(block, pageId); sendBlockUpdateOperation({ @@ -69,11 +69,10 @@ export const useBlockOptionSelect = ({ pageId, }); - setEditorState((prev) => ({ + setEditorState({ clock: editorCRDT.clock, linkedList: editorCRDT.LinkedList, - currentBlock: blockId || prev.currentBlock, - })); + }); }; const handleCopySelect = (blockId: BlockId) => { @@ -112,11 +111,11 @@ export const useBlockOptionSelect = ({ pageId, }); - setEditorState((prev) => ({ + editorCRDT.currentBlock = operation.node; + setEditorState({ clock: editorCRDT.clock, linkedList: editorCRDT.LinkedList, - currentBlock: operation.node.id || prev.currentBlock, - })); + }); }; const handleDeleteSelect = (blockId: BlockId) => { @@ -125,11 +124,18 @@ export const useBlockOptionSelect = ({ ); sendBlockDeleteOperation(editorCRDT.localDelete(currentIndex, undefined, pageId)); - setEditorState((prev) => ({ + // 삭제할 블록이 현재 활성화된 블록인 경우 + if (editorCRDT.currentBlock?.id.equals(blockId)) { + // 다음 블록이나 이전 블록으로 currentBlock 설정 + const nextBlock = editorCRDT.LinkedList.findByIndex(currentIndex + 1); + const prevBlock = editorCRDT.LinkedList.findByIndex(currentIndex - 1); + editorCRDT.currentBlock = nextBlock || prevBlock || null; + } + + setEditorState({ clock: editorCRDT.clock, linkedList: editorCRDT.LinkedList, - currentBlock: prev.currentBlock, - })); + }); }; return { diff --git a/client/src/features/editor/hooks/useMarkdownGrammer.ts b/client/src/features/editor/hooks/useMarkdownGrammer.ts index 5d66022c..d38ccbd9 100644 --- a/client/src/features/editor/hooks/useMarkdownGrammer.ts +++ b/client/src/features/editor/hooks/useMarkdownGrammer.ts @@ -11,6 +11,7 @@ import { BlockId } from "@noctaCrdt/NodeId"; import { useCallback } from "react"; import { EditorStateProps } from "@features/editor/Editor"; import { checkMarkdownPattern } from "@src/features/editor/utils/markdownPatterns"; +import { setCaretPosition } from "@src/utils/caretUtils"; interface useMarkdownGrammerProps { editorCRDT: EditorCRDT; @@ -19,7 +20,6 @@ interface useMarkdownGrammerProps { React.SetStateAction<{ clock: number; linkedList: BlockLinkedList; - currentBlock: BlockId | null; }> >; pageId: string; @@ -44,7 +44,6 @@ export const useMarkdownGrammer = ({ const handleKeyDown = useCallback( (e: React.KeyboardEvent) => { const createNewBlock = (index: number): RemoteBlockInsertOperation => { - console.log("createNewBlock"); const operation = editorCRDT.localInsert(index, ""); // TODO: 블록 타입이 초기화가 안됨??? operation.node.type = "p"; @@ -52,14 +51,19 @@ export const useMarkdownGrammer = ({ }; const updateEditorState = (newBlockId: BlockId | null = null) => { - setEditorState((prev) => ({ + if ( + newBlockId !== null && + editorCRDT.currentBlock !== editorCRDT.LinkedList.getNode(newBlockId) + ) { + editorCRDT.currentBlock = editorCRDT.LinkedList.getNode(newBlockId); + } + setEditorState({ clock: editorCRDT.clock, linkedList: editorCRDT.LinkedList, - currentBlock: newBlockId || prev.currentBlock, - })); + }); }; - const currentBlockId = editorState.currentBlock; + const currentBlockId = editorCRDT.currentBlock ? editorCRDT.currentBlock.id : null; if (!currentBlockId) return; const currentBlock = editorCRDT.LinkedList.getNode(currentBlockId); @@ -73,15 +77,16 @@ export const useMarkdownGrammer = ({ case "Enter": { if (e.nativeEvent.isComposing) return; e.preventDefault(); - const caretPosition = currentBlock.crdt.currentCaret; + const selection = window.getSelection(); + if (!selection) return; + const caretPosition = selection.focusOffset; const currentContent = currentBlock.crdt.read(); const afterText = currentContent.slice(caretPosition); if (!currentContent && currentBlock.type !== "p") { currentBlock.type = "p"; - // TODO: Update요청 - // remote update 보내기 - sendBlockUpdateOperation({ node: currentBlock, pageId }); + sendBlockUpdateOperation(editorCRDT.localUpdate(currentBlock, pageId)); + editorCRDT.currentBlock = currentBlock; updateEditorState(); break; } @@ -93,7 +98,6 @@ export const useMarkdownGrammer = ({ operation.node.crdt = new BlockCRDT(editorCRDT.client); sendBlockInsertOperation({ node: operation.node, pageId }); - updateEditorState(operation.node.id); break; } @@ -102,8 +106,6 @@ export const useMarkdownGrammer = ({ if (afterText) { // 캐럿 이후의 텍스트만 제거 for (let i = currentContent.length - 1; i >= caretPosition; i--) { - // char remote delete 보내기 - // currentBlock.crdt.localDelete(i, currentBlock.id); sendCharDeleteOperation(currentBlock.crdt.localDelete(i, currentBlock.id, pageId)); } } @@ -116,8 +118,6 @@ export const useMarkdownGrammer = ({ // 캐럿 이후의 텍스트 있으면 새 블록에 추가 if (afterText) { afterText.split("").forEach((char, i) => { - // char remote insert 보내기 - // newBlock.crdt.localInsert(i, char, newBlock.id); sendCharInsertOperation( operation.node.crdt.localInsert(i, char, operation.node.id, pageId), ); @@ -127,24 +127,22 @@ export const useMarkdownGrammer = ({ // 현재 블록이 li나 checkbox면 동일한 타입으로 생성 if (["ul", "ol", "checkbox"].includes(currentBlock.type)) { operation.node.type = currentBlock.type; - // TODO: Update요청 - // remote update 보내기 - sendBlockUpdateOperation({ node: operation.node, pageId }); + sendBlockUpdateOperation(editorCRDT.localUpdate(operation.node, pageId)); } - // !! TODO socket.update updateEditorState(operation.node.id); break; } case "Backspace": { + const selection = window.getSelection(); + const caretPosition = selection?.focusOffset || 0; const currentContent = currentBlock.crdt.read(); if (currentContent === "") { e.preventDefault(); if (currentBlock.indent > 0) { currentBlock.indent -= 1; - // TODO: Update요청 - // remote update 보내기 - sendBlockUpdateOperation({ node: currentBlock, pageId }); + sendBlockUpdateOperation(editorCRDT.localUpdate(currentBlock, pageId)); + editorCRDT.currentBlock = currentBlock; updateEditorState(); break; } @@ -152,9 +150,8 @@ export const useMarkdownGrammer = ({ if (currentBlock.type !== "p") { // 마지막 블록이면 기본 블록으로 변경 currentBlock.type = "p"; - // TODO: Update요청 - // remote update 보내기 - sendBlockUpdateOperation({ node: currentBlock, pageId }); + sendBlockUpdateOperation(editorCRDT.localUpdate(currentBlock, pageId)); + editorCRDT.currentBlock = currentBlock; updateEditorState(); break; } @@ -162,8 +159,6 @@ export const useMarkdownGrammer = ({ const prevBlock = currentIndex > 0 ? editorCRDT.LinkedList.findByIndex(currentIndex - 1) : null; if (prevBlock) { - // remote delete 보내기 - // editorCRDT.localDelete(currentIndex, undefined, pageId); sendBlockDeleteOperation(editorCRDT.localDelete(currentIndex, undefined, pageId)); prevBlock.crdt.currentCaret = prevBlock.crdt.read().length; editorCRDT.currentBlock = prevBlock; @@ -171,21 +166,18 @@ export const useMarkdownGrammer = ({ } break; } else { - const { currentCaret } = currentBlock.crdt; - if (currentCaret === 0) { + if (caretPosition === 0) { if (currentBlock.indent > 0) { currentBlock.indent -= 1; - // TODO: Update요청 - // remote update 보내기 - sendBlockUpdateOperation({ node: currentBlock, pageId }); + sendBlockUpdateOperation(editorCRDT.localUpdate(currentBlock, pageId)); + editorCRDT.currentBlock = currentBlock; updateEditorState(); break; } if (currentBlock.type !== "p") { currentBlock.type = "p"; - // TODO: Update요청 - // remote update 보내기 - sendBlockUpdateOperation({ node: currentBlock, pageId }); + sendBlockUpdateOperation(editorCRDT.localUpdate(currentBlock, pageId)); + editorCRDT.currentBlock = currentBlock; updateEditorState(); // FIX: 서윤님 피드백 반영 } else { @@ -194,8 +186,6 @@ export const useMarkdownGrammer = ({ if (prevBlock) { const prevBlockEndCaret = prevBlock.crdt.read().length; currentContent.split("").forEach((char) => { - // char remote insert 보내기 - // prevBlock.crdt.localInsert(prevBlock.crdt.read().length, char, prevBlock.id); sendCharInsertOperation( prevBlock.crdt.localInsert( prevBlock.crdt.read().length, @@ -205,12 +195,10 @@ export const useMarkdownGrammer = ({ ), ); sendCharDeleteOperation( - currentBlock.crdt.localDelete(currentCaret, currentBlock.id, pageId), + currentBlock.crdt.localDelete(caretPosition, currentBlock.id, pageId), ); }); prevBlock.crdt.currentCaret = prevBlockEndCaret; - // remote delete 보내기 - // editorCRDT.localDelete(currentIndex, undefined, pageId); sendBlockDeleteOperation(editorCRDT.localDelete(currentIndex, undefined, pageId)); updateEditorState(prevBlock.id); e.preventDefault(); @@ -229,21 +217,18 @@ export const useMarkdownGrammer = ({ // shift + tab: 들여쓰기 감소 if (currentBlock.indent > 0) { currentBlock.indent -= 1; - // TODO: Update요청 - // remote update 보내기 - sendBlockUpdateOperation({ node: currentBlock, pageId }); - updateEditorState(currentBlock.id); + sendBlockUpdateOperation(editorCRDT.localUpdate(currentBlock, pageId)); + editorCRDT.currentBlock = currentBlock; + updateEditorState(); } } else { // tab: 들여쓰기 증가 - console.log("tab"); const maxIndent = 3; if (currentBlock.indent < maxIndent) { currentBlock.indent += 1; - // TODO: Update요청 - // remote update 보내기 - sendBlockUpdateOperation({ node: currentBlock, pageId }); - updateEditorState(currentBlock.id); + sendBlockUpdateOperation(editorCRDT.localUpdate(currentBlock, pageId)); + editorCRDT.currentBlock = currentBlock; + updateEditorState(); } } } @@ -251,6 +236,9 @@ export const useMarkdownGrammer = ({ } case " ": { + // 여기 수정함 + const selection = window.getSelection(); + if (!selection) return; const currentContent = currentBlock.crdt.read(); const markdownElement = checkMarkdownPattern(currentContent); if (markdownElement && currentBlock.type === "p") { @@ -259,17 +247,13 @@ export const useMarkdownGrammer = ({ currentBlock.type = markdownElement.type; let deleteCount = 0; while (deleteCount < markdownElement.length) { - // char remote delete 보내기 - // currentBlock.crdt.localDelete(0, currentBlock.id); sendCharDeleteOperation(currentBlock.crdt.localDelete(0, currentBlock.id, pageId)); deleteCount += 1; } - // TODO: Update요청 - // remote update 보내기 - sendBlockUpdateOperation({ node: currentBlock, pageId }); - // !!TODO emit 송신 + sendBlockUpdateOperation(editorCRDT.localUpdate(currentBlock, pageId)); currentBlock.crdt.currentCaret = 0; - updateEditorState(currentBlock.id); + editorCRDT.currentBlock = currentBlock; + updateEditorState(); } break; @@ -277,57 +261,82 @@ export const useMarkdownGrammer = ({ case "ArrowUp": case "ArrowDown": { - e.preventDefault(); - const blocks = editorCRDT.LinkedList.spread(); - // 이전/다음 블록 존재 여부 확인 const hasPrevBlock = currentIndex > 0; - const hasNextBlock = currentIndex < blocks.length - 1; - // 방향키에 따라 이동 가능 여부 확인 - if (e.key === "ArrowUp" && !hasPrevBlock) return; - if (e.key === "ArrowDown" && !hasNextBlock) return; + const hasNextBlock = currentIndex < editorCRDT.LinkedList.spread().length - 1; + if (e.key === "ArrowUp" && !hasPrevBlock) { + e.preventDefault(); + return; + } + if (e.key === "ArrowDown" && !hasNextBlock) { + e.preventDefault(); + return; + } - const targetIndex = e.key === "ArrowUp" ? currentIndex - 1 : currentIndex + 1; + const selection = window.getSelection(); + const caretPosition = selection?.focusOffset || 0; - const targetBlock = blocks[targetIndex]; - targetBlock.crdt.currentCaret = Math.min( - currentBlock.crdt.currentCaret, - targetBlock.crdt.read().length, - ); - updateEditorState(targetBlock.id); + // 이동할 블록 결정 + const targetIndex = e.key === "ArrowUp" ? currentIndex - 1 : currentIndex + 1; + const targetBlock = editorCRDT.LinkedList.findByIndex(targetIndex); + if (!targetBlock) return; + e.preventDefault(); + targetBlock.crdt.currentCaret = Math.min(caretPosition, targetBlock.crdt.read().length); + editorCRDT.currentBlock = targetBlock; + setCaretPosition({ + blockId: targetBlock.id, + linkedList: editorCRDT.LinkedList, + position: Math.min(caretPosition, targetBlock.crdt.read().length), + }); break; } - // TODO: ArrowLeft, ArrowRight 블록 이동시 캐럿 이상하게 움직이는 것 수정 case "ArrowLeft": case "ArrowRight": { - const { currentCaret } = currentBlock.crdt; + const selection = window.getSelection(); + const caretPosition = selection?.focusOffset || 0; const textLength = currentBlock.crdt.read().length; - if (e.key === "ArrowLeft" && currentCaret > 0) { - // currentBlock.crdt.currentCaret = currentCaret - 1; - currentBlock.crdt.currentCaret -= 1; - break; - } - if (e.key === "ArrowRight" && currentCaret < textLength) { - // currentBlock.crdt.currentCaret = currentCaret + 1; - currentBlock.crdt.currentCaret += 1; - break; - } - if (e.key === "ArrowLeft" && currentCaret === 0 && currentIndex > 0) { + + // 왼쪽 끝에서 이전 블록으로 + if (e.key === "ArrowLeft" && caretPosition === 0 && currentIndex > 0) { + e.preventDefault(); // 기본 동작 방지 const prevBlock = editorCRDT.LinkedList.findByIndex(currentIndex - 1); if (prevBlock) { prevBlock.crdt.currentCaret = prevBlock.crdt.read().length; - updateEditorState(prevBlock.id); + editorCRDT.currentBlock = prevBlock; + setCaretPosition({ + blockId: prevBlock.id, + linkedList: editorCRDT.LinkedList, + position: prevBlock.crdt.read().length, + }); } break; - } - if (e.key === "ArrowRight" && currentCaret === textLength) { - // TODO: 다음 블록 없을 때 처리 crdt에 추가 - const nextBlock = editorCRDT.LinkedList.findByIndex(currentIndex + 1); + // 오른쪽 끝에서 다음 블록으로 + } else if ( + e.key === "ArrowRight" && + caretPosition === textLength && + currentIndex < editorCRDT.LinkedList.spread().length - 1 + ) { + e.preventDefault(); // 기본 동작 방지 + const nextBlock = editorState.linkedList.findByIndex(currentIndex + 1); if (nextBlock) { nextBlock.crdt.currentCaret = 0; - updateEditorState(nextBlock.id); + editorCRDT.currentBlock = nextBlock; + setCaretPosition({ + blockId: nextBlock.id, + linkedList: editorCRDT.LinkedList, + position: 0, + }); } break; + // 블록 내에서 이동하는 경우 + } else { + if (e.key === "ArrowLeft") { + currentBlock.crdt.currentCaret -= 1; + } else { + currentBlock.crdt.currentCaret += 1; + } } + + break; } } }, diff --git a/client/src/utils/caretUtils.ts b/client/src/utils/caretUtils.ts new file mode 100644 index 00000000..577a4015 --- /dev/null +++ b/client/src/utils/caretUtils.ts @@ -0,0 +1,71 @@ +import { BlockLinkedList, TextLinkedList } from "@noctaCrdt/LinkedList"; +import { BlockId } from "@noctaCrdt/NodeId"; + +interface SetCaretPositionProps { + blockId: BlockId; + linkedList: BlockLinkedList | TextLinkedList; + clientX?: number; + clientY?: number; + position?: number; // 특정 위치로 캐럿을 설정하고 싶을 때 사용 +} + +export const setCaretPosition = ({ + blockId, + linkedList, + clientX, + clientY, + position, +}: SetCaretPositionProps): boolean => { + try { + const selection = window.getSelection(); + if (!selection) return false; + + const blockElements = Array.from( + document.querySelectorAll('.d_flex.pos_relative.w_full[data-group="true"]'), + ); + const currentIndex = linkedList.spread().findIndex((b) => b.id === blockId); + const targetElement = blockElements[currentIndex]; + if (!targetElement) return false; + + const editableDiv = targetElement.querySelector('[contenteditable="true"]') as HTMLDivElement; + if (!editableDiv) return false; + + editableDiv.focus(); + + let range: Range; + + if (clientX !== undefined && clientY !== undefined) { + // 클릭 위치에 따른 캐럿 설정 + const clickRange = document.caretRangeFromPoint(clientX, clientY); + if (!clickRange) return false; + range = clickRange; + } else if (position !== undefined) { + // 특정 위치에 캐럿 설정 + range = document.createRange(); + const textNode = + Array.from(editableDiv.childNodes).find((node) => node.nodeType === Node.TEXT_NODE) || null; + if (!textNode) { + // 텍스트 노드가 없으면 새로운 텍스트 노드를 추가 + const newTextNode = document.createTextNode(""); + editableDiv.appendChild(newTextNode); + range.setStart(newTextNode, 0); + } else { + // position이 텍스트 길이를 초과하지 않도록 조정 + const safePosition = Math.min(position, textNode.textContent?.length || 0); + range.setStart(textNode, safePosition); + } + + range.collapse(true); + } else { + return false; + } + + selection.removeAllRanges(); + selection.addRange(range); + + return true; + } catch (error) { + console.error("Error setting caret position:", error); + return false; + } +};