issue #37
All checks were successful
Test CI / build (push) Successful in 16s

- 기능 구현 1차 완료(동작 확인 필요)
This commit is contained in:
geonhee-min
2025-12-02 16:50:24 +09:00
parent eec883ac32
commit af3fa26f3b
24 changed files with 519 additions and 97 deletions

View File

@@ -0,0 +1,177 @@
import { Card, CardContent, CardHeader, CardFooter } from '@/components/ui/card';
import { LoginSchema } from '@/data/form';
import { Field, FieldError, FieldLabel } from '@/components/ui/field';
import { Button } from '@/components/ui/button';
import { Input } from '@/components/ui/input';
import { zodResolver } from '@hookform/resolvers/zod';
import { useCallback, useState } from 'react';
import { Controller, useForm } from 'react-hook-form';
import { useNavigate } from 'react-router-dom';
import { PageRouting } from '@/const/PageRouting';
import * as z from 'zod';
import { Separator } from '@/components/ui/separator';
import { Validator } from '@/util/Validator';
import { LoginRequest } from '@/data/request/account/LoginRequest';
import { AccountNetwork } from '@/network/AccountNetwork';
import { toast } from 'sonner';
import { useAuthStore } from '@/store/authStore';
export default function LoginPage() {
const [isLoading, setIsLoading] = useState<boolean>(false);
const { login } = useAuthStore();
const navigate = useNavigate();
const accountNetwork = new AccountNetwork();
const loginForm = useForm<z.infer<typeof LoginSchema>>({
resolver: zodResolver(LoginSchema),
defaultValues: {
id: "",
password: ""
}
});
const { id, password } = { id: loginForm.watch('id'), password: loginForm.watch('password') };
const moveToSignUpPage = useCallback(() => {
navigate(PageRouting["SIGN_UP"].path);
}, []);
const moveToResetPasswordPage = useCallback(() => {
navigate(PageRouting["RESET_PASSWORD"].path);
}, []);
const moveToHomePage = useCallback(() => {
navigate(PageRouting["HOME"].path);
}, []);
// TODO 33 로그인 기능 구현
const reqLogin = async () => {
if (isLoading) return;
const type = Validator.isEmail(id) ? 'email' : 'accountId';
const data: LoginRequest = new LoginRequest(type, id, password);
setIsLoading(true);
const loginPromise = accountNetwork.login(data);
toast.promise<{ message?: string }>(
() => new Promise(async (resolve, reject) => {
try {
loginPromise.then((res) => {
if (res.data.success) {
resolve({message: ''});
} else {
reject(res.data.message);
}
})
} catch (err) {
reject ("서버 에러 발생");
}
}),
{
loading: "로그인 중입니다.",
success: "로그인이 완료되었습니다.",
error: (err) => `${err}`
}
);
loginPromise
.then((res) => {
if (res.data.success) {
const data = {
accessToken: res.data.accessToken!,
refreshToken: res.data.refreshToken!
}
login({ ...data });
moveToHomePage();
}
})
.finally(() => setIsLoading(false));
}
const TextSeparator = ({ text }: { text: string }) => {
return (
<div className="w-full flex flex-row items-center justify-center">
<Separator className="flex-1" />
<span className="text-gray-500 px-3 text-sm text-muted-foregroud">{text}</span>
<Separator className="flex-1" />
</div>
)
}
return (
<div className="w-full h-full flex flex-col justify-center items-center">
<Card className="w-md pl-2 pr-2">
<CardHeader>
</CardHeader>
<CardContent>
<form id="form-login" className="w-full flex flex-col gap-2.5">
<Controller
name="id"
control={loginForm.control}
render={({ field, fieldState }) => (
<Field data-invalid={fieldState.invalid}>
<FieldLabel htmlFor="form-login-id"> </FieldLabel>
<Input
{...field}
type="text"
id="form-login-id"
aria-invalid={fieldState.invalid}
/>
<FieldError errors={[fieldState.error]} />
</Field>
)}
>
</Controller>
<Controller
name="password"
control={loginForm.control}
render={({ field, fieldState }) => (
<Field data-invalid={fieldState.invalid}>
<div className="w-full flex flex-row justify-between items-end">
<FieldLabel className="w-fit" htmlFor="form-login-password"></FieldLabel>
<Button
className="p-0 bg-transparent hover:bg-transparent h-fit w-fit text-xs text-gray-400 hover:text-gray-500 cursor-pointer"
onClick={moveToResetPasswordPage}
type="button"
>
?
</Button>
</div>
<Input
{...field}
type="password"
id="form-login-password"
aria-invalid={fieldState.invalid}
/>
<FieldError errors={[fieldState.error]} />
</Field>
)}
>
</Controller>
</form>
</CardContent>
<CardFooter
className="w-full flex flex-col items-center gap-5"
>
<Button
className="w-full bg-indigo-500 hover:bg-indigo-400"
type="button"
disabled={id.trim().length < 1 || password.trim().length < 1}
onClick={reqLogin}
>
</Button>
<TextSeparator text="또는" />
<Button
className="w-full text-violet-500 bg-white border border-violet-500 hover:bg-violet-500 hover:text-white"
type="button"
onClick={moveToSignUpPage}
>
</Button>
</CardFooter>
</Card>
</div>
)
}

