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

[Refactor] ApplyPage 로직 분리 - 1 #441

Merged
merged 11 commits into from
Sep 9, 2024
11 changes: 6 additions & 5 deletions src/App.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import { sessionReplayPlugin } from '@amplitude/plugin-session-replay-browser';
import { MutationCache, QueryCache, QueryClient, QueryClientProvider } from '@tanstack/react-query';
import { ReactQueryDevtools } from '@tanstack/react-query-devtools';
import { AxiosError } from 'axios';
import { lazy, Suspense, useEffect, useRef, useState } from 'react';
import { lazy, Suspense, useEffect, useState } from 'react';
import { RouterProvider, createBrowserRouter } from 'react-router-dom';

import Layout from '@components/Layout';
Expand All @@ -14,6 +14,7 @@ import { dark, light } from 'styles/theme.css';
import BigLoading from 'views/loadings/BigLoding';

import 'styles/reset.css';
import useDialog from '@hooks/useDialog';

const SessionExpiredDialog = lazy(() =>
import('views/dialogs').then(({ SessionExpiredDialog }) => ({ default: SessionExpiredDialog })),
Expand Down Expand Up @@ -55,7 +56,7 @@ const App = () => {
// }
// }, []);

const sessionRef = useRef<HTMLDialogElement>(null);
const { ref: sessionExpiredDialogRef, handleShowDialog: handleShowSessionExpiredDialog } = useDialog();
const [isAmplitudeInitialized, setIsAmplitudeInitialized] = useState(false);
const { isLight } = useTheme();

Expand All @@ -74,7 +75,7 @@ const App = () => {
const axiosError = error as AxiosError;

if (axiosError.response?.status === 401) {
sessionRef.current?.showModal();
handleShowSessionExpiredDialog();
} else if (axiosError.response?.status === 500) {
window.location.href = '/error';
}
Expand All @@ -85,7 +86,7 @@ const App = () => {
const axiosError = error as AxiosError;

if (axiosError.response?.status === 401) {
sessionRef.current?.showModal();
handleShowSessionExpiredDialog();
} else if (axiosError.response?.status === 500) {
window.location.href = '/error';
}
Expand All @@ -110,7 +111,7 @@ const App = () => {

return (
<>
<SessionExpiredDialog ref={sessionRef} />
<SessionExpiredDialog ref={sessionExpiredDialogRef} />
<ThemeProvider>
<DeviceTypeProvider>
<RecruitingInfoProvider>
Expand Down
17 changes: 17 additions & 0 deletions src/common/hooks/useDialog.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
import { useRef } from 'react';

const useDialog = () => {
const ref = useRef<HTMLDialogElement>(null);

const handleShowDialog = () => {
ref.current?.showModal();
};

const handleCloseDialog = () => {
ref.current?.close();
};
Comment on lines +6 to +12
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

저는 항상 커스텀 훅에서 반환하는 handler에게는 useCallback을 붙혀주는 편인데, 해당 커스텀 훅을 사용하는 컴포넌트 쪽에서 최적화할 수 있는 여지를 줄 수 있기 때문입니다 !

커스텀 훅에서 반환하는 핸들러를 의존성에 주입하더라도, useCallback을 통해 메모제이션해주었기 때문에 불필요한 렌더링을 방지할 수 있다는 장점이 있을 것 같습니다 !

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

오홍 좋은 거 같아요~

근데 useCallback을 이용하는 것도 매번 이전과 달라진 게 있는지 비교를 해줘야 해서 리소스가 발생되는 작업인데 붙여주는게 항상 이득인가요??

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

음 ... 사실 저도 항상! 이라는 말에는 자신있게 대답할 수 있지는 못하는 것 같습니다. 언석님 말씀대로 메모이제이션 또한 비용이니까요 !

하지만 리액트 공식문서에서도 useCallback을 커스텀 훅에 적용하여 의존성 비교 시 렌더링 최적화를 권장하는 것처럼, useDialog와 같은 자주 사용되는 커먼 훅에 적용한다면 이점이 더 많을 것 같습니다 !

Copy link
Member

@lydiacho lydiacho Sep 6, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

제 갠적인 의견을 남겨보자면, , ,
useCallback 이 친구가 효과를 보여줄 수 있는 경우는 복잡한 연산이 반복적으로 일어날때 -> 근데 이런 케이스는 사실 많이 없고, 대부분 주용님이 말씀하신 것처럼 의존성에 들어갈때 라고 생각하는데요

그렇다면 만들어주는 모든 공통 훅에 useCallback을 적용해주기보다, 디폴트로 일반적인 함수로 작성하구
이후 핸들러를 의존성 배열에 주입시키게 될 경우 그때 추가적으로 useCallback으로 감싸주는 작업을 해주는게 좋지 않을까유??
이러면 불필요한 비용이 안생길 것 같아요

특히나 지금 같이 dialog를 껐/켰하는 핸들러들은
사용자 액션으로 인해 발생하는 이벤트시 실행되는 애들이기 때문에 더더욱 useEffect에서 사용될 일 없는, 즉 의존성 배열에 들어갈 일도 (아마도)없을 친구들이라고 생각해요!!

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

음 ..
사실 특정 로직을 커스텀 훅으로 분리한다, 메모이제이션을 적용한다, 등등 다양한 부분에서 "일종의 배팅" ? 이 존재한다고 생각해요.

예를 들어서, 우리가 스크롤을 맨 위로 보내는 로직이 중복된다고 생각해서 useScrollToTop이라는 커스텀 훅을 만들었다고 가정해볼게요. 근데 이 커스텀 훅을 하나의 파일로 분리하고 보니 딱 한번만 사용이 되는거에요. 그렇다면 결국 리소스 낭비라고 생각이 들 수도 있어요. 즉 우리가 "중복된 로직을 하나의 훅으로 분리"한 이유는 이게 다른 곳들에서 많이 사용되면서 리소스를 줄여줄 수 있도록 "배팅"을 한다는 말이죠.

이 메모이제이션 또한 비슷하다고 생각합니다. 결국엔 다른 파일에서 최적화할 수 있다는 여지를 제공하면서 일종의 "배팅"을 하는 거라고 생각합니다. 저는 이 "여지"가 중요한 것 같아요. 리액트 공식문서에서도 말하듯이 결국엔 다른 컴포넌트들이 최적화할 수 있는 가능성을 열어주는 것이니까요. 만약 나중에 리렌더링이 되는 이유를 분석하여 결국에는 해당 커스텀 훅을 찾아가서 변경하는 리소스보다 분명 더 낮은 리소스를 투자하여 미래를 대비할 수 있다는 생각입니다.

승희님이 말씀해주신 것처럼, 단순 dialog를 껐다 켜는 핸들러도 마찬가지에요. 단순한 핸들러일수록, 변경되는 일이 없고, 따라서 메모이제이션을 적용해주더라도 최적화 입장에서 이점을 볼 수 있으면 있지 이로 인한 손실이 많다고는 못느끼겠어요. 의존성이 없으면 없을수록 !

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

두 분 의견 다 이해했습니다!

사실 제가 useCallback에 대해 완전 깊게 이해하고 있지 못해서 이렇다 저렇다 할 수가 없는 거 같아요
그런 상황에서 바로 코드를 바꾸기 보다는 좀 더 공부해서 이해한 다음 그에 맞춰 코드를 수정하는 게 더 좋을 거 같단 생각이 들었어요!

이 부분에 대해서는 공부 완료한 후 자료랑 다 첨부해서 다시 제 의견 말씀드릴게요!!

Copy link
Member Author

@eonseok-jeon eonseok-jeon Sep 9, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@lydiacho @wuzoo

이것 저것 하다 보니 오래 걸렸네요 😓

제 의견 먼저 말씀드리면 useCallback을 사용할 필요는 없어 보입니다!

useCallback는 복잡한 함수의 경우 불필요한 함수 생성과 하위 컴포넌트의 재렌더링을 막기 위해서 사용이 되는데요

  1. useDialog 함수들이 복잡한 기능을 가지는가? -> x
  2. useDialog의 handler 함수를 prop으로 가지는 자식 컴포넌트들이 최적화가 필요할 만큼 무거운가? -> x

둘 다 해당하지 않아 사용하지 않는 것으로 결정했어요

실제로도 useCallback을 적용하고 안 하고 차이를 비교했을 때, 적용했을 때가 0.2ms 정도 빨리 렌더링 되는 것으로 확인이 되었는데요
이는 0.0002초로 정말 미비한 차이이므로 useCallback을 사용한다 해도 사실 유의미하지는 않음을 확인할 수 있었어요!

useCallback 적용 전
스크린샷 2024-09-09 오후 6 11 56

useCallback 적용 후
스크린샷 2024-09-09 오후 6 12 59

useCallback – React
javascript - useCallback vs simple function - Stack Overflow
javascript - What is useCallback in React and when to use it? - Stack Overflow

물론 주용님 말씀대로 위험을 예방하자는 건 너무 좋은 거 같아요!! 저 역시도 확장에 유연한 개발을 지향하거든요 ㅎㅎ 하지만 현재 dialog와 관련해서 추가 기능이 개발될 가능성은 조금 낮을 것으로 예상이 돼요 애초에 지원서 자체에 추가 기능이 더 발생하지 않을 가능성이 높거든요! 그래서 앞으로도 useCallback을 추가함으로써 유의미한 결과를 나타나진 않을 것으로 예상됩니다!

추가로 제 의견을 덧붙이자면
필요없는 코드는 사실 존재하지 않는 게 가장 베스트라고 생각해요
유지보수할 때도,
나중에 저희 후임자가 와서 코드를 읽을 때도 말이죠
실제로 react 공식문서에서도 useCallback을 많이 사용하면 가독성이 떨어질 수 있다고 주의하고 있어요

그래서 추가하지 않기로 결정했습니다!!

이대로 가는 것에 괜찮으시다면 이모지 달아주세요~ 두 분 다 확인되면 머지할게요!
추가로 더 하실 말 있으시면 편하게 comment 남겨주세요 :)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

언석님 의견도 충분히 이해갑니다 ! 고생하셨습니다 ~ 🚀🚀

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

완전 동의합니다 ~~
구체적인 의견 제시 감사드려요 ㅎㅎ 🚀 🚀 🚀 🚀


return { ref, handleShowDialog, handleCloseDialog };
};

export default useDialog;
17 changes: 17 additions & 0 deletions src/views/ApplyPage/hooks/useBeforeExitPageAlert.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
import { useEffect } from 'react';

/** 페이지 이탈 시 alert 띄우기 */
const useBeforeExitPageAlert = () => {
useEffect(() => {
const handleBeforeUnload = (e: BeforeUnloadEvent) => {
e.preventDefault();
e.returnValue = ''; // Included for legacy support, e.g. Chrome/Edeg < 119;
};

window.addEventListener('beforeunload', handleBeforeUnload);

return () => window.removeEventListener('beforeunload', handleBeforeUnload);
}, []);
};
eonseok-jeon marked this conversation as resolved.
Show resolved Hide resolved

export default useBeforeExitPageAlert;
168 changes: 85 additions & 83 deletions src/views/ApplyPage/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@ import useCheckBrowser from '@hooks/useCheckBrowser';
import useDate from '@hooks/useDate';
import useScrollToHash from '@hooks/useScrollToHash';
import { useDeviceType } from 'contexts/DeviceTypeProvider';
import BigLoading from 'views/loadings/BigLoding';

import ApplyCategory from './components/ApplyCategory';
import ApplyHeader from './components/ApplyHeader';
Expand All @@ -19,58 +18,70 @@ import CommonSection from './components/CommonSection';
import DefaultSection from './components/DefaultSection';
import PartSection from './components/PartSection';
import { SELECT_OPTIONS } from './constant';
import useBeforeExitPageAlert from './hooks/useBeforeExitPageAlert';
import useGetDraft from './hooks/useGetDraft';
import useGetQuestions from './hooks/useGetQuestions';
import useMutateDraft from './hooks/useMutateDraft';
import useMutateSubmit from './hooks/useMutateSubmit';
import { buttonWrapper, container, formContainerVar } from './style.css';

import type { ApplyRequest } from './types';
import useDialog from '@hooks/useDialog';

const DraftDialog = lazy(() => import('views/dialogs').then(({ DraftDialog }) => ({ default: DraftDialog })));
const PreventApplyDialog = lazy(() =>
import('views/dialogs').then(({ PreventApplyDialog }) => ({ default: PreventApplyDialog })),
);
const SubmitDialog = lazy(() => import('views/dialogs').then(({ SubmitDialog }) => ({ default: SubmitDialog })));
const NoMore = lazy(() => import('views/ErrorPage/components/NoMore'));
const IS_REVIEW = false;
eonseok-jeon marked this conversation as resolved.
Show resolved Hide resolved

interface ApplyPageProps {
onSetComplete?: () => void;
}

const ApplyPage = ({ onSetComplete }: ApplyPageProps) => {
// 반응형 페이지
const { deviceType } = useDeviceType();
useCheckBrowser(); // 크롬 브라우저 권장 알럿

const draftDialog = useRef<HTMLDialogElement>(null);
const preventApplyDialog = useRef<HTMLDialogElement>(null);
const submitDialog = useRef<HTMLDialogElement>(null);
const sectionsRef = useRef<HTMLSelectElement[]>([]);
// 2. 모달 관련 ref
const { ref: draftDialogRef, handleShowDialog: handleShowDraftDialog } = useDialog();
const { ref: preventApplyDialogRef, handleShowDialog: handleShowPreventApplyDialog } = useDialog();
const {
ref: submitDialogRef,
handleShowDialog: handleShowSubmitDialog,
handleCloseDialog: handleCloseSubmitDialog,
} = useDialog();

// 3. category active 상태 관리
useScrollToHash(); // scrollTo 카테고리
const [isInView, setIsInView] = useState([true, false, false]);
const [sectionsUpdated, setSectionsUpdated] = useState(false);

const navigate = useNavigate();

const isReview = false;
const minIndex = isInView.findIndex((value) => value === true);

useScrollToHash(); // scrollTo 카테고리

const { NoMoreApply, isLoading, isMakers } = useDate();
const { draftData, draftIsLoading } = useGetDraft();
// 4. 데이터 불러오기
const { draftData } = useGetDraft();
const {
applicant: applicantDraft,
commonQuestions: commonQuestionsDraft,
partQuestions: partQuestionsDraft,
} = draftData?.data || {};

const { questionsData, questionsIsLoading } = useGetQuestions(applicantDraft);
const { questionsData } = useGetQuestions(applicantDraft);
const { commonQuestions, partQuestions, questionTypes } = questionsData?.data || {};

const { draftMutate, draftIsPending } = useMutateDraft({ onSuccess: () => draftDialog.current?.showModal() });
// 5. 임시저장된 part data 붙이기
useEffect(() => {
if (applicantDraft?.part) {
setValue('part', applicantDraft?.part);
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [applicantDraft]);
eonseok-jeon marked this conversation as resolved.
Show resolved Hide resolved

// 6. 데이터 보내기
const { draftMutate, draftIsPending } = useMutateDraft({ onSuccess: () => handleShowDraftDialog });
eonseok-jeon marked this conversation as resolved.
Show resolved Hide resolved
const { submitMutate, submitIsPending } = useMutateSubmit({ onSuccess: onSetComplete });

// 7. react hook form method 생성
const methods = useForm({ mode: 'onBlur' });
const {
handleSubmit,
Expand Down Expand Up @@ -102,12 +113,9 @@ const ApplyPage = ({ onSetComplete }: ApplyPageProps) => {
...rest
} = getValues();

useEffect(() => {
if (applicantDraft?.part) {
setValue('part', applicantDraft?.part);
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [applicantDraft]);
// 8. intersection observer 연결
eonseok-jeon marked this conversation as resolved.
Show resolved Hide resolved
const sectionsRef = useRef<HTMLSelectElement[]>([]);
const [sectionsUpdated, setSectionsUpdated] = useState(false);

const refCallback = useCallback((element: HTMLSelectElement) => {
if (element) {
Expand Down Expand Up @@ -147,6 +155,9 @@ const ApplyPage = ({ onSetComplete }: ApplyPageProps) => {
};
}, [sectionsUpdated]);

// 9. 입력값 에러 관련 로직
const navigate = useNavigate();

useEffect(() => {
if (errors.picture) {
navigate('#default');
Expand Down Expand Up @@ -174,30 +185,21 @@ const ApplyPage = ({ onSetComplete }: ApplyPageProps) => {
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [errors.attendance, errors.personalInformation]);

useEffect(() => {
if (isReview) return;

const handleBeforeUnload = (e: BeforeUnloadEvent) => {
e.preventDefault();
e.returnValue = ''; // Included for legacy support, e.g. Chrome/Edeg < 119;
};

window.addEventListener('beforeunload', handleBeforeUnload);

return () => window.removeEventListener('beforeunload', handleBeforeUnload);
}, [isReview]);
useBeforeExitPageAlert();
eonseok-jeon marked this conversation as resolved.
Show resolved Hide resolved

if (questionsIsLoading || isLoading || draftIsLoading) return <BigLoading />;
// 11. 지원 기간 아니면 에러 페이지 띄우기
const { NoMoreApply, isMakers } = useDate();
if (NoMoreApply) return <NoMore isMakers={isMakers} content="모집 기간이 아니에요" />;

// 13. data 전송 로직
const selectedPartId = questionTypes?.find((type) => type.typeKr === getValues('part'))?.id;
const partQuestionsData = partQuestions?.find((part) => part.recruitingQuestionTypeId === selectedPartId);
const partQuestionIds = partQuestionsData?.questions.map((question) => question.id);
const commonQuestionIds = commonQuestions?.questions.map((question) => question.id);

const handleSendData = (type: 'draft' | 'submit') => {
if (NoMoreApply) {
preventApplyDialog.current?.showModal();
handleShowPreventApplyDialog();

return;
}
Expand Down Expand Up @@ -278,13 +280,13 @@ const ApplyPage = ({ onSetComplete }: ApplyPageProps) => {

const handleApplySubmit = () => {
track('click-apply-submit');
submitDialog.current?.showModal();
handleShowSubmitDialog();
};

return (
<FormProvider {...methods}>
<DraftDialog ref={draftDialog} />
<PreventApplyDialog ref={preventApplyDialog} />
<>
<DraftDialog ref={draftDialogRef} />
<PreventApplyDialog ref={preventApplyDialogRef} />
<SubmitDialog
userInfo={{
name,
Expand All @@ -293,48 +295,48 @@ const ApplyPage = ({ onSetComplete }: ApplyPageProps) => {
part,
}}
dataIsPending={submitIsPending}
ref={submitDialog}
ref={submitDialogRef}
onSendData={() => {
handleSendData('submit');
submitDialog.current?.close();
handleCloseSubmitDialog();
}}
/>
<div className={container}>
<ApplyHeader
isReview={isReview}
isLoading={draftIsPending || submitIsPending}
onSaveDraft={() => handleSendData('draft')}
onSubmitData={handleSubmit(handleApplySubmit)}
/>
<ApplyInfo isReview={isReview} />
<ApplyCategory isReview={isReview} minIndex={minIndex} />
<form
id="apply-form"
name="apply-form"
onSubmit={handleSubmit(handleApplySubmit)}
className={formContainerVar[deviceType]}>
<DefaultSection
isMakers={isMakers}
isReview={isReview}
refCallback={refCallback}
applicantDraft={applicantDraft}
/>
<CommonSection
isReview={isReview}
refCallback={refCallback}
questions={commonQuestions?.questions}
commonQuestionsDraft={commonQuestionsDraft}
/>
<PartSection
isReview={isReview}
refCallback={refCallback}
part={applicantDraft?.part}
questions={partQuestions}
partQuestionsDraft={partQuestionsDraft}
questionTypes={questionTypes}
<FormProvider {...methods}>
<div className={container}>
<ApplyHeader
isReview={IS_REVIEW}
isLoading={draftIsPending || submitIsPending}
onSaveDraft={() => handleSendData('draft')}
onSubmitData={handleSubmit(handleApplySubmit)}
/>
<BottomSection isReview={isReview} knownPath={applicantDraft?.knownPath} />
{!isReview && (
<ApplyInfo isReview={IS_REVIEW} />
<ApplyCategory isReview={IS_REVIEW} minIndex={minIndex} />
<form
id="apply-form"
name="apply-form"
onSubmit={handleSubmit(handleApplySubmit)}
className={formContainerVar[deviceType]}>
<DefaultSection
isMakers={isMakers}
isReview={IS_REVIEW}
refCallback={refCallback}
applicantDraft={applicantDraft}
/>
<CommonSection
isReview={IS_REVIEW}
refCallback={refCallback}
questions={commonQuestions?.questions}
commonQuestionsDraft={commonQuestionsDraft}
/>
<PartSection
isReview={IS_REVIEW}
refCallback={refCallback}
part={applicantDraft?.part}
questions={partQuestions}
partQuestionsDraft={partQuestionsDraft}
questionTypes={questionTypes}
/>
<BottomSection isReview={IS_REVIEW} knownPath={applicantDraft?.knownPath} />
<div className={buttonWrapper}>
<Button
isLoading={draftIsPending || submitIsPending}
Expand All @@ -346,11 +348,11 @@ const ApplyPage = ({ onSetComplete }: ApplyPageProps) => {
제출하기
</Button>
</div>
)}
</form>
</div>
<Footer />
</FormProvider>
</form>
</div>
<Footer />
</FormProvider>
</>
);
};

Expand Down
Loading