oh-my-design-cli 1.3.0 → 1.3.9

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -37,6 +37,49 @@ description: "선택된 reference brand의 라이브 사이트에서 디자인
37
37
  - 명시: "X 에셋 가져와줘", "X 사이트 패칭해줘", "X 라이브 스타일 추출", "X reference 캡쳐"
38
38
  - 묵시: omd:harness 안에서 reference 선정 후, 또는 사용자가 "X처럼 만들어줘" 요청 시 omd:init 후속 작업으로 자동 제안
39
39
 
40
+ ## Phase 0 — (v1.3.3 폐기) Mode 선택
41
+
42
+ 이전 버전(v1.3.2)은 `clone` vs `inspired` 두 mode를 제공했으나 v1.3.3에서 폐기. 시각적 동일성은 brand creative work을 사용자 product에 reproduce해야 가능하고 그건 IP 영역. 단일 mode 흐름으로 통일 — brand 토큰·구조·voice는 가져오되, brand 자체 자산(mascot·로고·마케팅 사진)은 reference로만 보존하고 사용자 product에는 자체 자산 자리(`[YOUR LOGO]` placeholder 등)를 둠. 결과물의 시각 polish는 무료 라이선스 자산 라이브러리(Open Peeps / Lucide / Heroicons 등 CC0/MIT/SIL OFL)로 채움 — 자세한 카탈로그는 `skills/omd-harness/SKILL.md` Step 4 master prompt rule 6 참조.
43
+
44
+ 다음 모든 Phase는 단일 흐름. (구버전 Phase 0 - clone/inspired ask 제거됨.)
45
+
46
+ ## (legacy reference) — 이전 mode 선택 텍스트 (참고용, 동작 안 함)
47
+
48
+ reference-capture가 어디까지 가져올지는 사용자 의도에 따라 두 갈래. 호출 진입 시점에 mode가 결정되지 않았다면 사용자에게 한 번에 묻기:
49
+
50
+ ```
51
+ <id>를 어떻게 활용할까요?
52
+
53
+ 1. clone — 거의 똑같이 시작. 실제 로고·일러스트·폰트 받아와서 dev scaffold 구성.
54
+ landing이 라이브 사이트와 시각적으로 매우 비슷하게 시작됩니다.
55
+ ⚠ 자동으로 CLONE-MODE.md 배너 + replace-checklist.md가 생성되고,
56
+ "사용자 product에 ship 전에 brand 자산을 자체 자산으로 교체 필요"라고 표시됩니다.
57
+
58
+ 2. inspired — 톤·체계만 가져옴. [YOUR LOGO] placeholder, 일러스트는 generic placeholder.
59
+ 브랜드의 voice·원칙·팔레트 철학만 적용. 바로 ship 가능한 상태로 산출.
60
+
61
+ 답: clone / inspired (기본값: inspired)
62
+ ```
63
+
64
+ 이 선택은 `.omd/init-context.json`의 `mode` 필드에 저장되어 후속 omd:init / omd:harness / omd:apply가 일관되게 사용한다.
65
+
66
+ 이미 omd:harness Step 3.7에서 mode를 묻고 진입했으면 Phase 0 skip.
67
+
68
+ ### Mode별 동작 요약
69
+
70
+ | 단계 | clone | inspired |
71
+ |---|---|---|
72
+ | LICENSE-NOTE.md | 작성 ✓ | 작성 ✓ |
73
+ | tokens.json (atomic facts) | 캡쳐 ✓ | 캡쳐 ✓ |
74
+ | structure.json (composition facts) | 캡쳐 ✓ | 캡쳐 ✓ |
75
+ | logo.<ext> | 캡쳐 + product `<img>`로 사용 가능 (banner 의무) | 캡쳐만 (product에 미사용, placeholder 강제) |
76
+ | screenshots/ | 캡쳐 ✓ | 캡쳐 ✓ |
77
+ | fonts.json (CDN URLs) | 캡쳐 + 자동 `<link>` 로드 강제 | 캡쳐만 (수동 로드) |
78
+ | hero illustration assets | (있고 publicly accessible면) URL 기록 + 사용 가능 | URL만 기록 |
79
+ | attribution.md | 작성 + 사용 표시 | 작성 |
80
+ | **CLONE-MODE.md (project root)** | **mandatory 작성** | 미작성 |
81
+ | **replace-checklist.md (project root)** | **mandatory 작성** | 미작성 |
82
+
40
83
  ## 전체 플로우
