-
Notifications
You must be signed in to change notification settings - Fork 5
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
✨ feat: 최신포스트 자동 업데이트 및 NEW태그 추가 #318
base: main
Are you sure you want to change the base?
Changes from all commits
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,93 @@ | ||
import { useEffect, useState } from "react"; | ||
|
||
import { RotateCw } from "lucide-react"; | ||
|
||
import { useUpdatePost } from "@/hooks/queries/useUpdatePost"; | ||
|
||
import { UpdatePostsApiResponse } from "@/types/post"; | ||
import { Post } from "@/types/post"; | ||
import { useQueryClient } from "@tanstack/react-query"; | ||
|
||
export default function LatestSectionTimer() { | ||
const queryClient = useQueryClient(); | ||
const update = useUpdatePost(); | ||
const [timer, setTimer] = useState<number>(0); | ||
|
||
const calculateTime = () => { | ||
const now = new Date(); | ||
const currentMinutes = now.getMinutes(); | ||
|
||
const targetMinutes = currentMinutes < 31 ? 31 : 1; | ||
let targetHours = now.getHours(); | ||
|
||
if (currentMinutes >= 31) { | ||
targetHours = (targetHours + 1) % 24; | ||
} | ||
|
||
const targetTime = new Date(now.getFullYear(), now.getMonth(), now.getDate(), targetHours, targetMinutes, 0); | ||
const remainingTime = Math.floor((targetTime.getTime() - now.getTime()) / 1000); | ||
|
||
return remainingTime; | ||
}; | ||
|
||
useEffect(() => { | ||
setTimer(calculateTime()); | ||
const interval = setInterval(() => { | ||
const time = calculateTime(); | ||
setTimer(time); | ||
if (time === 0) { | ||
handleUpdate(); | ||
setTimer(calculateTime()); | ||
} | ||
}, 1000); | ||
return () => clearInterval(interval); | ||
}, []); | ||
|
||
const handleUpdate = async () => { | ||
const oldPosts = queryClient.getQueryData<{ | ||
pageParams: number; | ||
pages: { hasMore: boolean; lastId: string; result: Post[] }[]; | ||
}>(["latest-posts"]); | ||
if (update.data && oldPosts) { | ||
oldPosts.pages.forEach((oldPost) => | ||
oldPost.result.forEach((post) => { | ||
post.isNew = false; | ||
}) | ||
); | ||
queryClient.setQueryData(["latest-posts"], { | ||
...oldPosts, | ||
pages: [ | ||
{ | ||
...oldPosts.pages[0], | ||
result: [...update.data.data, ...oldPosts.pages[0].result], | ||
}, | ||
], | ||
}); | ||
} | ||
}; | ||
Comment on lines
+46
to
+67
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. P4: 51~56번 라인에서는 React Query를 통해 가져온 React Query의 데이터는 기본적으로 불변성을 지키도록 권장된다고 합니다. 불변성을 지키지 않는다면 처음에는 마치 정상적으로 동작하는 듯 해 보여도, 참조가 동일한곳에 머물러 있기 때문에 더이상 리액트 쿼리가 옵저버들에게 변경사항을 알릴 수 없다고 합니다. - 참고 그리하여 기존 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. 해당 코드를 작성할때 불변성에 대해서는 크게 고민을 안했던거 같습니다. getQuery로 데이터를 불러오고 반환값은 새로운 배열을 만들어서 반환하기 때문에 크게 의미가 있을까 하는 생각에 forEach를 이용해서 isNew를 false로 바꿨었는데 지금보니 forEach보다는 map을 사용해서 새로운 배열을 만들고 데이터를 바꿔주는게 더 괜찮아 보이네요 찾아주셔서 감사합니다! |
||
|
||
return <Timer time={timer} handleUpdate={handleUpdate} update={update} />; | ||
} | ||
|
||
function Timer({ | ||
time, | ||
handleUpdate, | ||
update, | ||
}: { | ||
time: number; | ||
handleUpdate: () => void; | ||
update: { data?: UpdatePostsApiResponse; isLoading: boolean; error: Error | null }; | ||
}) { | ||
const formatTime = (time: number) => { | ||
const minutes = Math.floor(time / 60); | ||
const seconds = time % 60; | ||
return `${minutes.toString().padStart(2, "0")}:${seconds.toString().padStart(2, "0")} 초`; | ||
}; | ||
|
||
return ( | ||
<span className="text-sm text-gray-500 bg-gray-50 p-2 rounded-lg mr-5"> | ||
{update.error && <button onClick={handleUpdate}>reload</button>} | ||
{update.isLoading ? <RotateCw /> : <span>{formatTime(time)} 후 업데이트</span>} | ||
</span> | ||
); | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
import { posts } from "@/api/services/posts"; | ||
import { useQuery } from "@tanstack/react-query"; | ||
|
||
export const useUpdatePost = () => { | ||
const { data, isLoading, error } = useQuery({ queryKey: ["update-posts"], queryFn: posts.update, retry: 1 }); | ||
return { data, isLoading, error }; | ||
}; | ||
Comment on lines
+4
to
+7
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. P5: 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. 리액트 쿼리에서 API요청에 대한 데이터 식별을 위한 변수랑 비슷한 역할로 쿼리키를 사용합니다. 다른 코드에서 getQuery를 사용하거나 데이터 캐싱에 사용됩니다! |
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.
P4: 전체적으로 axios동작을 관리하는 부분의 코드들에 API 자원 경로 상수들이 문자열 하드코딩 되어있는데, 이 부분을 별도 상수로 분리해 보는게 문자열에 덜 의존적일 것 같습니다!
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.
따로 상수처리해서 하나의 파일에서 관리하는게 효율적이게 보이긴 하네요! 준혁님이랑 이야기해보고 수정해보겠습니다.