-
Notifications
You must be signed in to change notification settings - Fork 2
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
[FE] feat: NavigationTab 컴포넌트 구현 #1037
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
쑤쑤 구현한 것 잘 봤어요.
쿼리 스트링 구현
쿼리 스트링 값으로 탭을 구현한다는 아이디어 좋은 것 같아요
이거를 리뷰미에 적용하려면, 회원 페이지는 아래처럼 쿼리 스트리밍 같에 따라서 다른 children 을 넣어줘야겠네요.
<>
<NavigationTab
tabInfoList={[
{ name: '리뷰 링크 관리', path: `/test`, param: 'manage-links' },
{ name: '작성한 리뷰 확인', path: `/test`, param: 'written-reviews' },
]}
/>
{children}
</>
스타일 컴포넌트 Props, 타입과 탭 하단 바 애니메이션 관련 코멘트 남겼으니 확인 부탁드려요
const queryParams = new URLSearchParams(location.search); | ||
const tabParam = queryParams.get('tab'); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
useSearchParamAndQuery 훅 이용할 수 있어요
background-color: ${({ theme }) => theme.colors.primary}; | ||
border-radius: 0.1rem; | ||
|
||
transition: ${({ isTransitionEnabled }) => (isTransitionEnabled ? 'all 0.2s ease-in-out' : 'none')}; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
목록, 모아보기 페이지 OptionSwitch
처럼 각 페이지마다 NavigationTab
을 추가한다면, 애니메이션 로직이 있어도... 보이지 않아서 코드를 없앨 예정이에요:)
하단바 오차는 자식 요소의 left 값을 그대로 사용하고 있어서 생긴 오차같아요. 자식 요소의 left 값에서 부모 요소의 left 값을 빼면 오차없이 제대로 계산이 돼요.
const currentTab = currentItemRef.current.children[currentIndex];
const parentLeft = currentItemRef.current.getBoundingClientRect().left;
const currentLeft = currentTab.getBoundingClientRect().left - parentLeft;
추가로! 하단바를 넣은 이유가 애니메이션을 넣고 싶어서..였기 때문에😅 애니메이션이 없어진다면 하단바도 없어지고 버튼의 border-bottom으로 할 것 같아요!
<S.NavContainer> | ||
<S.NavList ref={currentItemRef}> | ||
{tabInfoList.map((item, index) => ( | ||
<S.NavItem key={index} selected={currentIndex === index}> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
key에 index를 사용하지 않는게 좋아서 다른 방법으로 key를 구별했으면 좋겠어요
@@ -0,0 +1,59 @@ | |||
import styled from '@emotion/styled'; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
스타일 컴포넌트에 props들 에 모두 $ 표시 없네요.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
추가하겠습니다....😅
} | ||
`; | ||
|
||
export const CurrentNavBar = styled.div<{ width: number; left: number; isTransitionEnabled: boolean }>` |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
스타일 컴포넌트 Props 가 두 개 이상일 경우 따로 타입으로 빼주세요
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
페이지 구조에서 탭의 위치, 선택된 탭에 대한 상태 관리 방식에 대한 코멘트 남겼어요~
전자는 디코에 썼는데 혹시 몰라 여기에 한번 더 남겨요!
(제가 이해한 게 맞다면) 링크 관리 페이지에서 작성한 리뷰 확인 페이지를 갖고 있는 건데, 두 페이지는 상하관계가 아니라 형제관계라 Collection 페이지와 비슷한 구조로 가야할 것 같아요~
쑤쑤도 새해 복 많이 받아요 😆😆
const NavigationTab = ({ tabInfoList }: NavigationTabProps) => { | ||
const activeTab = sessionStorage.getItem('activeTab'); | ||
|
||
const [currentIndex, setCurrentIndex] = useState(Number(activeTab) || 0); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
탭에서 선택 상태를 관리하고, 또 그 관리 방법이 useState인 이유가 있는지 궁금해요.
(아마 애니메이션 추가 + 페이지를 각자 다른 사람이 만들고, 탭 위치에 대해서 제대로 얘기한 적이 없어서 해석에 차이가 나는 것 같아요!)
현재 OptionSwtich의 구현을 보면 현재 선택된 옵션 상태가 컴포넌트 외부(훅)로 분리되어 있고, useState가 아닌 단순 js 변수를 사용해요. 그리고 선택된 요소를 url을 통해 판단하고 있어요.
const { pathname } = useLocation();
const { param: reviewRequestCode } = useSearchParamAndQuery({
paramKey: 'reviewRequestCode',
});
// 여기서는 includes만 1번 사용하고 있어서 확장성이 조금 떨어지지만
// 기존 breadcrumb과 유사한 방식으로 변경하면 괜찮을 것 같아요
const isReviewCollection = pathname.includes(ROUTE.reviewCollection);
그리고 이 상태를 이용해 선택 상태를 제어하기 때문에 새로고침이 일어나도 기존 페이지를 불러올 수 있어요.
const reviewDisplayLayoutOptions: OptionSwitchOption[] = [
{
label: '목록보기',
isChecked: !isReviewCollection,
handleOptionClick: navigateReviewListPage,
},
...
뒤로가기 동작 또한 방문했던 url 기반이기 때문에 자연스럽게 동작합니다.
url 기반 상태관리와 useState 기반 관리 모두 각자 장단점이 있지만, 지금 리뷰미에서는 url을 적극적으로 이용하고 있고(reviewRequestCode를 url에서 가져와 사용하는 등) 이 방식을 사용해도 중요한 정보가 url에 노출되지 않기 때문에, 또 탭 자체도 url과 연관이 깊기 때문에 url 방식이 더 낫다는 의견입니다~!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
(제가 이해한 게 맞다면) 링크 관리 페이지에서 작성한 리뷰 확인 페이지를 갖고 있는 건데, 두 페이지는 상하관계가 아니라 형제관계라 Collection 페이지와 비슷한 구조로 가야할 것 같아요~
제가 디코에서 말을 잘못 전달한 것 같아요..😭 디코에서는 제가 링크 관리 페이지
에 네비게이션 탭을 두고, 작성한 리뷰 확인
탭을 클릭하면 해당 페이지 안에서 작성한 리뷰 확인
페이지를 불러오는 방식이라고 설명해서 상하 구조로 오해가 생긴 것 같아요. 사실 제가 생각했던 방식은 링크 관리 페이지
와 작성한 리뷰 확인
페이지를 포함하는 부모 페이지를 만들어서, 네비게이션 탭은 부모 페이지에만 두고 탭을 클릭하면 네비게이션 탭은 그대로 유지되면서, 탭 아래의 컨텐츠가 전환되는 방식을 생각했어요. 그래서 부모 페이지를 만들어 하나의 탭만 넣을지, 두 개의 페이지에 각각 탭을 넣을지? 고민했었어요🤔
부모 페이지(/my-page)를 만들어서 리뷰 링크 관리 탭을 클릭하면 /my-page?tab=manage-links
로 이동하는 방식?
탭에서 선택 상태를 관리하고, 또 그 관리 방법이 useState인 이유가 있는지 궁금해요.
제가 원했던 탭 방식이 각 탭을 클릭하면 하단바가 애니메이션을 통해 부드럽게 이동하는 것이였어요. 그래서 하단바가 선택된 탭 상태에 따라 width와 left를 바로 계산해서 동적으로 변경해 줘야 하는데 새로고침이나 탭 간 이동 시, 하단바가 깜빡이거나 늦게 나타나는 문제가 있어서 useState와 세션 스토리지를 결합해서 사용했어요..😅
return ( | ||
<S.NavContainer> | ||
<S.NavList ref={currentItemRef}> | ||
{tabInfoList.map((item, index) => ( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
프로필 탭 리뷰에서도 남겼지만, 합성 컴포넌트 방식을 사용해서 map 대신 태그를 통해 어떤 요소들이 렌더링되는지 바로 알 수 있도록 만드는 방법은 어떨까요?? 링크
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
각 탭의 정보를 배열로 담아서 map으로 돌리는 방식이 아닌 명시적으로 하나하나 넣어주는 방식인가요?🤔
return (
<NavContainer>
<NavList>
<NavItem
label="리뷰 링크 관리"
$isSelected={currentTabIndex === index}
onClick={() => navigate('/user/review-link-management')}
/>
<NavItem
label="작성한 리뷰 확인"
$isSelected={currentTabIndex === index}
onClick={() => navigate('/user/written-review-confirm')}
/>
</NavList>
</NavContainer>
);
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
네 맞습니다!! 자식 요소 개수가 적고 정적인 경우에 사용하면 좋을 것 같아요. (옆동네에서 추천받은 방식입니다ㅋㅋ) 지금 리뷰미에서는 배열을 map으로 렌더하는 코드가 대다수지만요.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
쿼리 파라미터로 구현한 코드 잘 봤어요🔥
탭 hover 했을 때, 배경색이 아닌 글자색을 변경하는 게 깔끔해 보이는데, 여러분들은 어떻게 생각하시나요???🤔
실제로 구현된 모습을 비교해 보니 글자색만 변경해도 괜찮을 것 같네요!
쿼리 파라미터를 사용한 만큼, 세션 스토리지를 사용하지 않고도 활성화된 탭 상태를 유지할 수 있지 않을까요?
간단하게 코멘트 남겼어요. 확인 부탁해요!
const handleTabClick = (path: string, index: number) => { | ||
setCurrentIndex(index); | ||
setIsTransitionEnabled(true); | ||
navigate(`${path}?tab=${tabInfoList[index].param}`); | ||
}; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
현재 활성화된 탭을 클릭할 경우 불필요한 상태 업데이트가 발생할 것 같아요. index와 currentIndex가 다른 경우에만 업데이트하는 것은 어떨까요?
⭐️ 변경 사항 v0.0.1 ⭐️
|
const navigateReviewLinkManagementPage = () => { | ||
navigate('/user/review-link-management'); | ||
}; | ||
|
||
const navigateWrittenReviewConfirmPage = () => { | ||
navigate('/user/written-review-confirm'); | ||
}; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
아직 path가 확정이 아니라서, ROUTE 상수를 이용하지 않은 건가요?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
넵,,, 일단 임시로,, 넣어놨어요
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
경로 거의 확정된 것 같아서 추가했습니다~
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
기존 방식이 날아가서 아쉽지만 🥲 나중에 URL과 관계없거나 동적인 탭을 만들 때 유용하게 사용할 수 있을 거예요!!!
여러 방법으로 구현하느라 수고 많았어요 👍
{ | ||
label: '리뷰 링크 관리', | ||
path: '/user/review-link-management', | ||
handleTabClick: navigateReviewLinkManagementPage, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
tabList의 요소안에 path가 있으니, tabList 사용시 onClick={()=> navigate(tab.path)}로 해도 될 것 같은데, tabList에 handleTabClick으로 선언한 이유가 있을까요?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
저도 같은 의문이 있었는데, navigate 코드에 더해 페이지 방문 amplitude 코드를 추가한다고 생각하면 nav 함수를 따로 파도 괜찮을 것 같아요. 지금 useReviewDisplayLayoutOptions
도 이런 식으로 되어 있습니다!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
tabList의 요소안에 path가 있으니, tabList 사용시 onClick={()=> navigate(tab.path)}로 해도 될 것 같은데, tabList에 handleTabClick으로 선언한 이유가 있을까요?
amplitude 코드 추가될 것 같아서 함수 따로 팠어요~
지금 useReviewDisplayLayoutOptions도 이런 식으로 되어 있습니다!
맞아요. useReviewDisplayLayoutOptions
참고했어요!
⭐️ 변경 사항 v0.0.2 ⭐️
export const ROUTE = {
// ....
reviewLinks: 'user/review-links',
writtenReview: 'user/written-review',
};
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
route 상수 추가된거 확인했어요
충돌이 생겼네요. 확인 부탁해요!
리뷰 링크 페이지에 height이 없어서 이걸 감싸는 Contents에 center가 적용되면서 중앙에 배치되고 있었네요. 리뷰 링크 페이지에 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
수정 사항 확인했어요! UI 정말 만족스럽네요 고생했어요👏
🚀 어떤 기능을 구현했나요 ?
리뷰 링크 관리
와작성한 리뷰 확인
페이지를 탭으로 제공하여 쉽게 전환할 수 있도록 네비게이션 탭을 구현했습니다.🔥 어떻게 해결했나요 ?
일단, 네비게이션 탭을 구현하면서 집중적으로 고려한 부분은 아래와 같습니다.
새로고침 시, 탭 유지
useState
로 탭 상태를 관리할 경우 새로고침 시 상태가 초기화되어 기본값인 0번 인덱스 탭으로 돌아가게 됩니다. 그래서 활성화된 탭의 인덱스를 세션 스토리지에 저장하고, 새로고침 후에도 저장된 탭 인덱스를 불러와 이전에 선택한 탭을 유지하도록 했습니다.브라우저 뒤로가기, 앞으로 가기 클릭 시, 해당 탭으로 이동
리뷰 링크 관리
페이지에서는 쿼리 파라미터를tab=manage-links
로 설정하고,location.search
로 URL의 쿼리 값을 확인합니다. 그 후, 해당 값에 맞는 탭 인덱스를currentIndex
로 업데이트해서 브라우저의 뒤로가기나 앞으로 가기 버튼을 클릭했을 때 이전에 선택한 탭으로 자동으로 이동하도록 구현했습니다.모바일 환경에서는 탭의 길이에 따라 각 탭이 같은 영역을 차지하도록 반응형 로직을 추가했습니다.
📝 어떤 부분에 집중해서 리뷰해야 할까요?
탭 hover 했을 때, 배경색이 아닌 글자색을 변경하는 게 깔끔해 보이는데, 여러분들은 어떻게 생각하시나요???🤔
📚 참고 자료, 할 말
To. 리뷰미 여러분🍀
2025년 새해가 밝았네요. 새해 복 많이 받으세요~ 두 배!! 세 배!!! 네 배!!!! 더더더더 받으세요😊
늘 행복하시고, 원하는 거 다 이루시길 바랍니다. 날씨가 쌀쌀하니 감기 조심하세요~