launchframe 0.4.3 → 0.4.5

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.
@@ -26,7 +26,7 @@ Before reconnaissance, write or update:
26
26
 
27
27
  ## SaaS copy overlay (Phase 4 assembly and final polish)
28
28
 
29
- After structure and styles match the reference, apply the **SaaS idea** to hero, headings, and primary CTAs where the reference uses interchangeable marketing copy, **without** changing layout grids, spacing, or motion from extracted specs. Respect third-party brands.
29
+ After structure and styles match the reference, apply the **SaaS idea** to hero, headings, and primary CTAs where the reference uses interchangeable marketing copy, **without** changing layout grids, spacing, or motion from extracted specs. **Brand identity** (below) must be **original** for anything you ship as the user’s product — never pass off the reference company’s trademarks or distinctive marks.
30
30
 
31
31
  ---
32
32
 
@@ -38,12 +38,13 @@ The target page(s) are the URL(s) you parsed in Pre-Flight. Clone exactly what's
38
38
  - **In scope:** Visual layout and styling, component structure and interactions, responsive design, mock data for demo purposes
39
39
  - **Out of scope:** Real backend / database, authentication, real-time features, SEO optimization, accessibility audit
40
40
  - **Customization:** Structure and visuals — pure emulation of the reference. **Marketing copy** — apply the parsed **SaaS idea** where hero/headlines/CTAs are interchangeable (see “SaaS copy overlay” above); do not invent a different product than the user’s idea.
41
+ - **Brand identity:** The reference is a **pattern** for layout, motion, and UI craft — **not** permission to ship their brand. Unless the user **explicitly** asks for a faithful copy of the reference brand (e.g. licensed work, clearly labeled internal mock, private design audit), **invent an original brand** aligned with the SaaS idea: product name, wordmark or simple logomark (SVG or styled text) sized to the same logo slot, favicon / app icon, OG imagery, and a cohesive palette. Do **not** reuse their trademarked logo paths, mascot art, or distinctive illustrative brand assets; use originals or functional UI icons instead. Hero or lifestyle images that center the reference brand should be replaced with **original** imagery or neutral compositions that keep the **same layout rhythm**. Note in `docs/research/LAUNCHFRAME.md` which marks and assets are **original brand** versus **layout-only** extraction.
41
42
 
42
43
  If the user provides additional instructions (specific fidelity level, customizations, extra context), honor those over the defaults.
43
44
 
44
45
  ## Pre-Flight
45
46
 
46
- 1. **Browser automation is required.** Check for available browser MCP tools (Chrome MCP, Playwright MCP, Browserbase MCP, Puppeteer MCP, etc.). Use whichever is available — if multiple exist, prefer Chrome MCP. If none are detected, ask the user which browser tool they have and how to connect it. This skill cannot work without browser automation.
47
+ 1. **Browser automation is required.** Check for available browser MCP tools (Chrome MCP, Playwright MCP, Browserbase MCP, Puppeteer MCP, etc.). Use whichever is available — if multiple exist, **prefer Chrome DevTools MCP** for inspection: it maps to the same engine the user sees and preserves **computed `animation` / `transition` / scroll-driven** values accurately. If none are detected, ask the user which browser tool they have and how to connect it. This skill cannot work without browser automation. **Pixel-perfect MCP:** every extract and QA pass must follow **Browser MCP: pixel-perfect capture** (Phase 1) — locked viewports, **verbatim** `getComputedStyle` strings (including fractional `px`, `rem`, `oklch`/`rgba`), and **geometry** from `getBoundingClientRect()` where layout matters. **Motion parity:** every pass must treat **motion as first-class** — same **durations, delays, easings (`cubic-bezier` / steps), iteration counts, fill modes, directions, keyframe percentages, and scroll/view-timeline bindings** as the live site, not “similar” motion.
47
48
  2. **Parse arguments** — extract every `http://` / `https://` URL token (there may be several). **SaaS idea** = the remaining non-URL text (trim outer quotes). Normalize and validate each URL; if any are invalid, or the SaaS idea is missing, ask the user once. For each valid URL, verify it is accessible via your browser MCP tool.
48
49
  3. Verify the base project builds: `npm run build`. The Next.js + shadcn/ui + Tailwind v4 scaffold should already be in place. If not, tell the user to set it up first.
49
50
  4. Create the output directories if they don't exist: `docs/research/`, `docs/research/components/`, `docs/design-references/`, `scripts/`. For multiple clones, also prepare per-site folders like `docs/research/<hostname>/` and `docs/design-references/<hostname>/`.
@@ -59,7 +60,7 @@ When you traverse the DOM and the Network panel, do **not** treat all nodes equa
59
60
 
60
61
  1. **Images (raster + video stills)** — Enumerate `<img>`, `<picture>`, responsive `srcset`, `data-*` lazy URLs, **computed `background-image`** on the element and parents (including pseudo-elements), mask images, `<video poster>`, hero media. **Scrape:** download binary assets to `public/images/` (or `public/videos/`) with stable paths referenced in specs. **Create** a replacement PNG/WebP/SVG only when the asset is blocked (CORS, auth cookie, 403), ephemeral, or impossible to URL-fetch — use a high-DPI screenshot crop, traced artwork, or CSS gradient approximation, and mark `ASSET_SOURCE: generated` in the component spec with a short reason.
61
62
  2. **SVGs & iconography** — Inline `<svg>`, sprite `symbol` defs, **SVG used as masks/filters**, icon fonts (prefer path extraction). Convert to `@/components/icons.tsx` (or section-local components) with meaningful names. Prioritize crisp edges and correct `viewBox` over shrinking bundle size during emulation.
