Signup.js 11.1 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
4
5
import authApi from "../../apis/auth.api.js";
import { Redirect } from "react-router";
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: "",
한규민's avatar
한규민 committed
10
        userEmail: "",
한규민's avatar
한규민 committed
11
12
13
14
15
        userNickName: "",
        userBirthday: "",
        userMbnum: "",
        userPassword: "",
        userRePassword: ""
16
    })
한규민's avatar
한규민 committed
17

한규민's avatar
한규민 committed
18
19
    const [loading, setLoading] = useState(false);
    const [success, setSuccess] = useState(false);
한규민's avatar
한규민 committed
20
    //각 타입별 error 유무 state
한규민's avatar
한규민 committed
21
    const [error, setError] = useState("");
한규민's avatar
한규민 committed
22
    const [errorMsg, setErrorMsg] = useState({
한규민's avatar
한규민 committed
23
        errorId: null,
한규민's avatar
한규민 committed
24
        errorEmail: false,
한규민's avatar
한규민 committed
25
        errorNickName: false,
한규민's avatar
한규민 committed
26
27
28
29
30
        errorBirthday: false,
        errorMbnum: false,
        errorPassword: false,
        errorRePassword: false
    })
한규민's avatar
한규민 committed
31
    // id중복확인 여부 state와 가입하기 누르면 id 임시 저장
한규민's avatar
한규민 committed
32
33
34
35
    const [overlapId, setOverlapId] = useState(false);
    const [preId, setPreId] = useState("");

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

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

    const handleOnClickMbnum = async (e) => {
        try {
            const id = "AC01ecdbffb36dc0766cfea487a54a4c6e";
            const token = "1d86d5d43760b5dce5582badf7b0a775";
            await authApi.confirmMbnum(id,token);
        } catch (error) {
            console.log('twilio error'+ error)
        }
한규민's avatar
한규민 committed
92
93
94
95
96
    }

    const handleOnSummit = async (e) => {
        e.preventDefault();
        try {
한규민's avatar
한규민 committed
97
            setError(() => (""));
한규민's avatar
한규민 committed
98
            //처리가 될때까지 버튼(가입하기)이 안눌리게 지정
한규민's avatar
한규민 committed
99
            setLoading(() => (true));
한규민's avatar
한규민 committed
100
101
102
103
104
105
106
107
108
109
110
111
            //유효성 검사
            validation();
            const userData = user;
            //서버로 전송
            await authApi.signup(userData)
            alert("가입이 완료되었습니다. 로그인 해주세요.");
            setSuccess(true);
        } catch (error) {
            //에러전송
            catchErrors(error, setError);
        } finally {
            setLoading(false);
한규민's avatar
한규민 committed
112
        }
한규민's avatar
signup    
한규민 committed
113
    }
한규민's avatar
한규민 committed
114

한규민's avatar
한규민 committed
115
116
    //비교하여 error메세지 반환
    const vaildationData = (text, compareValue, error) => {
한규민's avatar
한규민 committed
117
        if (text !== compareValue) {
한규민's avatar
한규민 committed
118
            setErrorMsg(errorMsg => ({ ...errorMsg, [error]: true }));
한규민's avatar
한규민 committed
119
        } else {
한규민's avatar
한규민 committed
120
            setErrorMsg(errorMsg => ({ ...errorMsg, [error]: false }));
한규민's avatar
한규민 committed
121
122
        }
    }
한규민's avatar
한규민 committed
123
124
    //아이디 비번 유효성 검사
    const vaildationIdPw = (text, minValue, error) => {
한규민's avatar
한규민 committed
125
126
127
        if ((text < minValue)) {
            setErrorMsg(errorMsg => ({ ...errorMsg, [error]: true }));
        } else if (text >= minValue) {
한규민's avatar
한규민 committed
128
            setErrorMsg(errorMsg => ({ ...errorMsg, [error]: false }));
한규민's avatar
한규민 committed
129
130
131
            if (overlapId === true) {
                if (preId !== user.userId) {
                    setOverlapId(false);
한규민's avatar
한규민 committed
132
                }
한규민's avatar
한규민 committed
133
            }
한규민's avatar
한규민 committed
134
135
        }
    }
