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

[강대원] sprint9 #129

Merged

Conversation

Daewony
Copy link
Collaborator

@Daewony Daewony commented Dec 22, 2024

기본 요구사항

공통

로그인/회원가입 페이지

로그인 페이지

  • "회원 가입하기"를 클릭하면 회원가입 페이지로 이동해 주세요.
  • 로그인 실패 시, 이메일 input 아래에 "이메일을 확인해 주세요.", 비밀번호 input 아래에 "비밀번호를 확인해 주세요." 에러 메시지를 표시해 주세요.
  • 로그인 버튼이 활성화된 후, 로그인 버튼 클릭 또는 Enter키 입력으로 로그인 실행합니다.
  • "/auth/signIn"으로 POST 요청해서 성공 응답을 받으면 중고 마켓 페이지로 이동합니다. (JWT 사용)
  • 실패할 경우, 실패 메시지를 모달을 통해 표시합니다.

회원가입 페이지

  • "회원 가입하기"를 클릭하면 '/signin' 페이지로 이동합니다.
  • 회원가입 버튼 클릭 또는 Enter키 입력으로 회원가입을 실행합니다.
  • 비밀번호 input과 비밀번호 확인 input의 값이 다른 경우, 비밀번호 확인 input 아래에 "비밀번호가 일치하지 않아요." 에러 메시지를 표시해 주세요.
  • 버튼이 활성화된 후, 회원가입은 "/auth/signUp" POST 요청해서 진행합니다. (JWT 사용)
  • 회원가입 성공 응답을 받으면 중고마켓 페이지로 이동합니다.
  • 실패할 경우, 실패 메시지를 모달을 통해 표시합니다.

로그인, 회원가입 페이지 공통

  • 눈 모양 아이콘 클릭 시 비밀번호의 문자열이 보이기도 하고, 가려집니다.
  • 비밀번호의 문자열이 가려질 때는 눈 모양 아이콘에 사선이 그어져 있고, 비밀번호의 문자열이 보일 때는 사선이 없는 눈 모양 아이콘이 보입니다.
  • 소셜 로그인에 구글 아이콘 클릭 시 'https://www.google.com', 카카오 아이콘 클릭 시 'https://www.kakaocorp.com/page'로 이동합니다.
  • 로그인/회원가입 시 성공 응답으로 받은 accessToken을 로컬 스토리지에 저장합니다.
  • 로그인/회원가입 페이지에 접근 시 로컬 스토리지에 accessToken이 있는 경우 '/items' 페이지로 이동합니다.

GNB (상단 내비게이션 바)

  • 프로필 영역은 인가된 경우, 유저 정보 API를 활용해 주세요.
  • 인가되지 않았을 경우 "로그인" 버튼이 보이게 해 주세요.

상품 상세 페이지

  • PC, Tablet, Mobile 디자인에 해당하는 상품 상세 페이지를 만들어 주세요.
  • 상품 상세 페이지 URL path는 "/items/{itemId}"로 설정하세요.
  • '목록으로 돌아가기' 버튼 클릭 시 중고마켓 페이지 "/items"로 이동합니다.
  • 상품 상세 데이터는 '/products/{productId}' GET 메서드 사용해 불러오세요. (인가된 사용자만 이용 가능)
  • 상품에 대한 댓글 조회도 가능합니다.
  • 상품 수정 및 삭제 기능을 API를 활용해 구현합니다. (인가된 사용자만 이용 가능)
  • 상품 수정은 '/products/{productId}' PATCH을 사용합니다.
  • 상품 삭제는 '/products/{productId}' DELETE를 사용합니다.
  • 상품 삭제 전, 확인 모달을 띄워주세요.
  • 상품에 대한 좋아요 및 좋아요 취소 기능을 '/products/{productId}/favorite' POST & DELETE 활용해 구현합니다. (인가된 사용자만 이용 가능)
  • 댓글 생성 및 삭제 기능을 API를 활용해 구현합니다. (인가된 사용자만 이용 가능)
  • 댓글 수정은 '/comments/{commentId}' PATCH을 사용합니다.
  • 댓글 삭제는 '/comments/{commentId}' DELETE를 사용합니다.

심화 요구사항