41
84
 
42
85
  ```
@@ -145,8 +188,156 @@ This note is generated by `omd:reference-capture`. Do not edit by hand
145
188
  — rerun the skill to refresh.
146
189
  ```
147
190
 
191
+ ## Phase 3.5 — Font 캡쳐 (fonts.json, 라이브 폰트가 실제 로드되도록)
192
+
193
+ 캡쳐된 brand가 web font(Pretendard / BM JUA / Inter / Noto Sans KR 등 시스템 기본이 아닌 폰트)를 쓰면, 토큰만 잡아도 생성기가 폰트를 **로드 안 하면** 결과가 시스템 fallback으로 바뀐다 (가장 흔한 증상: macOS 시스템 폰트가 둥글둥글하게 렌더되는 mismatch).
194
+
195
+ `assets/_reference/<id>/fonts.json`:
196
+
197
+ ```json
198
+ {
199
+ "captured_at": "<ISO-8601>",
200
+ "fonts": [
201
+ {
202
+ "family": "Pretendard",
203
+ "license": "SIL OFL 1.1 (open-source, free for commercial use)",
204
+ "cdn_url": "https://cdn.jsdelivr.net/gh/orioncactus/pretendard/dist/web/static/pretendard.css",
205
+ "html_link": "<link rel=\"stylesheet\" href=\"https://cdn.jsdelivr.net/gh/orioncactus/pretendard/dist/web/static/pretendard.css\" />",
206
+ "live_observed": true,
207
+ "role": "body + heading"
208
+ },
209
+ {
210
+ "family": "Apple SD Gothic Neo",
211
+ "license": "system (macOS/iOS)",
212
+ "cdn_url": null,
213
+ "html_link": null,
214
+ "live_observed": false,
215
+ "role": "system fallback"
216
+ }
217
+ ]
218
+ }
219
+ ```
220
+
221
+ ### 3.5.1 추출 절차
222
+
223
+ 1. live homepage의 computed style에서 `body` / `h1-h3` / 주요 buttons의 `font-family` 첫 항목 추출 (fallback chain 무시, 첫 토큰만)
224
+ 2. 추출된 family를 known-font registry에 매칭:
225
+
226
+ | 추출된 family | license | CDN URL |
227
+ |---|---|---|
228
+ | `Pretendard` | SIL OFL 1.1 | `https://cdn.jsdelivr.net/gh/orioncactus/pretendard/dist/web/static/pretendard.css` |
229
+ | `Pretendard Variable` | SIL OFL 1.1 | `https://cdn.jsdelivr.net/gh/orioncactus/pretendard/dist/web/variable/pretendardvariable.css` |
230
+ | `Noto Sans KR` | SIL OFL 1.1 | `https://fonts.googleapis.com/css2?family=Noto+Sans+KR:wght@300;400;500;700;900&display=swap` |
231
+ | `Inter` | SIL OFL 1.1 | `https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700&display=swap` |
232
+ | `Wanted Sans` | SIL OFL 1.1 | `https://cdn.jsdelivr.net/gh/wanteddev/wanted-sans@latest/packages/wanted-sans/fonts/webfonts/static/complete/WantedSans.css` |
233
+ | `BM JUA` (배민 주아체) | 배달의민족 폰트 라이선스 (개인·기업 무료, **재배포·판매 금지**) | (CDN 미공식 — `html_link: null` 처리, 사용자에게 "BM JUA는 라이브에서 관측 안 됨; 로컬 fallback" 알림) |
234
+ | `Apple SD Gothic Neo`, `Malgun Gothic`, `system-ui`, `-apple-system` | system | `html_link: null` |
235
+
236
+ registry에 없는 family는 `html_link: null` + `notes`에 "CDN 미확인, 수동 로드 필요" 기록.
237
+
238
+ ### 3.5.2 BM JUA 같은 misapplication 가드
239
+
240
+ canonical DESIGN.md가 "BM JUA를 landing/promo accent로"라고 적어뒀더라도, **라이브 inspect에서 실제 BM JUA 사용이 관측 안 되면** `live_observed: false`로 기록하고 후속 generator가 적용 안 하도록 신호. omd:init Phase 5B의 priority rule이 `live_overrides` 우선 처리.
241
+
242
+ ### 3.5.3 Clone mode 강제 로드
243
+
244
+ mode=clone이면 omd:init/omd:harness가 생성 HTML `<head>`에 fonts.json의 `live_observed: true` 항목 `html_link`를 **반드시** 박는다. mode=inspired면 폰트 로드는 사용자 선택.
245
+
246
+ ## Phase 3.9 — Browser harness 자동 선택 (v1.3.6 신설)
247
+
248
+ 라이브 캡쳐 도구는 두 가지 옵션. 환경 detect 후 빠른 쪽 사용:
249
+
250
+ ### Fast-path: browser-harness (있으면 우선)
251
+
252
+ `shutil.which("browser-harness")` 또는 `command -v browser-harness`로 detect. 있고 `browser-harness --doctor` 가 `[ok ] chrome running` 띄우면 fast-path:
253
+
254
+ - **장점**: heredoc 단일 호출 + Chrome remote-debugging CDP 직접 → playwright MCP 대비 3-5x 빠름 (실측 3.5s/페이지). sub-agent의 "did live inspect" 거짓 보고 불가능 (heredoc 실행 자체가 evidence).
255
+ - **prerequisite**: Python 3.11+, uv 또는 pip, Chrome with `--remote-debugging-port=9222` 또는 `chrome://inspect#remote-debugging` 활성.
256
+ - **호출 패턴**:
257
+ ```bash
258
+ BU_CDP_URL="http://localhost:9222" browser-harness <<'PY'
259
+ new_tab("https://<brand-domain>")
260
+ wait_for_load()
261
+ import time; time.sleep(1.5) # SPA hydration
262
+ result = js("""
263
+ (() => {
264
+ const body = getComputedStyle(document.body);
265
+ const ctas = [...document.querySelectorAll('button, a[role="button"], [class*="button"]')]
266
+ .filter(el => {
267
+ const r = el.getBoundingClientRect();
268
+ return r.height >= 32 && r.height <= 80 && r.top < 1200;
269
+ })
270
+ .slice(0, 8).map(el => {
271
+ const cs = getComputedStyle(el);
272
+ return { bg: cs.backgroundColor, color: cs.color, radius: cs.borderRadius,
273
+ padding: cs.padding, fontWeight: cs.fontWeight, h: Math.round(el.getBoundingClientRect().height) };
274
+ });
275
+ return JSON.stringify({
276
+ bodyFont: body.fontFamily.split(',')[0].replace(/['"]/g,'').trim(),
277
+ ctas, scrollH: document.documentElement.scrollHeight
278
+ });
279
+ })()
280
+ """)
281
+ print(result)
282
+ capture_screenshot(path="<assets_dir>/screenshots/hero-desktop.png", full=False)
283
+ PY
284
+ ```
285
+ - 결과를 `tokens.json#live_overrides` + `.live-inspect-proof.json` raw_samples에 그대로 박음.
286
+
287
+ ### Fallback: playwright MCP (default)
288
+
289
+ browser-harness 미설치면 `mcp__playwright__browser_navigate` + `browser_evaluate` + `browser_take_screenshot` 사용 (이전 버전 동작 그대로).
290
+
291
+ ### 자동 선택 logic
292
+
293
+ ```bash
294
+ if command -v browser-harness >/dev/null && browser-harness --doctor 2>&1 | grep -q "ok.*chrome running"; then
295
+ echo "MODE=harness"
296
+ else
297
+ echo "MODE=playwright"
298
+ fi
299
+ ```
300
+
301
+ ### 사용자 안내 (선택)
302
+
303
+ skill 진입 시 사용자에게 한 줄로 알림:
304
+ > "browser-harness 감지됨 — fast-path 사용 (3.5s/page)" 또는 "playwright MCP 사용 — browser-harness 설치하면 3x 빠름: github.com/browser-use/browser-harness"
305
+
306
+ 이 안내는 informational only, 사용자 action 요구 X.
307
+
148
308
  ## Phase 4 — 토큰 캡쳐 (facts)
