호진방 블로그
기술

응집도와 결합도를 고려한 회원가입 설계와 유저가 지치지 않는 UI

# Offispace 프로젝트의 회원가입은 어떻게 진행될까요?

2024년 05월 02일

회원가입 재설계 배경

Offispace 프로젝트 회원가입 페이지에서는 총 3단계를 통해 회원가입을 진행된다.

각 단계별로 돌아다니면서 정보를 모으게 되고, 해당 정보를 합쳐서 서버로 보내게 되는데, 마치 MBTI 검사 페이지에서 질문 리스트를 여러 페이지로 걸쳐서 받게 되고, 저장된 답변에 따라 검사 결과를 출력하는 형식과 유사하다.


위와 같은 형식의 장점은 각 페이지에 필요한 데이터, 그리고 데이터가 변화하는 로직, 그리고 해당 데이터를 변화시키는 코드까지 독립된 페이지에서 관리 할 수 있다. 그 코드들은 각각의 검사 페이지를 위한 코드가 되고, 두번째 세번째 페이지에서도 독립된 코드 형식을 사용하여 각 페이지마다 곂치지 않게 된다.


❗ 다만 이와 같은 형식은 단점이 존재하는데, 페이지들을 돌아다니면서 하나의 데이터를 완성시키는 과정이 각 페이지에서 이뤄지기 때문에 어떤 데이터가 언제 변하는지 알기 위해서는 각 페이지들을 돌아다니면서, 어떤 시점에 어떤 데이터를 바꿔야 하는지 찾아다녀야 한다. 또한 디버깅을 하고 싶어도 각 페이지에 데이터가 바뀌는 부분에 대해 모두 로그를 심어야 하는 단점이 존재한다. 하나의 관심사를 갖고 있지만 너무 코드가 분리 되었기 때문이다.

즉, 해당 방식은 관련된 기능들이 하나의 모듈이나 페이지에 모여 있지 않아 응집도가 낮고, 한 페이지의 변경이 다른 페이지 부분에 영향을 미치는 정도가 매우 커 결합도가 높다고 얘기 할 수 있다.

❗️ 응집도와 결합도란?

응집도와 결합도 이야기

응집도가 낮은 이유

결합도가 높은 이유

해결 아이디어

💡 내가 생각한 방식은 한 페이지 안에서 step 상태를 이용해 각 단계별 컴포넌트로 분기처리 하는 방식이다. 해당 방식은 하나의 회원가입 페이지가 존재하고, 모든 회원가입에 필요한 데이터는 회원가입 페이지에서만 관리된다.

각 데이터는 step에 따라서 분기가 되는데, 예를들어 step1 휴대폰 인증에 관한 코드는 회원가입 페이지 하위 휴대폰 인증 컴포넌트에서 실행되고, 해당 데이터가 정상적으로 수집됐다면, 부모 컴포넌트인 회원가입 페이지의 step을 증가시켜 다음 단계인 이메일 인증에 필요한 데이터를 받게 된다.


해당 방식은 응집도와 결합도의 관점에서도 유리하다고 생각한다.

응집도가 높다는 것은 관련된 기능들이 하나의 모듈이나 컴포넌트 내에 모여 있어 모듈의 책임이 명확하고, 유지보수가 용이하다는 것을 의미하는데, 회원가입과 관련된 모든 데이터가 회원가입 하나의 페이지에서 관리되며, 각 단계별로 분리된 컴포넌트는 자신이 가진 데이터 (전화번호, 이메일, etc)만 관심사를 두고 있다

결합도가 낮다는 것은 모듈이나 컴포넌트 간의 의존성이 적어 변경이 용이하다는 것을 의미한다. 각 단계별 컴포넌트가 부모 컴포넌트와 step이란 상태를 통해 상호작용하므로, 각 컴포넌트는 다른 컴포넌트와 상호작용 없이 독립적으로 동작한다.


구현

src > pages > signup > index;
 
const SignUpPage = () => {
  const [applyValues, setApplyValues] = useState<Partial<ApplyValues>>({
    step: 0,
  });
 
  const handlePhoneNumber = (phoneNumber: ApplyValues['memberPhone']) => {
    setApplyValues((prev) => ({
      ...prev,
      memberPhone: phoneNumber,
      step: (prev.step as number) + 1,
    }));
  };
 
  const handleNameAndEmail = (name: ApplyValues['memberName'], email: ApplyValues['password']) => {
    setApplyValues((prev) => ({
      ...prev,
      memberName: name,
      email: email,
      step: (prev.step as number) + 1,
    }));
  };
 
  const handleRemainData = (
    password: ApplyValues['password'],
    job: ApplyValues['memberJob'],
    smsAgree: ApplyValues['memberSmsAgree']
  ) => {
    setApplyValues((prev) => ({
      ...prev,
      password: password,
      memberJob: job,
      memberSmsAgree: smsAgree,
      step: (prev.step as number) + 1,
    }));
  };
 
  useEffect(() => {
    if (applyValues.step === 3) {
      signUpReq({
        email: applyValues.email as string,
        password: applyValues.password as string,
        memberName: applyValues.memberName as string,
        memberJob: applyValues.memberJob as string,
        memberPhone: applyValues.memberPhone as string,
        memberSmsAgree: applyValues.memberSmsAgree as boolean,
      });
    }
  }, [applyValues, signUpReq]);
 
  return (
    <MainContainer>
      {applyValues.step === 0 ? <PhoneCertification onNext={handlePhoneNumber} /> : null}
      {applyValues.step === 1 ? <EmailVerification onNext={handleNameAndEmail} /> : null}
      {applyValues.step === 2 ? (
        <PasswordVerification onNext={handleRemainData} applyValues={applyValues} />
      ) : null}
      {applyValues.step === 4 ? <SignupDone /> : null}
    </MainContainer>
  );
};
 
