diff --git a/src/api/nickname.ts b/src/api/nickname.ts new file mode 100644 index 0000000..179cc42 --- /dev/null +++ b/src/api/nickname.ts @@ -0,0 +1,19 @@ +import axiosInstance from "./axiosInstance" + +export interface NickNameModificationResData { + data: { + id: 0 + nickname: "string" + } +} + +export const modifyNickName = async (uid: number, newNickName: string): Promise => { + try { + const res = await axiosInstance.put(`/users/${uid}/nickname`, { + nickname: newNickName, + }) + return res.data + } catch (e) { + throw e + } +} diff --git a/src/assets/icons/arrow-small-right.svg b/src/assets/icons/arrow-small-right.svg new file mode 100644 index 0000000..bb2df7c --- /dev/null +++ b/src/assets/icons/arrow-small-right.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/components/Modal/Modals.tsx b/src/components/Modal/Modals.tsx index 0719b29..030bc95 100644 --- a/src/components/Modal/Modals.tsx +++ b/src/components/Modal/Modals.tsx @@ -8,6 +8,7 @@ import JoinCrewModal from "./JoinCrewModal" import ReportModal from "./ReportModal" import ToWithdrawModal from "./ToWithdrawModal" import WithdrawCrewModal from "./WithdrawCrewModal" +import NickNameModal from "./NickNameModal" export const modals = { createCrewModal: CreateCrewModal, @@ -17,6 +18,7 @@ export const modals = { ToWithdrawModal: ToWithdrawModal, postureGuideModal: GoodPostureGuidePopupModal, reportModal: ReportModal, + nickNameModal: NickNameModal, } const Modals = (): React.ReactNode => { @@ -45,9 +47,9 @@ const Modals = (): React.ReactNode => { close(Component) } - const handleSubmit = async (): Promise => { + const handleSubmit = async (value?: any): Promise => { if (typeof onSubmit === "function") { - await onSubmit() + await onSubmit(value) } handleClose() } diff --git a/src/components/Modal/NickNameModal.tsx b/src/components/Modal/NickNameModal.tsx new file mode 100644 index 0000000..a7a0813 --- /dev/null +++ b/src/components/Modal/NickNameModal.tsx @@ -0,0 +1,55 @@ +import { ModalProps } from "@/contexts/ModalsContext" +import ModalContainer from "@components/ModalContainer" +import { useState } from "react" + +const NickNameModal = (props: ModalProps & { id: string }): React.ReactElement => { + const { onClose, onSubmit } = props + const [nickName, setNickName] = useState("") + + const onClickModifyNickNameButton = () => { + if (nickName && nickName.length <= 200) { + onSubmit?.(nickName) + } + } + + return ( + +
+ {/* header */} +
+
닉네임
+
+ +
+ 200 + ? "border-red-500 focus:border-red-500 focus:ring-red-200" + : "focus:border-blue-500 focus:ring-blue-200" + }`} + placeholder="닉네임을 입력해 주세요." + onChange={(e) => setNickName(e.target.value)} + /> +
+ {nickName.length > 200 && ( + 닉네임은 200자를 초과할 수 없어요. + )} +
+
+ + {/* button */} +
+ +
+
+
+ ) +} + +export default NickNameModal diff --git a/src/components/SideNav.tsx b/src/components/SideNav.tsx index 0d3e52b..59ec856 100644 --- a/src/components/SideNav.tsx +++ b/src/components/SideNav.tsx @@ -5,10 +5,12 @@ import LogoImage from "@assets/icons/side-nav-logo.svg?react" import AnalysisIcon from "@assets/icons/side-nav-analysis-icon.svg?react" import CrewIcon from "@assets/icons/side-nav-crew-icon.svg?react" import MonitoringIcon from "@assets/icons/side-nav-monitor-icon.svg?react" +import RightSmallArrow from "@assets/icons/arrow-small-right.svg?react" import { useMemo } from "react" import { Link, useLocation, useNavigate } from "react-router-dom" import { useModals } from "@/hooks/useModals" import { modals } from "./Modal/Modals" +import RoutePath from "@/constants/routes.json" const navItems = [ { @@ -86,12 +88,14 @@ export default function SideNav(): React.ReactElement { {/* User Info */} -
-
바른자세 똑딱똑딱
-
- {nickname} - -
+
+
바른자세 똑딱똑딱
+ + + {nickname} + + +
{/* Navigation Links */} @@ -100,7 +104,7 @@ export default function SideNav(): React.ReactElement { {navItems.map(({ icon: Icon, label, link }) => { const isActive = location.pathname.includes(link) return ( -
  • +
  • {label} @@ -126,7 +130,7 @@ export default function SideNav(): React.ReactElement {
  • {/* Footer Text */} -
    +
    © 2024 주인공.
    All rights reserved. diff --git a/src/constants/routes.json b/src/constants/routes.json index 805e521..5d7834e 100644 --- a/src/constants/routes.json +++ b/src/constants/routes.json @@ -6,5 +6,6 @@ "CREW": "/crew", "MYCREW": "/crew/my", "HOME": "/home", - "SOCKET": "/socket" + "SOCKET": "/socket", + "MYPAGE": "/mypage" } diff --git a/src/contexts/ModalsContext.ts b/src/contexts/ModalsContext.ts index 13e7df8..550944d 100644 --- a/src/contexts/ModalsContext.ts +++ b/src/contexts/ModalsContext.ts @@ -3,7 +3,7 @@ import { ComponentType, createContext } from "react" export type ModalProps = { id?: number onClose?: () => void - onSubmit?: () => void + onSubmit?: (value?: string) => void } export type ModalComponent = ComponentType diff --git a/src/pages/MyPage.tsx b/src/pages/MyPage.tsx new file mode 100644 index 0000000..809a3f3 --- /dev/null +++ b/src/pages/MyPage.tsx @@ -0,0 +1,48 @@ +import { modifyNickName } from "@/api/nickname" +import { modals } from "@/components/Modal/Modals" +import { useModals } from "@/hooks/useModals" +import { useAuthStore } from "@/store" + +export default function MyPage() { + const { user, setNickName } = useAuthStore() + + const { openModal } = useModals() + + const onClickModifyNickName = () => { + openModal(modals.nickNameModal, { + onSubmit: (newNickName) => { + if (user && newNickName) { + modifyNickName(user.uid, newNickName).then(({ data }) => { + setNickName(data.nickname) + }) + } + }, + }) + } + + return ( +
    +

    마이 페이지

    +
    +
    +
    +
    닉네임
    +
    {user?.nickname}
    +
    +
    + +
    +
    +
    +
    로그인 수단
    +
    카카오 계정
    +
    +
    +
    + ) +} diff --git a/src/routes/Router.tsx b/src/routes/Router.tsx index 1115e49..9392a6e 100644 --- a/src/routes/Router.tsx +++ b/src/routes/Router.tsx @@ -5,6 +5,7 @@ import BaseLayout from "@/layouts/BaseLayout" import MonitoringLayout from "@/layouts/MonitoringLayout" import { AnalysisDashboard, AuthPage, Crew, HomePage, MonitoringPage } from "@/pages" import MyCrew from "@/pages/MyCrew" +import MyPage from "@/pages/MyPage" import AuthRoute from "@/routes/AuthRoute" import { useAuthStore } from "@/store/AuthStore" import React from "react" @@ -23,6 +24,7 @@ const Router: React.FC = () => { }> }> + } /> }> } /> diff --git a/src/store/AuthStore.ts b/src/store/AuthStore.ts index eb55966..4217ab7 100644 --- a/src/store/AuthStore.ts +++ b/src/store/AuthStore.ts @@ -1,11 +1,16 @@ import { create } from "zustand" import { persist } from "zustand/middleware" +interface UserInfo { + uid: number + nickname: string +} interface AuthState { isAuthenticated: boolean - user: any + user: UserInfo | null accessToken: string - setUser: (user: any, accessToken: string) => void + setUser: (user: UserInfo, accessToken: string) => void + setNickName: (nickname: string) => void logout: (callback: () => void) => void } @@ -16,12 +21,16 @@ export const useAuthStore = create( isAuthenticated: false, user: null, accessToken: "", - setUser: (user: any, accessToken: string) => + setUser: (user: UserInfo, accessToken: string) => set({ user, isAuthenticated: true, accessToken, }), + setNickName: (nickname: string) => + set((state) => ({ + user: state.user ? { ...state.user, nickname } : null, + })), logout: (callback) => { set({ user: null,