149
309
 
310
+ **Proof gate (위반 = regression)**: 이 Phase는 실제 playwright navigate + computed style 추출이 일어났음을 후속 generator에게 증명하는 artifact를 남겨야 함. 이전 버전(v1.3.3)에서 sub-agent가 "라이브 inspect 했다"고 보고하지만 실제로는 canonical 값만 복사하는 결함이 관측됨. proof gate가 이를 차단:
311
+
312
+ ### 4.0 — `.live-inspect-proof.json` 작성 (REQUIRED, 이 파일 없으면 후속 generator는 live_overrides 무시)
313
+
314
+ playwright `browser_navigate` 직후 `browser_evaluate`로 raw computed style 5개 이상 추출하고 결과를 `assets/_reference/<id>/.live-inspect-proof.json`에 저장:
315
+
316
+ ```json
317
+ {
318
+ "navigated_at": "<ISO-8601 with seconds>",
319
+ "source_url": "https://www.banksalad.com",
320
+ "viewport": "1280x800",
321
+ "tool_used": "playwright_mcp_browser_evaluate",
322
+ "raw_samples": [
323
+ { "element_selector": "body", "font-family": "Pretendard, ...", "color": "rgb(0, 0, 0)", "background": "rgba(0, 0, 0, 0)" },
324
+ { "element_selector": "header nav a:first-child", "font-size": "14px", "font-weight": "500" },
325
+ { "element_selector": "button:contains('앱 다운로드')", "background-color": "rgb(19, 189, 126)", "border-radius": "41px" },
326
+ { "element_selector": ".hero-section section", "background-color": "rgba(0, 0, 0, 0)" },
327
+ { "element_selector": "footer", "background-color": "rgb(255, 255, 255)" }
328
+ ],
329
+ "playwright_response_hash": "<5+ char of returned evaluate result>"
330
+ }
331
+ ```
332
+
333
+ 후속 generator (omd:init / omd:harness)는 이 파일이 **존재하고 raw_samples 길이 ≥5** 이어야만 `tokens.json#live_overrides`를 신뢰. 파일 없거나 비어있으면 canonical만 사용.
334
+
335
+ ### 4.1 — drift detection (proof와 tokens.json#live_overrides 일관성)
336
+
337
+ tokens.json의 `live_overrides` 값이 캡쳐된 canonical 값과 **byte-for-byte 동일**하면 live inspect가 실제로는 안 돈 것 → live_overrides 블록을 통째로 삭제하고 사용자에게 "live inspect 결과가 canonical과 동일 — 차이 없거나 inspect 실패. tokens.json#live_overrides 미사용."이라고 알림. metadata note만 적고 실제 hex override 없는 거짓 live_overrides 금지.
338
+
339
+ ### 4.2 — token 캡쳐 본문
340
+
150
341
  playwright MCP를 사용해 homepage에 navigate한 뒤 computed styles를 추출. 결과는 `assets/_reference/<id>/tokens.json`:
