Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
30 commits
Select commit Hold shift + click to select a range
5e62bbd
feat: 기본 코드 추가
JunilHwang Sep 26, 2025
60fecb4
과제 제출을 위한 빈 커밋 만들기
jy0813 Nov 10, 2025
2c1d988
chore: prettier 및 절대경로 설정
jy0813 Nov 10, 2025
12046e4
chore: gh-pages 설치
jy0813 Nov 10, 2025
c1cdc92
feat: 기본 코드 추가
JunilHwang Sep 26, 2025
f30e838
chore: vite base 경로 및 serviceWorker url 옵션 추가
jy0813 Nov 10, 2025
ec5dbcb
chore: gh-pages github actions 설정 완료
jy0813 Nov 10, 2025
798c3c9
Merge remote-tracking branch 'upstream/main'
jy0813 Nov 10, 2025
fc3fa88
chore: prettier 정리
jy0813 Nov 10, 2025
2ffd0c8
feat: 상품 리스트 및 상품상세 컴포넌트 나누기 완료
jy0813 Nov 10, 2025
0c72e4e
fix; router 이동 경로 변경
jy0813 Nov 10, 2025
1327867
refactor: components 구조 변경
jy0813 Nov 11, 2025
39772da
feat: SearchForm 카테고리 loading 추가
jy0813 Nov 11, 2025
9628beb
feat: categories api 연결 및 limit 옵션 작업
jy0813 Nov 11, 2025
d3ed7e6
feat: 상품 limit 및 정렬 기능 완료
jy0813 Nov 11, 2025
a9e4f5f
feat: 무한 스크롤 구현
jy0813 Nov 11, 2025
f5d7f8e
feat: 상품을 장바구니에 담기 기능 완료
jy0813 Nov 12, 2025
351f04a
feat: 상품 검색 기능 완료
jy0813 Nov 12, 2025
3803d22
feat: 카테고리 선택 및 카테고리 네비게이션, 현재 상품 수 표시 완료
jy0813 Nov 12, 2025
ad33324
feat: 장바구니 관련 기능 전체 완료
jy0813 Nov 12, 2025
d13b068
feat: 상품로딩 중 에러 처리 완료
jy0813 Nov 12, 2025
be4bad0
feat: 상품상세 - 장바구니 담기 완료
jy0813 Nov 13, 2025
531cd80
feat: header 상품목록 상품상세 분기처리
jy0813 Nov 13, 2025
4d067ee
refactor: improt 경로 변경
jy0813 Nov 13, 2025
80ff023
feat: 관련 상품 기능 및 상품 목록으로 돌아가는 버튼 완료
jy0813 Nov 13, 2025
56f3863
feat: a 태그 pushState 이동으로 변경
jy0813 Nov 13, 2025
1a35794
feat: 상품상세 브레드크럼을 통해 카테고리별 상품 목록으로 이동할 수 있다 완료
jy0813 Nov 13, 2025
548039d
refactor: push 호출 코드 일관성 적용
jy0813 Nov 13, 2025
a3f2b29
fix: 브라우저 뒤로가기/앞으로가기가 올바르게 작동한다 테스트 해결
jy0813 Nov 13, 2025
d25a8d5
feat: 404 페이지 추가 및 href=/ 는 BASE_URL로 이동 추가
jy0813 Nov 13, 2025
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
269 changes: 207 additions & 62 deletions .github/pull_request_template.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,111 +4,256 @@

<!--
배포 링크를 적어주세요
예시: https://<username>.github.io/front-5th-chapter1-1/
예시: https://<username>.github.io/front-7th-chapter2-1/

배포가 완료되지 않으면 과제를 통과할 수 없습니다.
배포 후에 정상 작동하는지 확인해주세요.
-->


### 기본과제

#### 1) 라우팅 구현:
- [ ] History API를 사용하여 SPA 라우터 구현
- [ ] '/' (홈 페이지)
- [ ] '/login' (로그인 페이지)
- [ ] '/profile' (프로필 페이지)
- [ ] 각 라우트에 해당하는 컴포넌트 렌더링 함수 작성
- [ ] 네비게이션 이벤트 처리 (링크 클릭 시 페이지 전환)
- [ ] 주소가 변경되어도 새로고침이 발생하지 않아야 한다.