62
- 3. **Motion & animation** — CSS `@keyframes`, `animation`, `animation-timeline`, `transition`, `transform`, will-change hints; JS-driven motion (carousel timing, IntersectionObserver reveals); libraries (GSAP, Framer, Lottie JSON, Lenis). Capture **numbers** (ms, easing curves, stagger, scroll thresholds), not adjectives. Include **reduced-motion** behavior if present.
63
+ 3. **Motion & animation** — CSS `@keyframes`, `animation`, `animation-timeline` / `view-timeline` / `animation-range`, `transition`, `transform`, `transform-origin`, will-change; JS-driven motion (carousel timing, IntersectionObserver reveals); libraries (GSAP, Framer, Lottie JSON, Lenis). Capture **numbers** (ms, `cubic-bezier()`, `steps()`, stagger, scroll thresholds, **named keyframe blocks**), not adjectives — paste **verbatim** `cssText` for relevant `@keyframes` into specs when obtainable. Include **reduced-motion** (`matchMedia('(prefers-reduced-motion: reduce)'))` behavior if present.
63
64
 
64
65
  Only after the above are accounted for should you spend cycle time on minor text or non-visual refactors. A perfect grid with missing hero art and dead animation still fails the clone.
65
66
 
@@ -152,11 +153,48 @@ Navigate to the target URL with browser MCP.
152
153
 
153
154
  Follow **§0 (Visual crawl priority)** during the entire reconnaissance pass: images and backgrounds → SVGs/icons → motion/animations — before spending time on secondary copy tweaks.
154
155
 
156
+ ### Multi-pass DOM reconnaissance (mandatory)
157
+
158
+ **A single scroll is not enough.** You must run the live page **multiple full cycles** in browser MCP — each time walking the **DOM**, the **asset graph**, and the **motion flow** — until what you write in specs is **exactly** what the reference does, not a remembered approximation.
159
+
160
+ **Minimum passes** (run **Pass 1–4 at 1440px first**; **repeat Pass 1–3 at 768px and 390px** wherever layout or node order differs; reconciliation can be width-specific):
161
+
162
+ 1. **Structure pass (DOM, top → bottom):** Traverse from **`document.body`** in **depth-first order** (or section-by-section using regions you will later name in `PAGE_TOPOLOGY.md`). For every node that affects layout or visible content, record **tag**, **selector hint** (stable `id` / first meaningful classes), **`childNodes` order** (siblings matter — flex/grid gaps and wrappers are not optional), landmarks, and **which element is the scroll container** (`overflow` / `overscroll-behavior`). Produce a **literal hierarchy** in `docs/research/DOM_SKETCH.md` (whole page or per-section files under `docs/research/`) so builders cannot silently reorder or collapse wrapper divs. **Include “invisible” structure** (spacers, gradient overlays, absolutely positioned siblings).
163
+ 2. **Asset pass (images & layered media):** **Second full traversal** focused on §0 priority 1–2: every `<img>`, `<picture>` / `srcset`, `<video>` / `poster`, computed **`background-image`** on the element **and** ancestors, CSS masks, inline SVG / sprites. Cross-check with the Network panel. Every URL must end up **downloaded to `public/`** with a documented path, or marked **blocked / generated** with a substitute and reason.
164
+ 3. **Motion & flow pass:** **Third full journey** — slow scroll **top to bottom**, then **scroll again** (different speed) to catch lazy-mounted nodes, staggered `IntersectionObserver` reveals, and scroll-snapped stops. Run **`motionAudit`** (below) **per major section root** after scrolling that section into view. **Re-do click/hover sweeps** here while sampling `getComputedStyle` so **transitions and keyframe-driven flow** match the reference’s **order and timing**, not a generic fade-in.
165
+ 4. **Reconciliation pass:** **Fourth pass** — read `DOM_SKETCH.md`, `BEHAVIORS.md`, asset inventory, and `MOTION.md` together; list **gaps** (missing wrapper, second image layer, unnamed animation, wrong child order). For every gap, return to the site and **repeat Pass 1–3** for that subtree until the spec is closed.
166
+
167
+ **Hard rule:** Every fact in a component spec must trace to **which pass** produced it. If you “kind of remember” a structure from an earlier visit, **run the page again** — do not guess.
168
+
155
169
  ### Screenshots
156
170
  - Take **full-page screenshots** at desktop (1440px) and mobile (390px) viewports
157
171
  - Save to `docs/design-references/` with descriptive names
158
172
  - These are your master reference — builders will receive section-specific crops/screenshots later
159
173
 
174
+ ### Browser MCP: pixel-perfect capture
175
+
176
+ MCP-driven inspection must be **pixel-perfect**, not “close.” Apply this on **every** reconnaissance width and again in Phase 5.
177
+
178
+ - **Viewport lock:** Resize the MCP-controlled window to exact **CSS widths** used for this project: **1440**, **768**, and **390** px (height large enough to avoid accidental mobile chrome quirks). After each resize, record **`window.innerWidth`**, **`document.documentElement.clientWidth`**, and **`devicePixelRatio`** (run the snippet below). Extraction and QA **use the same widths** so diffs are valid.
179
+ - **Zoom:** Ensure **browser zoom is 100%** (not 90%/110%). Zoom breaks `px` parity and screenshots.
180
+ - **Computed styles are verbatim:** When copying from `getComputedStyle()`, paste **exact returned strings** — including **fractional** lengths (`12.8px`), **negative** values, **`oklch()`/`lch()`/`rgba()`**, `letter-spacing`, `line-height`, `border`/`outline`, and **`calc()`**/`min()` that the cascade resolves to. **Do not round** for “clean” tokens (no `~16px`; no swapping `0.875rem` for `14px` unless the computed value truly equals that).
181
+ - **Geometry:** When alignment, gutters, or sticky offsets matter, record **`getBoundingClientRect()`** (`x`, `y`, `width`, `height`) at the relevant scroll position and width — not only padding/margin strings. Fixed/sticky layers need **viewport-relative** positions.
182
+ - **Pseudos & layers:** Extract **`::before` / `::after`** (and other pseudos the reference uses) with `getComputedStyle(el, '::before')` etc.; they often hold gradients, borders, or icons that define the pixel look.
183
+ - **Anti-aliasing & text:** Prefer **real Chrome** via MCP so **font rasterization** matches; load the **same font files/weights** in Next.js as on the reference. Note `font-feature-settings` / `font-variation-settings` / `-webkit-font-smoothing` if non-default.
184
+ - **Screenshots vs. DOM:** Screenshots are for **diffing**; the **spec values** still come from **computed style + rect**, not from eyeballing pixels.
185
+
186
+ **Viewport diagnostic** (run via browser MCP after each resize):
187
+
188
+ ```javascript
189
+ JSON.stringify({
190
+ innerWidth: window.innerWidth,
191
+ clientWidth: document.documentElement.clientWidth,
192
+ clientHeight: document.documentElement.clientHeight,
193
+ devicePixelRatio: window.devicePixelRatio,
194
+ scrollY: window.scrollY
195
+ });
196
+ ```
197
+
160
198
  ### Global Extraction
161
199
  Extract these from the page before doing anything else:
162
200
 
@@ -196,6 +234,106 @@ This is a dedicated pass AFTER screenshots and BEFORE anything else. Its purpose
196
234
 
197
235
  Save all findings to `docs/research/BEHAVIORS.md`. This is your behavior bible — reference it when writing every component spec.
198
236
 
237
+ ### Chrome MCP: precise motion & animation extraction
238
+
239
+ Use **evaluate script** (or equivalent) in Chrome MCP **during** the scroll/click/hover sweeps so you capture **live computed values**, not guesses from DevTools screenshots alone.
240
+
241
+ **Rules:**
242
+
243
+ - **No hand-waving** — if the reference uses `0.45s cubic-bezier(0.4, 0, 0.2, 1)`, the spec and implementation use that exact string (or equivalent in Tailwind only when it truly matches).
244
+ - **`@keyframes` source of truth** — for each animated element, record `getComputedStyle(el).animationName` and resolve the **full `@keyframes` rule** from stylesheets when allowed. Many sites inline CSS or use same-origin sheets: iterate `document.styleSheets` / `cssRules` inside `try/catch`; for **cross-origin** sheets that throw on `cssRules`, copy the rule from the **Network** response or **Sources** panel in Chrome and paste into `docs/research/MOTION.md` / the component spec. Never substitute a different easing or keyframe shape “because it looks close.”
245
+ - **Scroll-driven animations** — for `animation-timeline: view()` / named timelines, capture **`view-timeline` / `scroll-timeline` on ancestors**, `animation-range`, `scroll-padding` / snap containers, and **which scroll container** is the timeline root (often `html`, sometimes a nested `overflow-y: auto` div).
246
+ - **Before/after pairs** — for scroll- or hover-driven motion, capture computed `transform`, `opacity`, and longhand `animation-*` / `transition-*` **in both states** (see §7) and the **exact trigger** (px, ratio, or event).
247
+ - **Libraries** — if Lenis/GSAP/Framer is present, document **version/hooks** from `package` on the page or script URLs, class hooks (e.g. `.lenis`), and any **inline style** the library sets at rest vs. mid-animation (sample via computed style after the interaction).
248
+ - **Deliverable** — maintain `docs/research/MOTION.md`: page-level keyframes inventory, global scroll/smooth-scroll setup, and per-section pointers into component specs. Large pages may scope the audit script to a **section root selector** to avoid noise.
249
+
250
+ **Motion audit script** (run via Chrome MCP; optional `rootSelector` limits to a subtree):
251
+
252
+ ```javascript
253
+ (function motionAudit(rootSelector) {
254
+ const root = rootSelector ? document.querySelector(rootSelector) : document.body;
255
+ if (!root) return JSON.stringify({ error: 'root not found: ' + rootSelector });
256
+
257
+ function safeRules(sheet) {
258
+ try {
259
+ return [...sheet.cssRules];
260
+ } catch {
261
+ return [];
262
+ }
263
+ }
264
+
265
+ const walkKeyframes = (rules, out) => {
266
+ for (const rule of rules) {
267
+ if (rule.type === CSSRule.KEYFRAMES_RULE) {
268
+ out.push({ name: rule.name, cssText: rule.cssText });
269
+ } else if (rule.type === CSSRule.MEDIA_RULE && rule.cssRules) {
270
+ walkKeyframes(rule.cssRules, out);
271
+ }
272
+ }
273
+ };
274
+
275
+ const keyframes = [];
276
+ for (const sheet of document.styleSheets) {
277
+ walkKeyframes(safeRules(sheet), keyframes);
278
+ }
279
+
280
+ const animated = [];
281
+ const scope = root.querySelectorAll('*');
282
+ for (const el of scope) {
283
+ const cs = getComputedStyle(el);
284
+ const animName = cs.animationName;
285
+ const hasAnim = animName && animName !== 'none';
286
+ const transDurs = (cs.transitionDuration || '')
287
+ .split(',')
288
+ .map((s) => s.trim())
289
+ .filter(Boolean);
290
+ const hasTrans =
291
+ cs.transitionProperty &&
292
+ cs.transitionProperty !== 'none' &&
293
+ transDurs.some((d) => d !== '0s');
294
+ if (!hasAnim && !hasTrans) continue;
295
+
296
+ animated.push({
297
+ hint: el.tagName.toLowerCase() + (el.className
298
+ ? '.' + String(el.className).trim().split(/\s+/).slice(0, 4).join('.')
299
+ : ''),
300
+ animation: cs.animation,
301
+ animationName: cs.animationName,
302
+ animationDuration: cs.animationDuration,
303
+ animationTimingFunction: cs.animationTimingFunction,
304
+ animationDelay: cs.animationDelay,
305
+ animationIterationCount: cs.animationIterationCount,
306
+ animationDirection: cs.animationDirection,
307
+ animationFillMode: cs.animationFillMode,
308
+ animationPlayState: cs.animationPlayState,
309
+ animationTimeline: cs.animationTimeline,
310
+ animationRange: cs.animationRange,
311
+ transition: cs.transition,
312
+ transitionProperty: cs.transitionProperty,
313
+ transitionDuration: cs.transitionDuration,
314
+ transitionTimingFunction: cs.transitionTimingFunction,
315
+ transitionDelay: cs.transitionDelay,
316
+ transform: cs.transform,
317
+ transformOrigin: cs.transformOrigin,
318
+ opacity: cs.opacity,
319
+ willChange: cs.willChange,
320
+ viewTimelineName: cs.viewTimelineName,
321
+ scrollTimelineName: cs.scrollTimelineName
322
+ });
323
+ }
324
+
325
+ return JSON.stringify(
326
+ {
327
+ prefersReducedMotion: window.matchMedia('(prefers-reduced-motion: reduce)').matches,
328
+ keyframes,
329
+ animatedElements: animated
330
+ },
331
+ null,
332
+ 2
333
+ );
334
+ })(''); // Pass section selector string, e.g. 'section.hero', instead of ''
335
+ ```
336
+
199
337
  ### Page Topology
200
338
  Map out every distinct section of the page from top to bottom. Give each a working name. Document:
201
339
  - Their visual order
@@ -286,7 +424,8 @@ For each section, use browser MCP to extract everything:
286
424
  'borderRadius','border','borderTop','borderBottom','borderLeft','borderRight',
287
425
  'boxShadow','overflow','overflowX','overflowY',
288
426
  'position','top','right','bottom','left','zIndex',
289
- 'opacity','transform','transition','cursor',
427
+ 'opacity','transform','transformOrigin','transition','transitionProperty','transitionDuration','transitionTimingFunction','transitionDelay',
428
+ 'animation','animationName','animationDuration','animationTimingFunction','animationDelay','animationIterationCount','animationDirection','animationFillMode','animationPlayState','animationTimeline','animationRange','viewTimelineName','scrollTimelineName','cursor',
290
429
  'objectFit','objectPosition','mixBlendMode','filter','backdropFilter',
291
430
  'whiteSpace','textOverflow','WebkitLineClamp'
292
431
  ];
