-
Notifications
You must be signed in to change notification settings - Fork 2
Expand file tree
/
Copy pathSignUp.tsx
More file actions
123 lines (108 loc) · 4.29 KB
/
SignUp.tsx
File metadata and controls
123 lines (108 loc) · 4.29 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
import { useState } from 'react';
import { Link, useLocation, useNavigate } from 'react-router-dom';
import { IcCheckboxChecked, IcCheckboxDefault, IcEssentialDot } from '@/assets/svg';
import { BasicInfoSection, SurveySection } from '@/page/signup';
import { SignUpButton } from '@/page/signup/component';
import * as styles from '@/page/signup/SignUp.css';
import { useSignUpForm } from '@/page/signup/hook/useSignUpForm';
import { PATH } from '@/route';
import { usePostSignUp } from '@/api/domain/signup/hook/usePostSignup';
import type { SignupRequest } from '@/api/domain/signup/type/SignupRequest';
const SIGNUP_MESSAGE = '회원가입 후 NiNE DOT를 만나보세요!';
const FIT_INFO_MESSAGE = '내 성향을 선택하고 맞춤형 목표 추천을 받아보세요';
const PERSONAL_INFO_AGREEMENT = '(필수) 개인정보 수집 및 이용약관 동의';
const SignUp = () => {
const navigate = useNavigate();
const location = useLocation();
const userData = location.state?.userData;
const [answers, setAnswers] = useState<Record<number, number>>({});
const { formState, actions, computed } = useSignUpForm(userData);
const { name, email, birth, selectedJob, inputJob, isChecked } = formState;
const { setName, setEmail, setBirth, setSelectedJob, setInputJob, setIsChecked } = actions;
const { finalJob, isValid } = computed;
const CheckIcon = isChecked ? IcCheckboxChecked : IcCheckboxDefault;
const { mutate: signUp } = usePostSignUp();
const handleSignUp = () => {
const payload: SignupRequest = {
socialProvider: userData.socialProvider,
socialToken: userData.socialToken,
name,
email,
birthday: birth,
job: finalJob,
profileImageUrl: userData.profileImageUrl,
answers: Object.entries(answers).map(([questionId, choiceId]) => ({
questionId: Number(questionId),
choiceId,
})),
};
signUp(payload, {
onSuccess: (data) => {
if (data.accessToken) {
localStorage.setItem('accessToken', data.accessToken);
}
navigate(PATH.INTRO, { state: { pageState: 'MANDALART' } });
},
onError: () => {
console.error('회원가입 실패');
},
});
};
return (
<main className={styles.mainContainer}>
<div className={styles.layoutContainer}>
<header className={styles.headerContainer}>
<h1 className={styles.headerText}>회원가입</h1>
<p className={styles.descriptionText}>{SIGNUP_MESSAGE}</p>
</header>
<section>
<div className={styles.basicInfoContainer}>
<h2 className={styles.infoText}>기본 정보</h2>
<span>
<IcEssentialDot className={styles.essentialIcon} />
<span className={styles.essentialText}>필수 입력 항목</span>
</span>
</div>
<div className={styles.basicInfoSection}>
<BasicInfoSection
name={name}
email={email}
birth={birth}
setName={setName}
setEmail={setEmail}
setBirth={setBirth}
selectedJob={selectedJob}
setSelectedJob={setSelectedJob}
inputJob={inputJob}
setInputJob={setInputJob}
/>
</div>
</section>
<section>
<div className={styles.fitInfoContainer}>
<span className={styles.infoText}>맞춤 정보</span>
<p className={styles.fitInfoText}>{FIT_INFO_MESSAGE}</p>
</div>
<div className={styles.surveySection}>
<SurveySection answers={answers} setAnswers={setAnswers} />
</div>
</section>
<div className={styles.agreementContainer}>
<button onClick={() => setIsChecked(!isChecked)}>
<CheckIcon className={styles.checkboxIcon} />
</button>
<p className={styles.agreeText}>{PERSONAL_INFO_AGREEMENT}</p>
<Link to={import.meta.env.VITE_TOS_LINK} target="_blank" className={styles.seeText}>
보기
</Link>
</div>
<div className={styles.buttonContainer}>
<SignUpButton onClick={handleSignUp} disabled={!isValid || !userData}>
가입하기
</SignUpButton>
</div>
</div>
</main>
);
};
export default SignUp;