launchframe 0.3.1 → 0.4.1
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.
- package/.amazonq/cli-agents/clone-website.json +9 -0
- package/.amazonq/cli-agents/launchframe.json +9 -0
- package/.amazonq/rules/project.md +158 -0
- package/{template/.augment → .augment}/commands/clone-website.md +35 -112
- package/.augment/commands/launchframe.md +46 -0
- package/.claude/skills/clone-website/SKILL.md +487 -0
- package/.claude/skills/launchframe/SKILL.md +45 -0
- package/.clinerules +158 -0
- package/.codex/skills/clone-website/SKILL.md +487 -0
- package/.codex/skills/launchframe/SKILL.md +45 -0
- package/{template/.continue → .continue}/commands/clone-website.md +35 -112
- package/.continue/commands/launchframe.md +47 -0
- package/.continue/rules/project.md +162 -0
- package/{template/.cursor → .cursor}/commands/clone-website.md +35 -112
- package/.cursor/commands/launchframe.md +42 -0
- package/.cursor/rules/project.mdc +7 -0
- package/{template/.gemini → .gemini}/commands/clone-website.toml +35 -112
- package/.gemini/commands/launchframe.toml +48 -0
- package/.github/copilot-instructions.md +158 -0
- package/.github/skills/clone-website/SKILL.md +487 -0
- package/.github/skills/launchframe/SKILL.md +45 -0
- package/.gitignore +49 -0
- package/{template/.opencode → .opencode}/commands/clone-website.md +35 -112
- package/.opencode/commands/launchframe.md +45 -0
- package/.windsurf/workflows/clone-website.md +484 -0
- package/.windsurf/workflows/launchframe.md +42 -0
- package/AGENTS.md +66 -0
- package/README.md +149 -31
- package/bin/launchframe.mjs +348 -315
- package/docs/research/INSPECTION_GUIDE.md +90 -0
- package/package.json +73 -26
- package/scripts/sync-skills.mjs +124 -0
- package/{template/src → src}/app/globals.css +1 -93
- package/{template/src → src}/app/layout.tsx +16 -5
- package/src/app/page.tsx +40 -0
- package/src/lib/launchframe-config.ts +8 -0
- package/template/.amazonq/cli-agents/clone-website.json +0 -9
- package/template/.amazonq/rules/project.md +0 -281
- package/template/.claude/skills/clone-website/SKILL.md +0 -564
- package/template/.claude/skills/marketing-social-proof-motion/SKILL.md +0 -47
- package/template/.clinerules +0 -285
- package/template/.codex/skills/clone-website/SKILL.md +0 -564
- package/template/.continue/rules/project.md +0 -285
- package/template/.cursor/commands/marketing-social-proof-motion.md +0 -42
- package/template/.cursor/rules/project.mdc +0 -22
- package/template/.github/copilot-instructions.md +0 -281
- package/template/.github/skills/clone-website/SKILL.md +0 -564
- package/template/.nvmrc +0 -1
- package/template/.windsurf/workflows/clone-website.md +0 -561
- package/template/AGENTS.md +0 -160
- package/template/LICENSE +0 -21
- package/template/README.md +0 -121
- package/template/START_HERE.md +0 -15
- package/template/docs/design-references/playwright-example.com-1440px.png +0 -0
- package/template/docs/design-references/playwright-example.com-390px.png +0 -0
- package/template/docs/research/INSPECTION_GUIDE.md +0 -124
- package/template/launchframe.config.json +0 -14
- package/template/package-lock.json +0 -9873
- package/template/package.json +0 -54
- package/template/scripts/.gitkeep +0 -0
- package/template/scripts/recon-playwright.mjs +0 -396
- package/template/scripts/sync-skills.mjs +0 -111
- package/template/src/app/page.tsx +0 -5
- package/template/src/components/marketing/scribewise-landing.tsx +0 -34
- package/template/src/hooks/.gitkeep +0 -0
- package/template/src/types/.gitkeep +0 -0
- /package/{template/.aider.conf.yml → .aider.conf.yml} +0 -0
- /package/{template/.dockerignore → .dockerignore} +0 -0
- /package/{template/.gitattributes → .gitattributes} +0 -0
- /package/{template/.github → .github}/ISSUE_TEMPLATE/bug_report.yml +0 -0
- /package/{template/.github → .github}/ISSUE_TEMPLATE/config.yml +0 -0
- /package/{template/.github → .github}/ISSUE_TEMPLATE/feature_request.yml +0 -0
- /package/{template/.github → .github}/PULL_REQUEST_TEMPLATE.md +0 -0
- /package/{template/.github → .github}/copilot-setup-steps.yml +0 -0
- /package/{template/.github → .github}/workflows/ci.yml +0 -0
- /package/{template/.windsurfrules → .windsurfrules} +0 -0
- /package/{template/CLAUDE.md → CLAUDE.md} +0 -0
- /package/{template/Dockerfile → Dockerfile} +0 -0
- /package/{template/Dockerfile.dev → Dockerfile.dev} +0 -0
- /package/{template/GEMINI.md → GEMINI.md} +0 -0
- /package/{template/components.json → components.json} +0 -0
- /package/{template/docker-compose.yml → docker-compose.yml} +0 -0
- /package/{template/docs → docs}/design-references/.gitkeep +0 -0
- /package/{template/docs → docs}/design-references/comparison.png +0 -0
- /package/{template/eslint.config.mjs → eslint.config.mjs} +0 -0
- /package/{template/next.config.ts → next.config.ts} +0 -0
- /package/{template/postcss.config.mjs → postcss.config.mjs} +0 -0
- /package/{template/public/images → scripts}/.gitkeep +0 -0
- /package/{template/scripts → scripts}/sync-agent-rules.sh +0 -0
- /package/{template/src → src}/app/favicon.ico +0 -0
- /package/{template/src → src}/components/ui/button.tsx +0 -0
- /package/{template/public/seo → src/hooks}/.gitkeep +0 -0
- /package/{template/src → src}/lib/utils.ts +0 -0
- /package/{template/public/videos → src/types}/.gitkeep +0 -0
- /package/{template/tsconfig.json → tsconfig.json} +0 -0
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: launchframe
|
|
3
|
+
description: Scaffold a new Next.js project with npx launchframe@latest using a reference URL and SaaS landing copy. Invoked as /launchframe.
|
|
4
|
+
argument-hint: "<url> \"<saas-idea>\""
|
|
5
|
+
user-invocable: true
|
|
6
|
+
---
|
|
7
|
+
|
|
8
|
+
# Launchframe
|
|
9
|
+
|
|
10
|
+
You are helping the user scaffold a **new** project from **$ARGUMENTS**:
|
|
11
|
+
|
|
12
|
+
- A **reference URL** (the website to copy later with `/clone-website`)
|
|
13
|
+
- A **SaaS idea** string (headline / positioning for the generated landing page)
|
|
14
|
+
|
|
15
|
+
## Parse inputs
|
|
16
|
+
|
|
17
|
+
1. Split **$ARGUMENTS** into:
|
|
18
|
+
- **URL** — First `http://` or `https://` URL. Normalize (trim, validate). If invalid or missing, ask once for a correct URL.
|
|
19
|
+
- **SaaS idea** — Everything after the URL, typically in quotes. Strip surrounding quotes. If empty, ask for a short product pitch.
|
|
20
|
+
|
|
21
|
+
2. Optional flags the user might pass (same as the CLI): `--dir` / `-o` for output folder, `--skip-install` to skip `npm install`.
|
|
22
|
+
|
|
23
|
+
## Run the published CLI
|
|
24
|
+
|
|
25
|
+
Execute the scaffold using the **latest** package from npm (exact intent: **`launchframe@latest`**):
|
|
26
|
+
|
|
27
|
+
```bash
|
|
28
|
+
npx launchframe@latest "<url>" "<saas-idea>" [--dir <folder>] [--skip-install]
|
|
29
|
+
```
|
|
30
|
+
|
|
31
|
+
- Use shell-appropriate quoting (PowerShell vs bash). Escape embedded quotes in the SaaS idea.
|
|
32
|
+
- **Output path:** The Launchframe CLI must not write **inside** the npm package / template directory itself (that would recurse). If the open workspace is this repository, run `npx` with `--dir` pointing to a **sibling** path (e.g. `..\\my-app` on Windows, `../my-app` on macOS/Linux) or ask the user for a folder **outside** the repo.
|
|
33
|
+
- Prefer letting the CLI run `npm install` unless the user passed `--skip-install`.
|
|
34
|
+
|
|
35
|
+
## After scaffold
|
|
36
|
+
|
|
37
|
+
Tell the user:
|
|
38
|
+
|
|
39
|
+
1. `cd` into the new project directory.
|
|
40
|
+
2. `npm run dev` to preview the SaaS landing shell.
|
|
41
|
+
3. Run **`/clone-website <same-reference-url>`** in that project so the agent can reverse-engineer the reference site into components, while keeping the SaaS idea from `launchframe.context.json` / `src/lib/launchframe-config.ts`.
|
|
42
|
+
|
|
43
|
+
## Fallback (local dev only)
|
|
44
|
+
|
|
45
|
+
If `npx launchframe@latest` is unavailable (offline), from a **checkout of this repo** you may run `node bin/launchframe.mjs "<url>" "<saas-idea>" ...` with the same rules for `--dir` outside the package root.
|
|
@@ -9,18 +9,7 @@ invokable: true
|
|
|
9
9
|
|
|
10
10
|
# Clone Website
|
|
11
11
|
|
|
12
|
-
You are about to reverse-engineer and rebuild
|
|
13
|
-
|
|
14
|
-
**Launchframe shorthand:** If the user only says **Build it**, **Go**, **Ship it**, **Clone the site**, or **Run launchframe** with no URL in the message, treat that as an invocation of this skill with empty `$ARGUMENTS` — **`launchframe.config.json` alone** supplies `url` and `idea`. Proceed without asking them to repeat those values unless the file is missing or invalid.
|
|
15
|
-
|
|
16
|
-
## Step 0: Read `launchframe.config.json`
|
|
17
|
-
|
|
18
|
-
**Before doing anything else**, read `launchframe.config.json` at the project root. This file was written by the `launchframe` CLI when the project was scaffolded and is the authoritative source of:
|
|
19
|
-
|
|
20
|
-
- `url` — the visual source-of-truth you are cloning
|
|
21
|
-
- `idea` — the user's SaaS idea, which becomes the rebranding directive applied after the pixel-perfect clone
|
|
22
|
-
|
|
23
|
-
If `$ARGUMENTS` is non-empty, treat the arguments as additional URLs (or an override) and merge them with the config — explicit CLI args win on conflict. If `launchframe.config.json` is missing, fall back to `$ARGUMENTS` and ask the user for an idea if one wasn't provided.
|
|
12
|
+
You are about to reverse-engineer and rebuild **$ARGUMENTS** as pixel-perfect clones.
|
|
24
13
|
|
|
25
14
|
When multiple URLs are provided, process them independently and in parallel where possible, while keeping each site's extraction artifacts isolated in dedicated folders (for example, `docs/research/<hostname>/`).
|
|
26
15
|
|
|
@@ -28,36 +17,36 @@ This is not a two-phase process (inspect then build). You are a **foreman walkin
|
|
|
28
17
|
|
|
29
18
|
## Scope Defaults
|
|
30
19
|
|
|
31
|
-
The target is
|
|
20
|
+
The target is whatever page `$ARGUMENTS` resolves to. Clone exactly what's visible at that URL. Unless the user specifies otherwise, use these defaults:
|
|
32
21
|
|
|
33
|
-
- **Fidelity level
|
|
34
|
-
- **
|
|
35
|
-
- **In scope:** Visual layout and styling, component structure and interactions, responsive design, mock data shaped for the SaaS idea
|
|
22
|
+
- **Fidelity level:** Pixel-perfect — exact match in colors, spacing, typography, animations
|
|
23
|
+
- **In scope:** Visual layout and styling, component structure and interactions, responsive design, mock data for demo purposes
|
|
36
24
|
- **Out of scope:** Real backend / database, authentication, real-time features, SEO optimization, accessibility audit
|
|
37
|
-
- **Customization
|
|
25
|
+
- **Customization:** None — pure emulation
|
|
38
26
|
|
|
39
|
-
If the user provides additional instructions (specific fidelity level,
|
|
27
|
+
If the user provides additional instructions (specific fidelity level, customizations, extra context), honor those over the defaults.
|
|
40
28
|
|
|
41
29
|
## Pre-Flight
|
|
42
30
|
|
|
43
|
-
1. **
|
|
44
|
-
2.
|
|
45
|
-
3.
|
|
46
|
-
4.
|
|
47
|
-
5.
|
|
48
|
-
6. When working with multiple sites in one command, optionally confirm whether to run them in parallel (recommended, if resources allow) or sequentially to avoid overload.
|
|
31
|
+
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.
|
|
32
|
+
2. Parse `$ARGUMENTS` as one or more URLs. Normalize and validate each URL; if any are invalid, ask the user to correct them before proceeding. For each valid URL, verify it is accessible via your browser MCP tool.
|
|
33
|
+
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.
|
|
34
|
+
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>/`.
|
|
35
|
+
5. When working with multiple sites in one command, optionally confirm whether to run them in parallel (recommended, if resources allow) or sequentially to avoid overload.
|
|
49
36
|
|
|
50
37
|
## Guiding Principles
|
|
51
38
|
|
|
52
39
|
These are the truths that separate a successful clone from a "close enough" mess. Internalize them — they should inform every decision you make.
|
|
53
40
|
|
|
54
|
-
### 0.
|
|
41
|
+
### 0. Visual crawl priority (images, SVGs, motion — first)
|
|
55
42
|
|
|
56
|
-
|
|
43
|
+
When you traverse the DOM and the Network panel, do **not** treat all nodes equally. Work in this **priority order** so the clone feels like the original, not a wireframe:
|
|
57
44
|
|
|
58
|
-
**
|
|
45
|
+
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.
|
|
46
|
+
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.
|
|
47
|
+
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.
|
|
59
48
|
|
|
60
|
-
|
|
49
|
+
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.
|
|
61
50
|
|
|
62
51
|
### 1. Completeness Beats Speed
|
|
63
52
|
|
|
@@ -75,6 +64,8 @@ Look at each section and judge its complexity. A simple banner with a heading an
|
|
|
75
64
|
|
|
76
65
|
Extract the actual text, images, videos, and SVGs from the live site. This is a clone, not a mockup. Use `element.textContent`, download every `<img>` and `<video>`, extract inline `<svg>` elements as React components. The only time you generate content is when something is clearly server-generated and unique per session.
|
|
77
66
|
|
|
67
|
+
**Prioritize** (see §0): downloadable imagery and backgrounds first, then SVG/icon layers, then motion. If you must **fabricate** an asset, prefer screenshot-based exports or traced vectors tied to measured box sizes — avoid unrelated stock art.
|
|
68
|
+
|
|
78
69
|
**Layered assets matter.** A section that looks like one image is often multiple layers — a background watercolor/gradient, a foreground UI mockup PNG, an overlay icon. Inspect each container's full DOM tree and enumerate ALL `<img>` elements and background images within it, including absolutely-positioned overlays. Missing an overlay image makes the clone look empty even if the background is correct.
|
|
79
70
|
|
|
80
71
|
### 4. Foundation First
|
|
@@ -144,12 +135,12 @@ Every builder agent must verify `npx tsc --noEmit` passes before finishing. Afte
|
|
|
144
135
|
|
|
145
136
|
Navigate to the target URL with browser MCP.
|
|
146
137
|
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
- Take
|
|
151
|
-
-
|
|
152
|
-
- These
|
|
138
|
+
Follow **§0 (Visual crawl priority)** during the entire reconnaissance pass: images and backgrounds → SVGs/icons → motion/animations — before spending time on secondary copy tweaks.
|
|
139
|
+
|
|
140
|
+
### Screenshots
|
|
141
|
+
- Take **full-page screenshots** at desktop (1440px) and mobile (390px) viewports
|
|
142
|
+
- Save to `docs/design-references/` with descriptive names
|
|
143
|
+
- These are your master reference — builders will receive section-specific crops/screenshots later
|
|
153
144
|
|
|
154
145
|
### Global Extraction
|
|
155
146
|
Extract these from the page before doing anything else:
|
|
@@ -205,12 +196,11 @@ Save this as `docs/research/PAGE_TOPOLOGY.md` — it becomes your assembly bluep
|
|
|
205
196
|
This is sequential. Do it yourself (not delegated to an agent) since it touches many files:
|
|
206
197
|
|
|
207
198
|
1. **Update fonts** in `layout.tsx` to match the target site's actual fonts
|
|
208
|
-
2. **
|
|
209
|
-
3. **
|
|
210
|
-
4. **
|
|
211
|
-
5. **
|
|
212
|
-
6.
|
|
213
|
-
7. Verify: `npm run build` passes
|
|
199
|
+
2. **Update globals.css** with the target's color tokens, spacing values, keyframe animations, utility classes, and any **global scroll behaviors** (Lenis, smooth scroll CSS, scroll-snap on body)
|
|
200
|
+
3. **Create TypeScript interfaces** in `src/types/` for the content structures you've observed
|
|
201
|
+
4. **Extract SVG icons** — find all inline `<svg>` elements on the page, deduplicate them, and save as named React components in `src/components/icons.tsx`. Name them by visual function (e.g., `SearchIcon`, `ArrowRightIcon`, `LogoIcon`).
|
|
202
|
+
5. **Download global assets** — write and run a Node.js script (`scripts/download-assets.mjs`) that downloads all images, videos, and other binary assets from the page to `public/`. Preserve meaningful directory structure.
|
|
203
|
+
6. Verify: `npm run build` passes
|
|
214
204
|
|
|
215
205
|
### Asset Discovery Script Pattern
|
|
216
206
|
|
|
@@ -367,25 +357,11 @@ For each section (or sub-component, if you're breaking it up), create a spec fil
|
|
|
367
357
|
- **State A (before):** maxWidth: 100vw, boxShadow: none, borderRadius: 0
|
|
368
358
|
- **State B (after):** maxWidth: 1200px, boxShadow: 0 4px 20px rgba(0,0,0,0.1), borderRadius: 16px
|
|
369
359
|
- **Transition:** transition: all 0.3s ease
|
|
370
|
-
- **Implementation approach:** <CSS transition + scroll listener | IntersectionObserver | CSS animation-timeline |
|
|
360
|
+
- **Implementation approach:** <CSS transition + scroll listener | IntersectionObserver | CSS animation-timeline | etc.>
|
|
371
361
|
|
|
372
362
|
### Hover states
|
|
373
363
|
- **<Element>:** <property>: <before> → <after>, transition: <value>
|
|
374
364
|
|
|
375
|
-
## Motion (Framer Motion vs CSS)
|
|
376
|
-
- **Tiers A–E:** <which tiers apply — load stagger, scroll reveal, ambient loop, interaction lifts, decorative parallax — list triggers>
|
|
377
|
-
- **Reduced motion:** <what disables or simplifies when user prefers reduced motion>
|
|
378
|
-
- **Entrance / scroll reveals:** <e.g. fade+translateY, staggerChildren — specify duration, easing, delay, viewport `once`/`margin`>
|
|
379
|
-
- **Library:** <`framer-motion` | CSS-only — justify if CSS-only>
|
|
380
|
-
- **Keyframes / springs:** <if any — match target curve>
|
|
381
|
-
|
|
382
|
-
## Illustration & pixel art *(production uplift — idea-native)*
|
|
383
|
-
- **Idea tie-in:** <one sentence — why this asset belongs only to this product>
|
|
384
|
-
- **Metaphor link:** <which keyword from `idea` / metaphor list this illustrates>
|
|
385
|
-
- **SVG components:** <named exports, paths using currentColor vs fixed fills>
|
|
386
|
-
- **Pixel motif:** <palette hex table, grid, scaling — or N/A>
|
|
387
|
-
- **Motif thread:** <how this echoes logo / OG / favicon for this SaaS>
|
|
388
|
-
|
|
389
365
|
## Per-State Content (if applicable)
|
|
390
366
|
|
|
391
367
|
### State: "Featured"
|
|
@@ -397,10 +373,9 @@ For each section (or sub-component, if you're breaking it up), create a spec fil
|
|
|
397
373
|
- Title: "..."
|
|
398
374
|
- Cards: [...]
|
|
399
375
|
|
|
400
|
-
## Assets
|
|
401
|
-
-
|
|
402
|
-
-
|
|
403
|
-
- Background layers: which div uses `background-image` and resolved URL → local path
|
|
376
|
+
## Assets
|
|
377
|
+
- Background image: `public/images/<file>.webp`
|
|
378
|
+
- Overlay image: `public/images/<file>.png`
|
|
404
379
|
- Icons used: <ArrowIcon>, <SearchIcon> from icons.tsx
|
|
405
380
|
|
|
406
381
|
## Text Content (verbatim)
|
|
@@ -453,49 +428,6 @@ After all sections are built and merged, wire everything together in `src/app/pa
|
|
|
453
428
|
- Implement page-level behaviors: scroll snap, scroll-driven animations, dark-to-light transitions, intersection observers, smooth scroll (Lenis etc.)
|
|
454
429
|
- Verify: `npm run build` passes clean
|
|
455
430
|
|
|
456
|
-
## Phase 4.5: SaaS Rebrand Pass
|
|
457
|
-
|
|
458
|
-
The pixel-perfect clone is done — now re-skin it for the SaaS idea from `launchframe.config.json`.
|
|
459
|
-
|
|
460
|
-
**Guiding rule:** swap words and brand marks, leave structure untouched. The original site's visual hierarchy was already validated by a real product team. Your job is to put the user's product into that proven shell, not to redesign it.
|
|
461
|
-
|
|
462
|
-
For every section, replace:
|
|
463
|
-
|
|
464
|
-
1. **Product name & logo** — wherever the original brand appears, use the SaaS idea's name (derive a short product name from the `idea` string if one isn't supplied — keep it 1–2 words, easy to lockup). Replace the wordmark text in place. For the logo glyph, either reuse the original SVG silhouette with a fresh fill, or use a Lucide icon that matches the SaaS category (e.g., `Brain` for AI, `Workflow` for automation, `Sparkles` for generative tooling). Do NOT keep the original brand's actual logo file.
|
|
465
|
-
2. **Hero headline & sub-headline** — write fresh copy that pitches the SaaS idea, using the original line lengths and tone as constraints. If the original is 6 words, write 6 words. If it's 14, write 14. Match emphasis, line breaks, and any inline highlighted phrase.
|
|
466
|
-
3. **Feature/section copy** — rewrite each feature card, callout, stat, and testimonial to fit the SaaS idea. Preserve the count and shape of items (3 feature cards stay 3 feature cards; a 4-column logo bar stays 4 columns). Generate plausible customer-logo names — never use real company names you haven't been authorized to use.
|
|
467
|
-
4. **CTA labels** — adapt button text to the SaaS idea ("Start free", "Get a demo", "Try it free", etc.). Keep the CTA hierarchy (primary/secondary) identical to the original.
|
|
468
|
-
5. **Mock data** — for product UI mockups embedded in marketing screenshots (e.g., a fake dashboard inside a hero), generate mock data shaped for the SaaS idea: realistic-looking but fictional rows, charts, conversation logs, etc.
|
|
469
|
-
6. **Imagery** — Replace photography/screenshots that depict the original brand with visuals **authored for this SaaS idea**, not interchangeable decoration. Before designing: derive a **metaphor list** from `idea` (3–6 concrete hooks). Each hero/feature/OG asset gets an **Idea tie-in** sentence in `docs/research/REBRAND.md`. Prefer bespoke SVG scenes (`src/components/marketing/art/`) or raster under `public/images/marketing/` that preserve dimensions/aspect/shadows from the cloned layout. **Avoid:** unrelated filler (e.g. apparel, random lifestyle props when the product is notes/voice), generic gradient-only heroes, Lucide-icon piles unless the reference was already that minimal — those fail the uniqueness bar in `AGENTS.md`.
|
|
470
|
-
7. **Metadata** — update `<title>`, meta description, OG tags, and favicon manifest in `src/app/layout.tsx` to reflect the new SaaS. Generate a simple favicon (initial letter on a brand-colored square) if no asset is provided.
|
|
471
|
-
|
|
472
|
-
What you must NOT change in this pass:
|
|
473
|
-
- Spacing, padding, typography scale, color tokens, **animation timing & motion choreography** (including Framer Motion `variants` / `transition` props), responsive breakpoints — those are still 1:1 to the original
|
|
474
|
-
- Section order, section count, component structure
|
|
475
|
-
- Interaction models (scroll-driven stays scroll-driven, etc.)
|
|
476
|
-
- Any computed-style value extracted in Phase 3
|
|
477
|
-
|
|
478
|
-
After the rebrand pass, the codebase should look like the original site visually but read like the user's SaaS at a glance. Save a short `docs/research/REBRAND.md` summarizing the product name you chose, the headline rewrites, and any assets you swapped — so the user can audit what's clone-derived vs. authored.
|
|
479
|
-
|
|
480
|
-
## Phase 4.6: Production uplift *(sparse references / stronger idea-specific art)*
|
|
481
|
-
|
|
482
|
-
Run this pass when **any** of the following is true:
|
|
483
|
-
|
|
484
|
-
- The cloned reference is mostly typography with weak imagery (internal demos, minimalist SaaS shells).
|
|
485
|
-
- The user asks for **more unique images**, **illustration tailored to the idea**, **stronger visuals**, motion, pixel art, or production polish.
|
|
486
|
-
- Visual QA feels “correct but dead” — layout matches but art is generic, unrelated, or repeated stock metaphors.
|
|
487
|
-
|
|
488
|
-
**Do not contradict pixel-perfect emulation when cloning a rich reference** — this phase *adds* or swaps **idea-native** imagery only where the brief allows uplift or the reference was inherently flat.
|
|
489
|
-
|
|
490
|
-
Checklist (mirror `AGENTS.md` — **uniqueness first**):
|
|
491
|
-
|
|
492
|
-
1. **Idea-tailored imagery** — Metaphor list from `idea`; replace any asset that could belong to another vertical. Per-asset **Idea tie-in** in `docs/research/PRODUCTION_UPLIFT.md` alongside SVG/raster paths.
|
|
493
|
-
2. **Density** — Layered hero + distinct scene per feature card in **one shared visual language** (stroke/accent/grid), still idea-specific.
|
|
494
|
-
3. **Motion tiers** — Implement at minimum **A + B + one of C/D**, optionally **E**: staggered hero load, `whileInView` sections + card stagger, looping ambient (marquee / caret / SVG dash loop), hover lifts / nav scroll shrink, optional light parallax. Gate loops with **`useReducedMotion()`**.
|
|
495
|
-
4. **Brand** — Motif thread + accent tokens in `:root`; favicon / OG echo **this** product narrative.
|
|
496
|
-
|
|
497
|
-
Document deltas in `docs/research/PRODUCTION_UPLIFT.md`.
|
|
498
|
-
|
|
499
431
|
## Phase 5: Visual QA Diff
|
|
500
432
|
|
|
501
433
|
After assembly, do NOT declare the clone complete. Take side-by-side comparison screenshots:
|
|
@@ -523,9 +455,6 @@ Before dispatching ANY builder agent, verify you can check every box. If you can
|
|
|
523
455
|
- [ ] For scroll-driven components: trigger threshold, before/after styles, and transition are recorded
|
|
524
456
|
- [ ] For hover states: before/after values and transition timing are recorded
|
|
525
457
|
- [ ] All images in the section are identified (including overlays and layered compositions)
|
|
526
|
-
- [ ] Any `<video>` (and poster), Lottie, or canvas-driven hero is identified — not approximated as a static div
|
|
527
|
-
- [ ] **Motion** subsection filled: tiers **A–E** coverage (see `AGENTS.md`), CSS vs **framer-motion**, durations, easings, stagger, scroll triggers, **reduced-motion** fallback
|
|
528
|
-
- [ ] **Illustration** subsection filled when uplift applies: **Idea tie-in** + metaphor link per asset, SVG/pixel detail — or explicit **N/A** with justification only on strict clone parity jobs
|
|
529
458
|
- [ ] Responsive behavior is documented for at least desktop and mobile
|
|
530
459
|
- [ ] Text content is verbatim from the site, not paraphrased
|
|
531
460
|
- [ ] The builder prompt is under ~150 lines of spec; if over, the section needs to be split
|
|
@@ -542,25 +471,19 @@ These are lessons from previous failed clones — each one cost hours of rework:
|
|
|
542
471
|
- **Don't build everything in one monolithic commit.** The whole point of this pipeline is incremental progress with verified builds at each step.
|
|
543
472
|
- **Don't reference docs from builder prompts.** Each builder gets the CSS spec inline in its prompt — never "see DESIGN_TOKENS.md for colors." The builder should have zero need to read external docs.
|
|
544
473
|
- **Don't skip asset extraction.** Without real images, videos, and fonts, the clone will always look fake regardless of how perfect the CSS is.
|
|
545
|
-
- **Don't defer image/video download to the end.** Run `MEDIA_MANIFEST.md` + `download-assets.mjs` during foundation so components reference real `public/` paths from the first build.
|
|
546
|
-
- **Don't fake complex motion with a single CSS `transition` when the target uses staggered, scroll-scrubbed, or layout-driven animation** — use **`framer-motion`** (`motion`, `whileInView`, `variants`, `staggerChildren`) and match duration/easing from extraction.
|
|
547
474
|
- **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.
|
|
548
475
|
- **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.
|
|
549
476
|
- **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.
|
|
550
477
|
- **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.
|
|
551
|
-
- **Don't ship interchangeable marketing art.** Random objects, unrelated lifestyle stock, or generic gradients that could match any SaaS violate the **idea-tailored** rule in `AGENTS.md` — every major visual needs a metaphor from `launchframe.config.json#idea`.
|
|
552
478
|
- **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.
|
|
553
479
|
|
|
554
480
|
## Completion
|
|
555
481
|
|
|
556
482
|
When done, report:
|
|
557
|
-
- Source URL cloned (from `launchframe.config.json` or `$ARGUMENTS`)
|
|
558
|
-
- SaaS idea applied (from `launchframe.config.json`) and the product name you chose
|
|
559
483
|
- Total sections built
|
|
560
484
|
- Total components created
|
|
561
485
|
- Total spec files written (should match components)
|
|
562
|
-
- Total assets downloaded (images, videos, SVGs, fonts)
|
|
563
|
-
- Rebrand summary (path to `docs/research/REBRAND.md`)
|
|
486
|
+
- Total assets downloaded (images, videos, SVGs, fonts)
|
|
564
487
|
- Build status (`npm run build` result)
|
|
565
488
|
- Visual QA results (any remaining discrepancies)
|
|
566
489
|
- Any known gaps or limitations
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: launchframe
|
|
3
|
+
description: "Scaffold a Next.js app with npx launchframe@latest from a reference URL and SaaS idea"
|
|
4
|
+
invokable: true
|
|
5
|
+
---
|
|
6
|
+
<!-- AUTO-GENERATED from .claude/skills/launchframe/SKILL.md — do not edit directly.
|
|
7
|
+
Run `node scripts/sync-skills.mjs` to regenerate. -->
|
|
8
|
+
|
|
9
|
+
|
|
10
|
+
# Launchframe
|
|
11
|
+
|
|
12
|
+
You are helping the user scaffold a **new** project from **$ARGUMENTS**:
|
|
13
|
+
|
|
14
|
+
- A **reference URL** (the website to copy later with `/clone-website`)
|
|
15
|
+
- A **SaaS idea** string (headline / positioning for the generated landing page)
|
|
16
|
+
|
|
17
|
+
## Parse inputs
|
|
18
|
+
|
|
19
|
+
1. Split **$ARGUMENTS** into:
|
|
20
|
+
- **URL** — First `http://` or `https://` URL. Normalize (trim, validate). If invalid or missing, ask once for a correct URL.
|
|
21
|
+
- **SaaS idea** — Everything after the URL, typically in quotes. Strip surrounding quotes. If empty, ask for a short product pitch.
|
|
22
|
+
|
|
23
|
+
2. Optional flags the user might pass (same as the CLI): `--dir` / `-o` for output folder, `--skip-install` to skip `npm install`.
|
|
24
|
+
|
|
25
|
+
## Run the published CLI
|
|
26
|
+
|
|
27
|
+
Execute the scaffold using the **latest** package from npm (exact intent: **`launchframe@latest`**):
|
|
28
|
+
|
|
29
|
+
```bash
|
|
30
|
+
npx launchframe@latest "<url>" "<saas-idea>" [--dir <folder>] [--skip-install]
|
|
31
|
+
```
|
|
32
|
+
|
|
33
|
+
- Use shell-appropriate quoting (PowerShell vs bash). Escape embedded quotes in the SaaS idea.
|
|
34
|
+
- **Output path:** The Launchframe CLI must not write **inside** the npm package / template directory itself (that would recurse). If the open workspace is this repository, run `npx` with `--dir` pointing to a **sibling** path (e.g. `..\\my-app` on Windows, `../my-app` on macOS/Linux) or ask the user for a folder **outside** the repo.
|
|
35
|
+
- Prefer letting the CLI run `npm install` unless the user passed `--skip-install`.
|
|
36
|
+
|
|
37
|
+
## After scaffold
|
|
38
|
+
|
|
39
|
+
Tell the user:
|
|
40
|
+
|
|
41
|
+
1. `cd` into the new project directory.
|
|
42
|
+
2. `npm run dev` to preview the SaaS landing shell.
|
|
43
|
+
3. Run **`/clone-website <same-reference-url>`** in that project so the agent can reverse-engineer the reference site into components, while keeping the SaaS idea from `launchframe.context.json` / `src/lib/launchframe-config.ts`.
|
|
44
|
+
|
|
45
|
+
## Fallback (local dev only)
|
|
46
|
+
|
|
47
|
+
If `npx launchframe@latest` is unavailable (offline), from a **checkout of this repo** you may run `node bin/launchframe.mjs "<url>" "<saas-idea>" ...` with the same rules for `--dir` outside the package root.
|
|
@@ -0,0 +1,162 @@
|
|
|
1
|
+
<!-- AUTO-GENERATED from AGENTS.md — do not edit directly.
|
|
2
|
+
Run `bash scripts/sync-agent-rules.sh` to regenerate. -->
|
|
3
|
+
|
|
4
|
+
---
|
|
5
|
+
description: Project conventions for AI Website Clone Template
|
|
6
|
+
alwaysApply: true
|
|
7
|
+
---
|
|
8
|
+
<!-- BEGIN:nextjs-agent-rules -->
|
|
9
|
+
# This is NOT the Next.js you know
|
|
10
|
+
|
|
11
|
+
This version has breaking changes — APIs, conventions, and file structure may all differ from your training data. Read the relevant guide in `node_modules/next/dist/docs/` before writing any code. Heed deprecation notices.
|
|
12
|
+
<!-- END:nextjs-agent-rules -->
|
|
13
|
+
|
|
14
|
+
# Website Reverse-Engineer Template
|
|
15
|
+
|
|
16
|
+
## What This Is
|
|
17
|
+
A reusable template for reverse-engineering any website into a clean, modern Next.js codebase using AI coding agents. The Next.js + shadcn/ui + Tailwind v4 base is pre-scaffolded — start a **new** project with **`/launchframe <url> "your saas idea"`** (runs **`npx launchframe@latest`**), or work in the current repo and run **`/clone-website <url1> [<url2> ...]`** to clone into this tree.
|
|
18
|
+
|
|
19
|
+
## Tech Stack
|
|
20
|
+
- **Framework:** Next.js 16 (App Router, React 19, TypeScript strict)
|
|
21
|
+
- **UI:** shadcn/ui (Radix primitives, Tailwind CSS v4, `cn()` utility)
|
|
22
|
+
- **Icons:** Lucide React (default — will be replaced/supplemented by extracted SVGs)
|
|
23
|
+
- **Styling:** Tailwind CSS v4 with oklch design tokens
|
|
24
|
+
- **Deployment:** Vercel
|
|
25
|
+
|
|
26
|
+
## Commands
|
|
27
|
+
- `npm run dev` — Start dev server
|
|
28
|
+
- `npm run build` — Production build
|
|
29
|
+
- `npm run lint` — ESLint check
|
|
30
|
+
- `npm run typecheck` — TypeScript check
|
|
31
|
+
- `npm run check` — Run lint + typecheck + build
|
|
32
|
+
|
|
33
|
+
## Code Style
|
|
34
|
+
- TypeScript strict mode, no `any`
|
|
35
|
+
- Named exports, PascalCase components, camelCase utils
|
|
36
|
+
- Tailwind utility classes, no inline styles
|
|
37
|
+
- 2-space indentation
|
|
38
|
+
- Responsive: mobile-first
|
|
39
|
+
|
|
40
|
+
## Design Principles
|
|
41
|
+
- **Pixel-perfect emulation** — match the target's spacing, colors, typography exactly
|
|
42
|
+
- **No personal aesthetic changes during emulation phase** — match 1:1 first, customize later
|
|
43
|
+
- **Real content** — use actual text and assets from the target site, not placeholders
|
|
44
|
+
- **Beauty-first** — every pixel matters
|
|
45
|
+
- **DOM crawl priority** — when walking the target page, emphasize **images** (raster, responsive sources, CSS backgrounds), **SVGs** (inline icons, sprites, masks), then **motion** (keyframes, transitions, scroll/time-driven animation, libraries). Scrape and save real assets from the DOM/network first; **create** a stand-in image or vector only when fetch/blocking prevents extraction, and label substitutes clearly in research notes so the clone stays auditable
|
|
46
|
+
|
|
47
|
+
## Project Structure
|
|
48
|
+
```
|
|
49
|
+
src/
|
|
50
|
+
app/ # Next.js routes
|
|
51
|
+
components/ # React components
|
|
52
|
+
ui/ # shadcn/ui primitives
|
|
53
|
+
icons.tsx # Extracted SVG icons as React components
|
|
54
|
+
lib/
|
|
55
|
+
utils.ts # cn() utility (shadcn)
|
|
56
|
+
types/ # TypeScript interfaces
|
|
57
|
+
hooks/ # Custom React hooks
|
|
58
|
+
public/
|
|
59
|
+
images/ # Downloaded images from target site
|
|
60
|
+
videos/ # Downloaded videos from target site
|
|
61
|
+
seo/ # Favicons, OG images, webmanifest
|
|
62
|
+
docs/
|
|
63
|
+
research/ # Inspection output (design tokens, components, layout)
|
|
64
|
+
design-references/ # Screenshots and visual references
|
|
65
|
+
scripts/ # Asset download scripts
|
|
66
|
+
```
|
|
67
|
+
|
|
68
|
+
## MOST IMPORTANT NOTES
|
|
69
|
+
- When launching Claude Code agent teams, ALWAYS have each teammate work in their own worktree branch and merge everyone's work at the end, resolving any merge conflicts smartly since you are basically serving the orchestrator role and have full context to our goals, work given, work achieved, and desired outcomes.
|
|
70
|
+
- After editing `AGENTS.md`, run `bash scripts/sync-agent-rules.sh` to regenerate platform-specific instruction files.
|
|
71
|
+
- After editing `.claude/skills/*/SKILL.md` (for example `clone-website` or `launchframe`), run `node scripts/sync-skills.mjs` to regenerate skill and command files for all platforms.
|
|
72
|
+
|
|
73
|
+
# Website Inspection Guide
|
|
74
|
+
|
|
75
|
+
## How to Reverse-Engineer Any Website
|
|
76
|
+
|
|
77
|
+
This guide outlines what to capture when inspecting a target website via Chrome MCP or browser DevTools.
|
|
78
|
+
|
|
79
|
+
## Priority: media, SVGs, and motion (do this early)
|
|
80
|
+
|
|
81
|
+
When crawling the DOM and network, **tackle these before fine-tuning copy or spacing**:
|
|
82
|
+
|
|
83
|
+
1. **Raster imagery** — Every `<img>`, `<picture>` / `source`, `srcset` / `sizes`, CDN URLs, lazy-loaded `data-src`, `loading="lazy"` nodes, **CSS `background-image`** on the element and ancestors (including `::before` / `::after`), masks that use `url()`, `<video>` still / poster frames. Prefer **downloading** originals via scripts or MCP; if a URL is blocked or session-gated, **export a screenshot** of the element’s bounding box at a crisp DPR and store it under `public/images/`, and note the substitute in the spec.
|
|
84
|
+
2. **SVGs** — Inline `<svg>`, `<use>` / sprite sheets, **SVG in CSS** (`mask-image`, `background-image`), favicons as SVG, logo marks. Prefer extracting path/viewBox into React components or static files under `public/` — **recreate** from a screenshot/trace only when the markup is obfuscated or blocked.
|
|
85
|
+
3. **Motion & animation** — Inspect Styles for `animation`, `animation-name`, `animation-timeline`, `transition`, `transform`, `@keyframes`; check for libraries (Framer Motion, GSAP, Lottie, Lenis). Capture **durations, easings, delays, fill-modes**, scroll/view triggers, and `prefers-reduced-motion` handling. Motion often defines perceived quality — do not leave it as an afterthought.
|
|
86
|
+
|
|
87
|
+
Then continue with typography, spacing, and component structure as usual.
|
|
88
|
+
|
|
89
|
+
## Phase 1: Visual Audit
|
|
90
|
+
|
|
91
|
+
### Screenshots to Capture
|
|
92
|
+
- [ ] Every distinct page — desktop, tablet, mobile
|
|
93
|
+
- [ ] Dark mode variants (if applicable)
|
|
94
|
+
- [ ] Light mode variants (if applicable)
|
|
95
|
+
- [ ] Key interaction states (hover, active, open menus, modals)
|
|
96
|
+
- [ ] Loading/skeleton states
|
|
97
|
+
- [ ] Empty states
|
|
98
|
+
- [ ] Error states
|
|
99
|
+
|
|
100
|
+
### Design Tokens to Extract
|
|
101
|
+
- [ ] **Colors** — background, text (primary/secondary/muted), accent, border, hover, error, success, warning
|
|
102
|
+
- [ ] **Typography** — font family, sizes (h1-h6, body, caption, label), weights, line heights, letter spacing
|
|
103
|
+
- [ ] **Spacing** — padding/margin patterns (look for a scale: 4px, 8px, 12px, 16px, 24px, 32px, etc.)
|
|
104
|
+
- [ ] **Border radius** — buttons, cards, avatars, inputs
|
|
105
|
+
- [ ] **Shadows/elevation** — card shadows, dropdown shadows, modal overlay
|
|
106
|
+
- [ ] **Breakpoints** — when does the layout shift? (inspect with DevTools responsive mode)
|
|
107
|
+
- [ ] **Icons** — which icon library? custom SVGs? sizes?
|
|
108
|
+
- [ ] **Avatars** — sizes, shapes, fallback behavior
|
|
109
|
+
- [ ] **Buttons** — all variants (primary, secondary, ghost, icon-only, danger)
|
|
110
|
+
- [ ] **Inputs** — text fields, textareas, selects, checkboxes, toggles
|
|
111
|
+
|
|
112
|
+
## Phase 2: Component Inventory
|
|
113
|
+
|
|
114
|
+
For each distinct UI component, document:
|
|
115
|
+
1. **Name** — what would you call this component?
|
|
116
|
+
2. **Structure** — what HTML elements / child components does it contain?
|
|
117
|
+
3. **Variants** — does it have different sizes, colors, or states?
|
|
118
|
+
4. **States** — default, hover, active, disabled, loading, error, empty
|
|
119
|
+
5. **Responsive behavior** — how does it change at different breakpoints?
|
|
120
|
+
6. **Interactions** — click, hover, focus, keyboard navigation
|
|
121
|
+
7. **Animations** — transitions, entrance/exit animations, micro-interactions
|
|
122
|
+
|
|
123
|
+
### Common Components to Look For
|
|
124
|
+
- Navigation (top bar, sidebar, bottom bar)
|
|
125
|
+
- Cards / list items
|
|
126
|
+
- Buttons and links
|
|
127
|
+
- Forms and inputs
|
|
128
|
+
- Modals and dialogs
|
|
129
|
+
- Dropdowns and menus
|
|
130
|
+
- Tabs and segmented controls
|
|
131
|
+
- Avatars and user badges
|
|
132
|
+
- Loading skeletons
|
|
133
|
+
- Toast notifications
|
|
134
|
+
- Tooltips and popovers
|
|
135
|
+
|
|
136
|
+
## Phase 3: Layout Architecture
|
|
137
|
+
|
|
138
|
+
- [ ] **Grid system** — CSS Grid? Flexbox? Fixed widths?
|
|
139
|
+
- [ ] **Column layout** — how many columns at each breakpoint?
|
|
140
|
+
- [ ] **Max-width** — main content area max-width
|
|
141
|
+
- [ ] **Sticky elements** — header, sidebar, floating buttons
|
|
142
|
+
- [ ] **Z-index layers** — navigation, modals, tooltips, overlays
|
|
143
|
+
- [ ] **Scroll behavior** — infinite scroll, pagination, virtual scrolling
|
|
144
|
+
|
|
145
|
+
## Phase 4: Technical Stack Analysis
|
|
146
|
+
|
|
147
|
+
- [ ] **Framework** — React? Vue? Angular? Check `__NEXT_DATA__`, `__NUXT__`, `ng-version`
|
|
148
|
+
- [ ] **CSS approach** — Tailwind (utility classes), CSS Modules, Styled Components, Emotion, vanilla CSS
|
|
149
|
+
- [ ] **State management** — Redux (check DevTools), React Query, Zustand, Pinia
|
|
150
|
+
- [ ] **API patterns** — REST, GraphQL (check network tab for `/graphql` requests)
|
|
151
|
+
- [ ] **Font loading** — Google Fonts, self-hosted, system fonts
|
|
152
|
+
- [ ] **Image strategy** — CDN, lazy loading, srcset, WebP/AVIF
|
|
153
|
+
- [ ] **Animation library** — Framer Motion, GSAP, CSS transitions only
|
|
154
|
+
|
|
155
|
+
## Phase 5: Documentation Output
|
|
156
|
+
|
|
157
|
+
After inspection, create these files in `docs/research/`:
|
|
158
|
+
1. `DESIGN_TOKENS.md` — All extracted colors, typography, spacing
|
|
159
|
+
2. `COMPONENT_INVENTORY.md` — Every component with structure notes
|
|
160
|
+
3. `LAYOUT_ARCHITECTURE.md` — Page layouts, grid system, responsive behavior
|
|
161
|
+
4. `INTERACTION_PATTERNS.md` — Animations, transitions, hover states
|
|
162
|
+
5. `TECH_STACK_ANALYSIS.md` — What the site uses and our chosen equivalents
|