#### 2) 사용자 관리 기능:
- [ ] LocalStorage를 사용한 간단한 사용자 데이터 관리
- [ ] 사용자 정보 저장 (이름, 간단한 소개)
- [ ] 로그인 상태 관리 (로그인/로그아웃 토글)
- [ ] 로그인 폼 구현
- [ ] 사용자 이름 입력 및 검증
- [ ] 로그인 버튼 클릭 시 LocalStorage에 사용자 정보 저장
- [ ] 로그아웃 기능 구현
- [ ] 로그아웃 버튼 클릭 시 LocalStorage에서 사용자 정보 제거

#### 3) 프로필 페이지 구현:
- [ ] 현재 로그인한 사용자의 정보 표시
- [ ] 사용자 이름
- [ ] 간단한 소개
- [ ] 프로필 수정 기능
- [ ] 사용자 소개 텍스트 수정 가능
- [ ] 수정된 정보 LocalStorage에 저장

#### 4) 컴포넌트 기반 구조 설계:
- [ ] 재사용 가능한 컴포넌트 작성
- [ ] Header 컴포넌트
- [ ] Footer 컴포넌트
- [ ] 페이지별 컴포넌트 작성
- [ ] HomePage 컴포넌트
- [ ] ProfilePage 컴포넌트
- [ ] NotFoundPage 컴포넌트

#### 5) 상태 관리 초기 구현:
- [ ] 간단한 상태 관리 시스템 설계
- [ ] 전역 상태 객체 생성 (예: 현재 로그인한 사용자 정보)
- [ ] 상태 변경 함수 구현
- [ ] 상태 업데이트 시 관련 컴포넌트 리렌더링

#### 6) 이벤트 처리 및 DOM 조작:
- [ ] 사용자 입력 처리 (로그인 폼, 프로필 수정 등)
- [ ] 동적 컨텐츠 렌더링 (사용자 정보 표시, 페이지 전환 등)

#### 7) 라우팅 예외 처리:
- [ ] 잘못된 라우트 접근 시 404 페이지 표시
#### 상품목록

**상품 목록 로딩**

- [ ] 페이지 접속 시 로딩 상태가 표시된다
- [ ] 데이터 로드 완료 후 상품 목록이 렌더링된다
- [ ] 로딩 실패 시 에러 상태가 표시된다
- [ ] 에러 발생 시 재시도 버튼이 제공된다

**상품 목록 조회**

- [ ] 각 상품의 기본 정보(이미지, 상품명, 가격)가 카드 형태로 표시된다

**한 페이지에 보여질 상품 수 선택**

- [ ] 드롭다운에서 10, 20, 50, 100개 중 선택할 수 있으며 기본 값은 20개 이다.
- [ ] 선택 변경 시 즉시 목록에 반영된다

**상품 정렬 기능**

- [ ] 상품을 가격순/이름순으로 오름차순/내림차순 정렬을 할 수 있다.
- [ ] 드롭다운을 통해 정렬 기준을 선택할 수 있다
- [ ] 정렬 변경 시 즉시 목록에 반영된다

**무한 스크롤 페이지네이션**

- [ ] 페이지 하단 근처 도달 시 다음 페이지 데이터가 자동 로드된다
- [ ] 스크롤에 따라 계속해서 새로운 상품들이 목록에 추가된다
- [ ] 새 데이터 로드 중일 때 로딩 인디케이터와 스켈레톤 UI가 표시된다
- [ ] 홈 페이지에서만 무한 스크롤이 활성화된다

**상품을 장바구니에 담기**

- [ ] 각 상품에 장바구니 추가 버튼이 있다
- [ ] 버튼 클릭 시 해당 상품이 장바구니에 추가된다
- [ ] 추가 완료 시 사용자에게 알림이 표시된다

**상품 검색**

- [ ] 상품명 기반 검색을 위한 텍스트 입력 필드가 있다
- [ ] 검색 버튼 클릭으로 검색이 수행된다
- [ ] Enter 키로 검색이 수행된다
- [ ] 검색어와 일치하는 상품들만 목록에 표시된다

**카테고리 선택**

- [ ] 사용 가능한 카테고리들을 선택할 수 있는 UI가 제공된다
- [ ] 선택된 카테고리에 해당하는 상품들만 표시된다
- [ ] 전체 상품 보기로 돌아갈 수 있다
- [ ] 2단계 카테고리 구조를 지원한다 (1depth, 2depth)

**카테고리 네비게이션**

- [ ] 현재 선택된 카테고리 경로가 브레드크럼으로 표시된다
- [ ] 브레드크럼의 각 단계를 클릭하여 상위 카테고리로 이동할 수 있다
- [ ] "전체" > "1depth 카테고리" > "2depth 카테고리" 형태로 표시된다

**현재 상품 수 표시**

