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
import Twilio from "twilio";
한규민's avatar
한규민 committed
7
8

const Signup = () => {
한규민's avatar
한규민 committed
9
    const [user, setUser] = useState({
한규민's avatar
한규민 committed
10
        userId: "",
한규민's avatar
한규민 committed
11
        userEmail: "",
한규민's avatar
한규민 committed
12
13
14
15
16
        userNickName: "",
        userBirthday: "",
        userMbnum: "",
        userPassword: "",
        userRePassword: ""
17
    })
한규민's avatar
한규민 committed
18

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

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

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

    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
93
94
95
96
97
    }

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

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

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

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

한규민's avatar
한규민 committed
167

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

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

                <div className="d-flex flex-column">
                    <div className={styles.inputContent}>
                        <label className={styles.signupLabel}>휴대폰 번호</label>
한규민's avatar
한규민 committed
211
212
213
214
                        <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
215
                    </div>
한규민's avatar
한규민 committed
216
                    {errorMsg.errorMbnum && <p className={styles.passwordConfirmError}>-없이 숫자 11자리를 입력해주세요.</p>}
217
                </div>
한규민's avatar
한규민 committed
218
219
220
                <div className="d-flex flex-column">
                    <div className={`${styles.inputContent}`}>
                        <label className={styles.signupLabel}>비밀번호</label>
한규민's avatar
한규민 committed
221
                        <input className={`${styles.input} ${styles.inputSize}`} type="password" name="userPassword" placeholder="8~11자리 사이" onChange={handleUserOnChange} maxLength="11" required />
한규민's avatar
한규민 committed
222
                    </div>
한규민's avatar
한규민 committed
223
                    {errorMsg.errorPassword && <p className={styles.passwordConfirmError}>8~11자리 사이로 입력해주세요.</p>}
224
                </div>
한규민's avatar
한규민 committed
225

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

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

export default Signup