export default SignUpPage;

여기서 또 중요한점은 applyValues를 객체 형식으로 관리한다는 점이다


const [step, setStep] = useState(0);
const [memberPhone, setMemberPhone] = useState('');
const [memberName, setMemberName] = useState('');
const [email, setEmail] = useState('');
const [password, setPassword] = useState('');
const [memberJob, setMemberJob] = useState('');
const [memberSmsAgree, setMemberSmsAgree] = useState(false);
 
// 각 상태를 업데이트할 때마다 여러 set 함수 호출 필요
setMemberPhone(phone);
setStep(step + 1);
const [applyValues, setApplyValues] = useState < Partial < ApplyValues >> { step: 0 };
 
// 한 번의 set 함수 호출로 여러 값을 업데이트 가능
setApplyValues((prev) => ({
  ...prev,
  memberPhone: phone,
  step: prev.step + 1,
}));

❗ 상태가 복잡해질수록, 여러 개의 useState 호출을 사용하는 것보다 하나의 객체로 상태를 관리하는 것이 더 효율적이고, 훨씬 체계적으로 상태를 관리할 수 있다. 또한 여러 상태 값을 개별적으로 관리하면, 서로 관련된 상태 값들 간의 동기화 문제가 발생할 수 있다. 하나의 객체로 상태를 관리하면, 한 번의 setApplyValues 호출로 여러 값을 동시에 업데이트할 수 있어 일관성을 유지하기가 쉬워진다.


export interface ISignIn {
  email: string;
  password: string;
}
 
export interface ISignUp extends ISignIn {
  memberName: string;
  memberJob: string;
  memberPhone: string;
  memberSmsAgree: boolean;
}
 
export interface ApplyValues extends ISignUp {
  step: number;
}

또한 회원가입 시 필수로 받아야 하는 모든 속성이 필수로 지정되어 있는데, Partial<ApplyValues>를 사용하여 모든 속성을 선택적으로 가지는 타입이 된다. 따라서, 초기 상태를 설정할 때 모든 속성을 제공하지 않아도 TypeScript에서 오류가 발생하지 않는다. 즉 step 속성만 초기화하고 나머지 속성들은 생략할 수 있다. 이는 각 단계별로 필요한 속성만 점진적으로 추가하고 관리하기 편리하게 만들어준다.

결과적으로. 이 페이지에서는 여러 단계의 폼을 통해 사용자의 정보를 수집하고 최종적으로 회원가입 요청을 서버에 보낸다. 각 단계는 별도의 폼 컴포넌트로 구성되어 있으며, 단계가 진행됨에 따라 applyValues 상태를 업데이트된다.


지치지 않는 UI

해당 코드 구조는 단계별로 사용자 정보를 수집하는 데 효율적이지만, 사용자 경험 측면에서 몇 가지 문제를 일으킬 수 있다. 특히, 사용자에게 현재 단계(step)를 명확히 전달하지 않으면, 사용자 입장에서 지루하거나 혼란스러울 수 있을것이다.

문제 상황 시나리오

앞서 예시에서의 MBTI 페이지에서는 진행상황을 시각적으로 표시하여 유저가 현재 어느정도 참고 끝까지 설문을 완료 할 수 있도록 해주었다.


❗ 따라서 UI/UX 팀에게 해당 문제를 제기하고, 회원가입 중 유저가 어느정도 진행됐는지 확인 할 수 있는 디자인을 요청했고, 프로젝트에 적용 시켜 혹시 모를 회원가입 유저 이탈을 방지 할 수 있도록 했다.

개발 할 때 기술적 완성도와 효율성만을 고려하는 경우가 종종 있다. 그러나 이번 회원가입 페이지 개선 작업을 통해, 사용자 입장을 먼저 생각해보며 혹시 모를 문제를 미리 방지한 경험이었다.

me
@banhogu
안녕하세요 배움을 나누며 함께 전진하는 1년차 주니어 개발자 방호진입니다.