- [ ] 현재 조건에서 조회된 총 상품 수가 화면에 표시된다
- [ ] 검색이나 필터 적용 시 상품 수가 실시간으로 업데이트된다

#### 장바구니

**장바구니 모달**

- [ ] 장바구니 아이콘 클릭 시 모달 형태로 장바구니가 열린다
- [ ] X 버튼이나 배경 클릭으로 모달을 닫을 수 있다
- [ ] ESC 키로 모달을 닫을 수 있다
- [ ] 모달에서 장바구니의 모든 기능을 사용할 수 있다

**장바구니 수량 조절**

- [ ] 각 장바구니 상품의 수량을 증가할 수 있다
- [ ] 각 장바구니 상품의 수량을 감소할 수 있다
- [ ] 수량 변경 시 총 금액이 실시간으로 업데이트된다

**장바구니 삭제**

- [ ] 각 상품에 삭제 버튼이 배치되어 있다
- [ ] 삭제 버튼 클릭 시 해당 상품이 장바구니에서 제거된다

**장바구니 선택 삭제**

- [ ] 각 상품에 선택을 위한 체크박스가 제공된다
- [ ] 선택 삭제 버튼이 있다
- [ ] 체크된 상품들만 일괄 삭제된다

**장바구니 전체 선택**

- [ ] 모든 상품을 한 번에 선택할 수 있는 마스터 체크박스가 있다
- [ ] 전체 선택 시 모든 상품의 체크박스가 선택된다
- [ ] 전체 해제 시 모든 상품의 체크박스가 해제된다

**장바구니 비우기**

- [ ] 장바구니에 있는 모든 상품을 한 번에 삭제할 수 있다

#### 상품 상세

**상품 클릭시 상세 페이지 이동**

- [ ] 상품 목록에서 상품 이미지나 상품 정보 클릭 시 상세 페이지로 이동한다
- [ ] URL이 `/product/{productId}` 형태로 변경된다
- [ ] 상품의 자세한 정보가 전용 페이지에서 표시된다

**상품 상세 페이지 기능**

- [ ] 상품 이미지, 설명, 가격 등의 상세 정보가 표시된다
- [ ] 전체 화면을 활용한 상세 정보 레이아웃이 제공된다

**상품 상세 - 장바구니 담기**

- [ ] 상품 상세 페이지에서 해당 상품을 장바구니에 추가할 수 있다
- [ ] 페이지 내에서 수량을 선택하여 장바구니에 추가할 수 있다
- [ ] 수량 증가/감소 버튼이 제공된다

**관련 상품 기능**

- [ ] 상품 상세 페이지에서 관련 상품들이 표시된다
- [ ] 같은 카테고리(category2)의 다른 상품들이 관련 상품으로 표시된다
- [ ] 관련 상품 클릭 시 해당 상품의 상세 페이지로 이동한다
- [ ] 현재 보고 있는 상품은 관련 상품에서 제외된다

**상품 상세 페이지 내 네비게이션**

- [ ] 상품 상세에서 상품 목록으로 돌아가는 버튼이 제공된다
- [ ] 브레드크럼을 통해 카테고리별 상품 목록으로 이동할 수 있다
- [ ] SPA 방식으로 페이지 간 이동이 부드럽게 처리된다

#### 사용자 피드백 시스템

**토스트 메시지**

- [ ] 장바구니 추가 시 성공 메시지가 토스트로 표시된다
- [ ] 장바구니 삭제, 선택 삭제, 전체 삭제 시 알림 메시지가 표시된다
- [ ] 토스트는 3초 후 자동으로 사라진다
- [ ] 토스트에 닫기 버튼이 제공된다
- [ ] 토스트 타입별로 다른 스타일이 적용된다 (success, info, error)

### 심화과제

#### 1) 해시 라우터 구현
- [ ] location.hash를 이용하여 SPA 라우터 구현
- [ ] '/#/' (홈 페이지)
- [ ] '/#/login' (로그인 페이지)
- [ ] '/#/profile' (프로필 페이지)

#### 2) 라우트 가드 구현
- [ ] 로그인 상태에 따른 접근 제어
- [ ] 비로그인 사용자의 특정 페이지 접근 시 로그인 페이지로 리다이렉션
#### SPA 네비게이션 및 URL 관리

**페이지 이동**

- [ ] 어플리케이션 내의 모든 페이지 이동(뒤로가기/앞으로가기를 포함)은 하여 새로고침이 발생하지 않아야 한다.

**상품 목록 - URL 쿼리 반영**

