Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
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
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -37,3 +37,4 @@ coverage

# OS
Thumbs.db
.claude/claude.md
85 changes: 58 additions & 27 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -26,16 +26,16 @@

## 1. ๊ธฐ์ˆ  ์Šคํƒ

| ๋ถ„๋ฅ˜ | ์‚ฌ์šฉ ๊ธฐ์ˆ  |
|---|---|
| Framework | React 18+ (Vite ๊ธฐ๋ฐ˜) |
| Language | JavaScript (ES2022+) ๋˜๋Š” TypeScript |
| 3D | Three.js, @react-three/fiber, @react-three/drei |
| Routing | react-router-dom v6 |
| State | Zustand (์ „์—ญ), useState/useReducer (์ง€์—ญ) |
| HTTP | Axios |
| Styling | Tailwind CSS (์„ ํƒ) ๋˜๋Š” CSS Modules |
| Lint/Format | ESLint + Prettier |
| ๋ถ„๋ฅ˜ | ์‚ฌ์šฉ ๊ธฐ์ˆ  |
| ----------- | ----------------------------------------------- |
| Framework | React 18+ (Vite ๊ธฐ๋ฐ˜) |
| Language | JavaScript (ES2022+) ๋˜๋Š” TypeScript |
| 3D | Three.js, @react-three/fiber, @react-three/drei |
| Routing | react-router-dom v6 |
| State | Zustand (์ „์—ญ), useState/useReducer (์ง€์—ญ) |
| HTTP | Axios |
| Styling | Tailwind CSS |
| Lint/Format | ESLint + Prettier |

> **๋ฒ„์ „ ๋ณ€๊ฒฝ ์‹œ PM ์Šน์ธ ํ•„์ˆ˜.** `package.json` ์˜์กด์„ฑ์„ ์ž„์˜๋กœ ์—…๊ทธ๋ ˆ์ด๋“œํ•˜์ง€ ์•Š์Šต๋‹ˆ๋‹ค.

