-
Notifications
You must be signed in to change notification settings - Fork 0
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[FE] feat: 꿀조합 댓글 기능 구현 #744
Changes from 21 commits
162d561
6a47996
7d849bc
5dc497f
e713a04
f5c90fb
33b36c0
9a09d11
b0a25cc
0e5fa37
02001e1
78a91ba
7dba7c3
89af090
67a109b
996e9d0
6b4e689
d6ed26b
a860bb6
838d38e
980e205
a08dd8e
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
import type { Meta, StoryObj } from '@storybook/react'; | ||
|
||
import CommentForm from './CommentForm'; | ||
|
||
const meta: Meta<typeof CommentForm> = { | ||
title: 'recipe/CommentForm', | ||
component: CommentForm, | ||
}; | ||
|
||
export default meta; | ||
type Story = StoryObj<typeof meta>; | ||
|
||
export const Default: Story = {}; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,101 @@ | ||
import { Button, Spacing, Text, Textarea, useTheme } from '@fun-eat/design-system'; | ||
import type { ChangeEventHandler, FormEventHandler } from 'react'; | ||
import { useState } from 'react'; | ||
import styled from 'styled-components'; | ||
|
||
import { SvgIcon } from '@/components/Common'; | ||
import { useToastActionContext } from '@/hooks/context'; | ||
import useRecipeCommentMutation from '@/hooks/queries/recipe/useRecipeCommentMutation'; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. index에 추가해주세요.! |
||
|
||
interface CommentFormProps { | ||
recipeId: number; | ||
} | ||
|
||
const MAX_COMMENT_LENGTH = 200; | ||
|
||
const CommentForm = ({ recipeId }: CommentFormProps) => { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 👍 |
||
const [commentValue, setCommentValue] = useState(''); | ||
const { mutate } = useRecipeCommentMutation(recipeId); | ||
|
||
const theme = useTheme(); | ||
const { toast } = useToastActionContext(); | ||
|
||
const handleCommentInput: ChangeEventHandler<HTMLTextAreaElement> = (e) => { | ||
setCommentValue(e.target.value); | ||
}; | ||
|
||
const handleSubmitComment: FormEventHandler<HTMLFormElement> = (e) => { | ||
e.preventDefault(); | ||
|
||
mutate( | ||
{ comment: commentValue }, | ||
{ | ||
onSuccess: () => { | ||
setCommentValue(''); | ||
toast.success('댓글이 등록되었습니다.'); | ||
}, | ||
onError: (error) => { | ||
if (error instanceof Error) { | ||
toast.error(error.message); | ||
return; | ||
} | ||
|
||
toast.error('댓글을 등록하는데 오류가 발생했습니다.'); | ||
}, | ||
} | ||
); | ||
}; | ||
|
||
return ( | ||
<CommentFormContainer> | ||
<Form onSubmit={handleSubmitComment}> | ||
<CommentTextarea | ||
placeholder="댓글을 입력하세요. (200자)" | ||
value={commentValue} | ||
onChange={handleCommentInput} | ||
maxLength={MAX_COMMENT_LENGTH} | ||
/> | ||
<SubmitButton variant="transparent" disabled={commentValue.length === 0}> | ||
<SvgIcon | ||
variant="plane" | ||
width={30} | ||
height={30} | ||
color={commentValue.length === 0 ? theme.colors.gray2 : theme.colors.gray4} | ||
/> | ||
</SubmitButton> | ||
</Form> | ||
<Spacing size={8} /> | ||
<Text size="xs" color={theme.textColors.info} align="right"> | ||
{commentValue.length}자 / {MAX_COMMENT_LENGTH}자 | ||
</Text> | ||
</CommentFormContainer> | ||
); | ||
}; | ||
|
||
export default CommentForm; | ||
|
||
const CommentFormContainer = styled.div` | ||
position: fixed; | ||
bottom: 0; | ||
width: calc(100% - 40px); | ||
max-width: 540px; | ||
padding: 16px 0; | ||
background: ${({ theme }) => theme.backgroundColors.default}; | ||
`; | ||
|
||
const Form = styled.form` | ||
display: flex; | ||
gap: 4px; | ||
justify-content: space-around; | ||
align-items: center; | ||
`; | ||
|
||
const CommentTextarea = styled(Textarea)` | ||
height: 50px; | ||
padding: 8px; | ||
font-size: 1.4rem; | ||
`; | ||
|
||
const SubmitButton = styled(Button)` | ||
cursor: ${({ disabled }) => (disabled ? 'not-allowed' : 'pointer')}; | ||
`; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
import type { Meta, StoryObj } from '@storybook/react'; | ||
|
||
import CommentItem from './CommentItem'; | ||
|
||
import comments from '@/mocks/data/comments.json'; | ||
|
||
const meta: Meta<typeof CommentItem> = { | ||
title: 'recipe/CommentItem', | ||
component: CommentItem, | ||
args: { | ||
recipeComment: comments.comments[0], | ||
}, | ||
}; | ||
|
||
export default meta; | ||
type Story = StoryObj<typeof meta>; | ||
|
||
export const Default: Story = {}; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,50 @@ | ||
import { Divider, Spacing, Text, useTheme } from '@fun-eat/design-system'; | ||
import styled from 'styled-components'; | ||
|
||
import type { Comment } from '@/types/recipe'; | ||
import { getFormattedDate } from '@/utils/date'; | ||
|
||
interface CommentItemProps { | ||
recipeComment: Comment; | ||
} | ||
|
||
const CommentItem = ({ recipeComment }: CommentItemProps) => { | ||
const theme = useTheme(); | ||
const { author, comment, createdAt } = recipeComment; | ||
|
||
return ( | ||
<> | ||
<AuthorWrapper> | ||
<AuthorProfileImage src={author.profileImage} alt={`${author.nickname}님의 프로필`} width={32} height={32} /> | ||
<div> | ||
<Text size="xs" color={theme.textColors.info}> | ||
{author.nickname} 님 | ||
</Text> | ||
<Text size="xs" color={theme.textColors.info}> | ||
{getFormattedDate(createdAt)} | ||
</Text> | ||
</div> | ||
</AuthorWrapper> | ||
<CommentContent size="sm">{comment}</CommentContent> | ||
<Divider variant="disabled" /> | ||
<Spacing size={16} /> | ||
</> | ||
); | ||
}; | ||
|
||
export default CommentItem; | ||
|
||
const AuthorWrapper = styled.div` | ||
display: flex; | ||
gap: 12px; | ||
align-items: center; | ||
`; | ||
|
||
const AuthorProfileImage = styled.img` | ||
border: 1px solid ${({ theme }) => theme.colors.primary}; | ||
border-radius: 50%; | ||
`; | ||
|
||
const CommentContent = styled(Text)` | ||
margin: 16px 0; | ||
`; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
import type { Meta, StoryObj } from '@storybook/react'; | ||
|
||
import CommentList from './CommentList'; | ||
|
||
const meta: Meta<typeof CommentList> = { | ||
title: 'recipe/CommentList', | ||
component: CommentList, | ||
}; | ||
|
||
export default meta; | ||
type Story = StoryObj<typeof meta>; | ||
|
||
export const Default: Story = {}; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
import { Heading, Spacing } from '@fun-eat/design-system'; | ||
import { useRef } from 'react'; | ||
|
||
import CommentItem from '../CommentItem/CommentItem'; | ||
|
||
import { useIntersectionObserver } from '@/hooks/common'; | ||
import { useInfiniteRecipeCommentQuery } from '@/hooks/queries/recipe'; | ||
|
||
interface CommentListProps { | ||
recipeId: number; | ||
} | ||
|
||
const CommentList = ({ recipeId }: CommentListProps) => { | ||
const scrollRef = useRef<HTMLDivElement>(null); | ||
|
||
const { fetchNextPage, hasNextPage, data } = useInfiniteRecipeCommentQuery(Number(recipeId)); | ||
useIntersectionObserver<HTMLDivElement>(fetchNextPage, scrollRef, hasNextPage); | ||
Comment on lines
+16
to
+17
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. ♾️👍 |
||
|
||
const comments = data.pages.flatMap((page) => page.comments); | ||
|
||
return ( | ||
<section> | ||
<Heading as="h3" size="lg"> | ||
댓글 ({comments.length}개) | ||
</Heading> | ||
<Spacing size={12} /> | ||
{comments.map((comment) => ( | ||
<CommentItem key={comment.id} recipeComment={comment} /> | ||
))} | ||
<div ref={scrollRef} style={{ height: '1px' }} aria-hidden /> | ||
</section> | ||
); | ||
}; | ||
|
||
export default CommentList; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,36 @@ | ||
import { useSuspendedInfiniteQuery } from '../useSuspendedInfiniteQuery'; | ||
|
||
import { recipeApi } from '@/apis'; | ||
import type { CommentResponse } from '@/types/response'; | ||
|
||
interface PageParam { | ||
lastId: number; | ||
totalElements: number | null; | ||
} | ||
|
||
const fetchRecipeComments = async (pageParam: PageParam, recipeId: number) => { | ||
const { lastId, totalElements } = pageParam; | ||
const response = await recipeApi.get({ | ||
params: `/${recipeId}/comments`, | ||
queries: `?lastId=${lastId}&totalElements=${totalElements}`, | ||
}); | ||
const data: CommentResponse = await response.json(); | ||
return data; | ||
}; | ||
|
||
const useInfiniteRecipeCommentQuery = (recipeId: number) => { | ||
return useSuspendedInfiniteQuery( | ||
['recipeComment', recipeId], | ||
({ pageParam = { lastId: 0, totalElements: null } }) => fetchRecipeComments(pageParam, recipeId), | ||
{ | ||
getNextPageParam: (prevResponse: CommentResponse) => { | ||
const lastId = prevResponse.comments[prevResponse.comments.length - 1].id; | ||
const totalElements = prevResponse.totalElements; | ||
const lastCursor = { lastId: lastId, totalElements: totalElements }; | ||
return prevResponse.hasNext ? lastCursor : undefined; | ||
Comment on lines
+26
to
+30
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 대박..👍 |
||
}, | ||
} | ||
); | ||
}; | ||
|
||
export default useInfiniteRecipeCommentQuery; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
import { useMutation, useQueryClient } from '@tanstack/react-query'; | ||
|
||
import { recipeApi } from '@/apis'; | ||
|
||
interface RecipeCommentRequestBody { | ||
comment: string; | ||
} | ||
|
||
const headers = { 'Content-Type': 'application/json' }; | ||
|
||
const postRecipeComment = (recipeId: number, body: RecipeCommentRequestBody) => { | ||
return recipeApi.post({ params: `/${recipeId}/comments`, credentials: true }, headers, body); | ||
}; | ||
|
||
const useRecipeCommentMutation = (recipeId: number) => { | ||
const queryClient = useQueryClient(); | ||
|
||
return useMutation({ | ||
mutationFn: (body: RecipeCommentRequestBody) => postRecipeComment(recipeId, body), | ||
onSuccess: () => queryClient.invalidateQueries({ queryKey: ['recipeComment', recipeId] }), | ||
}); | ||
}; | ||
|
||
export default useRecipeCommentMutation; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
짱👍