View File

@@ -0,0 +1,277 @@
import { useCallback, useEffect, useState } from 'react';
import { Button } from '@/components/ui/button';
import { Input } from '@/components/ui/input';
import {
Card,
CardContent,
CardHeader,
CardFooter
} from '@/components/ui/card';
import { Field, FieldError, FieldGroup, FieldLabel } from '@/components/ui/field';
import { SignUpSchema } from '@/data/form';
import { Controller, useForm } from 'react-hook-form';
import * as z from 'zod';
import { zodResolver } from '@hookform/resolvers/zod';
import EmailVerificationModal from '@/ui/component/modal/EmailVerificationModal';
import { CheckDuplicationRequest, SignupRequest } from '@/data/request';
import { AccountNetwork } from '@/network/AccountNetwork';
import { toast } from 'sonner';
import { useNavigate } from 'react-router-dom';
import { PageRouting } from '@/const/PageRouting';
export default function SignUpPage() {
const [emailVerificationModalOpen, setEmailVerificationModalOpen] = useState<boolean>(false);
const [isCheckedEmailDuplication, setIsCheckedEmailDuplication] = useState<boolean>(false);
const [isCheckedAccountIdDuplication, setIsCheckedAccountIdDuplication] = useState<boolean>(false);
const [duplicationCheckedEmail, setDuplicationCheckedEmail] = useState<string>("");
const [duplicationCheckedAccountId, setDuplicationCheckedAccountId] = useState<string>("");
const accountNetwork = new AccountNetwork();
const navigate = useNavigate();
const signUpForm = useForm<z.infer<typeof SignUpSchema>>({
resolver: zodResolver(SignUpSchema),
defaultValues: {
accountId: "",
email: "",
password: "",
passwordConfirm: "",
name: "",
nickname: "",
}
});
const goToLogin = useCallback(() => {
navigate(PageRouting["LOGIN"].path);
}, [navigate]);
const checkDuplication = async (type: 'email' | 'accountId', value: string) => {
const data: CheckDuplicationRequest = new CheckDuplicationRequest(type, value);
return await accountNetwork.checkDuplication(data);
}
const signup = async () => {
const { email, accountId, name, nickname, password } = signUpForm.getValues();
const data: SignupRequest = new SignupRequest(accountId, email, name, nickname, password);
const signupPromise = accountNetwork.signup(data);
toast.promise(
signupPromise,
{
loading: "회원가입 진행 중입니다.",
success: (res) => {
if (!res.data.success) return "회원가입에 실패하였습니다.\n잠시 후 다시 시도해주십시오.";
return <SuccessToast onClose={goToLogin} />
},
error: "회원가입에 실패하였습니다.\n잠시 후 다시 시도해주십시오.",
}
);
}
const handleOnChangeAccountId = (e: React.ChangeEvent<HTMLInputElement>) => {
setIsCheckedAccountIdDuplication(
e.currentTarget.value === duplicationCheckedAccountId
);
}
const handleOnChangeEmail = (e: React.ChangeEvent<HTMLInputElement>) => {
setIsCheckedEmailDuplication(
e.currentTarget.value === duplicationCheckedEmail
);
}
const handleDuplicationCheckButtonClick = async (type: 'email' | 'accountId') => {
const value = signUpForm.getValues(type);
const duplicatedMessage = type === 'email' ? '사용할 수 없는 이메일입니다.' : '사용할 수 없는 아이디입니다.';
if (!value) return;
const isDuplicated = (await checkDuplication(type, value)).data.isDuplicated;
if (isDuplicated) {
signUpForm.setError(type, { message: duplicatedMessage });
} else {
signUpForm.clearErrors(type);
if (type === 'email') {
setIsCheckedEmailDuplication(true);
setDuplicationCheckedEmail(value);
} else {
setIsCheckedAccountIdDuplication(true);
setDuplicationCheckedAccountId(value);
}
}
}
const handleOnSignUpButtonClick = () => {
if (!isCheckedAccountIdDuplication) {
signUpForm.setError("accountId", { message: "아이디 중복 확인이 필요합니다."});
return;
}
if (!isCheckedEmailDuplication) {
signUpForm.setError("email", { message: "이메일 중복 확인이 필요합니다." });
return;
}
setEmailVerificationModalOpen(true);
}
return (
<div className="w-full h-full flex flex-col justify-center items-center">
<Card className="w-md pl-2 pr-2">
<CardHeader></CardHeader>
<CardContent>
<form id="form-signup">
<FieldGroup>
<Controller
name="accountId"
control={signUpForm.control}
render={({ field, fieldState }) => (
<Field data-invalid={fieldState.invalid}>
<FieldLabel htmlFor="form-signup-account-id"></FieldLabel>
<div id="accountId-group" className="w-full flex flex-row justify-between gap-2.5">
<Input
{...field}
id="form-signup-account-id"
aria-invalid={fieldState.invalid}
onInput={handleOnChangeAccountId}
/>
<Button
type="button"
onClick={() => handleDuplicationCheckButtonClick('accountId')}
className="bg-indigo-500 hover:bg-indigo-400"
>
</Button>
</div>
{ isCheckedAccountIdDuplication && <p className="text-green-500 text-sm font-normal"> </p> }
<FieldError errors={[fieldState.error]}/>
</Field>
)}
/>
<Controller
name="name"
control={signUpForm.control}
render={({ field, fieldState }) => (
<Field data-invalid={fieldState.invalid}>
<FieldLabel htmlFor="form-signup-name"></FieldLabel>
<Input
{...field}
id="form-signup-name"
aria-invalid={fieldState.invalid}
/>
<FieldError errors={[fieldState.error]} />
</Field>
)}
/>
<Controller
name="nickname"
control={signUpForm.control}
render={({ field, fieldState }) => (
<Field data-invalid={fieldState.invalid}>
<FieldLabel htmlFor="form-signup-nickname"></FieldLabel>
<Input
{...field}
id="form-signup-nickname"
aria-invalid={fieldState.invalid}
/>
<FieldError errors={[fieldState.error]} />
</Field>
)}
/>
<Controller
name="email"
control={signUpForm.control}
render={({ field, fieldState }) => (
<Field data-invalid={fieldState.invalid}>
<FieldLabel htmlFor="form-signup-email"></FieldLabel>
<div id="email-group" className="w-full flex flex-row justify-between gap-2.5">
<Input
{...field}
id="form-signup-email"
aria-invalid={fieldState.invalid}
placeholder="example@domain.com"
type="email"
onInput={handleOnChangeEmail}
/>
<Button
type="button"
onClick={() => handleDuplicationCheckButtonClick('email')}
className="bg-indigo-500 hover:bg-indigo-400"
>
</Button>
</div>
{ isCheckedEmailDuplication && <p className="text-green-500 text-sm font-normal"> </p> }
<FieldError errors={[fieldState.error]}/>
</Field>
)}
/>
<Controller
name="password"
control={signUpForm.control}
render={({ field, fieldState }) => (
<Field data-invalid={fieldState.invalid}>
<FieldLabel htmlFor="form-signup-password"></FieldLabel>
<Input
{...field}
id="form-signup-password"
aria-invalid={fieldState.invalid}
type="password"
/>
<FieldError errors={[fieldState.error]} />
</Field>
)}
/>
<Controller
name="passwordConfirm"
control={signUpForm.control}
render={({ field, fieldState }) => (
<Field data-invalid={fieldState.invalid}>
<FieldLabel htmlFor="form-signup-password-confirm"> </FieldLabel>
<Input
{...field}
id="form-signup-password-confirm"
aria-invalid={fieldState.invalid}
type="password"
/>
<FieldError errors={[fieldState.error]} />
</Field>
)}
/>
</FieldGroup>
</form>
</CardContent>
<CardFooter>
<EmailVerificationModal
trigger={
<Button type="button" onClick={handleOnSignUpButtonClick} className="0">
</Button>
}
email={duplicationCheckedEmail}
open={emailVerificationModalOpen} // ✅ 부모 상태 연결
setOpen={setEmailVerificationModalOpen} // ✅ 부모 상태 변경 함수 전달
onVerifySuccess={signup} // ✅ 인증 성공 시 signup 호출
/>
</CardFooter>
</Card>
</div>
);
}
function SuccessToast({ onClose }: { onClose: () => void }) {
useEffect(() => {
const timer = setTimeout(() => onClose(), 3000); // 3초 후 이동
return () => clearTimeout(timer);
}, [onClose]);
return (
<div className="w-full flex flex-row justify-between items-center">
!
<button onClick={onClose}> </button>
</div>
);
}