From 7fcb3a2487c36ce57869ad9d4d7892f72276a6fb Mon Sep 17 00:00:00 2001
From: Jungu Lee <100949102+jobkaeHenry@users.noreply.github.com>
Date: Sat, 18 Nov 2023 23:17:04 +0900
Subject: [PATCH] =?UTF-8?q?=EC=9D=B8=ED=84=B0=ED=8E=98=EC=9D=B4=EC=8A=A4-?=
=?UTF-8?q?=EB=B3=80=EA=B2=BD=EC=97=90=EB=94=B0=EB=A5=B8-=EC=88=98?=
=?UTF-8?q?=EC=A0=95=20(#40)?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
* Refactor : 토큰을 받아오는 로직 함수로 분리
* Refactor : 네비게이션 바 컴포넌트 변경
* Refactor : 토큰을 받아오는 로직 함수로 분리
* Minor : 사용하지 않는 모듈 제거
* Fix : Suspense Query를 useQuery로 변경
* Refactor : Retry 를 하지않음
* Refactor : 유저명 클릭시 해당유저의 프로필로 이동
* Refactor : 인터페이스 변경
---
client/src/app/layout.tsx | 2 +-
client/src/app/page.tsx | 4 +-
client/src/app/search/page.tsx | 3 +-
.../components/Navigation/NavbarUserImage.tsx | 20 +++++
.../components/Navigation/NavigationBar.tsx | 82 ++++++++++++++++++
client/src/components/NavigationBar.tsx | 83 -------------------
client/src/components/post/PostCard.tsx | 49 +++++++----
.../queryClient/CustomQueryClientProvider.tsx | 1 +
client/src/const/clientPath.ts | 5 ++
client/src/queries/auth/useUserInfoQuery.tsx | 11 +--
.../src/queries/post/useDeletePostMutation.ts | 2 +-
.../queries/post/useGetPostDetailQuery.tsx | 4 +-
.../post/useGetPostListInfiniteQuery.tsx | 1 +
.../Components/Navigation/Navbar.stories.tsx | 2 +-
client/src/types/auth/myInfo.ts | 1 +
client/src/types/post/PostInterface.ts | 6 +-
client/src/utils/getTokenFromCookies.ts | 14 ++++
17 files changed, 179 insertions(+), 111 deletions(-)
create mode 100644 client/src/components/Navigation/NavbarUserImage.tsx
create mode 100644 client/src/components/Navigation/NavigationBar.tsx
delete mode 100644 client/src/components/NavigationBar.tsx
create mode 100644 client/src/utils/getTokenFromCookies.ts
diff --git a/client/src/app/layout.tsx b/client/src/app/layout.tsx
index f4d9abb..71e5e31 100644
--- a/client/src/app/layout.tsx
+++ b/client/src/app/layout.tsx
@@ -4,7 +4,7 @@ import { nameOfApp, oneLineMessage } from "@/const/brand";
import OverrideCSS from "@/const/overrideCSS";
import { Box, GlobalStyles } from "@mui/material";
import Pretendard from "~/assets/font/Pretendard";
-import NavigationBar from "~/components/NavigationBar";
+import NavigationBar from "~/components/Navigation/NavigationBar";
import "./globals.css";
import CustomQueryClientProvider from "@/components/queryClient/CustomQueryClientProvider";
import { ReactQueryDevtools } from "@tanstack/react-query-devtools";
diff --git a/client/src/app/page.tsx b/client/src/app/page.tsx
index 6aed334..7005b37 100644
--- a/client/src/app/page.tsx
+++ b/client/src/app/page.tsx
@@ -1,11 +1,11 @@
"use server";
import PostCardList from "@/components/post/PostCardList";
import { getPostListQueryFn } from "@/queries/post/useGetPostListInfiniteQuery";
+import getTokenFromCookies from "@/utils/getTokenFromCookies";
import { Container, Paper } from "@mui/material";
-import { cookies } from "next/headers";
export default async function Home() {
- const accessToken = cookies().get("accessToken")?.value;
+ const accessToken = await getTokenFromCookies()
const initialData = await getPostListQueryFn({
page: 0,
diff --git a/client/src/app/search/page.tsx b/client/src/app/search/page.tsx
index 03f6393..f77fa1e 100644
--- a/client/src/app/search/page.tsx
+++ b/client/src/app/search/page.tsx
@@ -1,6 +1,7 @@
"use server";
import SearchArea from "@/components/search/SearchArea";
import { getPostListQueryFn } from "@/queries/post/useGetPostListInfiniteQuery";
+import getTokenFromCookies from "@/utils/getTokenFromCookies";
import { Container } from "@mui/material";
import { cookies } from "next/headers";
@@ -9,7 +10,7 @@ const SearchPage = async ({
}: {
searchParams?: { [key: string]: string | undefined };
}) => {
- const accessToken = cookies().get("accessToken")?.value;
+ const accessToken = await getTokenFromCookies()
const initialData = await getPostListQueryFn({
searchKeyword: searchParams?.keyword,
headers: { Authorization: accessToken },
diff --git a/client/src/components/Navigation/NavbarUserImage.tsx b/client/src/components/Navigation/NavbarUserImage.tsx
new file mode 100644
index 0000000..2101e99
--- /dev/null
+++ b/client/src/components/Navigation/NavbarUserImage.tsx
@@ -0,0 +1,20 @@
+"use client";
+import { useUserInfoQuery } from "@/queries/auth/useUserInfoQuery";
+import UserAvatar from "../user/info/UserAvatar";
+import MyIcon from "~/assets/icons/MyIcon.svg";
+const NavbarUserImage = () => {
+ try {
+ const { data } = useUserInfoQuery();
+ return (
+
+ );
+ } catch (err) {
+ return ;
+ }
+};
+
+export default NavbarUserImage;
diff --git a/client/src/components/Navigation/NavigationBar.tsx b/client/src/components/Navigation/NavigationBar.tsx
new file mode 100644
index 0000000..4ca22fe
--- /dev/null
+++ b/client/src/components/Navigation/NavigationBar.tsx
@@ -0,0 +1,82 @@
+"use client";
+import { BottomNavigation, BottomNavigationAction, Paper } from "@mui/material";
+
+import HomeIcon from "~/assets/icons/HomeIcon.svg";
+import SearchIcon from "~/assets/icons/SearchIcon.svg";
+import PostIcon from "~/assets/icons/PostIcon.svg";
+import BeverageIcon from "~/assets/icons/BeverageIcon.svg";
+
+import HOME, { MY_PROFILE, NEW_POST, SEARCH, WIKI } from "@/const/clientPath";
+import Link from "next/link";
+import { usePathname } from "next/navigation";
+import NavbarUserImage from "@/components/Navigation/NavbarUserImage";
+import { useMemo } from "react";
+
+const NavigationBar = () => {
+ const path = usePathname();
+ const NavbarData = useMemo(
+ () => [
+ {
+ iconComponent: ,
+ label: "홈",
+ href: HOME,
+ },
+ {
+ iconComponent: ,
+ label: "검색",
+ href: SEARCH,
+ },
+ {
+ iconComponent: ,
+ href: NEW_POST,
+ },
+ {
+ iconComponent: ,
+ label: "술과사전",
+ href: WIKI,
+ },
+ {
+ iconComponent: ,
+ label: "내 정보",
+ href: MY_PROFILE,
+ },
+ ],
+ []
+ );
+ return (
+
+
+ {NavbarData.map(({ label, href, iconComponent, ...others }) => {
+ return (
+
+ );
+ })}
+
+
+ );
+};
+
+const WrapperStyle = {
+ position: "fixed",
+ bottom: 0,
+ left: 0,
+ right: 0,
+ borderRadius: 0,
+};
+const BtnStyle = {
+ borderRadius: "12px 12px 0 0",
+ border: "1px solid",
+ borderBottom: "none",
+ borderColor: "gray.secondary",
+ boxSizing: "border-box",
+};
+
+export default NavigationBar;
diff --git a/client/src/components/NavigationBar.tsx b/client/src/components/NavigationBar.tsx
deleted file mode 100644
index 4f0cf11..0000000
--- a/client/src/components/NavigationBar.tsx
+++ /dev/null
@@ -1,83 +0,0 @@
-"use client";
-import { BottomNavigation, BottomNavigationAction, Paper } from "@mui/material";
-
-import HomeIcon from "~/assets/icons/HomeIcon.svg";
-import SearchIcon from "~/assets/icons/SearchIcon.svg";
-import PostIcon from "~/assets/icons/PostIcon.svg";
-import BeverageIcon from "~/assets/icons/BeverageIcon.svg";
-import MyIcon from "~/assets/icons/MyIcon.svg";
-import HOME, { MY_PROFILE, NEW_POST, SEARCH, WIKI } from "@/const/clientPath";
-import Link from "next/link";
-import { usePathname } from "next/navigation";
-
-const NavbarData = [
- {
- iconComponent: ,
- label: "홈",
- href: HOME,
- },
- {
- iconComponent: ,
- label: "검색",
- href: SEARCH,
- },
- {
- iconComponent: ,
- href: NEW_POST,
- },
- {
- iconComponent: ,
- label: "술과사전",
- href: WIKI,
- },
- {
- iconComponent: ,
- label: "내 정보",
- //FIXME 실제 URL로 변경
- href: MY_PROFILE+"/1",
- },
-];
-
-const NavigationBar = () => {
- const path = usePathname();
- return (
-
-
- {NavbarData.map(({ label, href, iconComponent, ...others }) => {
- return (
-
- );
- })}
-
-
- );
-};
-
-export default NavigationBar;
diff --git a/client/src/components/post/PostCard.tsx b/client/src/components/post/PostCard.tsx
index b7b7e31..bc6f82f 100644
--- a/client/src/components/post/PostCard.tsx
+++ b/client/src/components/post/PostCard.tsx
@@ -25,13 +25,17 @@ import useUnLikePostMutation from "@/queries/post/useUnLikePostMutation";
import "../newpost/quill.mention.css";
import { sanitize } from "isomorphic-dompurify";
import UserAvatar from "../user/info/UserAvatar";
+import Link from "next/link";
+import { USER_PAGE } from "@/const/clientPath";
+import { useUserInfoQuery } from "@/queries/auth/useUserInfoQuery";
const PostCard = ({
postAttachUrls,
id,
+ createdBy,
nickname,
postContent,
- updateDt,
+ lastModifiedDate,
tagList,
postNo,
likeCount,
@@ -46,13 +50,21 @@ const PostCard = ({
const hasImage = useMemo(() => postAttachUrls.length !== 0, [postAttachUrls]);
const { mutate: likeHandler } = useLikePostMutation();
const { mutate: unLikeHandler } = useUnLikePostMutation();
+ const { data: currentUser } = useUserInfoQuery();
+
+ const isMyPost = useMemo(
+ () => currentUser?.userNo === createdBy,
+ [currentUser]
+ );
return (
-
+
+
+
{/* Header */}
{/* 타이틀 */}
- {nickname}
- {`@${id}`}
+
+ {nickname}
+
+
+ {`@${id}`}
+
- {dayjs(updateDt).format("MM.DD")}
+ {dayjs(lastModifiedDate).format("MM.DD")}
-
-
+
+ {isMyPost && }
+
{alcoholName && (
)}
@@ -110,7 +127,11 @@ const PostCard = ({
onClick={() => openPostDetailPage(id, String(postNo))}
image={postAttachUrls[0].attachUrl}
alt={`${id}의 포스트`}
- sx={{ borderRadius: 2, bgcolor: "background.default" }}
+ sx={{
+ borderRadius: 2,
+ bgcolor: "background.default",
+ cursor: "pointer",
+ }}
/>
)}
{/* CTA */}
diff --git a/client/src/components/queryClient/CustomQueryClientProvider.tsx b/client/src/components/queryClient/CustomQueryClientProvider.tsx
index 4843009..0c6faf7 100644
--- a/client/src/components/queryClient/CustomQueryClientProvider.tsx
+++ b/client/src/components/queryClient/CustomQueryClientProvider.tsx
@@ -13,6 +13,7 @@ export default function CustomQueryClientProvider({
new QueryClient({
defaultOptions: {
queries: {
+ retry:false,
staleTime: 60 * 1000,
},
},
diff --git a/client/src/const/clientPath.ts b/client/src/const/clientPath.ts
index 480b2bd..255107d 100644
--- a/client/src/const/clientPath.ts
+++ b/client/src/const/clientPath.ts
@@ -15,6 +15,11 @@ export const FORGOTPASSWORD = "/auth/forgot-password" as const;
* 로그인 했을 경우만 접근 가능한 마이페이지
*/
export const MY_PROFILE = "/user" as const;
+/**
+ * 유저의 PK를 입력받아 해당유저의 프로필 페이지로 이동하는 URL
+ */
+export const USER_PAGE = (pk:string|number)=>`/user/${pk}`
+
/**
* 술과사전 페이지 라우트
*/
diff --git a/client/src/queries/auth/useUserInfoQuery.tsx b/client/src/queries/auth/useUserInfoQuery.tsx
index 2122f49..9f74b1e 100644
--- a/client/src/queries/auth/useUserInfoQuery.tsx
+++ b/client/src/queries/auth/useUserInfoQuery.tsx
@@ -1,20 +1,21 @@
"use client";
import { MY_INFO } from "@/const/serverPath";
-import {axiosPrivate} from "@/libs/axios";
+import { axiosPrivate } from "@/libs/axios";
import { MyInfoInterface } from "@/types/auth/myInfo";
import { SigninRequirement } from "@/types/auth/signinRequirement";
-import { useSuspenseQuery } from "@tanstack/react-query";
+import getTokenFromLocalStorage from "@/utils/getTokenFromLocalStorage";
+import { useQuery } from "@tanstack/react-query";
export const useUserInfoQuery = () =>
- useSuspenseQuery({
+ useQuery({
queryKey: userInfoQueryKeys.all,
queryFn: getMyInfoByLocalStorage,
});
export const getMyInfoByLocalStorage = async () => {
- // const accessToken = localStorage.get("accessToken");
+ const accessToken = getTokenFromLocalStorage();
const { data } = await axiosPrivate.get(MY_INFO, {
- // headers: { Authorization: accessToken },
+ headers: { Authorization: accessToken },
});
return data;
};
diff --git a/client/src/queries/post/useDeletePostMutation.ts b/client/src/queries/post/useDeletePostMutation.ts
index 41bc682..b209aef 100644
--- a/client/src/queries/post/useDeletePostMutation.ts
+++ b/client/src/queries/post/useDeletePostMutation.ts
@@ -1,6 +1,6 @@
import { REMOVE_POST } from "@/const/serverPath";
import { axiosPrivate } from "@/libs/axios";
-import { useMutation, useQueryClient } from "@tanstack/react-query";
+import { useMutation } from "@tanstack/react-query";
import { useInvalidatePostList } from "./useGetPostListInfiniteQuery";
export const useDeletePostMutation = () => {
diff --git a/client/src/queries/post/useGetPostDetailQuery.tsx b/client/src/queries/post/useGetPostDetailQuery.tsx
index 9054e2f..2133e18 100644
--- a/client/src/queries/post/useGetPostDetailQuery.tsx
+++ b/client/src/queries/post/useGetPostDetailQuery.tsx
@@ -1,4 +1,4 @@
-import { useSuspenseQuery } from "@tanstack/react-query";
+import { useQuery } from "@tanstack/react-query";
import axios from "@/libs/axios";
import { PostInterface } from "@/types/post/PostInterface";
@@ -6,7 +6,7 @@ const useGetPostDetailQuery = (
postId: string,
options?: { initialData: PostInterface }
) => {
- return useSuspenseQuery({
+ return useQuery({
queryKey: ["post", postId],
queryFn: () => getPostDetailQueryFn(postId),
initialData: options?.initialData,
diff --git a/client/src/queries/post/useGetPostListInfiniteQuery.tsx b/client/src/queries/post/useGetPostListInfiniteQuery.tsx
index 38f554e..5667dbd 100644
--- a/client/src/queries/post/useGetPostListInfiniteQuery.tsx
+++ b/client/src/queries/post/useGetPostListInfiniteQuery.tsx
@@ -49,6 +49,7 @@ export interface GetPostListOptions {
page?: number;
size?: number;
searchKeyword?: string;
+ searchUserNos?:string;
}
/**
* 실제 서버에서 응답해주는 값
diff --git a/client/src/stories/Components/Navigation/Navbar.stories.tsx b/client/src/stories/Components/Navigation/Navbar.stories.tsx
index 562477d..6bbfd2e 100644
--- a/client/src/stories/Components/Navigation/Navbar.stories.tsx
+++ b/client/src/stories/Components/Navigation/Navbar.stories.tsx
@@ -1,4 +1,4 @@
-import NavigationBar from "@/components/NavigationBar";
+import NavigationBar from "@/components/Navigation/NavigationBar";
import { Meta, StoryObj } from "@storybook/react";
const meta = {
diff --git a/client/src/types/auth/myInfo.ts b/client/src/types/auth/myInfo.ts
index 59c1962..46f356a 100644
--- a/client/src/types/auth/myInfo.ts
+++ b/client/src/types/auth/myInfo.ts
@@ -1,5 +1,6 @@
export interface MyInfoInterface {
id: string;
+ userNo:string;
nickname: string;
profileImages: ProfileImagesType[];
introduction: string;
diff --git a/client/src/types/post/PostInterface.ts b/client/src/types/post/PostInterface.ts
index 6f492d4..52d33a0 100644
--- a/client/src/types/post/PostInterface.ts
+++ b/client/src/types/post/PostInterface.ts
@@ -9,8 +9,12 @@ export interface PostInterface {
/**
* 업데이트 된 날짜
*/
- updateDt?: string;
+ lastModifiedDate?: string;
/**
+ * 해당 게시글을 생성한유저PK
+ */
+ createdBy:string;
+ /**
* 수정 됬는지 여부
*/
edited?: boolean;
diff --git a/client/src/utils/getTokenFromCookies.ts b/client/src/utils/getTokenFromCookies.ts
new file mode 100644
index 0000000..9b0c662
--- /dev/null
+++ b/client/src/utils/getTokenFromCookies.ts
@@ -0,0 +1,14 @@
+import { cookies } from "next/headers";
+
+/**
+ * 쿠키에서 AccessToken을 불러오는 유틸함수
+ * 서버단에서만 동작
+ */
+export default async function getTokenFromCookies() {
+ "use server";
+ if (typeof window !== "undefined") {
+ return;
+ }
+ const accessToken = cookies().get("accessToken")?.value;
+ return accessToken;
+}