Signup.js 12.5 KB
Newer Older
한규민's avatar
한규민 committed
1
import styles from "./signup.module.scss";
한규민's avatar
한규민 committed
2
import { useState } from "react";
한규민's avatar
한규민 committed
3
import authApi from "../../apis/auth.api.js";
한규민's avatar
한규민 committed
4
import { Redirect } from "react-router-dom";
한규민's avatar
한규민 committed
5
import catchErrors from "../../utils/catchErrors.js";
한규민's avatar
한규민 committed
6
7

const Signup = () => {
한규민's avatar
한규민 committed
8
    const [user, setUser] = useState({
한규민's avatar
한규민 committed
9
        userId: "",
10
        userName:"",
한규민's avatar
한규민 committed
11
        userEmail: "",
한규민's avatar
한규민 committed
12
13
14
15
16
        userNickName: "",
        userBirthday: "",
        userMbnum: "",
        userPassword: "",
        userRePassword: ""
17
    })
18
    const [startTime, setStartTime] = useState("");
한규민's avatar
한규민 committed
19
    const [number, setNumber] = useState(null);
한규민's avatar
한규민 committed
20
21
    const [loading, setLoading] = useState(false);
    const [success, setSuccess] = useState(false);
한규민's avatar
한규민 committed
22
    //각 타입별 error 유무 state
한규민's avatar
한규민 committed
23
    const [mbError,setMbError] = useState(false);
한규민's avatar
한규민 committed
24
    const [error, setError] = useState("");
한규민's avatar
한규민 committed
25
    const [errorMsg, setErrorMsg] = useState({
한규민's avatar
한규민 committed
26
        errorId: null,
27
        errorName: false,
한규민's avatar
한규민 committed
28
        errorEmail: false,
한규민's avatar
한규민 committed
29
        errorNickName: false,
한규민's avatar
한규민 committed
30
31
32
33
        errorBirthday: false,
        errorMbnum: false,
        errorPassword: false,
    })
한규민's avatar
한규민 committed
34
    // id중복확인 여부 state와 가입하기 누르면 id 임시 저장
한규민's avatar
한규민 committed
35
    const [overlapId, setOverlapId] = useState(false);
한규민's avatar
한규민 committed
36
    const [confirmMb, setConfirmMb] = useState(false);
한규민's avatar
한규민 committed
37
38
39
    const [preId, setPreId] = useState("");

    //입력할때마다 state에 저장
40
    const handleUserOnChange = (e) => {
한규민's avatar
한규민 committed
41
42
        setUser({
            ...user,
43
44
            [e.target.name]: e.target.value
        })
한규민's avatar
한규민 committed
45
        if (e.target.name === "userBirthday" || e.target.name === "userMbum") {
한규민's avatar
한규민 committed
46
47
48
49
50
            setUser({
                ...user,
                [e.target.name]: String(e.target.value)
            })
        }
한규민's avatar
한규민 committed
51
    }
52

한규민's avatar
한규민 committed
53
    //id(중복확인 체크, 형식 에러)
한규민's avatar
한규민 committed
54
55
    const handleOnClickId = async (e) => {
        e.preventDefault();
한규민's avatar
한규민 committed
56
        try {
한규민's avatar
한규민 committed
57
58
59
60
61
62
63
64
65
66
67
            setError("");
            if (user.userId.length < 5) {
                setErrorMsg(errorMsg => ({
                    ...errorMsg,
                    [e.target.name]: true
                }));
                if (overlapId === true) {
                    setOverlapId(() => (false));
                };
            } else {
                const userId = user.userId;
한규민's avatar
한규민 committed
68
69
                const overlapId = await authApi.compareId(userId);
                if (!overlapId) {
한규민's avatar
한규민 committed
70
71
72
73
74
                    alert("이 아이디는 사용가능합니다.")
                    setErrorMsg(errorMsg => ({
                        ...errorMsg,
                        [e.target.name]: false
                    }));
한규민's avatar
한규민 committed
75
76
                    setOverlapId(() => (true));
                } else {
한규민's avatar
한규민 committed
77
                    alert("이미 사용중인 아이디입니다.")
한규민's avatar
한규민 committed
78
                    setOverlapId(() => (false));
한규민's avatar
한규민 committed
79
80
                }
            }
한규민's avatar
한규민 committed
81
82
83
        } catch (error) {
            catchErrors(error, setError)
        } finally {
한규민's avatar
한규민 committed
84
            setLoading(false);
한규민's avatar
한규민 committed
85
        }
한규민's avatar
한규민 committed
86
87
88
    }

    const handleOnClickMbnum = async (e) => {
한규민's avatar
한규민 committed
89
        e.preventDefault();
한규민's avatar
한규민 committed
90
        try {
91
            setStartTime("");
한규민's avatar
한규민 committed
92
93
94
95
96
97
            setError("");
            setLoading(true)
            const phone = user.userMbnum;
            const message = await authApi.confirmMbnum(phone);
            if(message.isSuccess){
            setMbError("보냄");
98
            setStartTime(message.startTime);
한규민's avatar
한규민 committed
99
            }
한규민's avatar
한규민 committed
100
        } catch (error) {
한규민's avatar
한규민 committed
101
102
103
104
105
106
107
108
109
110
111
112
113
114
            console.log('error'+ error)
        }finally {
            setLoading(false);
        }
    }

    const handleOnChangeMb = (e) => {
        setNumber(String(e.target.value));
    }

    const handleOnClickMbConfirm = async (e) => {
        e.preventDefault();
        try {
            setError("");
115
116
            setLoading(true);
            const confirmNum = {userMbnum : user.userMbnum, number : number, startTime : startTime};
한규민's avatar
한규민 committed
117
118
119
120
121
122
123
124
125
126
            const message = await authApi.confirmNum(confirmNum);
            setMbError(message);
            if(message === "성공"){
                setConfirmMb(true);
                console.log("인증완료");
            }
        } catch (error) {
            catchErrors(error, setError);
        }finally {
            setLoading(false);
한규민's avatar
한규민 committed
127
        }
한규민's avatar
한규민 committed
128
    }
한규민's avatar
한규민 committed
129
130
131
132
133
    const validationPw = () => {
        if(user.userPassword !== user.userRePassword){
            return false;
        }else{return true;}
    }
134

한규민's avatar
한규민 committed
135
136
137
    const handleOnSummit = async (e) => {
        e.preventDefault();
        try {
한규민's avatar
한규민 committed
138
            setError(() => (""));
한규민's avatar
한규민 committed
139
            //처리가 될때까지 버튼(가입하기)이 안눌리게 지정
한규민's avatar
한규민 committed
140
            setLoading(() => (true));
한규민's avatar
한규민 committed
141
142
143
144
145
146
147
148
149
150
151
152
            let validPw = validationPw();
            if(validPw){
                const userData = user;
                //서버로 전송
                const error = await authApi.signup(userData);
                setErrorMsg(error);
                if(error === "성공"){
                    setSuccess(true);
                }
            }else{
                throw new Error("비밀번호가 일치하지 않습니다.");
            }
한규민's avatar
한규민 committed
153
154
155
156
157
        } catch (error) {
            //에러전송
            catchErrors(error, setError);
        } finally {
            setLoading(false);
한규민's avatar
한규민 committed
158
        }
한규민's avatar
signup    
한규민 committed
159
    }
한규민's avatar
한규민 committed
160

한규민's avatar
한규민 committed
161
    if(success){
한규민's avatar
한규민 committed
162
163
        return <Redirect to="/login" />;
    }
한규민's avatar
한규민 committed
164

한규민's avatar
한규민 committed
165
    return (
한규민's avatar
한규민 committed
166
        // 데이터 입력과 유효성 검사 후 보이는 경고창
한규민's avatar
한규민 committed
167
        <form className={`d-flex col-md-6 col-12 justify-content-center`} onSubmit={handleOnSummit}>
한규민's avatar
한규민 committed
168
            <div className="d-flex flex-column">
한규민's avatar
signup    
한규민 committed
169
170
171
172
173
                <span className={styles.title}>회원가입</span>
                <div className="d-flex flex-column">
                    <div className={styles.inputContent}>
                        <label className={styles.signupLabel}>아이디</label>
                        <div className="d-flex col-md-auto">
한규민's avatar
한규민 committed
174
                            <input className={styles.input} type="text" name="userId" placeholder="5~10자리 사이" onChange={handleUserOnChange} maxLength="20" required />
한규민's avatar
한규민 committed
175
                            <button type="button" disabled={loading} name="errorId" className={`rounded-2 mt-2 ${styles.butterYellowAndBtn} ${styles.btnHover}`} onClick={handleOnClickId}>중복확인</button>
한규민's avatar
signup    
한규민 committed
176
                        </div>
한규민's avatar
한규민 committed
177
                    </div>
한규민's avatar
한규민 committed
178
179
180
                    {(overlapId === false) && errorMsg.errorId && <p className={styles.passwordConfirmError}>5~10자리 사이로 입력해주세요.</p>}
                    {overlapId && (errorMsg.errorId === false) && <p className={styles.passwordConfirmError}>아이디 중복이 확인되었습니다.</p>}
                    {(errorMsg.errorId === false) && (overlapId === false) && <p className={styles.passwordConfirmError}>아이디 중복확인을 해주세요.</p>}
181
                </div>
182
183
184
                <div className="d-flex flex-column">
                    <div className={styles.inputContent}>
                        <label className={styles.signupLabel}>이름</label>
한규민's avatar
한규민 committed
185
                        <input className={`${styles.input} ${styles.inputSize}`} type="text" name="userName" placeholder="이름을 입력해주세요" onChange={handleUserOnChange}  maxLength="20" required />
186
187
188
                    </div>
                    {errorMsg.errorName && <p className={styles.passwordConfirmError}>이름을 입력해주세요</p>}
                </div>
한규민's avatar
한규민 committed
189
190
191
                <div className="d-flex flex-column">
                    <div className={styles.inputContent}>
                        <label className={styles.signupLabel}>이메일</label>
한규민's avatar
한규민 committed
192
                        <input className={`${styles.input} ${styles.inputSize}`} type="email" name="userEmail" placeholder="이메일을 입력해주세요" onChange={handleUserOnChange} maxLength="20" required />
한규민's avatar
한규민 committed
193
194
195
                    </div>
                    {errorMsg.errorEmail && <p className={styles.passwordConfirmError}>이메일을 입력해주세요</p>}
                </div>
한규민's avatar
한규민 committed
196
197
198
                <div className="d-flex flex-column">
                    <div className={styles.inputContent}>
                        <label className={styles.signupLabel}>별명</label>
한규민's avatar
한규민 committed
199
                        <input className={`${styles.input} ${styles.inputSize}`} type="text" name="userNickName" placeholder="10자리 이내" onChange={handleUserOnChange} maxLength="20" required />
한규민's avatar
한규민 committed
200
                    </div>
한규민's avatar
한규민 committed
201
                    {errorMsg.errorNickName && <p className={styles.passwordConfirmError}>10 이내로 입력해주세요.</p>}
202
                </div>
한규민's avatar
한규민 committed
203
204
205
206

                <div className="d-flex flex-column">
                    <div className={styles.inputContent}>
                        <label className={styles.signupLabel}>생년월일</label>
한규민's avatar
한규민 committed
207
                        <input className={`${styles.input} ${styles.inputSize} ${styles.input.placeholder}`} type="number" name="userBirthday" placeholder="6자리(예시: 991225)" onChange={handleUserOnChange} min="0" max="999999" required />
한규민's avatar
한규민 committed
208
                    </div>
한규민's avatar
한규민 committed
209
                    {errorMsg.errorBirthday && <p className={styles.passwordConfirmError}>숫자 6자리를 입력해주세요.</p>}
210
                </div>
한규민's avatar
한규민 committed
211
212
213
214

                <div className="d-flex flex-column">
                    <div className={styles.inputContent}>
                        <label className={styles.signupLabel}>휴대폰 번호</label>
한규민's avatar
한규민 committed
215
                        <div className="d-flex col-md-auto">
한규민's avatar
한규민 committed
216
                            <input className={`${styles.input} ${styles.input2}`} type="number" name="userMbnum" placeholder="-없이 11자리 입력" onChange={handleUserOnChange} min="0" max="99999999999" required />
한규민's avatar
한규민 committed
217
                            <button type="button" disabled={loading} className={`rounded-2 mt-2 ${styles.butterYellowAndBtn} ${styles.btnHover}`} data-bs-toggle="collapse" data-bs-target="#collapseExample" aria-expanded="false" aria-controls="collapseExample" onClick={handleOnClickMbnum}>인증번호받기</button>
한규민's avatar
한규민 committed
218
                        </div>
한규민's avatar
한규민 committed
219
                    </div>
한규민's avatar
한규민 committed
220
                    {errorMsg.errorMbnum && <p className={styles.passwordConfirmError}>-없이 숫자 11자리를 입력해주세요.</p>}
한규민's avatar
한규민 committed
221
222
223
224
225
226

                    <div class="collapse" id="collapseExample">
                        {/* <div className="d-flex col-md-auto justify-content-end"> */}
                        <div className="d-flex justify-content-between mt-3">
                            <label className={`${styles.confirm}`}>인증하기</label>
                            <div>
227
                                <input className={`${styles.input} ${styles.input2}`} type="number" placeholder="인증번호를 입력" onChange={handleOnChangeMb} required/>
한규민's avatar
한규민 committed
228
229
230
231
232
233
234
235
236
                                <button type="button" className={`rounded-2 py-2 ${styles.butterYellowAndBtn} ${styles.btnHover}`} onClick={handleOnClickMbConfirm}>확인</button>
                                <button type="button" className={`rounded-2 py-2 ${styles.butterYellowAndBtn} ${styles.btnHover}`} onClick={handleOnClickMbnum}>재전송</button>
                            </div>
                        </div>
                        {(mbError === "재전송") && <p className={styles.passwordConfirmError}>유효시간이 만료되었습니다. 재전송해주세요.</p>}
                        {(mbError === "보냄") && <p className={styles.passwordConfirmError}>5분이내에 입력해주세요.</p>}
                        {(mbError === "성공") && <p className={styles.passwordConfirmError}>인증되었습니다.</p>}
                        {(mbError === "실패") && <p className={styles.passwordConfirmError}>인증번호를 다시 입력해주세요.</p>}
                    </div>
237
                </div>
한규민's avatar
한규민 committed
238

한규민's avatar
한규민 committed
239
240
241
                <div className="d-flex flex-column">
                    <div className={`${styles.inputContent}`}>
                        <label className={styles.signupLabel}>비밀번호</label>
한규민's avatar
한규민 committed
242
                        <input className={`${styles.input} ${styles.inputSize}`} type="password" name="userPassword" placeholder="8~11자리 사이" onChange={handleUserOnChange} maxLength="20" required />
한규민's avatar
한규민 committed
243
                    </div>
한규민's avatar
한규민 committed
244
                    {errorMsg.errorPassword && <p className={styles.passwordConfirmError}>8~11자리 사이로 입력해주세요.</p>}
245
                </div>
한규민's avatar
한규민 committed
246

한규민's avatar
한규민 committed
247
248
249
                <div className={`d-flex ${styles.inputContent}`}>
                    <label className={styles.signupLabel}>비밀번호 확인</label>
                    <input className={`${styles.input} ${styles.inputSize}`} type="password" name="userRePassword" placeholder="8~11자리 사이" onChange={handleUserOnChange} maxLength="20" required />
한규민's avatar
한규민 committed
250
251
                </div>

한규민's avatar
한규민 committed
252
                <button className={`rounded my-3 py-2 fs-5 ${styles.butterYellowAndBtn} ${styles.btnHover}`} type="submit" disabled={loading}>가입하기</button>
253
            </div>
한규민's avatar
한규민 committed
254
        </form>
한규민's avatar
한규민 committed
255
256
257
258
    )
}

export default Signup