Signup.js 8.99 KB
Newer Older
한규민's avatar
한규민 committed
1
import styles from "./signup.module.scss";
한규민's avatar
한규민 committed
2
3
4
5
import { useState, useEffect } from 'react';
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: '',
10
        userName: '',
한규민's avatar
한규민 committed
11
12
        userBirthday: '',
        userMbnum: '',
13
        userPassword: '',
한규민's avatar
한규민 committed
14
        userRePassword: ''
15
    })
한규민's avatar
한규민 committed
16
17
18
    const [loading, setLoading] = useState(false);
    const [success, setSuccess] = useState(false);

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

    //입력할때마다 state에 저장
34
    const handleUserOnChange = (e) => {
한규민's avatar
한규민 committed
35
36
        setUser({
            ...user,
37
38
            [e.target.name]: e.target.value
        })
한규민's avatar
한규민 committed
39
    }
40

한규민's avatar
한규민 committed
41
    //id(중복확인 체크, 형식 에러)
한규민's avatar
한규민 committed
42
43
44
45
46
47
    const handleOnClickId = async (e) => {
        e.preventDefault();
        const existId = await authApi.compareId(user.userId)
        if (user.userId.length < 5) {
            setErrorMsg(errorMsg => ({
                ...errorMsg,
한규민's avatar
한규민 committed
48
                [e.target.name]: true
한규민's avatar
한규민 committed
49
            }));
한규민's avatar
한규민 committed
50
51
            if (existId === true) {
                setOverlapId(() => (false));
한규민's avatar
한규민 committed
52
53
            };
        } else {
한규민's avatar
한규민 committed
54
55
            setErrorMsg(errorMsg => ({
                ...errorMsg,
한규민's avatar
한규민 committed
56
                [e.target.name]: false
한규민's avatar
한규민 committed
57
            }));
한규민's avatar
한규민 committed
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
            setOverlapId(() => (true));
            alert("이 아이디는 사용가능합니다.")
        }
    }

    const handleOnSummit = async (e) => {
        e.preventDefault();
        try {
            setError("");
            //처리가 될때까지 버튼(가입하기)이 안눌리게 지정
            setLoading(true);
            //유효성 검사
            validation();
            const userData = user;
            //서버로 전송
            await authApi.signup(userData)
            alert("가입이 완료되었습니다. 로그인 해주세요.");
            setSuccess(true);
        } catch (error) {
            //에러전송
            catchErrors(error, setError);
        } finally {
            setLoading(false);
한규민's avatar
한규민 committed
81
        }
한규민's avatar
signup    
한규민 committed
82
    }
한규민's avatar
한규민 committed
83

한규민's avatar
한규민 committed
84
85
86
    //유효성 검사 함수
    const vaildationData = (text, compareValue, error) =>{
        if (text !== compareValue) {
한규민's avatar
한규민 committed
87
            setErrorMsg(errorMsg => ({ ...errorMsg, [error]: true }));
한규민's avatar
한규민 committed
88
        } else{
한규민's avatar
한규민 committed
89
            setErrorMsg(errorMsg => ({ ...errorMsg, [error]: false }));
한규민's avatar
한규민 committed
90
91
        }
    }
한규민's avatar
한규민 committed
92
93
94
    //유효성 검사
    const validation = () => {
        setPreId(()=> (user.userId));
한규민's avatar
한규민 committed
95
        //아이디 유효성 검사
한규민's avatar
한규민 committed
96
97
98
99
        if ((user.userId.length < 5)) {
            setErrorMsg(errorMsg => ({ ...errorMsg, errorId: true }));
        } else if((user.userId.length >= 5) && (overlapId === true)){
            if(preId !== user.userId){
한규민's avatar
한규민 committed
100
101
102
103
                console.log(preId);
            setOverlapId(()=> (false));
            }
        } 
한규민's avatar
한규민 committed
104
105
        else if(user.userId >= 5){
            setErrorMsg(errorMsg => ({ ...errorMsg, errorId: false }));
한규민's avatar
한규민 committed
106
        }
한규민's avatar
한규민 committed
107
108
109
110
111
112
113
114
115
116
117

        //별명 유효성 검사
        vaildationData((user.userName.length === 0), false, "errorName");
        // 생일 유효성 검사
        vaildationData(user.userBirthday.length, 6, "errorBirthday");
        // 휴대폰 유효성 검사
        vaildationData(user.userMbnum.length, 11, "errorMbnum");
        // 비밀번호 유효성 검사
        vaildationData(user.userPassword.length, 8, "errorPassword");
        // 비밀번호 확인 유효성 검사
        vaildationData(user.userRePassword, user.userPassword, "errorRePassword");
한규민's avatar
한규민 committed
118
        
한규민's avatar
한규민 committed
119
        // 최종 유효성 검사
한규민's avatar
한규민 committed
120
121
122
123
124
        if (overlapId && (Object.values(errorMsg).some((element) => (element)) === false)) {
            console.log(";sadasda")
        }else{
            console.log("에러발생")
            throw new Error("유효하지 않은 데이터입니다.");
한규민's avatar
한규민 committed
125
        }
한규민's avatar
한규민 committed
126
    }
