Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Feat] 회원가입 페이지 중복확인 로직 구현 #150

Merged
merged 1 commit into from
Jan 24, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 0 additions & 2 deletions src/pages/signUp/SignUp.styled.ts
Original file line number Diff line number Diff line change
Expand Up @@ -28,8 +28,6 @@ export const LeftContainer = styled.section`

background-image: url('/images/background_img.png');
background-size: cover;

/* background-image: url(''); */
border-radius: 2rem 0 0 2rem;
`;

Expand Down
120 changes: 90 additions & 30 deletions src/pages/signUp/SignUp.tsx
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import { usePostNicknameCheck } from '@apis/users/queries';
import { ImgModalcheck } from '@assets/svgs';
import CircleButton from '@components/button/circleButton/CircleButton';
import { AlterModal } from '@components/modal';
Expand All @@ -14,17 +15,12 @@ import * as S from './SignUp.styled';
const SignUp = () => {
const navigate = useNavigate();
const [nickname, setNickname] = useState('');
const [nicknameState, setNicknameState] = useState<'default' | 'act' | 'error' | 'success'>('default');
const [nicknameMessage, setNicknameMessage] = useState<string>('');
const { mutateAsync: checkMutate } = usePostNicknameCheck();
const [isModalOpen, setIsModalOpen] = useState(false);
const [selectedAffiliation, setSelectedAffiliation] = useState<string | null>(null);
const [isSubmitting, setIsSubmitting] = useState(false); // 중복 클릭 방지용

const handleNicknameChange = (e: React.ChangeEvent<HTMLInputElement>) => {
setNickname(e.target.value);
};

const handleCloseModal = () => {
setIsModalOpen(false);
};
const [isSubmitting, setIsSubmitting] = useState(false);

useEffect(() => {
const userString = localStorage.getItem('user');
Expand All @@ -47,36 +43,94 @@ const SignUp = () => {
}
}, [navigate]);

// 닉네임 입력 핸들러
const handleNicknameChange = (e: React.ChangeEvent<HTMLInputElement>) => {
const value = e.target.value;
setNickname(value);

// 닉네임 변경 시 중복 확인 상태 초기화
if (value.length === 0) {
setNicknameState('default'); // 입력값 없음
} else {
setNicknameState('act'); // 입력값 있음
}
setNicknameMessage('');
};

// 닉네임 중복 확인
const handleNicknameCheck = async () => {
if (!nickname.trim()) {
setNicknameState('error');
setNicknameMessage('닉네임을 입력해주세요.');
return;
}

if (!/^[가-힣a-zA-Z0-9]+$/.test(nickname)) {
setNicknameState('error');
setNicknameMessage('초성 또는 모음만으로 구성된 닉네임은 사용할 수 없어요.');
return;
}

setNicknameState('act');
setNicknameMessage('중복 확인 중...');

try {
const checkResponse = await checkMutate(nickname);
if (checkResponse?.statusCode === 200 && checkResponse?.data) {
setNicknameState('error'); // 중복된 닉네임
setNicknameMessage('이미 사용 중인 닉네임이에요.');
} else {
setNicknameState('success'); // 사용 가능 상태
setNicknameMessage('사용 가능한 닉네임이에요.');
}
} catch (error) {
console.error('닉네임 체크 실패:', error);
setNicknameState('error'); // 사용 불가능 상태
setNicknameMessage('중복 확인 중 오류가 발생했어요.');
}
};

// 회원가입 버튼 활성화 조건
const isCircleBtnActive = nickname.length >= 1 && selectedAffiliation !== null && nicknameState === 'success';

// 회원가입 요청
const handleCircleBtnClick = async () => {
if (!nickname || !selectedAffiliation) {
alert('닉네임과 소속을 모두 입력해주세요.');
return;
}

const userString = localStorage.getItem('user');
if (userString) {
const user = JSON.parse(userString);
if (!user.email) {
alert('이메일 정보가 없습니다. 다시 로그인해주세요.');
return;
}
setIsSubmitting(true); // 버튼 중복 클릭 방지
try {
await signup({
nickname,
positions: selectedAffiliation,
email: user.email,
});
setIsModalOpen(true); // 회원가입 성공 시 모달 열기
} catch (error) {
console.error('회원가입 실패:', error);
} finally {
setIsSubmitting(false);
}
if (!userString) {
alert('로그인이 필요합니다. 다시 시도해주세요.');
return;
}

const user = JSON.parse(userString);
if (!user.email) {
alert('이메일 정보가 없습니다. 다시 로그인해주세요.');
return;
}

setIsSubmitting(true);
try {
await signup({
nickname,
positions: selectedAffiliation,
email: user.email,
});
setIsModalOpen(true); // 회원가입 성공 시 모달 열기
} catch (error) {
console.error('회원가입 실패:', error);
} finally {
setIsSubmitting(false);
}
};

const isCircleBtnActive = nickname.length >= 1 && selectedAffiliation !== null;
// 모달 닫기
const handleCloseModal = () => {
setIsModalOpen(false);
};

const modalProps = {
modalTitle: '회원가입이 완료되었어요.',
Expand Down Expand Up @@ -127,7 +181,13 @@ const SignUp = () => {
</S.AffiliationBtnBox>
</S.AffiliationBox>
<S.NicknameInputBox>
<NamingInput value={nickname} onChange={handleNicknameChange} />
<NamingInput
value={nickname}
state={nicknameState}
description={nicknameMessage}
onChange={handleNicknameChange}
onNicknameCheck={handleNicknameCheck}
/>
</S.NicknameInputBox>
<S.SignUpBtn>
<CircleButton size="mini" disabled={!isCircleBtnActive || isSubmitting} onClick={handleCircleBtnClick}>
Expand Down
13 changes: 13 additions & 0 deletions src/pages/signUp/apis/api.ts
Original file line number Diff line number Diff line change
Expand Up @@ -36,3 +36,16 @@ const signup = async (requestBody: SignupRequest): Promise<void> => {
};

export default signup;

export const postNicknameCheck = async (
nickname: string,
): Promise<{ statusCode: number; message: string; data: boolean } | undefined> => {
try {
const response = await post<{ statusCode: number; message: string; data: boolean }>(
`users/nickname?nickname=${nickname}`,
);
return response;
} catch (error) {
console.error('Error:', error);
}
};
10 changes: 5 additions & 5 deletions src/pages/signUp/components/namingInput/NamingInput.styled.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,9 +22,9 @@ export const InputBox = styled.div`
margin-bottom: 0.6rem;
`;

export const DescriptionBox = styled.div`
export const DescriptionBox = styled.div<{ $description: boolean }>`
display: flex;
flex-direction: row-reverse;
flex-direction: ${({ $description }) => ($description ? '' : 'row-reverse')};
justify-content: space-between;
width: 30.1rem;
height: 1.8rem;
Expand All @@ -40,11 +40,11 @@ export const Input = styled.input<{ state: 'default' | 'act' | 'error' | 'succes
border-color: ${({ state, theme }) => {
switch (state) {
case 'act':
return theme.colors.sys_green;
return theme.colors.gray1;
case 'error':
return theme.colors.sys_red;
case 'success':
return null;
return theme.colors.sys_green;
default:
return theme.colors.gray4;
}
Expand All @@ -60,7 +60,7 @@ export const Description = styled.span<{ state: 'default' | 'act' | 'error' | 's
color: ${({ state, theme }) => {
switch (state) {
case 'act':
return theme.colors.sys_green;
return theme.colors.gray1;
case 'error':
return theme.colors.sys_red;
case 'success':
Expand Down
8 changes: 5 additions & 3 deletions src/pages/signUp/components/namingInput/NamingInput.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ type NamingInputPropTypes = {
inputRestrictions?: string[];
value: string;
onChange?: (e: React.ChangeEvent<HTMLInputElement>) => void;
onNicknameCheck: () => void; // 닉네임 중복 확인 함수
} & InputHTMLAttributes<HTMLInputElement>;

const NamingInput = ({
Expand All @@ -23,6 +24,7 @@ const NamingInput = ({
],
value,
onChange,
onNicknameCheck,
...props
}: NamingInputPropTypes) => {
const placeholder = state === 'default' ? '닉네임을 입력해주세요.' : '';
Expand All @@ -34,16 +36,16 @@ const NamingInput = ({
}
};

const isActive = count > 0; // 입력값이 1글자 이상일 때만 active
const isActive = count > 0; // 입력값이 1글자 이상일 때만 활성화

return (
<S.InputWrapper>
{label && <S.Label>{label}</S.Label>}
<S.InputBox>
<S.Input state={state} value={value} onChange={handleInputChange} placeholder={placeholder} {...props} />
<ConfirmBtn isActive={isActive} />
<ConfirmBtn isActive={isActive} onClick={onNicknameCheck} />
</S.InputBox>
<S.DescriptionBox>
<S.DescriptionBox $description={!!description}>
{description && <S.Description state={state}>{description}</S.Description>}
<S.LetterCount>{count}/10</S.LetterCount>
</S.DescriptionBox>
Expand Down
Loading