@@ -365,6 +504,15 @@ For each section (or sub-component, if you're breaking it up), create a spec fil
365
504
  ### <Child element N>
366
505
  ...
367
506
 
507
+ ## Motion & animation (exact — from Chrome MCP `getComputedStyle` + stylesheet rules)
508
+
509
+ - **Keyframe names in use:** ...
510
+ - **Full `@keyframes` blocks** (paste `cssText` or equivalent); note **cross-origin** gaps and how you filled them
511
+ - **Per-element:** shorthand/longhand `animation` and `transition` values **verbatim** for resting state
512
+ - **Scroll/view timelines:** timeline name, range, scroll container, snap/observer thresholds
513
+ - **Libraries:** Lenis / GSAP / Framer / Lottie — hooks, init pattern, timing copied from reference
514
+ - **`prefers-reduced-motion`:** matched behavior on the reference
515
+
368
516
  ## States & Behaviors
369
517
 
370
518
  ### <Behavior name, e.g., "Scroll-triggered floating mode">
@@ -447,30 +595,34 @@ After all sections are built and merged, wire everything together in `src/app/pa
447
595
 
448
596
  After assembly, do NOT declare the clone complete. Take side-by-side comparison screenshots:
449
597
 
450
- 1. Open the original site and your clone side-by-side (or take screenshots at the same viewport widths)
598
+ 1. **Pixel-perfect comparison setup:** Match **Browser MCP: pixel-perfect capture** — same **CSS widths** (**1440**, **768**, **390**), **100% zoom**, and record `innerWidth`/`clientWidth`/`devicePixelRatio` on both reference and clone. Then open original and clone side-by-side or take paired screenshots at those widths.
451
599
  2. Compare section by section, top to bottom, at desktop (1440px)
452
- 3. Compare again at mobile (390px)
600
+ 3. Compare at tablet (768px) and again at mobile (390px)
453
601
  4. For each discrepancy found:
454
602
  - Check the component spec file — was the value extracted correctly?
455
603
  - If the spec was wrong: re-extract from browser MCP, update the spec, fix the component
456
604
  - If the spec was right but the builder got it wrong: fix the component to match the spec
457
605
  5. Test all interactive behaviors: scroll through the page, click every button/tab, hover over interactive elements
458
- 6. Verify smooth scroll feels right, header transitions work, tab switching works, animations play
606
+ 6. **Motion QA (Chrome MCP again if needed):** For every animated or transitioned element, re-run `getComputedStyle` on the reference vs. the clone at the **same scroll position / interaction state** and compare **`animation-*`, `transition-*`, `transform`, `opacity`** longhands — durations and easings must match character-for-character where the cascade specifies them. Re-record any `@keyframes` the reference still exposes so the clone can be patched until timings align.
607
+ 7. Verify smooth scroll feels right, header transitions work, tab switching works, stagger and scroll-driven reveals match the reference
459
608
 
460
- Only after this visual QA pass is the clone complete.
609
+ Only after this visual + motion QA pass is the clone complete.
461
610
 
462
611
  ## Pre-Dispatch Checklist
463
612
 
464
613
  Before dispatching ANY builder agent, verify you can check every box. If you can't, go back and extract more.
465
614
 
615
+ - [ ] **Multi-pass DOM reconnaissance:** structure (`DOM_SKETCH.md`) + asset sweep + motion/flow sweeps + reconciliation; repeated at **768 / 390** when layout or DOM order differs from 1440
466
616
  - [ ] Spec file written to `docs/research/components/<name>.spec.md` with ALL sections filled
467
- - [ ] Every CSS value in the spec is from `getComputedStyle()`, not estimated
617
+ - [ ] Every CSS value in the spec is from `getComputedStyle()`, not estimated — **verbatim**, including fractional px and color functions
618
+ - [ ] **MCP pixel-perfect pass:** viewports **1440 / 768 / 390**, **100% zoom**, viewport diagnostic recorded; **`getBoundingClientRect()`** and **`::before`/`::after`** captured where they affect layout or visuals
468
619
  - [ ] Interaction model is identified and documented (static / click / scroll / time)
469
620
  - [ ] For stateful components: every state's content and styles are captured
470
621
  - [ ] For scroll-driven components: trigger threshold, before/after styles, and transition are recorded
471
622
  - [ ] For hover states: before/after values and transition timing are recorded
623
+ - [ ] Motion: relevant `@keyframes` captured (or cross-origin gap documented with manual paste), `docs/research/MOTION.md` updated, component specs include **exact** `animation*` / `transition*` / timeline fields from Chrome MCP
472
624
  - [ ] All images in the section are identified (including overlays and layered compositions)
473
- - [ ] Responsive behavior is documented for at least desktop and mobile
625
+ - [ ] Responsive behavior is documented for **1440 / 768 / 390**
474
626
  - [ ] Text content is verbatim from the site, not paraphrased
475
627
  - [ ] The builder prompt is under ~150 lines of spec; if over, the section needs to be split
476
628
 
@@ -488,6 +640,8 @@ These are lessons from previous failed clones — each one cost hours of rework:
488
640
  - **Don't skip asset extraction.** Without real images, videos, and fonts, the clone will always look fake regardless of how perfect the CSS is.
489
641
  - **Don't give a builder agent too much scope.** If you're writing a builder prompt and it's getting long because the section is complex, that's a signal to break it into smaller tasks.
490
642
  - **Don't bundle unrelated sections into one agent.** A CTA section and a footer are different components with different designs — don't hand them both to one agent and hope for the best.
643
+ - **Don't round or “clean up” extracted values** for Tailwind guesses: `12.8px` stays `12.8px` in the spec unless the clone uses an exactly equivalent value.
644
+ - **Don't mix zoom levels or viewport widths** between reference extraction and QA — you will false-fail or false-pass pixel diffs.
491
645
  - **Don't skip responsive extraction.** If you only inspect at desktop width, the clone will break at tablet and mobile. Test at 1440, 768, and 390 during extraction.
492
646
  - **Don't forget smooth scroll libraries.** Check for Lenis (`.lenis` class), Locomotive Scroll, or similar. Default browser scrolling feels noticeably different and the user will spot it immediately.
493
647
  - **Don't dispatch builders without a spec file.** The spec file forces exhaustive extraction and creates an auditable artifact. Skipping it means the builder gets whatever you can fit in a prompt from memory.
@@ -495,6 +649,7 @@ These are lessons from previous failed clones — each one cost hours of rework:
495
649
  ## Completion
496
650
 
497
651
  When done, report:
652
+ - Multi-pass reconnaissance status (`docs/research/DOM_SKETCH.md` present; passes at **1440 / 768 / 390** as needed)
498
653
  - Total sections built
499
654
  - Total components created
500
655
  - Total spec files written (should match components)
@@ -21,7 +21,7 @@ Before reconnaissance, write or update:
21
21
 
22
22
  ## SaaS copy overlay (Phase 4 assembly and final polish)
23
23
 
24
- After structure and styles match the reference, apply the **SaaS idea** to hero, headings, and primary CTAs where the reference uses interchangeable marketing copy, **without** changing layout grids, spacing, or motion from extracted specs. Respect third-party brands.
24
+ After structure and styles match the reference, apply the **SaaS idea** to hero, headings, and primary CTAs where the reference uses interchangeable marketing copy, **without** changing layout grids, spacing, or motion from extracted specs. **Brand identity** (below) must be **original** for anything you ship as the user’s product — never pass off the reference company’s trademarks or distinctive marks.
25
25
 
26
26
  ---
27
27
 
@@ -33,12 +33,13 @@ The target page(s) are the URL(s) you parsed in Pre-Flight. Clone exactly what's
33
33
  - **In scope:** Visual layout and styling, component structure and interactions, responsive design, mock data for demo purposes
34
34
  - **Out of scope:** Real backend / database, authentication, real-time features, SEO optimization, accessibility audit
35
35
  - **Customization:** Structure and visuals — pure emulation of the reference. **Marketing copy** — apply the parsed **SaaS idea** where hero/headlines/CTAs are interchangeable (see “SaaS copy overlay” above); do not invent a different product than the user’s idea.
36
+ - **Brand identity:** The reference is a **pattern** for layout, motion, and UI craft — **not** permission to ship their brand. Unless the user **explicitly** asks for a faithful copy of the reference brand (e.g. licensed work, clearly labeled internal mock, private design audit), **invent an original brand** aligned with the SaaS idea: product name, wordmark or simple logomark (SVG or styled text) sized to the same logo slot, favicon / app icon, OG imagery, and a cohesive palette. Do **not** reuse their trademarked logo paths, mascot art, or distinctive illustrative brand assets; use originals or functional UI icons instead. Hero or lifestyle images that center the reference brand should be replaced with **original** imagery or neutral compositions that keep the **same layout rhythm**. Note in `docs/research/LAUNCHFRAME.md` which marks and assets are **original brand** versus **layout-only** extraction.
36
37
 
37
38
  If the user provides additional instructions (specific fidelity level, customizations, extra context), honor those over the defaults.
38
39
 
39
40
  ## Pre-Flight
40
41
 
41
- 1. **Browser automation is required.** Check for available browser MCP tools (Chrome MCP, Playwright MCP, Browserbase MCP, Puppeteer MCP, etc.). Use whichever is available — if multiple exist, prefer Chrome MCP. If none are detected, ask the user which browser tool they have and how to connect it. This skill cannot work without browser automation.
42
+ 1. **Browser automation is required.** Check for available browser MCP tools (Chrome MCP, Playwright MCP, Browserbase MCP, Puppeteer MCP, etc.). Use whichever is available — if multiple exist, **prefer Chrome DevTools MCP** for inspection: it maps to the same engine the user sees and preserves **computed `animation` / `transition` / scroll-driven** values accurately. If none are detected, ask the user which browser tool they have and how to connect it. This skill cannot work without browser automation. **Pixel-perfect MCP:** every extract and QA pass must follow **Browser MCP: pixel-perfect capture** (Phase 1) — locked viewports, **verbatim** `getComputedStyle` strings (including fractional `px`, `rem`, `oklch`/`rgba`), and **geometry** from `getBoundingClientRect()` where layout matters. **Motion parity:** every pass must treat **motion as first-class** — same **durations, delays, easings (`cubic-bezier` / steps), iteration counts, fill modes, directions, keyframe percentages, and scroll/view-timeline bindings** as the live site, not “similar” motion.
42
43
  2. **Parse arguments** — extract every `http://` / `https://` URL token (there may be several). **SaaS idea** = the remaining non-URL text (trim outer quotes). Normalize and validate each URL; if any are invalid, or the SaaS idea is missing, ask the user once. For each valid URL, verify it is accessible via your browser MCP tool.
43
44
  3. Verify the base project builds: `npm run build`. The Next.js + shadcn/ui + Tailwind v4 scaffold should already be in place. If not, tell the user to set it up first.
44
45
  4. Create the output directories if they don't exist: `docs/research/`, `docs/research/components/`, `docs/design-references/`, `scripts/`. For multiple clones, also prepare per-site folders like `docs/research/<hostname>/` and `docs/design-references/<hostname>/`.
@@ -54,7 +55,7 @@ When you traverse the DOM and the Network panel, do **not** treat all nodes equa
54
55
 
55
56
  1. **Images (raster + video stills)** — Enumerate `<img>`, `<picture>`, responsive `srcset`, `data-*` lazy URLs, **computed `background-image`** on the element and parents (including pseudo-elements), mask images, `<video poster>`, hero media. **Scrape:** download binary assets to `public/images/` (or `public/videos/`) with stable paths referenced in specs. **Create** a replacement PNG/WebP/SVG only when the asset is blocked (CORS, auth cookie, 403), ephemeral, or impossible to URL-fetch — use a high-DPI screenshot crop, traced artwork, or CSS gradient approximation, and mark `ASSET_SOURCE: generated` in the component spec with a short reason.
56
57
  2. **SVGs & iconography** — Inline `<svg>`, sprite `symbol` defs, **SVG used as masks/filters**, icon fonts (prefer path extraction). Convert to `@/components/icons.tsx` (or section-local components) with meaningful names. Prioritize crisp edges and correct `viewBox` over shrinking bundle size during emulation.
57
- 3. **Motion & animation** — CSS `@keyframes`, `animation`, `animation-timeline`, `transition`, `transform`, will-change hints; JS-driven motion (carousel timing, IntersectionObserver reveals); libraries (GSAP, Framer, Lottie JSON, Lenis). Capture **numbers** (ms, easing curves, stagger, scroll thresholds), not adjectives. Include **reduced-motion** behavior if present.
58
+ 3. **Motion & animation** — CSS `@keyframes`, `animation`, `animation-timeline` / `view-timeline` / `animation-range`, `transition`, `transform`, `transform-origin`, will-change; JS-driven motion (carousel timing, IntersectionObserver reveals); libraries (GSAP, Framer, Lottie JSON, Lenis). Capture **numbers** (ms, `cubic-bezier()`, `steps()`, stagger, scroll thresholds, **named keyframe blocks**), not adjectives — paste **verbatim** `cssText` for relevant `@keyframes` into specs when obtainable. Include **reduced-motion** (`matchMedia('(prefers-reduced-motion: reduce)'))` behavior if present.
58
59
 
59
60
  Only after the above are accounted for should you spend cycle time on minor text or non-visual refactors. A perfect grid with missing hero art and dead animation still fails the clone.
60
61
 
@@ -147,11 +148,48 @@ Navigate to the target URL with browser MCP.
147
148
 
148
149
  Follow **§0 (Visual crawl priority)** during the entire reconnaissance pass: images and backgrounds → SVGs/icons → motion/animations — before spending time on secondary copy tweaks.
149
150
 
151
+ ### Multi-pass DOM reconnaissance (mandatory)
152
+
153
+ **A single scroll is not enough.** You must run the live page **multiple full cycles** in browser MCP — each time walking the **DOM**, the **asset graph**, and the **motion flow** — until what you write in specs is **exactly** what the reference does, not a remembered approximation.
154
+
155
+ **Minimum passes** (run **Pass 1–4 at 1440px first**; **repeat Pass 1–3 at 768px and 390px** wherever layout or node order differs; reconciliation can be width-specific):
156
+
157
+ 1. **Structure pass (DOM, top → bottom):** Traverse from **`document.body`** in **depth-first order** (or section-by-section using regions you will later name in `PAGE_TOPOLOGY.md`). For every node that affects layout or visible content, record **tag**, **selector hint** (stable `id` / first meaningful classes), **`childNodes` order** (siblings matter — flex/grid gaps and wrappers are not optional), landmarks, and **which element is the scroll container** (`overflow` / `overscroll-behavior`). Produce a **literal hierarchy** in `docs/research/DOM_SKETCH.md` (whole page or per-section files under `docs/research/`) so builders cannot silently reorder or collapse wrapper divs. **Include “invisible” structure** (spacers, gradient overlays, absolutely positioned siblings).
158
+ 2. **Asset pass (images & layered media):** **Second full traversal** focused on §0 priority 1–2: every `<img>`, `<picture>` / `srcset`, `<video>` / `poster`, computed **`background-image`** on the element **and** ancestors, CSS masks, inline SVG / sprites. Cross-check with the Network panel. Every URL must end up **downloaded to `public/`** with a documented path, or marked **blocked / generated** with a substitute and reason.
159
+ 3. **Motion & flow pass:** **Third full journey** — slow scroll **top to bottom**, then **scroll again** (different speed) to catch lazy-mounted nodes, staggered `IntersectionObserver` reveals, and scroll-snapped stops. Run **`motionAudit`** (below) **per major section root** after scrolling that section into view. **Re-do click/hover sweeps** here while sampling `getComputedStyle` so **transitions and keyframe-driven flow** match the reference’s **order and timing**, not a generic fade-in.
160
+ 4. **Reconciliation pass:** **Fourth pass** — read `DOM_SKETCH.md`, `BEHAVIORS.md`, asset inventory, and `MOTION.md` together; list **gaps** (missing wrapper, second image layer, unnamed animation, wrong child order). For every gap, return to the site and **repeat Pass 1–3** for that subtree until the spec is closed.
161
+
162
+ **Hard rule:** Every fact in a component spec must trace to **which pass** produced it. If you “kind of remember” a structure from an earlier visit, **run the page again** — do not guess.
163
+
150
164
  ### Screenshots
151
165
  - Take **full-page screenshots** at desktop (1440px) and mobile (390px) viewports
152
166
  - Save to `docs/design-references/` with descriptive names
153
167
  - These are your master reference — builders will receive section-specific crops/screenshots later
154
168
 
169
+ ### Browser MCP: pixel-perfect capture
170
+
171
+ MCP-driven inspection must be **pixel-perfect**, not “close.” Apply this on **every** reconnaissance width and again in Phase 5.
172
+
173
+ - **Viewport lock:** Resize the MCP-controlled window to exact **CSS widths** used for this project: **1440**, **768**, and **390** px (height large enough to avoid accidental mobile chrome quirks). After each resize, record **`window.innerWidth`**, **`document.documentElement.clientWidth`**, and **`devicePixelRatio`** (run the snippet below). Extraction and QA **use the same widths** so diffs are valid.
174
+ - **Zoom:** Ensure **browser zoom is 100%** (not 90%/110%). Zoom breaks `px` parity and screenshots.
175
+ - **Computed styles are verbatim:** When copying from `getComputedStyle()`, paste **exact returned strings** — including **fractional** lengths (`12.8px`), **negative** values, **`oklch()`/`lch()`/`rgba()`**, `letter-spacing`, `line-height`, `border`/`outline`, and **`calc()`**/`min()` that the cascade resolves to. **Do not round** for “clean” tokens (no `~16px`; no swapping `0.875rem` for `14px` unless the computed value truly equals that).
176
+ - **Geometry:** When alignment, gutters, or sticky offsets matter, record **`getBoundingClientRect()`** (`x`, `y`, `width`, `height`) at the relevant scroll position and width — not only padding/margin strings. Fixed/sticky layers need **viewport-relative** positions.
177
+ - **Pseudos & layers:** Extract **`::before` / `::after`** (and other pseudos the reference uses) with `getComputedStyle(el, '::before')` etc.; they often hold gradients, borders, or icons that define the pixel look.
178
+ - **Anti-aliasing & text:** Prefer **real Chrome** via MCP so **font rasterization** matches; load the **same font files/weights** in Next.js as on the reference. Note `font-feature-settings` / `font-variation-settings` / `-webkit-font-smoothing` if non-default.
179
+ - **Screenshots vs. DOM:** Screenshots are for **diffing**; the **spec values** still come from **computed style + rect**, not from eyeballing pixels.
180
+
181
+ **Viewport diagnostic** (run via browser MCP after each resize):
182
+
183
+ ```javascript
184
+ JSON.stringify({
185
+ innerWidth: window.innerWidth,
186
+ clientWidth: document.documentElement.clientWidth,
187
+ clientHeight: document.documentElement.clientHeight,
188
+ devicePixelRatio: window.devicePixelRatio,
189
+ scrollY: window.scrollY
190
+ });
191
+ ```
192
+
155
193
  ### Global Extraction
156
194
  Extract these from the page before doing anything else:
157
195
 
@@ -191,6 +229,106 @@ This is a dedicated pass AFTER screenshots and BEFORE anything else. Its purpose
191
229
 
192
230
  Save all findings to `docs/research/BEHAVIORS.md`. This is your behavior bible — reference it when writing every component spec.
193
231
 
232
+ ### Chrome MCP: precise motion & animation extraction
233
+
234
+ Use **evaluate script** (or equivalent) in Chrome MCP **during** the scroll/click/hover sweeps so you capture **live computed values**, not guesses from DevTools screenshots alone.
235
+
236
+ **Rules:**
237
+
238
+ - **No hand-waving** — if the reference uses `0.45s cubic-bezier(0.4, 0, 0.2, 1)`, the spec and implementation use that exact string (or equivalent in Tailwind only when it truly matches).
239
+ - **`@keyframes` source of truth** — for each animated element, record `getComputedStyle(el).animationName` and resolve the **full `@keyframes` rule** from stylesheets when allowed. Many sites inline CSS or use same-origin sheets: iterate `document.styleSheets` / `cssRules` inside `try/catch`; for **cross-origin** sheets that throw on `cssRules`, copy the rule from the **Network** response or **Sources** panel in Chrome and paste into `docs/research/MOTION.md` / the component spec. Never substitute a different easing or keyframe shape “because it looks close.”
240
+ - **Scroll-driven animations** — for `animation-timeline: view()` / named timelines, capture **`view-timeline` / `scroll-timeline` on ancestors**, `animation-range`, `scroll-padding` / snap containers, and **which scroll container** is the timeline root (often `html`, sometimes a nested `overflow-y: auto` div).
241
+ - **Before/after pairs** — for scroll- or hover-driven motion, capture computed `transform`, `opacity`, and longhand `animation-*` / `transition-*` **in both states** (see §7) and the **exact trigger** (px, ratio, or event).
242
+ - **Libraries** — if Lenis/GSAP/Framer is present, document **version/hooks** from `package` on the page or script URLs, class hooks (e.g. `.lenis`), and any **inline style** the library sets at rest vs. mid-animation (sample via computed style after the interaction).
243
+ - **Deliverable** — maintain `docs/research/MOTION.md`: page-level keyframes inventory, global scroll/smooth-scroll setup, and per-section pointers into component specs. Large pages may scope the audit script to a **section root selector** to avoid noise.
244
+
245
+ **Motion audit script** (run via Chrome MCP; optional `rootSelector` limits to a subtree):
246
+
247
+ ```javascript
248
+ (function motionAudit(rootSelector) {
249
+ const root = rootSelector ? document.querySelector(rootSelector) : document.body;
250
+ if (!root) return JSON.stringify({ error: 'root not found: ' + rootSelector });
251
+
252
+ function safeRules(sheet) {
253
+ try {
254
+ return [...sheet.cssRules];
255
+ } catch {
256
+ return [];
257
+ }
258
+ }
259
+
260
+ const walkKeyframes = (rules, out) => {
261
+ for (const rule of rules) {
262
+ if (rule.type === CSSRule.KEYFRAMES_RULE) {
263
+ out.push({ name: rule.name, cssText: rule.cssText });
264
+ } else if (rule.type === CSSRule.MEDIA_RULE && rule.cssRules) {
265
+ walkKeyframes(rule.cssRules, out);
266
+ }
267
+ }
268
+ };
269
+
270
+ const keyframes = [];
271
+ for (const sheet of document.styleSheets) {
272
+ walkKeyframes(safeRules(sheet), keyframes);
273
+ }
274
+
275
+ const animated = [];
276
+ const scope = root.querySelectorAll('*');
277
+ for (const el of scope) {
278
+ const cs = getComputedStyle(el);
279
+ const animName = cs.animationName;
280
+ const hasAnim = animName && animName !== 'none';
281
+ const transDurs = (cs.transitionDuration || '')
282
+ .split(',')
283
+ .map((s) => s.trim())
284
+ .filter(Boolean);
285
+ const hasTrans =
286
+ cs.transitionProperty &&
287
+ cs.transitionProperty !== 'none' &&
288
+ transDurs.some((d) => d !== '0s');
289
+ if (!hasAnim && !hasTrans) continue;
290
+
291
+ animated.push({
292
+ hint: el.tagName.toLowerCase() + (el.className
293
+ ? '.' + String(el.className).trim().split(/\s+/).slice(0, 4).join('.')
294
+ : ''),
295
+ animation: cs.animation,
296
+ animationName: cs.animationName,
297
+ animationDuration: cs.animationDuration,
298
+ animationTimingFunction: cs.animationTimingFunction,
299
+ animationDelay: cs.animationDelay,
300
+ animationIterationCount: cs.animationIterationCount,
301
+ animationDirection: cs.animationDirection,
302
+ animationFillMode: cs.animationFillMode,
303
+ animationPlayState: cs.animationPlayState,
304
+ animationTimeline: cs.animationTimeline,
305
+ animationRange: cs.animationRange,
306
+ transition: cs.transition,
307
+ transitionProperty: cs.transitionProperty,
308
+ transitionDuration: cs.transitionDuration,
309
+ transitionTimingFunction: cs.transitionTimingFunction,
310
+ transitionDelay: cs.transitionDelay,
311
+ transform: cs.transform,
312
+ transformOrigin: cs.transformOrigin,
313
+ opacity: cs.opacity,
314
+ willChange: cs.willChange,
315
+ viewTimelineName: cs.viewTimelineName,
316
+ scrollTimelineName: cs.scrollTimelineName
317
+ });
318
+ }
319
+
320
+ return JSON.stringify(
321
+ {
322
+ prefersReducedMotion: window.matchMedia('(prefers-reduced-motion: reduce)').matches,
323
+ keyframes,
324
+ animatedElements: animated
325
+ },
326
+ null,
327
+ 2
328
+ );
329
+ })(''); // Pass section selector string, e.g. 'section.hero', instead of ''
330
+ ```
331
+
194
332
  ### Page Topology
195
333
  Map out every distinct section of the page from top to bottom. Give each a working name. Document:
196
334
  - Their visual order
@@ -281,7 +419,8 @@ For each section, use browser MCP to extract everything:
281
419
  'borderRadius','border','borderTop','borderBottom','borderLeft','borderRight',
282
420
  'boxShadow','overflow','overflowX','overflowY',
283
421
  'position','top','right','bottom','left','zIndex',
284
- 'opacity','transform','transition','cursor',
422
+ 'opacity','transform','transformOrigin','transition','transitionProperty','transitionDuration','transitionTimingFunction','transitionDelay',
423
+ 'animation','animationName','animationDuration','animationTimingFunction','animationDelay','animationIterationCount','animationDirection','animationFillMode','animationPlayState','animationTimeline','animationRange','viewTimelineName','scrollTimelineName','cursor',
285
424
  'objectFit','objectPosition','mixBlendMode','filter','backdropFilter',
286
425
  'whiteSpace','textOverflow','WebkitLineClamp'
287
426
  ];
@@ -360,6 +499,15 @@ For each section (or sub-component, if you're breaking it up), create a spec fil
360
499
  ### <Child element N>
361
500
  ...
362
501
 
502
+ ## Motion & animation (exact — from Chrome MCP `getComputedStyle` + stylesheet rules)
503
+
504
+ - **Keyframe names in use:** ...
505
+ - **Full `@keyframes` blocks** (paste `cssText` or equivalent); note **cross-origin** gaps and how you filled them
506
+ - **Per-element:** shorthand/longhand `animation` and `transition` values **verbatim** for resting state
507
+ - **Scroll/view timelines:** timeline name, range, scroll container, snap/observer thresholds
508
+ - **Libraries:** Lenis / GSAP / Framer / Lottie — hooks, init pattern, timing copied from reference
509
+ - **`prefers-reduced-motion`:** matched behavior on the reference
510
+
363
511
  ## States & Behaviors
364
512
 
365
513
  ### <Behavior name, e.g., "Scroll-triggered floating mode">
@@ -442,30 +590,34 @@ After all sections are built and merged, wire everything together in `src/app/pa
442
590
 
443
591
  After assembly, do NOT declare the clone complete. Take side-by-side comparison screenshots:
444
592
 
445
- 1. Open the original site and your clone side-by-side (or take screenshots at the same viewport widths)
593
+ 1. **Pixel-perfect comparison setup:** Match **Browser MCP: pixel-perfect capture** — same **CSS widths** (**1440**, **768**, **390**), **100% zoom**, and record `innerWidth`/`clientWidth`/`devicePixelRatio` on both reference and clone. Then open original and clone side-by-side or take paired screenshots at those widths.
446
594
  2. Compare section by section, top to bottom, at desktop (1440px)
447
- 3. Compare again at mobile (390px)
595
+ 3. Compare at tablet (768px) and again at mobile (390px)
448
596
  4. For each discrepancy found:
449
597
  - Check the component spec file — was the value extracted correctly?
450
598
  - If the spec was wrong: re-extract from browser MCP, update the spec, fix the component
451
599
  - If the spec was right but the builder got it wrong: fix the component to match the spec
452
600
  5. Test all interactive behaviors: scroll through the page, click every button/tab, hover over interactive elements
453
- 6. Verify smooth scroll feels right, header transitions work, tab switching works, animations play
601
+ 6. **Motion QA (Chrome MCP again if needed):** For every animated or transitioned element, re-run `getComputedStyle` on the reference vs. the clone at the **same scroll position / interaction state** and compare **`animation-*`, `transition-*`, `transform`, `opacity`** longhands — durations and easings must match character-for-character where the cascade specifies them. Re-record any `@keyframes` the reference still exposes so the clone can be patched until timings align.
602
+ 7. Verify smooth scroll feels right, header transitions work, tab switching works, stagger and scroll-driven reveals match the reference
454
603
 
455
- Only after this visual QA pass is the clone complete.
604
+ Only after this visual + motion QA pass is the clone complete.
456
605
 
457
606
  ## Pre-Dispatch Checklist
458
607
 
459
608
  Before dispatching ANY builder agent, verify you can check every box. If you can't, go back and extract more.
460
609
 
610
+ - [ ] **Multi-pass DOM reconnaissance:** structure (`DOM_SKETCH.md`) + asset sweep + motion/flow sweeps + reconciliation; repeated at **768 / 390** when layout or DOM order differs from 1440
461
611
  - [ ] Spec file written to `docs/research/components/<name>.spec.md` with ALL sections filled
462
- - [ ] Every CSS value in the spec is from `getComputedStyle()`, not estimated
612
+ - [ ] Every CSS value in the spec is from `getComputedStyle()`, not estimated — **verbatim**, including fractional px and color functions
613
+ - [ ] **MCP pixel-perfect pass:** viewports **1440 / 768 / 390**, **100% zoom**, viewport diagnostic recorded; **`getBoundingClientRect()`** and **`::before`/`::after`** captured where they affect layout or visuals
463
614
  - [ ] Interaction model is identified and documented (static / click / scroll / time)
464
615
  - [ ] For stateful components: every state's content and styles are captured
465
616
  - [ ] For scroll-driven components: trigger threshold, before/after styles, and transition are recorded
466
617
  - [ ] For hover states: before/after values and transition timing are recorded
618
+ - [ ] Motion: relevant `@keyframes` captured (or cross-origin gap documented with manual paste), `docs/research/MOTION.md` updated, component specs include **exact** `animation*` / `transition*` / timeline fields from Chrome MCP
467
619
  - [ ] All images in the section are identified (including overlays and layered compositions)
468
- - [ ] Responsive behavior is documented for at least desktop and mobile
620
+ - [ ] Responsive behavior is documented for **1440 / 768 / 390**
469
621
  - [ ] Text content is verbatim from the site, not paraphrased
470
622
  - [ ] The builder prompt is under ~150 lines of spec; if over, the section needs to be split
471
623
 
@@ -483,6 +635,8 @@ These are lessons from previous failed clones — each one cost hours of rework:
483
635
  - **Don't skip asset extraction.** Without real images, videos, and fonts, the clone will always look fake regardless of how perfect the CSS is.
484
636
  - **Don't give a builder agent too much scope.** If you're writing a builder prompt and it's getting long because the section is complex, that's a signal to break it into smaller tasks.
485
637
  - **Don't bundle unrelated sections into one agent.** A CTA section and a footer are different components with different designs — don't hand them both to one agent and hope for the best.
638
+ - **Don't round or “clean up” extracted values** for Tailwind guesses: `12.8px` stays `12.8px` in the spec unless the clone uses an exactly equivalent value.
639
+ - **Don't mix zoom levels or viewport widths** between reference extraction and QA — you will false-fail or false-pass pixel diffs.
486
640
  - **Don't skip responsive extraction.** If you only inspect at desktop width, the clone will break at tablet and mobile. Test at 1440, 768, and 390 during extraction.
487
641
  - **Don't forget smooth scroll libraries.** Check for Lenis (`.lenis` class), Locomotive Scroll, or similar. Default browser scrolling feels noticeably different and the user will spot it immediately.
488
642
  - **Don't dispatch builders without a spec file.** The spec file forces exhaustive extraction and creates an auditable artifact. Skipping it means the builder gets whatever you can fit in a prompt from memory.
@@ -490,6 +644,7 @@ These are lessons from previous failed clones — each one cost hours of rework:
490
644
  ## Completion
491
645
 
492
646
  When done, report:
647
+ - Multi-pass reconnaissance status (`docs/research/DOM_SKETCH.md` present; passes at **1440 / 768 / 390** as needed)
493
648
  - Total sections built
494
649
  - Total components created
495
650
  - Total spec files written (should match components)