import React, { useState, useRef, useCallback, memo, useMemo } from "react"; import { useParams } from "react-router-dom"; import { ActionIcon, Center, Divider, Group, Paper, Stack, Tabs, Badge, Text, ScrollArea, Tooltip, } from "@mantine/core"; import CommentListItem from "@/features/comment/components/comment-list-item"; import { useCommentsQuery, useCreateCommentMutation, } from "@/features/comment/queries/comment-query"; import CommentEditor from "@/features/comment/components/comment-editor"; import CommentActions from "@/features/comment/components/comment-actions"; import { useFocusWithin } from "@mantine/hooks"; import { IComment } from "@/features/comment/types/comment.types.ts"; import { usePageQuery } from "@/features/page/queries/page-query.ts"; import { extractPageSlugId } from "@/lib"; import { useTranslation } from "react-i18next"; import { useGetSpaceBySlugQuery } from "@/features/space/queries/space-query.ts"; import { IconArrowUp, IconMessageOff, IconX } from "@tabler/icons-react"; import { useAtom } from "jotai"; import { currentUserAtom } from "@/features/user/atoms/current-user-atom"; import { CustomAvatar } from "@/components/ui/custom-avatar.tsx"; interface CommentListWithTabsProps { onClose?: () => void; } // Index replies by their parent id once (O(n)), instead of an O(n^2) filter per // thread. Replies whose parent is not in `items` are still grouped under their // parentCommentId (they simply won't be reached by the top-level walk). // Exported for unit testing. export function buildChildrenByParent( items: IComment[] | undefined, ): Map { const m = new Map(); for (const c of items ?? []) { if (c.parentCommentId) { const arr = m.get(c.parentCommentId); if (arr) arr.push(c); else m.set(c.parentCommentId, [c]); } } return m; } function CommentListWithTabs({ onClose }: CommentListWithTabsProps) { const { t } = useTranslation(); const { pageSlug } = useParams(); const { data: page } = usePageQuery({ pageId: extractPageSlugId(pageSlug) }); const { data: comments, isLoading: isCommentsLoading, isError, } = useCommentsQuery({ pageId: page?.id }); const createCommentMutation = useCreateCommentMutation(); // mutateAsync is a stable reference across renders; depend on it (not the // mutation object) so the reply/comment callbacks stay stable. const createCommentAsync = createCommentMutation.mutateAsync; const { data: space } = useGetSpaceBySlugQuery(page?.space?.slug); const canEdit = page?.permissions?.canEdit ?? false; const canComment = canEdit || (space?.settings?.comments?.allowViewerComments === true); // Separate active and resolved comments const { activeComments, resolvedComments } = useMemo(() => { if (!comments?.items) { return { activeComments: [], resolvedComments: [] }; } const parentComments = comments.items.filter( (comment: IComment) => comment.parentCommentId === null, ); const active = parentComments.filter( (comment: IComment) => !comment.resolvedAt, ); const resolved = parentComments.filter( (comment: IComment) => comment.resolvedAt, ); return { activeComments: active, resolvedComments: resolved }; }, [comments]); // Index replies by their parent once, instead of an O(n^2) filter per thread. // The map ref changes on any comments update, so MemoizedChildComments re-runs // (cheap) and re-looks-up, while memoized CommentListItems skip unchanged items. const childrenByParent = useMemo( () => buildChildrenByParent(comments?.items), [comments?.items], ); const [isPageCommentLoading, setIsPageCommentLoading] = useState(false); const handleAddPageComment = useCallback( async (_commentId: string, content: string) => { try { setIsPageCommentLoading(true); const createdComment = await createCommentAsync({ pageId: page?.id, content: JSON.stringify(content), }); setTimeout(() => { const selector = `div[data-comment-id="${createdComment.id}"]`; const commentElement = document.querySelector(selector); commentElement?.scrollIntoView({ behavior: "smooth", block: "center", }); }, 400); } catch (error) { console.error("Failed to post comment:", error); } finally { setIsPageCommentLoading(false); } }, [createCommentAsync, page?.id], ); const handleAddReply = useCallback( async (commentId: string, content: string) => { // Pending state lives inside CommentEditorWithActions so sending a reply // does not churn renderComments and re-render the whole list. try { const commentData = { pageId: page?.id, parentCommentId: commentId, content: JSON.stringify(content), }; await createCommentAsync(commentData); } catch (error) { console.error("Failed to post comment:", error); } }, [createCommentAsync, page?.id], ); const renderComments = useCallback( (comment: IComment) => (
{!comment.resolvedAt && canComment && ( <> )}
), [ childrenByParent, handleAddReply, page?.id, space?.membership?.role, canComment, canEdit, ], ); if (isCommentsLoading) { return <>; } if (isError) { return
{t("Error loading comments.")}
; } const totalComments = activeComments.length + resolvedComments.length; const pageCommentInput = canComment ? ( ) : null; return (
Resolved -> Open switch. keepMounted={false} style={{ flex: "1 1 auto", display: "flex", flexDirection: "column", overflow: "hidden", }} > {/* Header row: full-width centered tab list with the close button overlaid on the right. */}
{activeComments.length} } > {t("Open")} {resolvedComments.length} } > {t("Resolved")} {onClose && ( )}
{/* keepMounted keeps the Open panel alive even while Resolved is active, so a lazily-mounted reply editor's draft (and an in-progress edit) is not discarded on tab switch. */} {activeComments.length === 0 ? (
{t("No open comments.")}
) : ( activeComments.map(renderComments) )}
{resolvedComments.length === 0 ? (
{t("No resolved comments.")}
) : ( resolvedComments.map(renderComments) )}
{pageCommentInput}
); } interface ChildCommentsProps { childrenByParent: Map; parentId: string; pageId: string; canComment: boolean; canEdit?: boolean; userSpaceRole?: string; } const ChildComments = ({ childrenByParent, parentId, pageId, canComment, canEdit, userSpaceRole, }: ChildCommentsProps) => { const children = childrenByParent.get(parentId) ?? []; return (
{children.map((childComment) => (
))}
); }; const MemoizedChildComments = memo(ChildComments); export const CommentEditorWithActions = ({ commentId, onSave, placeholder = undefined, }) => { const { t } = useTranslation(); // Lazily mount the TipTap reply editor: until the user interacts with the // stub, no editor instance is created for this thread. Once mounted it stays // mounted so the draft is preserved. const [mounted, setMounted] = useState(false); const [content, setContent] = useState(""); const [isSending, setIsSending] = useState(false); const { ref, focused } = useFocusWithin(); const commentEditorRef = useRef(null); const activate = useCallback(() => setMounted(true), []); const handleSave = useCallback(async () => { try { setIsSending(true); await onSave(commentId, content); setContent(""); commentEditorRef.current?.clearContent(); } finally { setIsSending(false); } }, [commentId, content, onSave]); if (!mounted) { return (
{ if (e.key === "Enter" || e.key === " ") { e.preventDefault(); activate(); } }} style={{ padding: "6px", fontSize: "var(--mantine-font-size-sm)", lineHeight: 1.4, color: "var(--mantine-color-placeholder)", cursor: "text", borderRadius: "var(--mantine-radius-sm)", }} > {placeholder || t("Reply...")}
); } return (
{focused && }
); }; const PageCommentInput = ({ onSave, isLoading }) => { const { t } = useTranslation(); const [content, setContent] = useState(""); const { ref, focused } = useFocusWithin(); const commentEditorRef = useRef(null); const [currentUser] = useAtom(currentUserAtom); const handleSave = useCallback(() => { onSave(null, content); setContent(""); commentEditorRef.current?.clearContent(); }, [content, onSave]); return (
{focused && ( e.preventDefault()} loading={isLoading} style={{ position: "absolute", right: 8, bottom: 15 }} > )}
); }; export default CommentListWithTabs;