127

한규민's avatar
한규민 committed
128
129
130
131
132
    if (success) {
        return <Redirect to="/login" />;
    }
    

한규민's avatar
한규민 committed
133
    return (
한규민's avatar
한규민 committed
134
        // 데이터 입력과 유효성 검사 후 보이는 경고창
한규민's avatar
한규민 committed
135
136
        <form className={`d-flex col-md-6 col-12 justify-content-center`} onSubmit={handleOnSummit}>
            {console.log("user==", user, errorMsg, overlapId)}
한규민's avatar
한규민 committed
137
            <div className="d-flex flex-column">
한규민's avatar
signup    
한규민 committed
138
139
140
141
142
                <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
143
                            <input className={styles.input} type="text" name="userId" id="userId" placeholder="5~10자리 사이" onChange={handleUserOnChange} maxLength="10" required />
한규민's avatar
한규민 committed
144
                            <button type="button" name="errorId" className={`rounded-2 mt-2 ${styles.butterYellowAndBtn} ${styles.btnHover}`} onClick={handleOnClickId}>중복확인</button>
한규민's avatar
signup    
한규민 committed
145
                        </div>
한규민's avatar
한규민 committed
146
                    </div>
한규민's avatar
한규민 committed
147
148
149
                    {(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>}
150
                </div>
한규민's avatar
한규민 committed
151
152
153
154
155
156

                <div className="d-flex flex-column">
                    <div className={styles.inputContent}>
                        <label className={styles.signupLabel}>별명</label>
                        <input className={`${styles.input} ${styles.inputSize}`} type="text" name="userName" id="userName" placeholder="10자리 이내" onChange={handleUserOnChange} maxLength="10" required />
                    </div>
한규민's avatar
한규민 committed
157
                    {errorMsg.errorName && <p className={styles.passwordConfirmError}>10 이내로 입력해주세요.</p>}
158
                </div>
한규민's avatar
한규민 committed
159
160
161
162
163
164

                <div className="d-flex flex-column">
                    <div className={styles.inputContent}>
                        <label className={styles.signupLabel}>생년월일</label>
                        <input className={`${styles.input} ${styles.inputSize} ${styles.input.placeholder}`} type="number" name="userBirthday" id="userBirthday" placeholder="6자리(예시: 991225)" onChange={handleUserOnChange} min="0" max="999999" required />
                    </div>
한규민's avatar
한규민 committed
165
                    {errorMsg.errorBirthday && <p className={styles.passwordConfirmError}>숫자 6자리를 입력해주세요.</p>}
166
                </div>
한규민's avatar
한규민 committed
167
168
169
170
171
172

                <div className="d-flex flex-column">
                    <div className={styles.inputContent}>
                        <label className={styles.signupLabel}>휴대폰 번호</label>
                        <input className={`${styles.input} ${styles.inputSize}`} type="number" name="userMbnum" id="userMbnum" placeholder="-없이 11자리 입력" onChange={handleUserOnChange} min="0" max="99999999999" required />
                    </div>
한규민's avatar
한규민 committed
173
                    {errorMsg.errorMbnum && <p className={styles.passwordConfirmError}>-없이 숫자 11자리를 입력해주세요.</p>}
174
                </div>
한규민's avatar
한규민 committed
175
176
177
178
179
180

                <div className="d-flex flex-column">
                    <div className={`${styles.inputContent}`}>
                        <label className={styles.signupLabel}>비밀번호</label>
                        <input className={`${styles.input} ${styles.inputSize}`} type="password" name="userPassword" id="password" placeholder="8자리 입력" onChange={handleUserOnChange} maxLength="8" required />
                    </div>
한규민's avatar
한규민 committed
181
                    {errorMsg.errorPassword && <p className={styles.passwordConfirmError}>8자리를 입력해주세요.</p>}
182
                </div>
한규민's avatar
한규민 committed
183

184
                <div className="d-flex flex-column">
한규민's avatar
한규민 committed
185
                    <div className={styles.inputContent}>
186
                        <label className={styles.signupLabel}>비밀번호 확인</label>
한규민's avatar
한규민 committed
187
                        <input className={`${styles.input} ${styles.inputSize}`} type="password" name="userRePassword" id="userRePassword" placeholder="8자리 입력" onChange={handleUserOnChange} maxLength="8" required />
한규민's avatar
한규민 committed
188
                    </div>
한규민's avatar
한규민 committed
189
                    {errorMsg.errorRePassword && <p className={styles.passwordConfirmError}>비밀번호가 일치하지 않습니다.</p>}
한규민's avatar
한규민 committed
190
191
                </div>

한규민's avatar
한규민 committed
192
                <button className={`rounded my-3 py-2 fs-5 ${styles.butterYellowAndBtn} ${styles.btnHover}`} type="submit" disabled={loading}>가입하기</button>
193
            </div>
한규민's avatar
한규민 committed
194
        </form>
한규민's avatar
한규민 committed
195
196
197
198
    )
}

export default Signup