로그인 및 회원가입 페이지 공통

  • 로그인, 회원가입 기능에 react-hook-form을 활용해 주세요.
  • 브라우저에 현재 보이는 화면의 영역(viewport) 너비를 기준으로 분기되는 반응형 디자인을 적용합니다.
    • PC: 1200px 이상
    • Tablet: 744px 이상 ~ 1199px 이하
    • Mobile: 375px 이상 ~ 743px 이하
    • 375px 미만 사이즈의 디자인은 고려하지 않습니다.

유저 기능

  • 리퀘스트 헤더에 인증 토큰을 첨부할 때 axios interceptors를 활용해 주세요. (axios를 사용하지 않는다면 이와 유사한 기능을 활용해 주세요.)

React-Query로 마이그레이션

  • fetch 혹은 axios로 구현된 기존의 API 요청 코드를 React-Query로 마이그레이션 합니다.

로딩 및 에러 핸들링

  • 로딩 인디케이터와 에러 메시지를 구현합니다.
  • 상품 목록 및 상품 상세 데이터를 Prefetching 합니다.

상품 데이터 캐싱 및 업데이트

  • React Query의 캐싱 기능을 활용하여 데이터 로딩 시간을 최소화합니다.
  • 상품 목록 페이지에서 데이터의 실시간 업데이트를 위해 적절한 Query Refresh 설정을 적용합니다.

@Daewony Daewony requested a review from pers0n4 December 22, 2024 15:01
@Daewony Daewony self-assigned this Dec 22, 2024
@Daewony Daewony added 매운맛🔥 뒤는 없습니다. 그냥 필터 없이 말해주세요. 책임은 제가 집니다. 미완성🫠 완성은 아니지만 제출합니다... 진행 중 🏃 스프린트 미션 진행중입니다. labels Dec 22, 2024
@Daewony Daewony added 최종 제출 스프린트미션 최종 제출본입니다. and removed 미완성🫠 완성은 아니지만 제출합니다... 진행 중 🏃 스프린트 미션 진행중입니다. labels Dec 24, 2024
Comment on lines +1 to +2
import image from "@/public/images/img_default.png";
export const DEFAULT_IMAGE = image;
Copy link
Collaborator

Choose a reason for hiding this comment

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

이런 케이스는 import한 이미지를 그대로 export 시켜버리기보다는 default 이미지의 경로를 상수로 정의해서 사용하는 케이스가 조금 더 메이지한 접근 방법인 것 같네요.

Comment on lines +76 to +80
<ul>
{articles?.list?.map((article) => (
<AllArticleCard key={article.id} article={article} />
))}
</ul>
Copy link
Collaborator

Choose a reason for hiding this comment

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

이 부분 전체를 wrapping하는 컴포넌트가 하나 있어도 좋을 것 같네요.

Comment on lines +22 to +25
staleTime: 1000 * 60 * 5,
gcTime: 1000 * 60 * 30,
refetchOnWindowFocus: false,
refetchInterval: 1000 * 60 * 10,
Copy link
Collaborator

Choose a reason for hiding this comment

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

하나 정도는 괜찮은데 이렇게 단위가 흩뿌려져 있으니 보기가 어려워지는군요. ㅋㅋ
이런 케이스는 미리 unit에 따라 생수를 정의해두고 필요에 따라 로드해서 사용하면 어떨까 싶네요.

e.g.

const SECONDS = 1000;
const MINUTES = 60 * SECONDS;

Comment on lines +13 to +17
type ProductFormProps = {
defaultValues?: ProductCreateRequest;
productId?: number;
isEdit?: boolean;
};
Copy link
Collaborator

Choose a reason for hiding this comment

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

컴포넌트의 모든 props가 nullable이라는 건 props의 실효성에 대해 의심해봐야 할 신호이기도 합니다.
그리고 혹시 productId와 isEdit을 하나로 합치는 건 불가능할까요? productId의 존재 여부에 따라 isEdit 값이 종속적으로 결정되는 것이 아닌지?

Comment on lines +16 to +28
const mutation = useMutation({
mutationFn: (content: string) => createComment(Number(productId), content),
onSuccess: () => {
// 댓글 등록 성공 시, 해당 제품의 댓글 리스트를 다시 불러옴
queryClient.invalidateQueries({
queryKey: ["comments", String(productId)],
});
setCommentContent(""); // 입력 필드 초기화
},
onError: (error) => {
console.error("댓글 등록 실패:", error);
},
});
Copy link
Collaborator

Choose a reason for hiding this comment

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

