From b6390a72ae35280dd2cc1c8f675e700ddea8774c Mon Sep 17 00:00:00 2001 From: Jungmin Date: Tue, 29 Oct 2024 01:54:49 +0900 Subject: [PATCH] =?UTF-8?q?=ED=9A=8C=EC=9B=90=EA=B0=80=EC=9E=85=20?= =?UTF-8?q?=ED=8F=BC=20=EB=A6=AC=ED=8C=A9=ED=86=A0=EB=A7=81=20=EB=B0=8F=20?= =?UTF-8?q?=EC=9C=A0=ED=9A=A8=EC=84=B1=20=EA=B2=80=EC=82=AC=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - 회원가입 폼에서 서비스 코드 분리 - tanstack query(react-query)를 사용하여 회원가입 로직 통합 - 코드 전반에 걸쳐 가독성 및 유지 보수성 향상 --- src/api/authAPI.ts | 10 +++ src/app/(auth)/components/SignupForm.tsx | 69 ++++++++----------- .../(auth)/components/SignupFormContainer.tsx | 4 +- src/hook/useSignup.tsx | 22 ++++++ 4 files changed, 62 insertions(+), 43 deletions(-) create mode 100644 src/api/authAPI.ts create mode 100644 src/hook/useSignup.tsx diff --git a/src/api/authAPI.ts b/src/api/authAPI.ts new file mode 100644 index 0000000..2ad7185 --- /dev/null +++ b/src/api/authAPI.ts @@ -0,0 +1,10 @@ +import api from "@/lib/api"; + +export const signup = async (data: { email: string; nickname: string; password: string }) => { + const response = await api.post("/user", { + email: data.email, + name: data.nickname, + password: data.password, + }); + return response.data; +}; diff --git a/src/app/(auth)/components/SignupForm.tsx b/src/app/(auth)/components/SignupForm.tsx index 4c8b946..5c6900d 100644 --- a/src/app/(auth)/components/SignupForm.tsx +++ b/src/app/(auth)/components/SignupForm.tsx @@ -1,13 +1,13 @@ "use client"; -import { toast } from "react-toastify"; import { useEffect } from "react"; -import { useRouter } from "next/navigation"; import axios from "axios"; import { z } from "zod"; import { useForm } from "react-hook-form"; import { zodResolver } from "@hookform/resolvers/zod"; +import { useSignup } from "@/hook/useSignup"; + import SubmitButton from "./SubmitButton"; import InputField from "./InputField"; @@ -16,67 +16,54 @@ type FormFields = "nickname" | "email" | "password" | "passwordConfirm"; type FormData = z.infer; export default function SignupForm() { - const router = useRouter(); - const { register, handleSubmit, formState: { errors, isSubmitting }, trigger, setError, - clearErrors, - reset, watch, } = useForm({ resolver: zodResolver(signupSchema), mode: "onChange", // 실시간 유효성 검사를 위해 추가 }); - useEffect(() => { - const firstInput = document.getElementById("nickname") as HTMLInputElement; - if (firstInput) { - firstInput.focus(); - } - }, []); + const signupMutation = useSignup(); const onSubmit = async (data: FormData) => { - try { - const response = await axios.post(`${process.env.NEXT_PUBLIC_API_URL}/user`, { - email: data.email, - name: data.nickname, - password: data.password, - }); - - if (response.data) { - clearErrors(); - reset(); - toast.success("회원가입이 완료되었습니다."); - router.push("/login"); - } - } catch (error) { - console.error("회원가입 서버 오류🚨", error); + signupMutation.mutate(data, { + onError: (error) => { + console.error("회원가입 서버 오류🚨", error); - if (axios.isAxiosError(error) && error.response) { - const errorCode = error.response.status; - const errorInfo = errorMessage[errorCode]; + if (axios.isAxiosError(error) && error.response) { + const errorCode = error.response.status; + const errorInfo = errorMessage[errorCode]; - if (errorCode) { - setError(errorInfo.field, { - type: "manual", - message: errorInfo.message, - }); + if (errorInfo) { + setError(errorInfo.field, { + type: "manual", + message: errorInfo.message, + }); + } } - } - } + }, + }); }; + useEffect(() => { + const firstInput = document.getElementById("nickname") as HTMLInputElement; + if (firstInput) { + firstInput.focus(); + } + }, []); + return (
브랜드
-

+

회원가입 -

+
diff --git a/src/hook/useSignup.tsx b/src/hook/useSignup.tsx new file mode 100644 index 0000000..0c85f9e --- /dev/null +++ b/src/hook/useSignup.tsx @@ -0,0 +1,22 @@ +import { useRouter } from "next/navigation"; +import { toast } from "react-toastify"; +import { useMutation } from "@tanstack/react-query"; + +import { signup } from "@/api/authAPI"; +import { ErrorType } from "@/api/goalAPI"; + +export const useSignup = () => { + const router = useRouter(); + console.log("1"); + + return useMutation({ + mutationFn: signup, + onSuccess: () => { + toast.success("회원가입이 완료되었습니다."); + router.push("/login"); + }, + onError: (error: ErrorType) => { + throw error; + }, + }); +};