151
342
 
152
343
  ```json
@@ -185,6 +376,113 @@ playwright MCP를 사용해 homepage에 navigate한 뒤 computed styles를 추
185
376
 
186
377
  inspect 패턴은 `.claude/skills/omd-add-reference/SKILL.md`의 "apple.com 류 라이브 inspect 패턴"과 동일. 5-10 elements 샘플링.
187
378
 
379
+ ## Phase 4.5 — 구조 cue 캡쳐 (structure.json)
380
+
381
+ Tokens는 색·radius·font 같은 원자 값이고, **structure**는 페이지가 어떻게 짜였는지의 관측 가능한 facts다. 토큰만 보면 같은 brand도 surface마다 다른 결과물이 나오는 mismatch가 생긴다 (예: app surface의 2px-radius advisor 톤 vs marketing landing의 41px-pill 일러스트 톤). structure.json은 후속 생성기가 **실제 라이브 사이트의 composition idiom**을 따라가도록 한다.
382
+
383
+ ### 4.5.1 추출 항목 (facts only — 저작권 영역 0)
384
+
385
+ `assets/_reference/<id>/structure.json`:
386
+
387
+ ```json
388
+ {
389
+ "captured_at": "<ISO-8601>",
390
+ "source_url": "https://www.banksalad.com",
391
+ "viewport": "1280x800",
392
+ "hero": {
393
+ "type": "illustration | photo | video | data-card | text-only | mixed",
394
+ "has_carousel": false,
395
+ "carousel_dot_count": 0,
396
+ "primary_visual_position": "right | left | center | full-width | background",
397
+ "background_ornament": "none | gradient-blob | geometric | radial-tint | photo-bleed",
398
+ "approx_height_vh": 75
399
+ },
400
+ "cta": {
401
+ "dominant_shape": "pill | rectangular | ghost-only | text-link",
402
+ "primary_radius_px_observed": 41,
403
+ "secondary_style": "ghost-outline | underline | none",
404
+ "vertical_stack": true
405
+ },
406
+ "nav": {
407
+ "structure": "horizontal-categories | mega-menu | hamburger-mobile | single-row-utility | top-tabs",
408
+ "approx_item_count": 11,
409
+ "has_search": false,
410
+ "has_locale_switch": false
411
+ },
412
+ "page": {
413
+ "viewport_heights": 3.9,
414
+ "section_count_observed": 8,
415
+ "predominant_alignment": "center | left | grid-asymmetric",
416
+ "image_density": "high | medium | low | none"
417
+ },
418
+ "section_rhythm": [
419
+ "hero",
420
+ "social-proof",
421
+ "feature-grid",
422
+ "testimonial",
423
+ "footer"
424
+ ],
425
+ "notes": [
426
+ "라이브 hero는 carousel — 3-5 슬라이드로 product 카테고리 순환",
427
+ "section_rhythm은 위에서 아래로 관측 순서"
428
+ ]
429
+ }
430
+ ```
431
+
432
+ ### 4.5.2 추출 방법
433
+
434
+ playwright로 homepage navigate 후 다음 JS 실행 (facts derivation only):
435
+
436
+ ```js
437
+ () => {
438
+ // hero 탐색 — 보통 첫 800px 이내 viewport에서 가장 큰 visual element
439
+ const heroCandidates = [...document.querySelectorAll('section, header + div, [class*="hero" i], [class*="Hero"]')]
440
+ .filter(el => {
441
+ const r = el.getBoundingClientRect();
442
+ return r.top < 200 && r.height > 300 && r.width > 800;
443
+ });
444
+ const heroEl = heroCandidates[0];
445
+
446
+ // type 판정
447
+ const heroImgCount = heroEl ? heroEl.querySelectorAll('img, svg, picture').length : 0;
448
+ const heroVideoCount = heroEl ? heroEl.querySelectorAll('video').length : 0;
449
+ const heroHasCarouselDots = heroEl ? !!heroEl.querySelector('[class*="dot" i], [class*="pagination" i], [class*="indicator" i]') : false;
450
+
451
+ // CTA 탐색 — viewport 위 절반의 button/cta
452
+ const ctas = [...document.querySelectorAll('button, a[role="button"], [class*="button" i] a')]
453
+ .filter(el => el.getBoundingClientRect().top < 800)
454
+ .map(el => {
455
+ const cs = getComputedStyle(el);
456
+ return { radius: cs.borderRadius, bg: cs.backgroundColor };
457
+ });
458
+ const dominantRadii = ctas.map(c => parseInt(c.radius)).filter(n => !isNaN(n));
459
+ const maxRadius = Math.max(...dominantRadii, 0);
460
+
461
+ // nav
462
+ const navEl = document.querySelector('header nav, [role="navigation"]');
463
+ const navItemCount = navEl ? navEl.querySelectorAll('a, button').length : 0;
464
+
465
+ return {
466
+ hero_img_count: heroImgCount,
467
+ hero_video_count: heroVideoCount,
468
+ hero_has_carousel_dots: heroHasCarouselDots,
469
+ cta_max_radius_observed: maxRadius,
470
+ cta_sample_count: ctas.length,
471
+ nav_item_count: navItemCount,
472
+ total_scroll_height_px: document.documentElement.scrollHeight,
473
+ viewport_height_px: window.innerHeight,
474
+ };
475
+ }
476
+ ```
477
+
478
+ 위 raw 측정값을 structure.json schema의 enum 값으로 매핑 (예: `hero_img_count >= 1 && hero_video_count === 0` → `hero.type: "illustration"`, `cta_max_radius_observed >= 40` → `cta.dominant_shape: "pill"`).
479
+
480
+ ### 4.5.3 IP 가드레일 재확인
481
+
482
+ structure.json에는 텍스트·이미지·copy·brand statement 일체 X. 오로지 **structural facts** (개수, radius, viewport-relative position). 저작권 영역 아님.
483
+
484
+ `notes` 필드의 자유 텍스트는 **본인 관찰**만 — 사이트의 마케팅 카피 인용 금지.
485
+
188
486
  ## Phase 5 — 시각 reference 캡쳐
189
487
 
190
488
  ### 5.1 로고 다운로드
@@ -229,6 +527,7 @@ Captured: **<ISO-8601>** via `omd:reference-capture`
229
527
  | File | Source URL | Captured | Notes |
230
528
  |---|---|---|---|
231
529
  | `tokens.json` | `<homepage_url>` | <date> | Computed styles via playwright, factual analysis |
530
+ | `structure.json` | `<homepage_url>` | <date> | Observable composition facts (hero type, CTA shape, nav structure, page rhythm) |
232
531
  | `logo.<ext>` | `<logo_url>` | <date> | Brand trademark — do not redistribute |
233
532
  | `screenshots/hero-desktop.png` | `<homepage_url>` (1280×800) | <date> | Reference for design alignment |
234
533
  | `screenshots/hero-mobile.png` | `<homepage_url>` (375×812) | <date> | Reference for design alignment |
@@ -250,6 +549,103 @@ directory). Captured tokens reflect the live site at capture time and
250
549
  may drift as the brand evolves.
251
550
  ```
