# 호령 사주 UI v2 리디자인 — 디자인 문서 - **상태**: Spec 단계 (brainstorming 종료, plan 대기) - **작성일**: 2026-05-26 - **대상 저장소**: `web-ui` (React + Vite, `/saju` 라우트 트리) - **참조 디자인 소스**: `C:\Users\jaeoh\Desktop\workspace\source\images\saju_page\사주풀이\` (백호 사주도사 프로토타입: babel/standalone JSX 11 파일 + styles.css) - **선행 시스템**: saju-lab UI v1 (`web-ui/src/pages/saju/`, 호령 캐릭터 7 PNG 자산 포함) - **백엔드 변경 없음**: saju-lab `/api/saju/*` API는 그대로 사용 --- ## 1. 목적 & 성공 기준 ### 목적 v1의 임시 구조(컴포넌트 12개 직렬 배치, 단일 SajuNav)를 한국 전통 명리학 미학에 충실한 **풀 디자인 시스템**으로 교체. 4 라우트(`/saju`, `/saju/result`, `/saju/today`, `/saju/compatibility`)를 동시 리디자인하고 신규 `/saju/me` placeholder 추가. ### 성공 기준 1. 4 라우트가 새 디자인 토큰/컴포넌트/네비로 일관되게 동작 2. 1024px breakpoint에서 모바일(BottomNav) ↔ 데스크탑(헤더 nav) 자동 전환 3. `useSajuReading` hook + 기존 API 호출 0개 변경, 응답 매핑만 추가 4. 호령 PNG 7개 자산 100% 재사용 (variant API로 추상화) 5. v1 컴포넌트 12개 + SajuNav 제거 — 두 디자인 시스템 동시 유지 X 6. 시각 QA: 골든 패스(메인→입력→result→today→compatibility) + 1024px ± 경계 + me placeholder 모두 정상 --- ## 2. 미학 방향 (Aesthetic Direction) **컨셉**: *한국 전통 명리학 + 차분한 호령 캐릭터*. 디자인 프로토타입이 이미 강하게 commit한 방향을 충실히 옮긴다. ### 2.1 타이포 - **Display**: Nanum Myeongjo (weight 800, `letter-spacing: -0.02em`) — 페이지 타이틀, h1, 큰 한자 - **Body**: Nanum Gothic (weight 400/700, `letter-spacing: -0.01em`) — 본문, 버튼, 캡션 - **Fallback serif**: Gowun Batang - Google Fonts CSS 로드는 `web-ui/index.html`에 link 추가 (페이지 import 대신 — preconnect로 LCP 개선) - Inter/Roboto/system-ui 같은 generic AI sans는 사용 금지 ### 2.2 컬러 시스템 (CSS 토큰) 디자인 프로토타입 `styles.css`의 `:root` 변수를 그대로 도입: | 토큰 | 값 | 용도 | |---|---|---| | `--navy` | `#1F2A44` | dominant body color, dark surface | | `--navy-deep` | `#141B30` | night-bg gradient 하단 | | `--navy-soft` | `#2E3B5A` | 보조 dark | | `--ivory` | `#F7F2E8` | paper 배경, dark surface 위 텍스트 | | `--ivory-soft` | `#FBF7EF` | 카드 배경 | | `--ivory-warm` | `#F0E9D9` | 액센트 배경 | | `--gold` | `#D4AF37` | sharp accent, 보더, ornament | | `--gold-soft` | `#E8C76B` | 활성 상태 텍스트 | | `--gold-dim` | `#B89530` | 비활성 골드 | | `--green` / `--green-soft` / `--green-bg` | 한국 전통 녹색 | 궁합 화면 accent | | `--purple` / `--purple-soft` / `--purple-bg` | `#6A4C7C` 계열 | 사주풀이 accent | | `--pink` / `--pink-deep` / `--pink-bg` | `#F2C7CD` 계열 | 보조 | | `--gray` / `--gray-soft` | `#6B6B6B` / `#9A968D` | 메타 텍스트 | | `--gray-line` / `--gray-line-strong` | 보더 | | `--shadow-card` / `--shadow-pop` / `--shadow-dark` | 그림자 단계 | **화면별 accent 단일 색** (팔레트 골고루 분산 안티패턴 회피): - 홈 (`/saju`) — navy - 오늘 (`/saju/today`) — gold - 궁합 (`/saju/compatibility`) — green - 사주풀이 (`/saju/result`) — purple - 마이 (`/saju/me`) — gray ### 2.3 배경 텍스처 - `.paper-bg` — radial gold/purple wash + 페이퍼 노이즈 (사주풀이, 오늘, 궁합, 마이) - `.night-bg` — 밤하늘 gradient (홈 hero) - `.mt-wash` — 데스크탑 헤더 산수화 SVG decoration (좌하단 + 우하단 산 outline, opacity 0.35) - 단색 배경은 카드 내부에서만 (`--ivory-soft`) ### 2.4 차별화 요소 (UNFORGETTABLE) 1. **OrnateFrame** — 한국 전통 더블 보더 + 4 코너 꺽쇠 SVG (``) 2. **MascotBubble** — 호령 발자국이 매 말풍선마다 `paw-bob` 2.4s ease infinite로 미세 bobbing 3. **OrnamentBloom** — 골드 꽃봉오리 SVG가 모든 섹션 타이틀 좌우 ornament 4. **TopRibbon** — 구름 SVG ribbon이 페이지 상단에 은은히 5. **CharBox** — 사주명식 천간/지지 한자 Nanum Myeongjo 800 + 원소별 색 (목=green, 화=red, 토=earth, 금=gold, 수=blue) ### 2.5 모션 - `screenIn` 0.3s `cubic-bezier(0.16,1,0.3,1)` translateY(6→0) — 라우트 진입 fade-up - `paw-bob` 2.4s ease infinite — 호령 발자국 - BottomNav 활성 항목 배경 색 전환 0.2s - 과한 마이크로 인터랙션 X — "페이지당 1 hero 모션" 원칙 --- ## 3. 아키텍처 & 라우팅 ### 3.1 라우트 매핑 | 라우트 | 디자인 화면 | 파일 | 상태 | |---|---|---|---| | `/saju` | HomeScreen | `Saju.jsx` | **교체** (v1 메인) | | `/saju/result?rid=N` | SajuScreen (4탭) | `SajuResult.jsx` | **교체** (v1 결과) | | `/saju/today?rid=N` | TodayScreen | `Today.jsx` | **교체** (v1 오늘) | | `/saju/compatibility` | MatchScreen | `Compatibility.jsx` | **placeholder → 본격 구현** | | `/saju/compatibility/result?cid=N` | (디자인에 없음) | `CompatibilityResult.jsx` | 디자인 토큰만 라이트 리스타일 | | `/saju/me` | MeScreen placeholder | `Me.jsx` | **신규** | 라우트 수: 5 신규 진입점 + 1 sub. `routes.jsx`에 `/saju/me` lazy import 추가. ### 3.2 디렉토리 구조 ``` web-ui/src/pages/saju/ ├── _shell/ # v2 디자인 시스템 + 네비 │ ├── tokens.css # CSS 변수 정의 │ ├── shell.css # paper-bg, night-bg, mt-wash, OrnateFrame, screenIn │ ├── useViewportMode.js # 1024px breakpoint hook │ ├── BottomNav.jsx # 모바일 5항목 (home/today/match/saju/me) │ ├── DesktopHeader.jsx # 데스크탑 horizontal nav + 로고 │ ├── Mascot.jsx # variant API: full|head|upper|greeting|thinking|pointing|happy │ ├── MascotBubble.jsx # tone: ivory|navy|purple|green │ ├── OrnateFrame.jsx │ ├── OrnamentBloom.jsx │ ├── TopRibbon.jsx │ ├── TitleBlock.jsx │ ├── PrimaryButton.jsx # gold inset shadow │ ├── GhostButton.jsx │ ├── Icons.jsx # 5 nav icon + IconPaw/IconChevron/IconSparkle/IconYinYang │ └── helpers/ │ ├── daeunLabel.js # age → 성장기/학습기/... │ ├── deriveTraits.js # elements + sipsin → 6 성향 │ └── hexA.js # hex → rgba(x,x,x,a) ├── Saju.jsx # routes 진입, useViewportMode → 분기 ├── SajuResult.jsx ├── Today.jsx ├── Compatibility.jsx ├── CompatibilityResult.jsx ├── Me.jsx └── views/ # mobile/desktop 컴포넌트 분리 ├── home.mobile.jsx ├── home.desktop.jsx ├── saju.mobile.jsx # 4탭 (basic/chart/flow/traits) ├── saju.desktop.jsx # 데스크탑은 4탭 그대로 vs 2-column 변형 — plan에서 결정 ├── today.mobile.jsx ├── today.desktop.jsx ├── match.mobile.jsx └── match.desktop.jsx ``` **Me 페이지는 mobile/desktop 분리 안 함** (placeholder라 단순 — `Me.jsx` 본문에 직접 구현). ``` 기존 v1 파일들: - `components/` 디렉토리 **전체 삭제** (SajuNav, HoryungMascot, SajuInputForm, ActionCard, SajuPillars, ElementBarChart, FortuneRing, ScoreCard, LuckyBox, InterpretAccordion, MonthlyFlow, HoryungQuote) - `hooks/useSajuForm.js`, `hooks/useSajuReading.js` 유지 (데이터 흐름) - `Saju.css` 신규 `_shell/tokens.css` + `_shell/shell.css`로 교체 --- ## 4. 컴포넌트 명세 ### 4.1 `useViewportMode()` ```js function useViewportMode() { const [mode, setMode] = useState(() => typeof window !== 'undefined' && window.innerWidth >= 1024 ? 'desktop' : 'mobile' ); useEffect(() => { const onResize = () => { const next = window.innerWidth >= 1024 ? 'desktop' : 'mobile'; setMode(prev => (prev === next ? prev : next)); }; window.addEventListener('resize', onResize); return () => window.removeEventListener('resize', onResize); }, []); return mode; } ``` - 디자인 프로토타입의 동일 hook 그대로 포팅 - SSR 안전 (typeof window 체크) — Vite 기본 CSR이라 항상 window 존재하지만 방어 - debounce 없음 — resize 빈도가 낮고 setState가 동일 값일 때 reflow 없음 (Object.is 비교) ### 4.2 `` | variant | 매핑 PNG (기존 v1 자산) | |---|---| | `full` | `/images/saju/horyung/horyung-main.png` | | `head` | `/images/saju/horyung/horyung-bust.png` (얼굴 중심 crop) | | `upper` | `/images/saju/horyung/horyung-front.png` | | `greeting` | `/images/saju/horyung/horyung-greeting.png` | | `thinking` | `/images/saju/horyung/horyung-thinking.png` | | `pointing` | `/images/saju/horyung/horyung-pointing.png` | | `happy` | `/images/saju/horyung/horyung-happy.png` | props: `variant`, `size` (px), `style` (override). ``. ### 4.3 `` - `position: fixed; bottom: 0` — iPhone frame이 아닌 실제 모바일 뷰포트의 하단 - 5 아이템: home/today/match/saju/me. NavLink 사용으로 라우트 매핑 (`useLocation`으로 current 결정) - theme: `'ivory'` (paper 배경) / `'navy'` (night 배경) — backdrop-filter blur 적용 - 활성 항목: 화면별 accent 색 배경(opacity 0.10~0.18) + 라벨 weight 700 ### 4.4 `` - `position: sticky; top: 0; z-index: 30` — 스크롤 시 상단 고정 - 좌측: 로고 (`壽` 한자 + "호령사주" Nanum Myeongjo) - 중앙: nav 5 링크 (BottomNav와 동일 항목, horizontal 배치) - 우측: 미사용 (향후 me 메뉴) - 배경: `--ivory-soft` + 하단 `--gray-line` 1px ### 4.5 `` - 디자인 프로토타입 `common.jsx`의 OrnateFrame 그대로 포팅 - `double=true`면 inset 4px 위치에 추가 보더 - 4 코너 꺽쇠 SVG (rotate 0/90/180/270) ### 4.6 `` - tone 팔레트 (`ivory`/`navy`/`green`/`purple`) → bg/border/text 색 - `paw=true`면 우하단 IconPaw + `paw-bob` 애니메이션 - `tail=true`면 풍선 꼬리 (rotate 45deg 사각형) ### 4.7 Buttons - `PrimaryButton`: gold inset shadow (`inset 0 1px 0 rgba(212,175,55,0.4)`) + 풀 너비 옵션 - `GhostButton`: 투명 배경 + 보더만, 동일 폰트/spacing ### 4.8 `Me.jsx` (placeholder, mobile/desktop 공통) - `paper-bg` + `` + `` + `` "곧 만나요" + 비활성 카드 4개 (이력/북마크/설정/문의 — disabled) - 뷰포트 분리 없이 단일 컴포넌트 (placeholder라 단순) ### 4.9 입력 폼 컴포넌트 (Home에서 사용) - `` — 디자인 토큰 적용된 단일 행 (label 좌측 64px + input 우측) - ``, ``, ``, `` (양/음력) — `useSajuForm` state와 연결 - Phase 2에서 신설. 기존 v1 `SajuInputForm.jsx`의 검증 로직만 이식, 시각 표현은 새 디자인 --- ## 5. 데이터 흐름 ### 5.1 hook 재사용 - `useSajuReading(rid)` — 그대로 유지. `api.js`의 `sajuGetReading(id)` 호출 → `reading` 객체 반환 - `useSajuForm()` — 그대로 유지. 입력 검증 + `sajuInterpret(body)` 호출 + navigate ### 5.2 매핑 헬퍼 (`_shell/helpers/`) #### `daeunLabel(age)` → string - `age < 10` → "성장기" - `age < 20` → "학습기" - `age < 30` → "도전기" - `age < 40` → "성장기" - `age < 50` → "전성기" - `age < 60` → "안정기" - `age < 70` → "정리기" - `age >= 70` → "여유기" #### `deriveTraits(elements, sipsin)` → `[{id, ko, icon, color}]` (최대 6개) - 강한 원소 1~2개 → 매칭 성향: - `fire >= 30` → `{id:'challenge', ko:'도전정신', color:'#C04A4A'}` - `metal >= 30` → `{id:'lead', ko:'리더십', color:'#D4AF37'}` - `wood >= 30` → `{id:'adapt', ko:'적응력', color:'#4E6B5C'}` - `water >= 30` → `{id:'wisdom', ko:'지혜', color:'#3A5A8C'}` - `earth >= 30` → `{id:'wealth', ko:'풍부함', color:'#A67B3F'}` - 일간 강도 (신강/신약) → `will` (의지) - 결과 6개 미만이면 다음으로 강한 원소 추가 - 순서: 강한 원소 점수 내림차순 #### `hexA(hex, alpha)` → `rgba(...)` 문자열 - 디자인 프로토타입 동일 헬퍼 ### 5.3 SAJU_DATA mock → 실제 API 매핑 표 | 디자인 mock 필드 (screen-saju.jsx) | API 응답 경로 (saju-lab) | 비고 | |---|---|---| | `name`, `birth`, `gender`, `birthTime`, `birthPlace` | `reading.input.*` | 직접 매핑 | | `sajuLabel` | `reading.label` | "경오년 신사월 갑자일 OO시" | | `ilgan` | `reading.ilgan` | `{ko, ch, element, sound}` | | `pillars[]` | `reading.pillars` | year/month/day/hour 4기둥 | | `pillars[].cheongan.color` | 원소→색 매핑 (`elementColor()`) | wood=green, fire=red, earth=earth, metal=gold, water=blue | | `pillars[].sipsin`, `jijang` | `reading.pillars[i].sipsin`, `jijang` | | | `ohaeng[]` | `reading.analysis.elements` | `{wood, fire, earth, metal, water}` → `[{id, ko, ch, value, color}]` 변환 | | `daeun[]` | `reading.daeun` (8개) | `label`은 `daeunLabel(age)` 헬퍼, `current`는 현재 나이 기반 derive | | `traits[]` | `deriveTraits(elements, sipsin)` | 헬퍼로 derive (API 응답에 직접 없음) | | TraitsTab `title`, `desc` | 상위 3 성향 → 정적 desc 사전 매핑 | YAGNI: 백엔드에 trait description 추가는 향후 작업 | | Today: `fortune_scores`, `lucky`, `monthly_flow` | API 응답에 이미 존재 | 그대로 사용 | ### 5.4 BottomNav active state ```jsx const { pathname } = useLocation(); const current = pathname === '/saju' ? 'home' : pathname.startsWith('/saju/today') ? 'today' : pathname.startsWith('/saju/compatibility') ? 'match' : pathname.startsWith('/saju/result') ? 'saju' : pathname.startsWith('/saju/me') ? 'me' : 'home'; ``` --- ## 6. 반응형 & 네비게이션 전략 ### 6.1 1024px breakpoint - `< 1024px` → 모바일: 페이지 컴포넌트가 `` 렌더, `` 표시 - `>= 1024px` → 데스크탑: `` 렌더, `` 표시 - 페이지 진입 시 `useViewportMode()`가 결정. resize 시 동적 전환 ### 6.2 iPhone frame 제거 - 디자인 프로토타입은 모바일 미리보기용으로 iPhone 외곽선을 그렸으나 실제 모바일 디바이스는 OS frame이 있으므로 frame DOM 제거 - StatusBar(`BrandStatusBar`)도 미사용 — 실제 디바이스 status bar 자연스럽게 사용 ### 6.3 컨테이너 max-width - 모바일: `100%` (BottomNav만 fixed) - 데스크탑: 콘텐츠 max-width 1200px, `margin: 0 auto`. mt-wash 배경은 viewport 풀 ### 6.4 transition between modes - 1024px 경계에서 mode 변경 시 컴포넌트가 unmount → 새 컴포넌트 mount → screenIn 0.3s 재생 - 폼 입력 중 transition 발생 시: useSajuForm 상태는 hook이 보관하므로 데이터 유실 X --- ## 7. 점진적 구현 단계 (Phase Plan) 각 Phase 끝에 `npm run dev`로 `http://localhost:3007/saju` 시각 확인 + git commit. PR은 Phase 1~3, 4~5, 6 (fixup) 3개로 분할 권장. | Phase | 산출물 | 검증 | |---|---|---| | **1. Shell + 토큰** | `_shell/` 전체 + `Me.jsx` + 라우트 `/saju/me` 추가 + Google Fonts link | `/saju/me` 진입 시 placeholder + BottomNav/Header 모두 정상. 기존 4 페이지 무손상 | | **2. Home** | `Saju.jsx` + `views/home.{mobile,desktop}.jsx` + 입력 폼 + 호령 hero | 모바일/데스크탑 모두 입력 → submit → `/saju/result?rid=N` 이동 | | **3. SajuResult** | `SajuResult.jsx` + `views/saju.{mobile,desktop}.jsx` 4탭 + 매핑 헬퍼 | 실제 reading 데이터로 4탭 모두 정상 표시. 일간 표시·오행 막대·대운 흐름·성향 derive 검증 | | **4. Today** | `Today.jsx` + `views/today.{mobile,desktop}.jsx` | fortune_scores·lucky·monthly_flow 표시. PrimaryButton "다른 운세 보기" → SajuResult 이동 | | **5. Compatibility** | `Compatibility.jsx` + `views/match.{mobile,desktop}.jsx` 본격 구현. `CompatibilityResult.jsx` 라이트 리스타일 | 두 사람 입력 폼 + compat API 호출 + 결과 화면 | | **6. QA + cleanup** | v1 `components/` 삭제, `Saju.css` 제거, 시각 QA, 1024px 경계 chrome devtools | 골든 패스 통과, dead code 없음 | --- ## 8. 에러 / 빈 상태 | 상황 | UI | |---|---| | API 실패 (네트워크/500) | `` + `` "아이고, 다시 시도해주세요" + `` 새로고침 | | `?rid=` 없이 `/saju/result` 직접 진입 | `` "사주를 먼저 입력해주세요" + `` "사주 입력하러 가기" → `/saju` | | `?rid=` 없이 `/saju/today` 직접 진입 | 동일 패턴, accent gold | | `?cid=` 없이 `/saju/compatibility/result` 진입 | 동일 패턴, accent green | | `/saju/me` | `` "곧 만나요" + 비활성 placeholder 카드 4개 | | 백엔드 timeout (사주 해석 30~60초) | 로딩 화면: `` + `` "호령이 풀이 중이에요..." + spinner | --- ## 9. 검증 전략 ### 9.1 자동 테스트 - `useViewportMode.test.js` — `vi.mock` window.innerWidth + resize 이벤트 dispatch, 1023/1024 경계 변환 확인 - `daeunLabel.test.js` — 8 구간 모두 정답 매핑 - `deriveTraits.test.js` — 강한 원소 1~5개 입력에 대한 정렬·중복 제거 확인 - `Mascot.test.jsx` — 7 variant 모두 올바른 src prop ### 9.2 시각 검증 (Phase 마다 dev server) 1. `npm run dev` → `http://localhost:3007/saju` 진입 2. 모바일 chrome devtools (375×667 iPhone SE, 390×844 iPhone 12) 3. 데스크탑 (1280×720 이상) 4. 1024px 경계 ± 1px (1023↔1024)에서 mode 전환 확인 5. 5 라우트 모두 BottomNav active 상태 + DesktopHeader active 상태 일치 6. 호령 PNG 7 variant 모두 로드 확인 (Network 탭) 7. 폰트 로드 (Nanum Myeongjo, Nanum Gothic, Gowun Batang) ### 9.3 회귀 - 기존 reading_id URL 호환 (`/saju/result?rid=N` 패턴 유지) - `useSajuReading` hook 응답 매핑이 v1과 동일 데이터 표시 - saju-lab API 호출 0개 변경 (네트워크 탭 비교) --- ## 10. YAGNI 명시 제외 다음은 이번 v2에서 의도적으로 **제외**: - i18n / 다국어 - 다크모드 토글 (디자인 자체가 화면별 light/dark scope 고정) - 호령 마스코트 드래그·물리 모션 (paw-bob bobbing만) - BottomNav 햅틱·진동 - 인증/로그인 (Me는 placeholder, 향후 별도 spec) - PWA / 오프라인 캐시 - 백엔드 trait description API (`deriveTraits` 프론트 헬퍼로 충분) - 디자인 프로토타입의 desktop-shell.jsx full conversion — DesktopHeader만 차용, shell 전체는 v2 컨테이너에 흡수 --- ## 11. 마이그레이션 노트 ### 11.1 삭제 대상 (Phase 6에서 일괄 정리) - `web-ui/src/pages/saju/components/` 전체 12 파일 - `web-ui/src/pages/saju/Saju.css` - v1 `Compatibility.jsx`의 placeholder 본문 (본격 구현으로 교체) ### 11.2 보존 대상 - `web-ui/src/pages/saju/hooks/useSajuForm.js`, `useSajuReading.js` (데이터 흐름) - `web-ui/public/images/saju/horyung/` 7 PNG 자산 (Mascot variant API가 매핑) - `web-ui/src/api.js` saju 헬퍼 함수들 ### 11.3 routes.jsx 변경 기존 import 라인 유지 + Me lazy import 추가: ```diff + const SajuMe = lazy(() => import('./pages/saju/Me')); ``` `children` 배열에 me 라우트 추가: ```diff path: '/saju', children: [ { index: true, element: }, { path: 'result', element: }, { path: 'today', element: }, { path: 'compatibility', element: }, { path: 'compatibility/result', element: }, + { path: 'me', element: }, ], ``` --- ## 12. Plan 단계로 넘길 결정 사항 다음은 plan 작성 시 구체화: - 각 view 파일별 line budget (현실적 500~800 라인 예상, 더 크면 sub 컴포넌트 분할) - 색→원소 매핑 함수 (`elementColor(elementId)`) 위치 — `_shell/helpers/` vs view 안 인라인 - 데스크탑 `saju.desktop.jsx`의 4탭 유지 vs 2-column 변형 (디자인 프로토타입의 `desktop-saju.jsx` 상세 검토 후 결정) - 데스크탑 헤더의 me 메뉴 (향후 인증 위치 — 현재는 nav 5번째 링크) - 시각 QA 시 사용자 직접 확인 단계 (Claude가 puppeteer로 자동화하지 않음 — 시각 판단은 사람) - `` 등 입력 컴포넌트의 상세 props 시그니처