` — 옅은 회색 본문 배경. PublicShell의 `pt-20` 헤더 여백을 통과해 mypage 자체 hero가 곧장 시작.
+
+### 4.2 사용자 헤더 (Dark Hero — 축소)
+
+**현재**(mypage page.tsx:302-325):
+```tsx
+
+
+
+
{user.email?.[0].toUpperCase()}
+
+
{user.email}
+
가입일: {...}
+
+
+
+
+
+
+
+```
+
+**변경**:
+```tsx
+
+
+
+
+ {user.email?.[0].toUpperCase()}
+
+
+
{user.email}
+
+ 가입일 {new Date(user.created_at).toLocaleDateString('ko-KR')}
+
+
+
+
+
+```
+
+차이점:
+- `bg-[#04102b]` → CSS var `var(--kx-surface)` (= #060e20, 더 깊음)
+- `bg-[#1a56db]` 아바타 → `var(--kx-primary)` (#cc97ff 보라)
+- 패딩 `py-10` → `py-8` (덜 압도적)
+- 아바타 크기 `w-14 h-14 text-xl` → `w-12 h-12 text-lg` (한 단계 축소)
+- 가입일 폰트 `text-sm` → `text-xs` 색 `text-blue-300/60` → `text-white/50` (덜 강조)
+- "가입일:" 콜론 → "가입일 " 공백 (kx 톤, 미니멀)
+- **로그아웃 버튼 완전 제거** — TopNav로 이동
+- 헤더 폰트에 `kx-display` 클래스 추가 (Jua + letter-spacing 정돈)
+
+### 4.3 탭 바 (7개로 확장)
+
+**현재 6개**: 프로젝트현황 / 의뢰내역 / 결제내역 / 내정보 / 구독관리 / 사주기록
+**추가 1개**: 구매한 팩
+**총 7개**
+
+**탭 디자인 변경**:
+- 컨테이너: `bg-white border border-[#dbe8ff] rounded-xl p-1` → `bg-white border border-slate-200 rounded-xl p-1`
+- 액티브 탭 배경: `bg-[#1a56db] text-white shadow` → `bg-violet-600 text-white shadow` (보라)
+- 비액티브 텍스트: `text-slate-500 hover:text-slate-700` → `text-slate-500 hover:text-violet-600`
+- 카운트 배지 액티브: `bg-white/20 text-white` 그대로
+- 카운트 배지 비액티브: `bg-slate-100 text-slate-600` 그대로
+
+**탭 7개 표시**:
+- 데스크톱: `flex` 한 줄 — 가능 (각 탭 평균 5~7글자)
+- 모바일: 현재도 `flex` 한 줄이나 7개에서 글자 잘림 위험. **`flex-wrap` 적용** + 각 탭 `min-width: 110px` 정도 부여 → 자연스럽게 2줄로 떨어짐
+- 또는 모바일에서 `grid-cols-3 sm:grid-cols-4 md:flex` 단계적 적용
+
+탭 순서 (구매한 팩 위치 결정):
+- "구매한 팩"을 `결제 내역` 다음에 배치 — 결제와 자료가 인접
+- 최종: `프로젝트현황 / 의뢰내역 / 결제내역 / 구매한 팩 / 내정보 / 구독관리 / 사주기록`
+
+### 4.4 본문 카드 토큰 마이그레이션
+
+전체 mypage `page.tsx`(~960행)에서 다음 색 토큰을 일괄 치환:
+
+| 현재 토큰 | 새 토큰 | 의미 |
+|---|---|---|
+| `bg-[#f0f5ff]` | `bg-slate-50` | 본문 배경 (또는 흰색) |
+| `bg-[#04102b]` (헤더 외 위치) | `bg-[#060e20]` 또는 `var(--kx-surface)` | 다크 강조 |
+| `bg-[#1a56db]`, `bg-blue-600` | `bg-violet-600` | 주 액센트 |
+| `hover:bg-[#1e4fc2]`, `hover:bg-blue-700` | `hover:bg-violet-500` | 액센트 hover |
+| `bg-[#1a56db]/20`, `text-[#1a56db]` | `text-violet-600`, `bg-violet-100` | 텍스트 액센트 |
+| `border-[#dbe8ff]` | `border-slate-200` | 중성 카드 보더 |
+| `bg-blue-50 border-blue-200` | `bg-violet-50 border-violet-200` | 강조 박스 |
+| `text-[#04102b]` | `text-slate-900` | 본문 다크 텍스트 |
+| `text-blue-300/60` (다크 hero 안) | `text-white/50` | 다크 위 옅은 텍스트 |
+
+**유지하는 색 (보조 의미를 가지므로 그대로)**:
+- `bg-emerald-*` 완료/이용중 status
+- `bg-orange-*` 해지 예정 warning
+- `bg-amber-*` 유료 배지
+- `bg-red-*` 에러/해지
+- `bg-rose-*`, `bg-pink-*` 사주 메타
+- `bg-cyan-*` 사주 메타
+- `bg-sky-*` 텔레그램 연동
+
+이들은 의미 색이라 톤 통일에서 제외.
+
+### 4.5 "구매한 팩" 탭 (신규 — Placeholder)
+
+#### 데이터 소스 — 정정 (코드 검증 결과)
+
+**확인된 결제 흐름** (`app/services/music/page.tsx:295` + `app/components/PurchaseAgreementModal.tsx:55-63`):
+- Music 팩 구매: `PurchaseAgreementModal`이 `/api/contact` POST → **`contact_requests` 테이블**(mypage에서 `orders` 변수로 fetch 중)
+- `service` 컬럼 = `"구매 신청: AI 음악 마스터 팩 · {tier.name}"` 형식 (예: `"구매 신청: AI 음악 마스터 팩 · 프로"`)
+- `payments` 테이블은 **PortOne PG 결제 전용** (현재 사주 1,000원만) — Music 팩과 무관
+
+→ **데이터 소스는 기존 `orders` (contact_requests)**. 추가 fetch 불필요 (mypage:129-136에서 이미 fetch 중).
+
+#### Tier 매핑 함수
+
+`lib/pack-assets.ts` (신규 파일):
+
+```ts
+export interface PackAsset {
+ name: string;
+ files: string[];
+}
+
+export const PACK_ASSETS: Record<'starter' | 'pro' | 'master', PackAsset> = {
+ starter: {
+ name: 'AI 음악 마스터 팩 (입문)',
+ files: [
+ 'Suno 프롬프트 북 PDF (40p)',
+ '구조 템플릿 PDF',
+ '저작권 가이드 기본판',
+ ],
+ },
+ pro: {
+ name: 'AI 음악 마스터 팩 (프로)',
+ files: [
+ '입문 자료 전체',
+ 'MV 워크플로우 가이드 (Runway · Luma · Pika)',
+ '샘플 프로젝트 1개 (.prj 파일 + 영상)',
+ '유튜브 SEO 템플릿',
+ '1:1 Q&A 1회 (이메일 응답)',
+ ],
+ },
+ master: {
+ name: 'AI 음악 마스터 팩 (마스터)',
+ files: [
+ '프로 자료 전체',
+ '샘플 프로젝트 장르별 3종',
+ '저작권 심화판 + 상업 이용 체크리스트',
+ '제작 레시피 영상 (우선 공개)',
+ ],
+ },
+};
+
+/**
+ * orders.service ("구매 신청: AI 음악 마스터 팩 · 프로") → tier key.
+ * 매칭 안 되면 null 반환 (Music 팩 외 의뢰).
+ */
+export function extractPackTier(service: string): 'starter' | 'pro' | 'master' | null {
+ if (!service.startsWith('구매 신청:')) return null;
+ if (service.includes('입문')) return 'starter';
+ if (service.includes('프로')) return 'pro';
+ if (service.includes('마스터')) return 'master';
+ return null;
+}
+```
+
+(파일 크기, 실제 파일 URL은 Phase 2에서 추가)
+
+#### Status 매핑
+
+`orders.status` 값별 표시:
+- `'completed'` — 자료 발송 완료. 자료 리스트 + (Phase 2)다운로드 버튼 노출
+- `'in_progress'` — 입금 확인 중. "결제 처리 중" 안내 + 자료 리스트는 보여주되 다운로드 비활성
+- `'pending'` 또는 그 외 — "입금 대기 중" + 자료 리스트는 미노출 (또는 흐리게)
+
+#### UI 컴포넌트 (구체)
+
+```tsx
+import { PACK_ASSETS, extractPackTier } from '@/lib/pack-assets';
+
+// ...
+
+{tab === 'packs' && (
+
+ {(() => {
+ const packOrders = orders
+ .map((o) => ({ order: o, tier: extractPackTier(o.service) }))
+ .filter((x): x is { order: Order; tier: 'starter' | 'pro' | 'master' } => x.tier !== null);
+
+ if (packOrders.length === 0) {
+ return (
+
+ );
+ }
+
+ return packOrders.map(({ order, tier }) => {
+ const asset = PACK_ASSETS[tier];
+ const statusLabel =
+ order.status === 'completed' ? '자료 발송 완료' :
+ order.status === 'in_progress' ? '결제 처리 중' :
+ '입금 대기';
+ const statusColor =
+ order.status === 'completed' ? 'bg-violet-50 text-violet-600 border-violet-200' :
+ order.status === 'in_progress' ? 'bg-amber-50 text-amber-600 border-amber-200' :
+ 'bg-slate-100 text-slate-500 border-slate-200';
+
+ return (
+
+
+
+
{asset.name}
+
+ {new Date(order.created_at).toLocaleDateString('ko-KR')} 신청
+
+
+
+ {statusLabel}
+
+
+
+
+
+ 📦 자료 패키지 ({asset.files.length}개)
+
+
+ {asset.files.map((file, i) => (
+ -
+ ·
+ {file}
+
+ ))}
+
+
+
+
+ 현재는 카톡 1:1로 자료를 보내드립니다. 자동 다운로드는 곧 활성화됩니다.
+
+
+ 카톡 오픈채팅 →
+
+
+
+
+ );
+ });
+ })()}
+
+)}
+```
+
+#### Phase 2에서 변경될 것
+- `disabled` 버튼 → `
` 활성화
+- "자료 준비 중" → "자료 다운로드"
+- 카톡 안내 문구 제거
+- `PACK_ASSETS` 상수는 그대로 유지 가능 (정적 자료 매핑은 Phase 2에서도 유효), Phase 2는 파일별 URL/HMAC 토큰만 추가
+- `order.status === 'completed'` 일 때만 다운로드 활성, 그 외엔 placeholder 유지
+
+### 4.6 Tab type 확장
+
+```ts
+type Tab = 'profile' | 'projects' | 'subscription' | 'saju' | 'payments' | 'orders' | 'packs';
+```
+
+`packs` 추가. `tabs` 배열에 정의 추가:
+```ts
+{ key: 'packs', label: '구매한 팩', count: packPayments.length || undefined }
+```
+
+순서: `projects → orders → payments → packs → profile → subscription → saju`
+
+### 4.7 사용자 빠른 메뉴 (`tab === 'profile'` 안)
+
+현재 mypage page.tsx:512-535 에 "빠른 메뉴" 섹션이 있음:
+- 사주 분석 (`/saju/input`)
+- 외주 의뢰 (`/freelance`)
+
+음악 통합 강화를 위해 다음 항목 추가:
+- **AI 스튜디오** (`/studio`) — 새 트랙 만들기. 아이콘은 음악 아이콘.
+- (선택) **AI 음악 팩** (`/services/music`) — 다른 팩 둘러보기
+
+이 항목들은 디자인 토큰 마이그레이션과 함께 처리. 현재 4.4의 `border-[#dbe8ff]`, `bg-blue-50/50` 등을 `border-slate-200`, `bg-violet-50/50`으로 치환.
+
+## 5. 보존되는 동작 (회귀 방지)
+
+이번 spec은 시각·구조 변경이 본질이고, 비즈니스 로직(데이터 fetch, 텔레그램 연결, 구독 해지, 견적 토큰 연결, 사주 기록 표시 등)은 그대로 유지한다. 회귀 점검 목록:
+
+- supabase auth getUser 후 `/login` redirect — 그대로
+- payments / orders / saju_records / projects fetch — 그대로
+- 텔레그램 deeplink 발급, 연결/해제 — 그대로
+- 구독 자동갱신 토글, 구독 해지 — 그대로
+- 견적서 토큰 연결 폼 — 그대로
+- 카카오 오픈채팅 플로팅 버튼 (현재 DashboardShell에 있음) — Sidebar 제거 후에도 PublicShell에서 작동해야 함
+ - **주의**: DashboardShell.tsx 의 카카오 버튼은 사이드바 분기(useSidebar=true) 안에 있음 (76-127행) — 이 분기 자체가 사라지면 카카오 버튼도 사라짐
+ - **수정 필요**: 카카오 플로팅 버튼을 PublicShell.tsx로 이동 (또는 별도 컴포넌트로 분리하여 두 shell 모두에 마운트)
+- TelegramGuideModal — 그대로
+
+## 6. Phase 2 미리보기 (별도 spec — 이번 plan 종료 후 작성)
+
+Phase 2가 다룰 것:
+
+1. **NAS `/media/packs/` 디렉토리 셋업** — `/volume1/docker/webpage/media/packs/{starter|pro|master}/`
+2. **nginx 설정** — `/media/packs/...` 정적 서빙. 토큰 검증 옵션은 다음 중 결정:
+ - (a) Next.js API에서 토큰 검증 후 file 응답 (메모리 부담)
+ - (b) nginx auth_request 모듈로 사전 검증
+3. **Next.js API** `/api/packs/[productId]` — supabase user auth + payments 확인 후 HMAC 토큰 발급, file URL 반환
+4. **DB 스키마** — `pack_files` 테이블 (또는 정적 매핑 유지). 파일 메타데이터(크기, 종류, 업데이트일).
+5. **admin 업로드 UI** — `/admin/services` 또는 별도 `/admin/packs` — 새 자료 등록 흐름
+6. **mypage** — `disabled` 버튼 → 활성 다운로드 링크로 교체
+7. **회귀 테스트** — 구매하지 않은 사용자가 토큰 우회 시도 시 차단
+
+## 7. 의도적 제외 (이번 spec 범위 밖)
+
+| 항목 | 이유 |
+|---|---|
+| Studio 트랙 DB 저장 | 음악 통합 옵션 A — 별도 plan 필요 |
+| `/admin` shell Liquid Glass 마이그레이션 | mypage와 별도 surface, 우선순위 낮음 |
+| 사주 1,000원 PG 결제 결정 | P0 brainstorm 부록 A에 보류 항목 |
+| URL 마이그레이션 (`/freelance` → `/work/freelance` 등) | P1 home-restructure plan |
+| `/work` 우산 페이지 | P1 home-restructure plan |
+| Music 팩 자체 페이지(`/services/music`) 디자인 변경 | 이미 P0(Liquid Glass)로 처리됨 |
+| 모바일 햄버거 오버레이 디테일 정돈 | 필요 시 P1 home-restructure plan에 흡수 |
+
+## 8. 다음 단계
+
+1. 이 spec 검토 (사용자)
+2. 승인 후 → `superpowers:writing-plans` 스킬로 implementation plan 작성
+3. plan 작성 후 → `superpowers:subagent-driven-development` 로 task별 실행
+4. 모두 종료 후 → 통합 final review + finishing-a-development-branch
+5. Phase 2 spec 별도 작성 시점은 CEO 결정 (운영상 자료 자동 다운로드 수요가 생길 때)
+
+## 부록 A. 영향받는 라우트 / 라우트별 사용 shell
+
+| 라우트 | 현재 shell | 변경 후 shell |
+|---|---|---|
+| `/` | PublicShell | PublicShell (변경 X) |
+| `/services/*` | PublicShell | PublicShell (변경 X) |
+| `/freelance`, `/saju`, `/studio` | PublicShell | PublicShell (변경 X) |
+| `/login`, `/signup` | Standalone | Standalone (변경 X) |
+| `/admin/*` | Standalone (자체 admin shell) | Standalone (변경 X) |
+| `/mypage` | Sidebar (DashboardShell 분기) | **PublicShell + TopNav** |
+| `/payment/*` | PublicShell | PublicShell (변경 X) |
+| `/legal/*`, `/portfolio/[token]`, `/quote/[token]` | PublicShell | PublicShell (변경 X) |
+
+## 부록 B. 카카오 플로팅 버튼 처리 결정
+
+**현재**: `DashboardShell.tsx:76-90`의 `useSidebar=true` 분기에만 mount → mypage 전용으로만 노출.
+
+**변경 후**: mypage가 PublicShell로 옮겨지면 카카오 버튼이 사라짐. 하지만 카카오 1:1 상담은 모든 페이지에서 유효한 도구.
+
+**결정**: 카카오 버튼을 `PublicShell.tsx`로 이동 → 모든 공개 페이지에서 노출 (mypage 포함).
+
+**구체 위치**: `PublicShell.tsx`의 `