252
551
 
552
+ ## Phase 6.5 — Clone mode 전용 산출 (mode=clone일 때만)
553
+
554
+ ### 6.5.1 `CLONE-MODE.md` (project root, mandatory)
555
+
556
+ clone mode 진입 시점에 프로젝트 루트(`./CLONE-MODE.md`)에 **다른 산출보다 먼저** 작성:
557
+
558
+ ```markdown
559
+ # ⚠ CLONE MODE — Reference Scaffold
560
+
561
+ This project was scaffolded in **clone mode** from the **<id>** reference
562
+ on **<ISO-date>**. The scaffold deliberately resembles <id>'s live
563
+ public surface to help you start fast.
564
+
565
+ ## What this means
566
+
567
+ - **Brand assets in this project are NOT yours.** The header logo, web
568
+ fonts, illustration patterns, and visual rhythm currently match
569
+ <id>'s public brand. They are placeholders/references, not part of
570
+ your product's identity.
571
+ - **You MUST replace them before shipping.** See `replace-checklist.md`
572
+ for the exact list of files and lines to swap.
573
+ - **Marketing copy** in this scaffold was generated fresh in the brand's
574
+ voice register — it is not copied verbatim from the live site. Edit
575
+ freely for your product's narrative.
576
+
577
+ ## Why clone mode exists
578
+
579
+ Designers and engineers commonly evaluate "what would an X-styled UI
580
+ look like for my product" by starting from a close visual reference.
581
+ clone mode automates that scaffold step. It is **not** a license to
582
+ ship the reference brand's identity as your own.
583
+
584
+ ## To switch back to inspired mode
585
+
586
+ Re-run with `mode=inspired` — all brand-identifying assets revert to
587
+ `[YOUR LOGO]` placeholders and generic illustration silhouettes, ready
588
+ to ship with your own brand.
589
+
590
+ ---
591
+
592
+ Generated by `omd:reference-capture`. Do not delete this banner until
593
+ `replace-checklist.md` is fully resolved.
594
+ ```
595
+
596
+ ### 6.5.2 `replace-checklist.md` (project root, mandatory)
597
+
598
+ clone mode가 만든 모든 brand-identifying 자산의 정확한 swap 지점 enumeration:
599
+
600
+ ```markdown
601
+ # Replace Checklist — pre-ship swap list
602
+
603
+ Before shipping, resolve every item below. Each item points to a
604
+ brand-cloned asset that must be replaced with your own product's
605
+ equivalent.
606
+
607
+ ## Logo
608
+
609
+ - [ ] `assets/_reference/<id>/logo.<ext>` — replace `<img src="assets/_reference/<id>/logo.<ext>">` references
610
+ in `landing.html` (line <N>), `app/layout.tsx` (line <N>), etc.
611
+ Suggested replacement: `assets/brand/logo.<ext>` (your own).
612
+
613
+ ## Web fonts
614
+
615
+ - [ ] `<link rel="stylesheet" href="<CDN URL>">` in `landing.html` head
616
+ — keep if you use the same font (Pretendard is open-source);
617
+ remove if you switch to a different typeface.
618
+
619
+ ## Hero illustration / imagery
620
+
621
+ - [ ] `landing.html` lines <N-M> — placeholder illustration matching
622
+ <id>'s visual idiom. Replace with your own illustration / photo.
623
+
624
+ ## Marketing copy
625
+
626
+ - (none — all copy was written fresh in the brand voice register;
627
+ edit for your product's specific narrative as needed)
628
+
629
+ ## Color tokens
630
+
631
+ - [ ] Live brand colors (e.g. `#13bd7e` for banksalad) are used as
632
+ `--brand-primary`. Adjust if your brand uses a different hue.
633
+
634
+ ## Final pre-ship check
635
+
636
+ ```bash
637
+ # Should return no matches before shipping:
638
+ grep -rn "_reference/<id>" .
639
+ ```
640
+
641
+ ## When this checklist is complete
642
+
643
+ Delete this file and `CLONE-MODE.md` from the project root. The
644
+ scaffold is now your product.
645
+ ```
646
+
647
+ 이 두 파일은 generator가 clone mode를 인지하는 마커이기도 함. 사용자가 의도적으로 삭제해야 inspired mode로 전환.
648
+
253
649
  ## Phase 7 — 사용자 요약
254
650
 
255
651
  prose로:
@@ -295,7 +691,8 @@ UI 생성 시 logo 사용 규칙 (omd:apply / omd:harness 둘 다 적용):
295
691
  assets/_reference/<id>/
296
692
  ├── LICENSE-NOTE.md (사전 작성)
297
693
  ├── attribution.md (마지막 작성)
298
- ├── tokens.json (facts)
694
+ ├── tokens.json (facts — atomic values)
695
+ ├── structure.json (facts — composition cues for surface idiom)
299
696
  ├── logo.<svg|png|ico> (brand mark — reference only)
300
697
  └── screenshots/
301
698
  ├── hero-desktop.png (1280×800, above fold)