Expand Down Expand Up @@ -80,26 +80,28 @@ src/
```

**๊ทœ์น™**

- ํ•œ ํด๋”์— ํŒŒ์ผ์ด 10๊ฐœ ์ด์ƒ ์Œ“์ด๋ฉด ํ•˜์œ„ ํด๋”๋กœ ๋ถ„๋ฆฌํ•ฉ๋‹ˆ๋‹ค.
- `pages/` ์•ˆ์—๋Š” ๋ผ์šฐํŠธ์™€ 1:1๋กœ ๋งค์นญ๋˜๋Š” ํŒŒ์ผ๋งŒ ๋‘ก๋‹ˆ๋‹ค. ๊ทธ ์™ธ ๋กœ์ง์€ `components/`, `hooks/`๋กœ ๋ถ„๋ฆฌํ•ฉ๋‹ˆ๋‹ค.

---

## 3. ๋„ค์ด๋ฐ ๊ทœ์น™

| ๋Œ€์ƒ | ๊ทœ์น™ | ์˜ˆ์‹œ |
|---|---|---|
| ์ปดํฌ๋„ŒํŠธ ํŒŒ์ผ | PascalCase + `.jsx` | `RecruitForm.jsx` |
| ์ผ๋ฐ˜ JS ํŒŒ์ผ | camelCase + `.js` | `formatDate.js` |
| ์ปค์Šคํ…€ ํ›… | `use`๋กœ ์‹œ์ž‘, camelCase | `useRecruitForm.js` |
| ์ƒ์ˆ˜ | UPPER_SNAKE_CASE | `MAX_FILE_SIZE` |
| ๋ณ€์ˆ˜/ํ•จ์ˆ˜ | camelCase | `getUserInfo` |
| Boolean ๋ณ€์ˆ˜ | `is`, `has`, `can`, `should` ์ ‘๋‘์‚ฌ | `isLoading`, `hasError` |
| ์ด๋ฒคํŠธ ํ•ธ๋“ค๋Ÿฌ | `handle` + ๋™์ž‘ | `handleSubmit`, `handleClick` |
| props๋กœ ๋ฐ›๋Š” ํ•ธ๋“ค๋Ÿฌ | `on` + ๋™์ž‘ | `onSubmit`, `onClose` |
| CSS ํด๋ž˜์Šค (CSS Modules) | camelCase | `submitButton` |
| ๋Œ€์ƒ | ๊ทœ์น™ | ์˜ˆ์‹œ |
| ------------------------ | ----------------------------------- | ----------------------------- |
| ์ปดํฌ๋„ŒํŠธ ํŒŒ์ผ | PascalCase + `.jsx` | `RecruitForm.jsx` |
| ์ผ๋ฐ˜ JS ํŒŒ์ผ | camelCase + `.js` | `formatDate.js` |
| ์ปค์Šคํ…€ ํ›… | `use`๋กœ ์‹œ์ž‘, camelCase | `useRecruitForm.js` |
| ์ƒ์ˆ˜ | UPPER_SNAKE_CASE | `MAX_FILE_SIZE` |
| ๋ณ€์ˆ˜/ํ•จ์ˆ˜ | camelCase | `getUserInfo` |
| Boolean ๋ณ€์ˆ˜ | `is`, `has`, `can`, `should` ์ ‘๋‘์‚ฌ | `isLoading`, `hasError` |
| ์ด๋ฒคํŠธ ํ•ธ๋“ค๋Ÿฌ | `handle` + ๋™์ž‘ | `handleSubmit`, `handleClick` |
| props๋กœ ๋ฐ›๋Š” ํ•ธ๋“ค๋Ÿฌ | `on` + ๋™์ž‘ | `onSubmit`, `onClose` |
| CSS ํด๋ž˜์Šค (CSS Modules) | camelCase | `submitButton` |

**๊ธˆ์ง€ ์‚ฌํ•ญ**

- ์˜๋ฏธ ์—†๋Š” ์•ฝ์–ด ์‚ฌ์šฉ ๊ธˆ์ง€ (`btn`, `usr`, `tmp` ๋“ฑ). `button`, `user`, `temporary`๋กœ ํ’€์–ด์“ฐ๊ธฐ.
- ํ•œ๊ธ€ ๋ณ€์ˆ˜๋ช… ๊ธˆ์ง€.
- `data`, `info`, `value`์ฒ˜๋Ÿผ ๋ชจํ˜ธํ•œ ์ด๋ฆ„ ๋‹จ๋… ์‚ฌ์šฉ ๊ธˆ์ง€. `userData`, `recruitInfo`์ฒ˜๋Ÿผ ๋งฅ๋ฝ ํฌํ•จ.
Expand All @@ -109,6 +111,7 @@ src/
## 4. ์ปดํฌ๋„ŒํŠธ ์ž‘์„ฑ ๊ทœ์น™

### 4.1 ๊ธฐ๋ณธ ์›์น™

- **ํ•จ์ˆ˜ํ˜• ์ปดํฌ๋„ŒํŠธ๋งŒ ์‚ฌ์šฉํ•ฉ๋‹ˆ๋‹ค.** Class ์ปดํฌ๋„ŒํŠธ ๊ธˆ์ง€.
- **ํ•œ ํŒŒ์ผ์— ํ•œ ์ปดํฌ๋„ŒํŠธ.** export default๋Š” ํŒŒ์ผ๋‹น ํ•˜๋‚˜.
- **150์ค„์„ ๋„˜์œผ๋ฉด ๋ถ„๋ฆฌ๋ฅผ ๊ณ ๋ฏผํ•ฉ๋‹ˆ๋‹ค.** 250์ค„์„ ๋„˜์œผ๋ฉด ๋ฌด์กฐ๊ฑด ๋ถ„๋ฆฌํ•ฉ๋‹ˆ๋‹ค.
Expand Down Expand Up @@ -159,6 +162,7 @@ export default RecruitForm;
```

### 4.3 ์ปดํฌ๋„ŒํŠธ ์ž‘์„ฑ ์ˆœ์„œ (์œ„์—์„œ ์•„๋ž˜๋กœ)

1. import
2. ์ปดํฌ๋„ŒํŠธ ํ•จ์ˆ˜ ์„ ์–ธ
3. ์ƒํƒœ(useState, useReducer)
Expand All @@ -170,12 +174,15 @@ export default RecruitForm;
9. export default

