mirror of
https://github.com/docmost/docmost
synced 2025-03-28 21:13:28 +00:00
fix comments
This commit is contained in:
parent
7b91b6d642
commit
eb95a619db
@ -1,5 +1,5 @@
|
||||
import { atom } from 'jotai';
|
||||
|
||||
export const showCommentPopupAtom = atom(false);
|
||||
export const activeCommentIdAtom = atom<string | null>(null);
|
||||
export const draftCommentIdAtom = atom<string | null>(null);
|
||||
export const showCommentPopupAtom = atom<boolean>(false);
|
||||
export const activeCommentIdAtom = atom<string>('');
|
||||
export const draftCommentIdAtom = atom<string>('');
|
||||
|
@ -1,22 +0,0 @@
|
||||
import { useParams } from 'react-router-dom';
|
||||
import React from 'react';
|
||||
import classes from '@/features/comment/components/comment.module.css';
|
||||
import { Text } from '@mantine/core';
|
||||
import CommentList from '@/features/comment/components/comment-list';
|
||||
import { useCommentsQuery } from '@/features/comment/queries/comment';
|
||||
|
||||
export default function Comments() {
|
||||
const { pageId } = useParams();
|
||||
const { data, isLoading, isError } = useCommentsQuery(pageId);
|
||||
|
||||
if (isLoading) {
|
||||
return <></>;
|
||||
}
|
||||
|
||||
return (
|
||||
<div className={classes.wrapper}>
|
||||
<Text mb="md" fw={500}>Comments</Text>
|
||||
{data ? <CommentList comments={data} /> : 'No comments yet'}
|
||||
</div>
|
||||
);
|
||||
}
|
@ -11,7 +11,8 @@ import { Editor } from '@tiptap/core';
|
||||
import CommentEditor from '@/features/comment/components/comment-editor';
|
||||
import CommentActions from '@/features/comment/components/comment-actions';
|
||||
import { currentUserAtom } from '@/features/user/atoms/current-user-atom';
|
||||
import { useCreateCommentMutation } from '@/features/comment/queries/comment';
|
||||
import { useCreateCommentMutation } from '@/features/comment/queries/comment-query';
|
||||
import { asideStateAtom } from '@/components/navbar/atoms/sidebar-atom';
|
||||
|
||||
interface CommentDialogProps {
|
||||
editor: Editor,
|
||||
@ -20,15 +21,16 @@ interface CommentDialogProps {
|
||||
|
||||
function CommentDialog({ editor, pageId }: CommentDialogProps) {
|
||||
const [comment, setComment] = useState('');
|
||||
const [, setShowCommentPopup] = useAtom<boolean>(showCommentPopupAtom);
|
||||
const [, setActiveCommentId] = useAtom<string | null>(activeCommentIdAtom);
|
||||
const [draftCommentId, setDraftCommentId] = useAtom<string | null>(draftCommentIdAtom);
|
||||
const [, setShowCommentPopup] = useAtom(showCommentPopupAtom);
|
||||
const [, setActiveCommentId] = useAtom(activeCommentIdAtom);
|
||||
const [draftCommentId, setDraftCommentId] = useAtom(draftCommentIdAtom);
|
||||
const [currentUser] = useAtom(currentUserAtom);
|
||||
const [, setAsideState] = useAtom(asideStateAtom);
|
||||
const useClickOutsideRef = useClickOutside(() => {
|
||||
handleDialogClose();
|
||||
});
|
||||
const createCommentMutation = useCreateCommentMutation();
|
||||
const { isLoading } = createCommentMutation;
|
||||
const { isPending } = createCommentMutation;
|
||||
|
||||
const handleDialogClose = () => {
|
||||
setShowCommentPopup(false);
|
||||
@ -54,6 +56,7 @@ function CommentDialog({ editor, pageId }: CommentDialogProps) {
|
||||
editor.chain().setComment(createdComment.id).unsetCommentDecoration().run();
|
||||
setActiveCommentId(createdComment.id);
|
||||
|
||||
setAsideState({ tab: 'comments', isAsideOpen: true });
|
||||
setTimeout(() => {
|
||||
const selector = `div[data-comment-id="${createdComment.id}"]`;
|
||||
const commentElement = document.querySelector(selector);
|
||||
@ -61,7 +64,7 @@ function CommentDialog({ editor, pageId }: CommentDialogProps) {
|
||||
});
|
||||
} finally {
|
||||
setShowCommentPopup(false);
|
||||
setDraftCommentId(null);
|
||||
setDraftCommentId('');
|
||||
}
|
||||
};
|
||||
|
||||
@ -86,7 +89,7 @@ function CommentDialog({ editor, pageId }: CommentDialogProps) {
|
||||
<CommentEditor onUpdate={handleCommentEditorChange} placeholder="Write a comment"
|
||||
editable={true} autofocus={true}
|
||||
/>
|
||||
<CommentActions onSave={handleAddComment} isLoading={isLoading}
|
||||
<CommentActions onSave={handleAddComment} isLoading={isPending}
|
||||
/>
|
||||
</Stack>
|
||||
</Dialog>
|
||||
|
@ -3,10 +3,10 @@ import { Placeholder } from '@tiptap/extension-placeholder';
|
||||
import { Underline } from '@tiptap/extension-underline';
|
||||
import { Link } from '@tiptap/extension-link';
|
||||
import { StarterKit } from '@tiptap/starter-kit';
|
||||
import React from 'react';
|
||||
import classes from './comment.module.css';
|
||||
import { useFocusWithin } from '@mantine/hooks';
|
||||
import clsx from 'clsx';
|
||||
import { forwardRef, useImperativeHandle } from 'react';
|
||||
|
||||
interface CommentEditorProps {
|
||||
defaultContent?: any;
|
||||
@ -16,7 +16,8 @@ interface CommentEditorProps {
|
||||
autofocus?: boolean;
|
||||
}
|
||||
|
||||
function CommentEditor({ defaultContent, onUpdate, editable, placeholder, autofocus }: CommentEditorProps) {
|
||||
const CommentEditor = forwardRef(({ defaultContent, onUpdate, editable, placeholder, autofocus }: CommentEditorProps,
|
||||
ref) => {
|
||||
const { ref: focusRef, focused } = useFocusWithin();
|
||||
|
||||
const commentEditor = useEditor({
|
||||
@ -39,6 +40,12 @@ function CommentEditor({ defaultContent, onUpdate, editable, placeholder, autofo
|
||||
autofocus: (autofocus && 'end') || false,
|
||||
});
|
||||
|
||||
useImperativeHandle(ref, () => ({
|
||||
clearContent: () => {
|
||||
commentEditor.commands.clearContent();
|
||||
},
|
||||
}));
|
||||
|
||||
return (
|
||||
<div ref={focusRef} className={classes.commentEditor}>
|
||||
<EditorContent editor={commentEditor}
|
||||
@ -46,7 +53,6 @@ function CommentEditor({ defaultContent, onUpdate, editable, placeholder, autofo
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
|
||||
}
|
||||
});
|
||||
|
||||
export default CommentEditor;
|
||||
|
@ -1,22 +1,23 @@
|
||||
import { Group, Avatar, Text, Box } from '@mantine/core';
|
||||
import { Group, Text, Box } from '@mantine/core';
|
||||
import React, { useState } from 'react';
|
||||
import classes from './comment.module.css';
|
||||
import { useAtomValue } from 'jotai';
|
||||
import { timeAgo } from '@/lib/time-ago';
|
||||
import { timeAgo } from '@/lib/time';
|
||||
import CommentEditor from '@/features/comment/components/comment-editor';
|
||||
import { editorAtom } from '@/features/editor/atoms/editorAtom';
|
||||
import CommentActions from '@/features/comment/components/comment-actions';
|
||||
import CommentMenu from '@/features/comment/components/comment-menu';
|
||||
import ResolveComment from '@/features/comment/components/resolve-comment';
|
||||
import { useHover } from '@mantine/hooks';
|
||||
import { useDeleteCommentMutation, useUpdateCommentMutation } from '@/features/comment/queries/comment';
|
||||
import { useDeleteCommentMutation, useUpdateCommentMutation } from '@/features/comment/queries/comment-query';
|
||||
import { IComment } from '@/features/comment/types/comment.types';
|
||||
import { UserAvatar } from '@/components/ui/user-avatar';
|
||||
|
||||
interface CommentListItemProps {
|
||||
comment: IComment;
|
||||
}
|
||||
|
||||
function CommentListItem({ comment }: CommentListItemProps) {
|
||||
|
||||
const { hovered, ref } = useHover();
|
||||
const [isEditing, setIsEditing] = useState(false);
|
||||
const [isLoading, setIsLoading] = useState(false);
|
||||
@ -58,28 +59,20 @@ function CommentListItem({ comment }: CommentListItemProps) {
|
||||
return (
|
||||
<Box ref={ref} pb="xs">
|
||||
<Group>
|
||||
{comment.creator.avatarUrl ? (
|
||||
<Avatar
|
||||
src={comment.creator.avatarUrl}
|
||||
alt={comment.creator.name}
|
||||
size="sm"
|
||||
radius="xl"
|
||||
/>) : (
|
||||
<Avatar size="sm" color="blue">{comment.creator.name.charAt(0)}</Avatar>
|
||||
)}
|
||||
<UserAvatar color="blue" size="sm" avatarUrl={comment.creator.avatarUrl}
|
||||
name={comment.creator.name}
|
||||
/>
|
||||
|
||||
<div style={{ flex: 1 }}>
|
||||
<Group justify="space-between" wrap="nowrap">
|
||||
<Text size="sm" fw={500} lineClamp={1}>{comment.creator.name}</Text>
|
||||
|
||||
<div style={{ visibility: hovered ? 'visible' : 'hidden' }}>
|
||||
{!comment.parentCommentId && (
|
||||
{/*!comment.parentCommentId && (
|
||||
<ResolveComment commentId={comment.id} pageId={comment.pageId} resolvedAt={comment.resolvedAt} />
|
||||
)}
|
||||
)*/}
|
||||
|
||||
<CommentMenu commentId={comment.id}
|
||||
onEditComment={handleEditToggle}
|
||||
onDeleteComment={handleDeleteComment} />
|
||||
<CommentMenu onEditComment={handleEditToggle} onDeleteComment={handleDeleteComment} />
|
||||
</div>
|
||||
</Group>
|
||||
|
||||
|
@ -1,55 +1,30 @@
|
||||
import { Divider, Paper, ScrollArea } from '@mantine/core';
|
||||
import React, { useState, useRef } from 'react';
|
||||
import { useParams } from 'react-router-dom';
|
||||
import { Divider, Paper } 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 React, { useState } from 'react';
|
||||
import CommentListItem from '@/features/comment/components/comment-list-item';
|
||||
import { IComment } from '@/features/comment/types/comment.types';
|
||||
import { useFocusWithin } from '@mantine/hooks';
|
||||
import { useCreateCommentMutation } from '@/features/comment/queries/comment';
|
||||
|
||||
interface CommentListProps {
|
||||
comments: IComment[];
|
||||
}
|
||||
|
||||
function CommentList({ comments }: CommentListProps) {
|
||||
function CommentList() {
|
||||
const { pageId } = useParams();
|
||||
const { data: comments, isLoading: isCommentsLoading, isError } = useCommentsQuery(pageId);
|
||||
const [isLoading, setIsLoading] = useState(false);
|
||||
const createCommentMutation = useCreateCommentMutation();
|
||||
const [isLoading, setIsLoading] = useState<boolean>(false);
|
||||
|
||||
const getChildComments = (parentId) => {
|
||||
return comments.filter(comment => comment.parentCommentId === parentId);
|
||||
};
|
||||
if (isCommentsLoading) {
|
||||
return <></>;
|
||||
}
|
||||
|
||||
const renderChildComments = (parentId) => {
|
||||
const children = getChildComments(parentId);
|
||||
return (
|
||||
<div>
|
||||
{children.map(childComment => (
|
||||
<div key={childComment.id}>
|
||||
<CommentListItem comment={childComment} />
|
||||
{renderChildComments(childComment.id)}
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
if (isError) {
|
||||
return <div>Error loading comments.</div>;
|
||||
}
|
||||
|
||||
const CommentEditorWithActions = ({ commentId, onSave, isLoading }) => {
|
||||
const [content, setContent] = useState('');
|
||||
const { ref, focused } = useFocusWithin();
|
||||
|
||||
const handleSave = () => {
|
||||
onSave(commentId, content);
|
||||
setContent('');
|
||||
};
|
||||
|
||||
return (
|
||||
<div ref={ref}>
|
||||
<CommentEditor onUpdate={setContent} editable={true} />
|
||||
|
||||
{focused && <CommentActions onSave={handleSave} isLoading={isLoading} />}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
if (!comments || comments.length === 0) {
|
||||
return <>No comments yet.</>;
|
||||
}
|
||||
|
||||
const renderComments = (comment) => {
|
||||
const handleAddReply = async (commentId, content) => {
|
||||
@ -63,41 +38,68 @@ function CommentList({ comments }: CommentListProps) {
|
||||
|
||||
await createCommentMutation.mutateAsync(commentData);
|
||||
} catch (error) {
|
||||
console.error('Failed to add reply:', error);
|
||||
console.error('Failed to post comment:', error);
|
||||
} finally {
|
||||
setIsLoading(false);
|
||||
}
|
||||
|
||||
//setCommentsAtom(prevComments => [...prevComments, createdComment]);
|
||||
};
|
||||
|
||||
return (
|
||||
<Paper shadow="sm" radius="md" p="sm" mb="sm" withBorder
|
||||
key={comment.id} data-comment-id={comment.id}
|
||||
>
|
||||
<Paper shadow="sm" radius="md" p="sm" mb="sm" withBorder key={comment.id} data-comment-id={comment.id}>
|
||||
<div>
|
||||
<CommentListItem comment={comment} />
|
||||
{renderChildComments(comment.id)}
|
||||
<ChildComments comments={comments} parentId={comment.id} />
|
||||
</div>
|
||||
|
||||
<Divider my={4} />
|
||||
|
||||
<CommentEditorWithActions onSave={handleAddReply} isLoading={isLoading} />
|
||||
|
||||
<CommentEditorWithActions commentId={comment.id} onSave={handleAddReply} isLoading={isLoading} />
|
||||
</Paper>
|
||||
);
|
||||
};
|
||||
|
||||
return (
|
||||
<ScrollArea style={{ height: '85vh' }} scrollbarSize={4} type="scroll">
|
||||
<div style={{ paddingBottom: '200px' }}>
|
||||
{comments
|
||||
.filter(comment => comment.parentCommentId === null)
|
||||
.map(comment => renderComments(comment))
|
||||
}
|
||||
</div>
|
||||
</ScrollArea>
|
||||
<>
|
||||
{comments.filter(comment => comment.parentCommentId === null).map(renderComments)}
|
||||
</>
|
||||
);
|
||||
}
|
||||
|
||||
const ChildComments = ({ comments, parentId }) => {
|
||||
const getChildComments = (parentId) => {
|
||||
return comments.filter(comment => comment.parentCommentId === parentId);
|
||||
};
|
||||
|
||||
return (
|
||||
<div>
|
||||
{getChildComments(parentId).map(childComment => (
|
||||
<div key={childComment.id}>
|
||||
<CommentListItem comment={childComment} />
|
||||
<ChildComments comments={comments} parentId={childComment.id} />
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
const CommentEditorWithActions = ({ commentId, onSave, isLoading }) => {
|
||||
const [content, setContent] = useState('');
|
||||
const { ref, focused } = useFocusWithin();
|
||||
const commentEditorRef = useRef();
|
||||
|
||||
const handleSave = () => {
|
||||
onSave(commentId, content);
|
||||
setContent('');
|
||||
commentEditorRef?.current.clearContent();
|
||||
};
|
||||
|
||||
return (
|
||||
<div ref={ref}>
|
||||
<CommentEditor ref={commentEditorRef} onUpdate={setContent} editable={true} />
|
||||
{focused && <CommentActions onSave={handleSave} isLoading={isLoading} />}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
|
||||
|
||||
export default CommentList;
|
||||
|
@ -1,7 +1,7 @@
|
||||
import { ActionIcon } from '@mantine/core';
|
||||
import { IconCircleCheck } from '@tabler/icons-react';
|
||||
import { modals } from '@mantine/modals';
|
||||
import { useResolveCommentMutation } from '@/features/comment/queries/comment';
|
||||
import { useResolveCommentMutation } from '@/features/comment/queries/comment-query';
|
||||
|
||||
function ResolveComment({ commentId, pageId, resolvedAt }) {
|
||||
const resolveCommentMutation = useResolveCommentMutation();
|
||||
|
@ -8,7 +8,7 @@ import {
|
||||
import { IComment, IResolveComment } from '@/features/comment/types/comment.types';
|
||||
import { notifications } from '@mantine/notifications';
|
||||
|
||||
export const RQ_KEY = (pageId: string) => ['comment', pageId];
|
||||
export const RQ_KEY = (pageId: string) => ['comments', pageId];
|
||||
|
||||
export function useCommentsQuery(pageId: string): UseQueryResult<IComment[], Error> {
|
||||
return useQuery({
|
||||
@ -57,7 +57,7 @@ export function useDeleteCommentMutation(pageId?: string) {
|
||||
return useMutation({
|
||||
mutationFn: (commentId: string) => deleteComment(commentId),
|
||||
onSuccess: (data, variables) => {
|
||||
let comments = queryClient.getQueryData(RQ_KEY(pageId));
|
||||
let comments = queryClient.getQueryData(RQ_KEY(pageId)) as IComment[];
|
||||
if (comments) {
|
||||
comments = comments.filter(comment => comment.id !== variables);
|
||||
queryClient.setQueryData(RQ_KEY(pageId), comments);
|
||||
@ -77,7 +77,7 @@ export function useResolveCommentMutation() {
|
||||
mutationFn: (data: IResolveComment) => resolveComment(data),
|
||||
onSuccess: (data: IComment, variables) => {
|
||||
|
||||
const currentComments = queryClient.getQueryData(RQ_KEY(data.pageId));
|
||||
const currentComments = queryClient.getQueryData(RQ_KEY(data.pageId)) as IComment[];
|
||||
|
||||
if (currentComments) {
|
||||
const updatedComments = currentComments.map((comment) =>
|
@ -43,7 +43,7 @@ export function useDeletePageMutation() {
|
||||
return useMutation({
|
||||
mutationFn: (pageId: string) => deletePage(pageId),
|
||||
onSuccess: () => {
|
||||
notifications.show({ title: 'Page deleted successfully' });
|
||||
notifications.show({ message: 'Page deleted successfully' });
|
||||
},
|
||||
});
|
||||
}
|
@ -1,5 +0,0 @@
|
||||
import { formatDistanceStrict } from 'date-fns';
|
||||
|
||||
export function timeAgo(date: Date){
|
||||
return formatDistanceStrict(new Date(date), new Date(), { addSuffix: true })
|
||||
}
|
16
client/src/lib/time.ts
Normal file
16
client/src/lib/time.ts
Normal file
@ -0,0 +1,16 @@
|
||||
import { formatDistanceStrict } from 'date-fns';
|
||||
import { format, isToday, isYesterday } from 'date-fns';
|
||||
|
||||
export function timeAgo(date: Date) {
|
||||
return formatDistanceStrict(new Date(date), new Date(), { addSuffix: true });
|
||||
}
|
||||
|
||||
export function formatDate(date: Date) {
|
||||
if (isToday(date)) {
|
||||
return `Today, ${format(date, 'h:mma')}`;
|
||||
} else if (isYesterday(date)) {
|
||||
return `Yesterday, ${format(date, 'h:mma')}`;
|
||||
} else {
|
||||
return format(date, 'MMM dd, yyyy, h:mma');
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user