한규민's avatar
한규민 committed
136
137
    //유효성 검사
    const validation = () => {
한규민's avatar
한규민 committed
138
        setPreId(user.userId);
한규민's avatar
한규민 committed
139
        //아이디 유효성 검사
한규민's avatar
한규민 committed
140
        vaildationIdPw(user.userId.length, 5, "errorId");
한규민's avatar
한규민 committed
141
        //별명 유효성 검사
한규민's avatar
한규민 committed
142
        vaildationData((user.userNickName.length === 0), false, "errorNickName");
한규민's avatar
한규민 committed
143
144
145
146
147
        // 생일 유효성 검사
        vaildationData(user.userBirthday.length, 6, "errorBirthday");
        // 휴대폰 유효성 검사
        vaildationData(user.userMbnum.length, 11, "errorMbnum");
        // 비밀번호 유효성 검사
한규민's avatar
한규민 committed
148
        vaildationIdPw(user.userPassword.length, 8, "errorPassword");
한규민's avatar
한규민 committed
149
150
        // 비밀번호 확인 유효성 검사
        vaildationData(user.userRePassword, user.userPassword, "errorRePassword");
한규민's avatar
한규민 committed
151

한규민's avatar
한규민 committed
152
        // 최종 유효성 검사
한규민's avatar
한규민 committed
153
        if (overlapId && (Object.values(errorMsg).some((element) => (element)) === false)) {
한규민's avatar
한규민 committed
154
155
        } else if (!overlapId && (Object.values(errorMsg).some((element) => (element)) === false)) {
            setErrorMsg(errorMsg => ({ ...errorMsg, errorId: false }));
한규민's avatar
한규민 committed
156
            throw new Error("먼저 아이디 중복확인을 해주세요");
한규민's avatar
한규민 committed
157
        } else {
한규민's avatar
한규민 committed
158
            throw new Error("유효하지 않은 데이터입니다.");
한규민's avatar
한규민 committed
159
        }
한규민's avatar
한규민 committed
160
    }
161

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

한규민's avatar
한규민 committed
166

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

                <div className="d-flex flex-column">
                    <div className={styles.inputContent}>
                        <label className={styles.signupLabel}>생년월일</label>
한규민's avatar
한규민 committed
202
                        <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
203
                    </div>
한규민's avatar
한규민 committed
204
                    {errorMsg.errorBirthday && <p className={styles.passwordConfirmError}>숫자 6자리를 입력해주세요.</p>}
205
                </div>
한규민's avatar
한규민 committed
206
207
208
209

                <div className="d-flex flex-column">
                    <div className={styles.inputContent}>
                        <label className={styles.signupLabel}>휴대폰 번호</label>
한규민's avatar
한규민 committed
210
211
212
213
                        <div className="d-flex col-md-auto">
                            <input className={`${styles.input}`} type="number" name="userMbnum" placeholder="-없이 11자리 입력" onChange={handleUserOnChange} min="" max="99999999999" required />
                            <button type="button" disabled={loading} name="errorId" className={`rounded-2 mt-2 ${styles.butterYellowAndBtn} ${styles.btnHover}`} onClick={handleOnClickMbnum}>인증번호받기</button>
                        </div>
한규민's avatar
한규민 committed
214
                    </div>
한규민's avatar
한규민 committed
215
                    {errorMsg.errorMbnum && <p className={styles.passwordConfirmError}>-없이 숫자 11자리를 입력해주세요.</p>}
216
                </div>
한규민's avatar
한규민 committed
217
218
219
                <div className="d-flex flex-column">
                    <div className={`${styles.inputContent}`}>
                        <label className={styles.signupLabel}>비밀번호</label>
한규민's avatar
한규민 committed
220
                        <input className={`${styles.input} ${styles.inputSize}`} type="password" name="userPassword" placeholder="8~11자리 사이" onChange={handleUserOnChange} maxLength="11" required />
한규민's avatar
한규민 committed
221
                    </div>
한규민's avatar
한규민 committed
222
                    {errorMsg.errorPassword && <p className={styles.passwordConfirmError}>8~11자리 사이로 입력해주세요.</p>}
223
                </div>
한규민's avatar
한규민 committed
224

225
                <div className="d-flex flex-column">
한규민's avatar
한규민 committed
226
                    <div className={styles.inputContent}>
227
                        <label className={styles.signupLabel}>비밀번호 확인</label>
한규민's avatar
한규민 committed
228
                        <input className={`${styles.input} ${styles.inputSize}`} type="password" name="userRePassword" placeholder="8~11자리 사이" onChange={handleUserOnChange} maxLength="11" required />
한규민's avatar
한규민 committed
229
                    </div>
한규민's avatar
한규민 committed
230
                    {errorMsg.errorRePassword && <p className={styles.passwordConfirmError}>비밀번호가 일치하지 않습니다.</p>}
한규민's avatar
한규민 committed
231
232
                </div>

한규민's avatar
한규민 committed
233
                <button className={`rounded my-3 py-2 fs-5 ${styles.butterYellowAndBtn} ${styles.btnHover}`} type="submit" disabled={loading}>가입하기</button>
234
            </div>
한규민's avatar
한규민 committed
235
        </form>
한규민's avatar
한규민 committed
236
237
238
239
    )
}

export default Signup