### 4.4 ์กฐ๊ฑด๋ถ€ ๋ Œ๋”๋ง

- ์‚ผํ•ญ ์—ฐ์‚ฐ์ž๋Š” 1๋‹จ๊นŒ์ง€๋งŒ. ์ค‘์ฒฉ ๊ธˆ์ง€.
- ๋ณต์žกํ•œ ๋ถ„๊ธฐ๋Š” ๋ณ€์ˆ˜๋กœ ๋นผ๊ฑฐ๋‚˜ early return ์‚ฌ์šฉ.

```jsx
// โŒ ๋‚˜์œ ์˜ˆ
{isLoading ? <Loading /> : error ? <Error /> : data ? <List /> : <Empty />}
{
isLoading ? <Loading /> : error ? <Error /> : data ? <List /> : <Empty />;
}

// โœ… ์ข‹์€ ์˜ˆ
if (isLoading) return <Loading />;
Expand All @@ -189,11 +196,13 @@ return <List />;
## 5. ์Šคํƒ€์ผ๋ง ๊ทœ์น™

### 5.1 ๋„๊ตฌ ์„ ํƒ

- **๊ธฐ๋ณธ์€ Tailwind CSS๋ฅผ ์‚ฌ์šฉํ•ฉ๋‹ˆ๋‹ค.** (์„ ํƒํ•œ ๊ฒฝ์šฐ)
- ๋ณต์žกํ•œ ์• ๋‹ˆ๋ฉ”์ด์…˜, ๊ธ€๋กœ๋ฒŒ ์Šคํƒ€์ผ์€ `styles/` ์•„๋ž˜ CSS ํŒŒ์ผ๋กœ ๋ถ„๋ฆฌ.
- ์ธ๋ผ์ธ style์€ ๋™์  ๊ฐ’(์˜ˆ: `transform: translateX(${x}px)`)์—๋งŒ ์‚ฌ์šฉ.

### 5.2 ์ƒ‰์ƒ๊ณผ ํฐํŠธ

- **ํ•˜๋“œ์ฝ”๋”ฉ ๊ธˆ์ง€.** ๋ชจ๋“  ์ƒ‰์ƒ์€ `tailwind.config.js` ๋˜๋Š” `src/constants/theme.js`์— ์ •์˜ ํ›„ ์‚ฌ์šฉ.

```js
Expand All @@ -207,6 +216,7 @@ export const COLORS = {
```

### 5.3 ๋ฐ˜์‘ํ˜•

- ๋ชจ๋ฐ”์ผ ์šฐ์„ (Mobile First). ๊ธฐ๋ณธ ์Šคํƒ€์ผ์€ ๋ชจ๋ฐ”์ผ, `md:`, `lg:` ๋“ฑ์œผ๋กœ ํ™•์žฅ.
- ๋ธŒ๋ ˆ์ดํฌํฌ์ธํŠธ: `sm: 640px`, `md: 768px`, `lg: 1024px`, `xl: 1280px` (Tailwind ๊ธฐ๋ณธ๊ฐ’ ์œ ์ง€)

Expand All @@ -215,6 +225,7 @@ export const COLORS = {
## 6. Three.js ์ž‘์„ฑ ๊ทœ์น™

### 6.1 ๊ธฐ๋ณธ ์›์น™

- **`@react-three/fiber`๋ฅผ ํ†ตํ•ด ์„ ์–ธ์ ์œผ๋กœ ์ž‘์„ฑํ•ฉ๋‹ˆ๋‹ค.** ๋ช…๋ นํ˜•(`new THREE.Scene()`)์€ ์ •๋ง ํ•„์š”ํ•œ ๊ฒฝ์šฐ์—๋งŒ.
- Three.js ๊ด€๋ จ ๋ชจ๋“  ์ฝ”๋“œ๋Š” `src/three/` ์•„๋ž˜์— ๋‘ก๋‹ˆ๋‹ค.
- 3D ๋ชจ๋ธ ํŒŒ์ผ์€ `assets/models/`์— ๋‘๊ณ , ํŒŒ์ผ๋ช…์€ kebab-case (`hero-scene.glb`).
Expand Down Expand Up @@ -249,13 +260,15 @@ export default HeroScene;
```

### 6.3 ์„ฑ๋Šฅ ๊ทœ์น™ (์ธ์ˆ˜์ธ๊ณ„์šฉ ํ•„์ˆ˜ ์‚ฌํ•ญ)

- **๋ฉ”๋ชจ๋ฆฌ ๋ˆ„์ˆ˜ ๋ฐฉ์ง€**: `useEffect` cleanup์—์„œ `geometry.dispose()`, `material.dispose()`, `texture.dispose()` ํ˜ธ์ถœ.
- **์žฌ์‚ฌ์šฉ ๊ฐ€๋Šฅํ•œ geometry/material์€ `useMemo`๋กœ ๊ฐ์Œ‰๋‹ˆ๋‹ค.** ๋งค ๋ Œ๋”๋งˆ๋‹ค ์ƒˆ๋กœ ๋งŒ๋“ค์ง€ ์•Š์Šต๋‹ˆ๋‹ค.
- **3D ๋ชจ๋ธ์€ `useGLTF.preload()`๋กœ ์‚ฌ์ „ ๋กœ๋“œ**ํ•˜์—ฌ ์ดˆ๊ธฐ ์ง„์ž… ์ง€์—ฐ์„ ์ค„์ž…๋‹ˆ๋‹ค.
- **๋ชจ๋ฐ”์ผ ๋Œ€์‘**: ๊ทธ๋ฆผ์ž, ์•ˆํ‹ฐ์•จ๋ฆฌ์–ด์‹ฑ์€ ๊ธฐ๋ณธ false. ํ•„์š” ์‹œ ๋””๋ฐ”์ด์Šค ์„ฑ๋Šฅ ๊ฐ์ง€ ํ›„ ํ™œ์„ฑํ™”.
- **๋ชจ๋ธ ํŒŒ์ผ ํฌ๊ธฐ๋Š” 5MB ์ดํ•˜** ๊ถŒ์žฅ. Draco ์••์ถ• ์‚ฌ์šฉ.

### 6.4 Three.js ๋งค์ง ๋„˜๋ฒ„ ๊ธˆ์ง€

์ขŒํ‘œ, FOV, ํšŒ์ „ ์†๋„ ๊ฐ™์€ ๊ฐ’์€ ์ปดํฌ๋„ŒํŠธ ์ƒ๋‹จ์— ์ƒ์ˆ˜๋กœ ๋ถ„๋ฆฌํ•ฉ๋‹ˆ๋‹ค.

```jsx
Expand All @@ -269,13 +282,15 @@ const LIGHT_INTENSITY = 0.8;
## 7. ์ƒํƒœ ๊ด€๋ฆฌ ๋ฐ API ํ†ต์‹ 

### 7.1 ์ƒํƒœ ๋ถ„๋ฅ˜
| ์ข…๋ฅ˜ | ๋„๊ตฌ |
|---|---|
| ์ปดํฌ๋„ŒํŠธ ๋‚ด๋ถ€ ์ƒํƒœ | `useState`, `useReducer` |
| ์ „์—ญ ์ƒํƒœ (๋กœ๊ทธ์ธ ์ •๋ณด, ํ…Œ๋งˆ ๋“ฑ) | Zustand |
| ์„œ๋ฒ„ ์ƒํƒœ (API ์‘๋‹ต ์บ์‹ฑ) | TanStack Query ๊ถŒ์žฅ / ๋˜๋Š” ์ง์ ‘ ๊ด€๋ฆฌ |

| ์ข…๋ฅ˜ | ๋„๊ตฌ |
| -------------------------------- | ------------------------------------ |
| ์ปดํฌ๋„ŒํŠธ ๋‚ด๋ถ€ ์ƒํƒœ | `useState`, `useReducer` |
| ์ „์—ญ ์ƒํƒœ (๋กœ๊ทธ์ธ ์ •๋ณด, ํ…Œ๋งˆ ๋“ฑ) | Zustand |
| ์„œ๋ฒ„ ์ƒํƒœ (API ์‘๋‹ต ์บ์‹ฑ) | TanStack Query ๊ถŒ์žฅ / ๋˜๋Š” ์ง์ ‘ ๊ด€๋ฆฌ |

### 7.2 Axios ์ธ์Šคํ„ด์Šค

๋ชจ๋“  API๋Š” `src/apis/instance.js`์˜ ๊ณตํ†ต ์ธ์Šคํ„ด์Šค๋ฅผ ํ†ตํ•ด ํ˜ธ์ถœํ•ฉ๋‹ˆ๋‹ค.

```js
Expand All @@ -297,6 +312,7 @@ export default instance;
```

### 7.3 API ํ•จ์ˆ˜ ๋ถ„๋ฆฌ

์ปดํฌ๋„ŒํŠธ์—์„œ ์ง์ ‘ `axios.get(...)`์„ ํ˜ธ์ถœํ•˜์ง€ ์•Š์Šต๋‹ˆ๋‹ค. ๋ฐ˜๋“œ์‹œ `apis/` ํด๋”์˜ ํ•จ์ˆ˜๋ฅผ ํ†ตํ•ด ํ˜ธ์ถœํ•ฉ๋‹ˆ๋‹ค.

```js
Expand All @@ -308,6 +324,7 @@ export const getRecruitList = () => instance.get('/admin/recruit');
```

### 7.4 ์—๋Ÿฌ ์ฒ˜๋ฆฌ

- ๋ชจ๋“  ๋น„๋™๊ธฐ ํ˜ธ์ถœ์€ `try-catch`๋กœ ๊ฐ์Œ‰๋‹ˆ๋‹ค.
- ์‚ฌ์šฉ์ž์—๊ฒŒ ๋ณด์ผ ์—๋Ÿฌ ๋ฉ”์‹œ์ง€๋Š” ์ƒ์ˆ˜๋กœ ๊ด€๋ฆฌํ•˜๊ณ , ์ฝ˜์†”์—๋Š” `[์ปดํฌ๋„ŒํŠธ๋ช…] ์—๋Ÿฌ๋‚ด์šฉ` ํ˜•์‹์œผ๋กœ ์ถœ๋ ฅํ•ฉ๋‹ˆ๋‹ค.

Expand Down Expand Up @@ -343,6 +360,7 @@ import logoImage from '@/assets/images/logo.png';
## 9. ์ฃผ์„ ์ž‘์„ฑ ๊ทœ์น™

### 9.1 ์ž‘์„ฑ ์›์น™

- **"๋ฌด์—‡์„" ํ•˜๋Š”์ง€๊ฐ€ ์•„๋‹ˆ๋ผ "์™œ" ํ•˜๋Š”์ง€๋ฅผ ์ ์Šต๋‹ˆ๋‹ค.** ์ฝ”๋“œ๋งŒ ๋ด์„œ ์•Œ ์ˆ˜ ์žˆ๋Š” ๊ฑด ์ฃผ์„ ๋‹ฌ์ง€ ์•Š์Šต๋‹ˆ๋‹ค.
- **์ธ์ˆ˜์ธ๊ณ„๋ฐ›๋Š” ์‚ฌ๋žŒ์„ ์ƒ๊ฐํ•˜๊ณ  ์ ์Šต๋‹ˆ๋‹ค.** "์ด๊ฑด ~ ๋•Œ๋ฌธ์— ์ด๋ ‡๊ฒŒ ํ–ˆ๋‹ค"๊ฐ€ ํ•ต์‹ฌ.
- ํ•œ๊ตญ์–ด๋กœ ์ž‘์„ฑ. ์˜์–ด ์„ž์ง€ ์•Š๊ธฐ.
Expand All @@ -364,6 +382,7 @@ const TEMP_BG_COLOR = '#CCCCCC';
```

### 9.3 JSDoc

์žฌ์‚ฌ์šฉ ์ปดํฌ๋„ŒํŠธ์™€ ์œ ํ‹ธ ํ•จ์ˆ˜์—๋Š” JSDoc ๊ถŒ์žฅ.

```js
Expand All @@ -380,6 +399,7 @@ export function formatKoreanDate(dateString) { ... }
## 10. Git ์ปจ๋ฒค์…˜

### 10.1 ๋ธŒ๋žœ์น˜ ์ „๋žต

```
main โ†’ ๋ฐฐํฌ์šฉ (PM๋งŒ ๋จธ์ง€)
develop โ†’ ๊ฐœ๋ฐœ ํ†ตํ•ฉ ๋ธŒ๋žœ์น˜
Expand All @@ -389,6 +409,7 @@ refactor/*โ†’ ๋ฆฌํŒฉํ† ๋ง
```

### 10.2 ์ปค๋ฐ‹ ๋ฉ”์‹œ์ง€ (Conventional Commits)

```
<type>: <์ œ๋ชฉ (ํ•œ๊ตญ์–ด, ๋ช…๋ นํ˜•, 50์ž ์ด๋‚ด)>

Expand All @@ -407,13 +428,15 @@ refactor/*โ†’ ๋ฆฌํŒฉํ† ๋ง
| `chore` | ๋นŒ๋“œ ์„ค์ •, ํŒจํ‚ค์ง€ ๋งค๋‹ˆ์ € ๋“ฑ |

**์˜ˆ์‹œ**

```
feat: ์‹ ์ž…๋ถ€์› ๋ชจ์ง‘ ์‹ ์ฒญ ํผ ๊ตฌํ˜„
fix: ๋ชจ๋ฐ”์ผ์—์„œ Three.js ์บ”๋ฒ„์Šค๊ฐ€ ์ž˜๋ฆฌ๋Š” ๋ฌธ์ œ ํ•ด๊ฒฐ
docs: README์— ํ™˜๊ฒฝ ๋ณ€์ˆ˜ ์„ค๋ช… ์ถ”๊ฐ€
```

### 10.3 PR ๊ทœ์น™

- **base ๋ธŒ๋žœ์น˜๋Š” ํ•ญ์ƒ `develop`.** `main`์— ์ง์ ‘ PR ๊ธˆ์ง€.
- ์ œ๋ชฉ: ์ปค๋ฐ‹ ์ปจ๋ฒค์…˜๊ณผ ๋™์ผํ•œ ํ˜•์‹
- ๋ณธ๋ฌธ์— ํฌํ•จํ•  ๊ฒƒ:
Expand Down Expand Up @@ -445,9 +468,11 @@ VITE_ADMIN_SECRET_KEY=your_key_here
## 12. ์ฝ”๋“œ ํ’ˆ์งˆ ๋„๊ตฌ

### 12.1 ESLint + Prettier

ํ”„๋กœ์ ํŠธ ๋ฃจํŠธ์˜ `.eslintrc`, `.prettierrc`๋ฅผ ๋”ฐ๋ฆ…๋‹ˆ๋‹ค. ๊ฐœ์ธ ์„ค์ •์œผ๋กœ ๋ฎ์–ด์“ฐ์ง€ ๋งˆ์„ธ์š”.

**ํ•„์ˆ˜ ์„ค์ •**

```json
// .prettierrc
{
Expand All @@ -461,13 +486,15 @@ VITE_ADMIN_SECRET_KEY=your_key_here
```

### 12.2 ์ปค๋ฐ‹ ์ „ ์ฒดํฌ๋ฆฌ์ŠคํŠธ

- [ ] `npm run lint` ํ†ต๊ณผ
- [ ] `npm run format` ์ ์šฉ
- [ ] `console.log` ์ œ๊ฑฐ (๋””๋ฒ„๊น… ํ›„)
- [ ] ์‚ฌ์šฉํ•˜์ง€ ์•Š๋Š” import ์ œ๊ฑฐ
- [ ] ์ฃผ์„ ์ฒ˜๋ฆฌ๋œ ์˜›๋‚  ์ฝ”๋“œ ์ œ๊ฑฐ

### 12.3 VSCode ํ™•์žฅ (๊ถŒ์žฅ)

- ESLint
- Prettier - Code formatter
- ES7+ React/Redux/React-Native snippets
Expand All @@ -480,25 +507,29 @@ VITE_ADMIN_SECRET_KEY=your_key_here
> **๋‹ค์Œ ๊ธฐ์ˆ˜์—๊ฒŒ ๋„˜๊ธฐ๊ธฐ ์ „์— ๋ฐ˜๋“œ์‹œ ํ™•์ธํ•  ํ•ญ๋ชฉ์ž…๋‹ˆ๋‹ค.**

### 13.1 ๋ฌธ์„œ

- [ ] ๋ณธ README๊ฐ€ ์ตœ์‹  ์ƒํƒœ์ธ์ง€ ํ™•์ธ
- [ ] `.env.example`์— ๋ชจ๋“  ํ™˜๊ฒฝ ๋ณ€์ˆ˜๊ฐ€ ๋ช…์‹œ๋˜์–ด ์žˆ๋Š”์ง€ ํ™•์ธ
- [ ] ์ฃผ์š” ๋””๋ ‰ํ† ๋ฆฌ์— `README.md`๊ฐ€ ์žˆ๋Š”์ง€ (์˜ˆ: `src/three/README.md`)
- [ ] API ๋ช…์„ธ์„œ ๋งํฌ๊ฐ€ README์— ํฌํ•จ๋˜์–ด ์žˆ๋Š”์ง€
- [ ] ๋””์ž์ธ ์‹œ์•ˆ ๋งํฌ (Figma ๋“ฑ) ํฌํ•จ

### 13.2 ์ฝ”๋“œ

- [ ] `TODO`, `FIXME` ์ฃผ์„ ์ •๋ฆฌ (์ž‘์„ฑ์ž/๋‚ ์งœ ๋ช…์‹œ)
- [ ] ์‚ฌ์šฉํ•˜์ง€ ์•Š๋Š” ํŒŒ์ผ/ํŒจํ‚ค์ง€ ์ œ๊ฑฐ
- [ ] ๋งค์ง ๋„˜๋ฒ„ ์ƒ์ˆ˜ํ™”
- [ ] ๋ชจ๋“  ํŽ˜์ด์ง€๊ฐ€ ์ •์ƒ ๋™์ž‘ํ•˜๋Š”์ง€ ์ˆ˜๋™ ํ…Œ์ŠคํŠธ

### 13.3 ๋ฐฐํฌ

- [ ] ๋นŒ๋“œ ๋ช…๋ น์–ด (`npm run build`) ์ •์ƒ ๋™์ž‘
- [ ] ๋ฐฐํฌ URL, ๋„๋ฉ”์ธ ์ •๋ณด ๋ฌธ์„œํ™”
- [ ] ํ˜ธ์ŠคํŒ… ๊ณ„์ • ์ธ์ˆ˜์ธ๊ณ„ (Vercel, Netlify ๋“ฑ)
- [ ] Firebase / ๋ฐฑ์—”๋“œ ์„œ๋ฒ„ ์ ‘๊ทผ ๊ถŒํ•œ ์ด์–‘

### 13.4 ํšŒ๊ณ 

- [ ] ์•Œ๋ ค์ง„ ๋ฒ„๊ทธ/์ด์Šˆ ๋ชฉ๋ก ์ž‘์„ฑ
- [ ] ๋‹ค์Œ ๊ธฐ์ˆ˜์—๊ฒŒ ์ „ํ•˜๊ณ  ์‹ถ์€ ๊ฐœ์„  ํฌ์ธํŠธ ์ •๋ฆฌ

Expand Down
6 changes: 6 additions & 0 deletions index.html
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,12 @@
<meta charset="UTF-8" />
<link rel="icon" type="image/svg+xml" href="/favicon.svg" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<link rel="preconnect" href="https://fonts.googleapis.com" />
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
<link
href="https://fonts.googleapis.com/css2?family=Noto+Sans+KR:wght@400;500;700&display=swap"
rel="stylesheet"
/>
<title>ONE | ์ปดํ“จํ„ฐ๊ณตํ•™๊ณผ ๋™์•„๋ฆฌ</title>
<meta name="description" content="ONE ๋™์•„๋ฆฌ ๊ณต์‹ ์›น์‚ฌ์ดํŠธ" />
</head>
Expand Down
Loading