diff --git a/client/src/features/editor/components/TextOptionModal/TextOptionModal.tsx b/client/src/features/editor/components/TextOptionModal/TextOptionModal.tsx index 30026afe..3312dda4 100644 --- a/client/src/features/editor/components/TextOptionModal/TextOptionModal.tsx +++ b/client/src/features/editor/components/TextOptionModal/TextOptionModal.tsx @@ -1,9 +1,11 @@ +import { Char } from "@noctaCrdt/Node"; import { motion } from "framer-motion"; import { useEffect, useRef, useState } from "react"; import { createPortal } from "react-dom"; import { modalContainer, optionButton, optionModal } from "./TextOptionModal.style"; interface SelectionModalProps { + selectedNodes: Array | null; isOpen: boolean; onBoldSelect: () => void; onItalicSelect: () => void; @@ -17,6 +19,7 @@ const ModalPortal = ({ children }: { children: React.ReactNode }) => { }; export const TextOptionModal = ({ + selectedNodes, isOpen, onBoldSelect, onItalicSelect, @@ -29,6 +32,12 @@ export const TextOptionModal = ({ top: 0, left: 0, }); + const [styleState, setStyleState] = useState({ + isBold: false, + isItalic: false, + isUnderline: false, + isStrike: false, + }); useEffect(() => { if (!isOpen) return; @@ -60,38 +69,141 @@ export const TextOptionModal = ({ return () => { document.removeEventListener("mousedown", handleClickOutside); + setStyleState({ + isBold: false, + isItalic: false, + isUnderline: false, + isStrike: false, + }); }; }, [isOpen, onClose]); + useEffect(() => { + if (!selectedNodes || selectedNodes.length === 0) { + setStyleState({ + isBold: false, + isItalic: false, + isUnderline: false, + isStrike: false, + }); + return; + } + + // 각 스타일의 출현 횟수를 계산 + const styleCounts = { + bold: 0, + italic: 0, + underline: 0, + strike: 0, + }; + + // 각 노드의 스타일을 확인하고 카운트 + selectedNodes.forEach((node) => { + // node.style이 undefined인 경우 빈 배열로 처리 + const nodeStyles = Array.isArray(node.style) ? node.style : node.style ? [node.style] : []; + + if (nodeStyles.includes("bold")) styleCounts.bold += 1; + if (nodeStyles.includes("italic")) styleCounts.italic += 1; + if (nodeStyles.includes("underline")) styleCounts.underline += 1; + if (nodeStyles.includes("strike")) styleCounts.strike += 1; + }); + + const totalNodes = selectedNodes.length; + + // 모든 노드가 해당 스타일을 가지고 있는 경우에만 true + setStyleState({ + isBold: styleCounts.bold === totalNodes, + isItalic: styleCounts.italic === totalNodes, + isUnderline: styleCounts.underline === totalNodes, + isStrike: styleCounts.strike === totalNodes, + }); + }, [selectedNodes]); + if (!isOpen) return; return ( - {isOpen && ( - -
- - + - + - + -
-
- )} + + + +
); }; diff --git a/client/src/features/editor/components/block/Block.tsx b/client/src/features/editor/components/block/Block.tsx index c2419ede..78cc5c07 100644 --- a/client/src/features/editor/components/block/Block.tsx +++ b/client/src/features/editor/components/block/Block.tsx @@ -217,6 +217,7 @@ export const Block: React.FC = memo( /> handleStyleSelect("bold")}