- [ ] 검색어가 URL 쿼리 파라미터에 저장된다
- [ ] 카테고리 선택이 URL 쿼리 파라미터에 저장된다
- [ ] 상품 옵션이 URL 쿼리 파라미터에 저장된다
- [ ] 정렬 조건이 URL 쿼리 파라미터에 저장된다
- [ ] 조건 변경 시 URL이 자동으로 업데이트된다
- [ ] URL을 통해 현재 검색/필터 상태를 공유할 수 있다

**상품 목록 - 새로고침 시 상태 유지**

- [ ] 새로고침 후 URL 쿼리에서 검색어가 복원된다
- [ ] 새로고침 후 URL 쿼리에서 카테고리가 복원된다
- [ ] 새로고침 후 URL 쿼리에서 옵션 설정이 복원된다
- [ ] 새로고침 후 URL 쿼리에서 정렬 조건이 복원된다
- [ ] 복원된 조건에 맞는 상품 데이터가 다시 로드된다

**장바구니 - 새로고침 시 데이터 유지**

- [ ] 장바구니 내용이 브라우저에 저장된다
- [ ] 새로고침 후에도 이전 장바구니 내용이 유지된다
- [ ] 장바구니의 선택 상태도 함께 유지된다

**상품 상세 - URL에 ID 반영**

- [ ] 상품 상세 페이지 이동 시 상품 ID가 URL 경로에 포함된다 (`/product/{productId}`)
- [ ] URL로 직접 접근 시 해당 상품의 상세 페이지가 자동으로 로드된다

#### 3) 이벤트 위임
**상품 상세 - 새로고침시 유지**

- [ ] 이벤트 위임 방식으로 이벤트를 관리하고 있다.
- [ ] 새로고침 후에도 URL의 상품 ID를 읽어서 해당 상품 상세 페이지가 유지된다

**404 페이지**

- [ ] 존재하지 않는 경로 접근 시 404 에러 페이지가 표시된다
- [ ] 홈으로 돌아가기 버튼이 제공된다

#### AI로 한 번 더 구현하기

- [ ] 기존에 구현한 기능을 AI로 다시 구현한다.
- [ ] 이 과정에서 직접 가공하는 것은 최대한 지양한다.

## 과제 셀프회고

<!-- 과제에 대한 회고를 작성해주세요 -->

### 기술적 성장

<!-- 예시
- 새로 학습한 개념
- 기존 지식의 재발견/심화
- 구현 과정에서의 기술적 도전과 해결
-->

### 코드 품질
### 자랑하고 싶은 코드

<!-- 예시
- 특히 만족스러운 구현
- 리팩토링이 필요한 부분
- 코드 설계 관련 고민과 결정
-->

### 개선이 필요하다고 생각하는 코드

<!-- 예시
- 특히 만족스러운 구현
- 리팩토링이 필요한 부분
- 코드 설계 관련 고민과 결정
-->

### 학습 효과 분석

<!-- 예시
- 가장 큰 배움이 있었던 부분
- 추가 학습이 필요한 영역
- 실무 적용 가능성
-->

### 과제 피드백

<!-- 예시
- 과제에서 모호하거나 애매했던 부분
- 과제에서 좋았던 부분
-->

### AI 활용 경험 공유하기

<!-- 예시
- 사용한 AI 도구 (예: ChatGPT, Copilot, Claude, Cursor, ...)
- 프롬프트를 작성한 과정
- AI가 일을 더 잘 하게 만든 방법
- 내가 작성한 코드와 비교하기
-->

## 리뷰 받고 싶은 내용

<!--
Expand Down
27 changes: 8 additions & 19 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -11,17 +11,6 @@ on:

jobs:
basic:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
with:
fetch-depth: 0
ref: ${{ github.event.pull_request.head.sha }}
- name: test basic
run: |
npm install
npm run test:basic
advacned:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
Expand All @@ -33,13 +22,13 @@ jobs:
version: latest
- uses: actions/setup-node@v4
with:
node-version: 20
node-version: 22
cache: 'pnpm'
- name: advanced-test
run: |
- run: |
pnpm install
pnpm run test:advanced
e2e:
pnpm exec playwright install
pnpm run test:e2e:basic
advanced:
timeout-minutes: 60
runs-on: ubuntu-latest
steps:
Expand All @@ -52,10 +41,10 @@ jobs:
version: latest
- uses: actions/setup-node@v4
with:
node-version: 20
node-version: 22
cache: 'pnpm'
- name: Install dependencies
run: |
pnpm install
npx playwright install --with-deps
pnpm run test:e2e
pnpm exec playwright install
pnpm run test:e2e:advanced
Loading