issue #33
Some checks failed
Test CI / build (push) Failing after 12s

- 로그인 화면 기능 로직 1차 구현 중
This commit is contained in:
2025-11-30 01:42:26 +09:00
parent 6943d3d6ab
commit de8381f094
25 changed files with 521 additions and 117 deletions

View File

@@ -1,5 +1,4 @@
import { useState } from 'react';
import { Label } from '@/components/ui/label';
import { useCallback, useEffect, useState } from 'react';
import { Button } from '@/components/ui/button';
import { Input } from '@/components/ui/input';
import {
@@ -8,21 +7,32 @@ import {
CardHeader,
CardFooter
} from '@/components/ui/card';
import { Field, FieldError, FieldGroup, FieldLabel, FieldLegend } from '@/components/ui/field';
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 '@/data/RoutingData';
export default function SignUpPage() {
const [isCheckedEmailDuplication, setIsCheckdEmailDupliation] = useState<boolean>(false);
const [isEmailVerificated, setIsEmailVerificated] = useState<boolean>(false);
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: "",
@@ -31,28 +41,81 @@ export default function SignUpPage() {
}
});
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>) => {
setIsCheckdEmailDupliation(
setIsCheckedEmailDuplication(
e.currentTarget.value === duplicationCheckedEmail
);
// setIsEmailVerificated(
// e.currentTarget.value === duplicationCheckedEmail
// );
}
const handleEmailDuplicationCheckButtonClick = () => {
console.log(signUpForm.getValues("email"));
setDuplicationCheckedEmail(signUpForm.getValues("email"));
setIsCheckdEmailDupliation(true);
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 handleOnSubmitSignUpForm = () => {
const handleOnSignUpButtonClick = () => {
if (!isCheckedAccountIdDuplication) {
signUpForm.setError("accountId", { message: "아이디 중복 확인이 필요합니다."});
return;
}
if (!isCheckedEmailDuplication) {
signUpForm.setError("email", { message: "이메일 중복 확인이 필요합니다." });
return;
}
// if (!isEmailVerificated) {
// signUpForm.setError("email", { message: "이메일 인증이 완료되지 않았습니다." });
// }
setEmailVerificationModalOpen(true);
}
return (
@@ -60,8 +123,34 @@ export default function SignUpPage() {
<Card className="w-md pl-2 pr-2">
<CardHeader></CardHeader>
<CardContent>
<form id="form-signup" onSubmit={signUpForm.handleSubmit(handleOnSubmitSignUpForm)}>
<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}
@@ -108,11 +197,14 @@ export default function SignUpPage() {
onInput={handleOnChangeEmail}
/>
<Button
onClick={handleEmailDuplicationCheckButtonClick}
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>
)}
@@ -155,16 +247,31 @@ export default function SignUpPage() {
<CardFooter>
<EmailVerificationModal
trigger={
<Button type="submit" form="form-signup">
<Button type="button" onClick={handleOnSignUpButtonClick} className="0">
</Button>
}
email={duplicationCheckedEmail}
handler={() => {}}
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>
);
}