코드를 보다 반복적으로 발견한 부분인데 id에 대해서 자꾸 불필요한 형 변환이 진행되고 있는 것 같습니다. 어떤 타입이든 하나의 타입으로 변환해서 고정하는 게 복잡성을 줄일 수 있을 것 같네요.
e.g. 서버에서 string 타입으로 반환 => 하위 컴포넌트에서도 전부 string으로만 사용 => 필요에 따라 한 번씩 캐스팅 (이런 경우는 거의 없을 것 같긴 합니다.)

});

const [dropdownStates, setDropdownStates] = useState<{
[key: string]: boolean;
Copy link
Collaborator

Choose a reason for hiding this comment

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

모든 타입이 그렇긴 하지만, 특히 이렇게 넓은 타입은 의미를 명확하게 전달하기 위해 alias를 해서 사용하면 좋습니다.

Comment on lines +16 to +25
export const getComments = async (
productId: number,
limit: number,
cursor: number,
) => {
const response = await api.get<CommentListResponse>(
`/products/${productId}/comments?limit=${limit}&cursor=${cursor}`,
);
return response.data;
};
Copy link
Collaborator

Choose a reason for hiding this comment

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

  1. axios로 요청을 보내는 부분은 params로 처리하면 좋을 것 같네요.
  2. 직접 정의한 getComments에서 limit과 cursor parameter를 받을 때도 positional하게 필수로 받는 것보다는 유연함을 위해 객체로 묶어서 처리하는 방식이 좋지 않을까 싶네요.

<div className="mb-[90px] max-w-[400px]">
<h1
className="mb-[4rem] max-w-[35rem] whitespace-normal break-keep text-center text-[4rem] font-bold leading-[5.6rem] sm:max-w-[24rem] sm:text-center sm:text-[3.2rem] sm:leading-[4rem] md:max-w-full"
dangerouslySetInnerHTML={{ __html: title }}
Copy link
Collaborator

Choose a reason for hiding this comment

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

보통 이 속성은 사용할 일이 거의 없기도 하지만, 꼭 필요한 사용하지 않는 것이 좋습니다. 🤔

{paginationArr.map((value) => (
<div
key={value}
className={`flex h-10 w-10 cursor-pointer items-center justify-center rounded-full border ${value === pageNo ? "border-[#E5E7EB] bg-[#2F80ED] text-[#F9FAFB]" : "border-[#E5E7EB] text-[#6B7280]"}`}
Copy link
Collaborator

Choose a reason for hiding this comment

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

컬러 코드를 직접 사용하는 패턴은 특히나 변경에 취약하기 때문에 앞으로 계속 사용될 것 같은 색상이라면 이름을 정의해두고 사용하는 걸 권장드립니다.

Comment on lines +19 to 50
type PostAndCommentActionsDropdownProps = {
id?: string | number; // 수정 시 필요한 id 값
basePath?: "/items" | "/articles"; // 경로 변경을 위한 추가 prop
onDelete: () => void;
onEdit?: () => void; // 댓글 수정 시 필요한 함수
type: "post" | "comment";
};

const PostAndCommentActionsDropdown = ({
id,
basePath,
onDelete,
onEdit,
type,
}: PostAndCommentActionsDropdownProps) => {
const router = useRouter();

const onClick = (action: string): void => {
switch (action) {
case "edit":
console.log("수정하기");
if (type === "post" && basePath && id) {
router.push(`${basePath}/${id}/edit`); // 수정 페이지로 이동
} else if (type === "comment" && onEdit) {
onEdit();
}

break;
case "delete":
console.log("삭제하기");
onDelete();
break;
default:
break;
Copy link
Collaborator

Choose a reason for hiding this comment

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

이름에서부터 And가 등장한다는 건 좋은 징조가 아닙니다. 액션을 보면 타입을 onClick 액션의 플래그로 사용하고 있는데, 이 경우라면 type을 받는 대신 action 자체를 외부에서 주입받고 Dropdown 컴포넌트를 type으로부터 독립시키는 게 더 좋을 것 같네요.

@pers0n4 pers0n4 merged commit 0e84453 into codeit-sprint-fullstack:next-강대원 Jan 2, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
매운맛🔥 뒤는 없습니다. 그냥 필터 없이 말해주세요. 책임은 제가 집니다. 최종 제출 스프린트미션 최종 제출본입니다.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants