mirror of
https://github.com/garrytan/gstack.git
synced 2026-05-01 19:25:10 +02:00
b805aa0113
* feat: add Confusion Protocol to preamble resolver Injects a high-stakes ambiguity gate at preamble tier >= 2 so all workflow skills get it. Fires when Claude encounters architectural decisions, data model changes, destructive operations, or contradictory requirements. Does NOT fire on routine coding. Addresses Karpathy failure mode #1 (wrong assumptions) with an inline STOP gate instead of relying on workflow skill invocation. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * feat: add Hermes and GBrain host configs Hermes: tool rewrites for terminal/read_file/patch/delegate_task, paths to ~/.hermes/skills/gstack, AGENTS.md config file. GBrain: coding skills become brain-aware when GBrain mod is installed. Same tool rewrites as OpenClaw (agents spawn Claude Code via ACP). GBRAIN_CONTEXT_LOAD and GBRAIN_SAVE_RESULTS NOT suppressed on gbrain host, enabling brain-first lookup and save-to-brain behavior. Both registered in hosts/index.ts with setup script redirect messages. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * feat: GBrain resolver — brain-first lookup and save-to-brain New scripts/resolvers/gbrain.ts with two resolver functions: - GBRAIN_CONTEXT_LOAD: search brain for context before skill starts - GBRAIN_SAVE_RESULTS: save skill output to brain after completion Placeholders added to 4 thinking skill templates (office-hours, investigate, plan-ceo-review, retro). Resolves to empty string on all hosts except gbrain via suppressedResolvers. GBRAIN suppression added to all 9 non-gbrain host configs. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * feat: wire slop:diff into /review as advisory diagnostic Adds Step 3.5 to the review template: runs bun run slop:diff against the base branch to catch AI code quality issues (empty catches, redundant return await, overcomplicated abstractions). Advisory only, never blocking. Skips silently if slop-scan is not installed. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * docs: add Karpathy compatibility note to README Positions gstack as the workflow enforcement layer for Karpathy-style CLAUDE.md rules (17K stars). Links to forrestchang/andrej-karpathy-skills. Maps each Karpathy failure mode to the gstack skill that addresses it. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * fix: improve native OpenClaw thinking skills office-hours: add design doc path visibility message after writing ceo-review: add HARD GATE reminder at review section transitions retro: add non-git context support (check memory for meeting notes) Mirrors template improvements to hand-crafted native skills. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * chore: update tests and golden fixtures for new hosts - Host count: 8 → 10 (hermes, gbrain) - OpenClaw adapter test: expects undefined (dead code removed) - Golden ship fixtures: updated with Confusion Protocol + vendoring Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * chore: regenerate all SKILL.md files Regenerated from templates after Confusion Protocol, GBrain resolver placeholders, slop:diff in review, HARD GATE reminders, investigation learnings, design doc visibility, and retro non-git context changes. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * docs: update project documentation for v0.18.0.0 - CHANGELOG: add v0.18.0.0 entry (Confusion Protocol, Hermes, GBrain, slop in review, Karpathy note, skill improvements) - CLAUDE.md: add hermes.ts and gbrain.ts to hosts listing - README.md: update agent count 8→10, add Hermes + GBrain to table - VERSION: bump to 0.18.0.0 Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * chore: sync package.json version to 0.18.0.0 Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * fix: extract Step 0 from review SKILL.md in E2E test The review-base-branch E2E test was copying the full 1493-line review/SKILL.md into the test fixture. The agent spent 8+ turns reading it in chunks, leaving only 7 turns for actual work, causing error_max_turns on every attempt. Now extracts only Step 0 (base branch detection, ~50 lines) which is all the test actually needs. Follows the CLAUDE.md rule: "NEVER copy a full SKILL.md file into an E2E test fixture." Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * feat: update GBrain and Hermes host configs for v0.10.0 integration GBrain: add 'triggers' to keepFields so generated skills pass checkResolvable() validation. Add version compat comment. Hermes: un-suppress GBRAIN_CONTEXT_LOAD and GBRAIN_SAVE_RESULTS. The resolvers handle GBrain-not-installed gracefully, so Hermes agents with GBrain as a mod get brain features automatically. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * feat: GBrain resolver DX improvements and preamble health check Resolver changes: - gbrain query → gbrain search (fast keyword search, not expensive hybrid) - Add keyword extraction guidance for agents - Show explicit gbrain put_page syntax with --title, --tags, heredoc - Add entity enrichment with false-positive filter - Name throttle error patterns (exit code 1, stderr keywords) - Add data-research routing for investigate skill - Expand skillSaveMap from 4 to 8 entries - Add brain operation telemetry summary Preamble changes: - Add gbrain doctor --fast --json health check for gbrain/hermes hosts - Parse check failures/warnings count - Show failing check details when score < 50 Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * fix: preserve keepFields in allowlist frontmatter mode The allowlist mode hard-coded name + description reconstruction but never iterated keepFields for additional fields. Adding 'triggers' to keepFields was a no-op because the field was silently stripped. Now iterates keepFields and preserves any field beyond name/description from the source template frontmatter, including YAML arrays. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * feat: add triggers to all 38 skill templates Multi-word, skill-specific trigger keywords for GBrain's RESOLVER.md router. Each skill gets 3-6 triggers derived from its "Use when asked to..." description text. Avoids single generic words that would collide across skills (e.g., "debug this" not "debug"). These are distinct from voice-triggers (speech-to-text aliases) and serve GBrain's checkResolvable() validation. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * chore: regenerate all SKILL.md files and update golden fixtures Regenerated from updated templates (triggers, brain placeholders, resolver DX improvements, preamble health check). Golden fixtures updated to match. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * fix: settings-hook remove exits 1 when nothing to remove gstack-settings-hook remove was exiting 0 when settings.json didn't exist, causing gstack-uninstall to report "SessionStart hook" as removed on clean systems where nothing was installed. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * docs: update project documentation for GBrain v0.10.0 integration ARCHITECTURE.md: added GBRAIN_CONTEXT_LOAD and GBRAIN_SAVE_RESULTS to resolver table. CHANGELOG.md: expanded v0.18.0.0 entry with GBrain v0.10.0 integration details (triggers, expanded brain-awareness, DX improvements, Hermes brain support), updated date. CLAUDE.md: added gbrain to resolvers/ directory comment. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * fix: routing E2E stops writing to user's ~/.claude/skills/ installSkills() was copying SKILL.md files to both project-level (.claude/skills/ in tmpDir) and user-level (~/.claude/skills/). Writing to the user's real install fails when symlinks point to different worktrees or dangling targets (ENOENT on copyFileSync). Now installs to project-level only. The test already sets cwd to the tmpDir, so project-level discovery works. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> * chore: scale Gemini E2E back to smoke test Gemini CLI gets lost in worktrees on complex tasks (review times out at 600s, discover-skill hits exit 124). Nobody uses Gemini for gstack skill execution. Replace the two failing tests (gemini-discover-skill and gemini-review-findings) with a single smoke test that verifies Gemini can start and read the README. 90s timeout, no skill invocation. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com> --------- Co-authored-by: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
601 lines
22 KiB
Cheetah
601 lines
22 KiB
Cheetah
---
|
|
name: design-html
|
|
preamble-tier: 2
|
|
version: 1.0.0
|
|
description: |
|
|
Design finalization: generates production-quality Pretext-native HTML/CSS.
|
|
Works with approved mockups from /design-shotgun, CEO plans from /plan-ceo-review,
|
|
design review context from /plan-design-review, or from scratch with a user
|
|
description. Text actually reflows, heights are computed, layouts are dynamic.
|
|
30KB overhead, zero deps. Smart API routing: picks the right Pretext patterns
|
|
for each design type. Use when: "finalize this design", "turn this into HTML",
|
|
"build me a page", "implement this design", or after any planning skill.
|
|
Proactively suggest when user has approved a design or has a plan ready. (gstack)
|
|
voice-triggers:
|
|
- "build the design"
|
|
- "code the mockup"
|
|
- "make it real"
|
|
triggers:
|
|
- build the design
|
|
- code the mockup
|
|
- make design real
|
|
allowed-tools:
|
|
- Bash
|
|
- Read
|
|
- Write
|
|
- Edit
|
|
- Glob
|
|
- Grep
|
|
- Agent
|
|
- AskUserQuestion
|
|
---
|
|
|
|
{{PREAMBLE}}
|
|
|
|
# /design-html: Pretext-Native HTML Engine
|
|
|
|
You generate production-quality HTML where text actually works correctly. Not CSS
|
|
approximations. Computed layout via Pretext. Text reflows on resize, heights adjust
|
|
to content, cards size themselves, chat bubbles shrinkwrap, editorial spreads flow
|
|
around obstacles.
|
|
|
|
{{DESIGN_SETUP}}
|
|
|
|
{{UX_PRINCIPLES}}
|
|
|
|
{{BROWSE_SETUP}}
|
|
|
|
---
|
|
|
|
## Step 0: Input Detection
|
|
|
|
```bash
|
|
eval "$(~/.claude/skills/gstack/bin/gstack-slug 2>/dev/null)"
|
|
```
|
|
|
|
Detect what design context exists for this project. Run all four checks:
|
|
|
|
```bash
|
|
setopt +o nomatch 2>/dev/null || true
|
|
_CEO=$(ls -t ~/.gstack/projects/$SLUG/ceo-plans/*.md 2>/dev/null | head -1)
|
|
[ -n "$_CEO" ] && echo "CEO_PLAN: $_CEO" || echo "NO_CEO_PLAN"
|
|
```
|
|
|
|
```bash
|
|
setopt +o nomatch 2>/dev/null || true
|
|
_APPROVED=$(ls -t ~/.gstack/projects/$SLUG/designs/*/approved.json 2>/dev/null | head -1)
|
|
[ -n "$_APPROVED" ] && echo "APPROVED: $_APPROVED" || echo "NO_APPROVED"
|
|
```
|
|
|
|
```bash
|
|
setopt +o nomatch 2>/dev/null || true
|
|
_VARIANTS=$(ls -t ~/.gstack/projects/$SLUG/designs/*/variant-*.png 2>/dev/null | head -1)
|
|
[ -n "$_VARIANTS" ] && echo "VARIANTS: $_VARIANTS" || echo "NO_VARIANTS"
|
|
```
|
|
|
|
```bash
|
|
setopt +o nomatch 2>/dev/null || true
|
|
_FINALIZED=$(ls -t ~/.gstack/projects/$SLUG/designs/*/finalized.html 2>/dev/null | head -1)
|
|
[ -n "$_FINALIZED" ] && echo "FINALIZED: $_FINALIZED" || echo "NO_FINALIZED"
|
|
[ -f DESIGN.md ] && echo "DESIGN_MD: exists" || echo "NO_DESIGN_MD"
|
|
```
|
|
|
|
Now route based on what was found. Check these cases in order:
|
|
|
|
### Case A: approved.json exists (design-shotgun ran)
|
|
|
|
If `APPROVED` was found, read it. Extract: approved variant PNG path, user feedback,
|
|
screen name. Also read the CEO plan if one exists (it adds strategic context).
|
|
|
|
Read `DESIGN.md` if it exists in the repo root. These tokens take priority for
|
|
system-level values (fonts, brand colors, spacing scale).
|
|
|
|
Then check for prior finalized.html. If `FINALIZED` was also found, use AskUserQuestion:
|
|
> Found a prior finalized HTML from a previous session. Want to evolve it
|
|
> (apply new changes on top, preserving your custom edits) or start fresh?
|
|
> A) Evolve — iterate on the existing HTML
|
|
> B) Start fresh — regenerate from the approved mockup
|
|
|
|
If evolve: read the existing HTML. Apply changes on top during Step 3.
|
|
If fresh or no finalized.html: proceed to Step 1 with the approved PNG as the
|
|
visual reference.
|
|
|
|
### Case B: CEO plan and/or design variants exist, but no approved.json
|
|
|
|
If `CEO_PLAN` or `VARIANTS` was found but no `APPROVED`:
|
|
|
|
Read whichever context exists:
|
|
- If CEO plan found: read it and summarize the product vision and design requirements.
|
|
- If variant PNGs found: show them inline using the Read tool.
|
|
- If DESIGN.md found: read it for design tokens and constraints.
|
|
|
|
Use AskUserQuestion:
|
|
> Found [CEO plan from /plan-ceo-review | design review variants from /plan-design-review | both]
|
|
> but no approved design mockup.
|
|
> A) Run /design-shotgun — explore design variants based on the existing plan context
|
|
> B) Skip mockups — I'll design the HTML directly from the plan context
|
|
> C) I have a PNG — let me provide the path
|
|
|
|
If A: tell the user to run /design-shotgun, then come back to /design-html.
|
|
If B: proceed to Step 1 in "plan-driven mode." There is no approved PNG, the plan is
|
|
the source of truth. Ask the user for a screen name to use for the output directory
|
|
(e.g., "landing-page", "dashboard", "pricing").
|
|
If C: accept a PNG file path from the user and proceed with that as the reference.
|
|
|
|
### Case C: Nothing found (clean slate)
|
|
|
|
If none of the above produced any context:
|
|
|
|
Use AskUserQuestion:
|
|
> No design context found for this project. How do you want to start?
|
|
> A) Run /plan-ceo-review first — think through the product strategy before designing
|
|
> B) Run /plan-design-review first — design review with visual mockups
|
|
> C) Run /design-shotgun — jump straight to visual design exploration
|
|
> D) Just describe it — tell me what you want and I'll design the HTML live
|
|
|
|
If A, B, or C: tell the user to run that skill, then come back to /design-html.
|
|
If D: proceed to Step 1 in "freeform mode." Ask the user for a screen name.
|
|
|
|
### Context summary
|
|
|
|
After routing, output a brief context summary:
|
|
- **Mode:** approved-mockup | plan-driven | freeform | evolve
|
|
- **Visual reference:** path to approved PNG, or "none (plan-driven)" or "none (freeform)"
|
|
- **CEO plan:** path or "none"
|
|
- **Design tokens:** "DESIGN.md" or "none"
|
|
- **Screen name:** from approved.json, user-provided, or inferred from CEO plan
|
|
|
|
---
|
|
|
|
## Step 1: Design Analysis
|
|
|
|
1. If `$D` is available (`DESIGN_READY`), extract a structured implementation spec:
|
|
```bash
|
|
$D prompt --image <approved-variant.png> --output json
|
|
```
|
|
This returns colors, typography, layout structure, and component inventory via GPT-4o vision.
|
|
|
|
2. If `$D` is not available, read the approved PNG inline using the Read tool.
|
|
Describe the visual layout, colors, typography, and component structure yourself.
|
|
|
|
3. If in plan-driven or freeform mode (no approved PNG), design from context:
|
|
- **Plan-driven:** read the CEO plan and/or design review notes. Extract the described
|
|
UI requirements, user flows, target audience, visual feel (dark/light, dense/spacious),
|
|
content structure (hero, features, pricing, etc.), and design constraints. Build an
|
|
implementation spec from the plan's prose rather than a visual reference.
|
|
- **Freeform:** use AskUserQuestion to gather what the user wants to build. Ask about:
|
|
purpose/audience, visual feel (dark/light, playful/serious, dense/spacious),
|
|
content structure (hero, features, pricing, etc.), and any reference sites they like.
|
|
In both cases, describe the intended visual layout, colors, typography, and
|
|
component structure as your implementation spec. Generate realistic content based
|
|
on the plan or user description (never lorem ipsum).
|
|
|
|
4. Read `DESIGN.md` tokens. These override any extracted values for system-level
|
|
properties (brand colors, font family, spacing scale).
|
|
|
|
5. Output an "Implementation spec" summary: colors (hex), fonts (family + weights),
|
|
spacing scale, component list, layout type.
|
|
|
|
---
|
|
|
|
## Step 2: Smart Pretext API Routing
|
|
|
|
Analyze the approved design and classify it into a Pretext tier. Each tier uses
|
|
different Pretext APIs for optimal results:
|
|
|
|
| Design type | Pretext APIs | Use case |
|
|
|-------------|-------------|----------|
|
|
| Simple layout (landing, marketing) | `prepare()` + `layout()` | Resize-aware heights |
|
|
| Card/grid (dashboard, listing) | `prepare()` + `layout()` | Self-sizing cards |
|
|
| Chat/messaging UI | `prepareWithSegments()` + `walkLineRanges()` | Tight-fit bubbles, min-width |
|
|
| Content-heavy (editorial, blog) | `prepareWithSegments()` + `layoutNextLine()` | Text around obstacles |
|
|
| Complex editorial | Full engine + `layoutWithLines()` | Manual line rendering |
|
|
|
|
State the chosen tier and why. Reference the specific Pretext APIs that will be used.
|
|
|
|
---
|
|
|
|
## Step 2.5: Framework Detection
|
|
|
|
Check if the user's project uses a frontend framework:
|
|
|
|
```bash
|
|
[ -f package.json ] && cat package.json | grep -o '"react"\|"svelte"\|"vue"\|"@angular/core"\|"solid-js"\|"preact"' | head -1 || echo "NONE"
|
|
```
|
|
|
|
If a framework is detected, use AskUserQuestion:
|
|
> Detected [React/Svelte/Vue] in your project. What format should the output be?
|
|
> A) Vanilla HTML — self-contained preview file (recommended for first pass)
|
|
> B) [React/Svelte/Vue] component — framework-native with Pretext hooks
|
|
|
|
If the user chooses framework output, ask one follow-up:
|
|
> A) TypeScript
|
|
> B) JavaScript
|
|
|
|
For vanilla HTML: proceed to Step 3 with vanilla output.
|
|
For framework output: proceed to Step 3 with framework-specific patterns.
|
|
If no framework detected: default to vanilla HTML, no question needed.
|
|
|
|
---
|
|
|
|
## Step 3: Generate Pretext-Native HTML
|
|
|
|
### Pretext Source Embedding
|
|
|
|
For **vanilla HTML output**, check for the vendored Pretext bundle:
|
|
```bash
|
|
_PRETEXT_VENDOR=""
|
|
_ROOT=$(git rev-parse --show-toplevel 2>/dev/null)
|
|
[ -n "$_ROOT" ] && [ -f "$_ROOT/.claude/skills/gstack/design-html/vendor/pretext.js" ] && _PRETEXT_VENDOR="$_ROOT/.claude/skills/gstack/design-html/vendor/pretext.js"
|
|
[ -z "$_PRETEXT_VENDOR" ] && [ -f ~/.claude/skills/gstack/design-html/vendor/pretext.js ] && _PRETEXT_VENDOR=~/.claude/skills/gstack/design-html/vendor/pretext.js
|
|
[ -n "$_PRETEXT_VENDOR" ] && echo "VENDOR: $_PRETEXT_VENDOR" || echo "VENDOR_MISSING"
|
|
```
|
|
|
|
- If `VENDOR` found: read the file and inline it in a `<script>` tag. The HTML file
|
|
is fully self-contained with zero network dependencies.
|
|
- If `VENDOR_MISSING`: use CDN import as fallback:
|
|
`<script type="module">import { prepare, layout, prepareWithSegments, walkLineRanges, layoutNextLine, layoutWithLines } from 'https://esm.sh/@chenglou/pretext'</script>`
|
|
Add a comment: `<!-- FALLBACK: vendor/pretext.js missing, using CDN -->`
|
|
|
|
For **framework output**, add to the project's dependencies instead:
|
|
```bash
|
|
# Detect package manager
|
|
[ -f bun.lockb ] && echo "bun add @chenglou/pretext" || \
|
|
[ -f pnpm-lock.yaml ] && echo "pnpm add @chenglou/pretext" || \
|
|
[ -f yarn.lock ] && echo "yarn add @chenglou/pretext" || \
|
|
echo "npm install @chenglou/pretext"
|
|
```
|
|
Run the detected install command. Then use standard imports in the component.
|
|
|
|
### HTML Generation
|
|
|
|
Write a single file using the Write tool. Save to:
|
|
`~/.gstack/projects/$SLUG/designs/<screen-name>-YYYYMMDD/finalized.html`
|
|
|
|
For framework output, save to:
|
|
`~/.gstack/projects/$SLUG/designs/<screen-name>-YYYYMMDD/finalized.[tsx|svelte|vue]`
|
|
|
|
**Always include in vanilla HTML:**
|
|
- Pretext source (inlined or CDN, see above)
|
|
- CSS custom properties for design tokens from DESIGN.md / Step 1 extraction
|
|
- Google Fonts via `<link>` tags + `document.fonts.ready` gate before first `prepare()`
|
|
- Semantic HTML5 (`<header>`, `<nav>`, `<main>`, `<section>`, `<footer>`)
|
|
- Responsive behavior via Pretext relayout (not just media queries)
|
|
- Breakpoint-specific adjustments at 375px, 768px, 1024px, 1440px
|
|
- ARIA attributes, heading hierarchy, focus-visible states
|
|
- `contenteditable` on text elements + MutationObserver to re-prepare + re-layout on edit
|
|
- ResizeObserver on containers to re-layout on resize
|
|
- `prefers-color-scheme` media query for dark mode
|
|
- `prefers-reduced-motion` for animation respect
|
|
- Real content extracted from the mockup (never lorem ipsum)
|
|
|
|
**Never include (AI slop blacklist):**
|
|
- Purple/blue gradients as default
|
|
- Generic 3-column feature grids
|
|
- Center-everything layouts with no visual hierarchy
|
|
- Decorative blobs, waves, or geometric patterns not in the mockup
|
|
- Stock photo placeholder divs
|
|
- "Get Started" / "Learn More" generic CTAs not from the mockup
|
|
- Rounded-corner cards with drop shadows as the default component
|
|
- Emoji as visual elements
|
|
- Generic testimonial sections
|
|
- Cookie-cutter hero sections with left-text right-image
|
|
|
|
### Pretext Wiring Patterns
|
|
|
|
Use these patterns based on the tier selected in Step 2. These are the correct
|
|
Pretext API usage patterns. Follow them exactly.
|
|
|
|
**Pattern 1: Basic height computation (Simple layout, Card/grid)**
|
|
```js
|
|
import { prepare, layout } from './pretext-inline.js'
|
|
// Or if inlined: const { prepare, layout } = window.Pretext
|
|
|
|
// 1. PREPARE — one-time, after fonts load
|
|
await document.fonts.ready
|
|
const elements = document.querySelectorAll('[data-pretext]')
|
|
const prepared = new Map()
|
|
|
|
for (const el of elements) {
|
|
const text = el.textContent
|
|
const font = getComputedStyle(el).font
|
|
prepared.set(el, prepare(text, font))
|
|
}
|
|
|
|
// 2. LAYOUT — cheap, call on every resize
|
|
function relayout() {
|
|
for (const [el, handle] of prepared) {
|
|
const { height } = layout(handle, el.clientWidth, parseFloat(getComputedStyle(el).lineHeight))
|
|
el.style.height = `${height}px`
|
|
}
|
|
}
|
|
|
|
// 3. RESIZE-AWARE
|
|
new ResizeObserver(() => relayout()).observe(document.body)
|
|
relayout()
|
|
|
|
// 4. CONTENT-EDITABLE — re-prepare when text changes
|
|
for (const el of elements) {
|
|
if (el.contentEditable === 'true') {
|
|
new MutationObserver(() => {
|
|
const font = getComputedStyle(el).font
|
|
prepared.set(el, prepare(el.textContent, font))
|
|
relayout()
|
|
}).observe(el, { characterData: true, subtree: true, childList: true })
|
|
}
|
|
}
|
|
```
|
|
|
|
**Pattern 2: Shrinkwrap / tight-fit containers (Chat bubbles)**
|
|
```js
|
|
import { prepareWithSegments, walkLineRanges } from './pretext-inline.js'
|
|
|
|
// Find the tightest width that produces the same line count
|
|
function shrinkwrap(text, font, maxWidth, lineHeight) {
|
|
const segs = prepareWithSegments(text, font)
|
|
let bestWidth = maxWidth
|
|
walkLineRanges(segs, maxWidth, (lineCount, startIdx, endIdx) => {
|
|
// walkLineRanges calls back with progressively narrower widths
|
|
// The first call gives us the line count at maxWidth
|
|
// We want the narrowest width that still produces this line count
|
|
})
|
|
// Binary search for tightest width with same line count
|
|
const { lineCount: targetLines } = layout(prepare(text, font), maxWidth, lineHeight)
|
|
let lo = 0, hi = maxWidth
|
|
while (hi - lo > 1) {
|
|
const mid = (lo + hi) / 2
|
|
const { lineCount } = layout(prepare(text, font), mid, lineHeight)
|
|
if (lineCount === targetLines) hi = mid
|
|
else lo = mid
|
|
}
|
|
return hi
|
|
}
|
|
```
|
|
|
|
**Pattern 3: Text around obstacles (Editorial layout)**
|
|
```js
|
|
import { prepareWithSegments, layoutNextLine } from './pretext-inline.js'
|
|
|
|
function layoutAroundObstacles(text, font, containerWidth, lineHeight, obstacles) {
|
|
const segs = prepareWithSegments(text, font)
|
|
let state = null
|
|
let y = 0
|
|
const lines = []
|
|
|
|
while (true) {
|
|
// Calculate available width at current y position, accounting for obstacles
|
|
let availWidth = containerWidth
|
|
for (const obs of obstacles) {
|
|
if (y >= obs.top && y < obs.top + obs.height) {
|
|
availWidth -= obs.width
|
|
}
|
|
}
|
|
|
|
const result = layoutNextLine(segs, state, availWidth, lineHeight)
|
|
if (!result) break
|
|
|
|
lines.push({ text: result.text, width: result.width, x: 0, y })
|
|
state = result.state
|
|
y += lineHeight
|
|
}
|
|
|
|
return { lines, totalHeight: y }
|
|
}
|
|
```
|
|
|
|
**Pattern 4: Full line-by-line rendering (Complex editorial)**
|
|
```js
|
|
import { prepareWithSegments, layoutWithLines } from './pretext-inline.js'
|
|
|
|
const segs = prepareWithSegments(text, font)
|
|
const { lines, height } = layoutWithLines(segs, containerWidth, lineHeight)
|
|
|
|
// lines = [{ text, width, x, y }, ...]
|
|
// Use for Canvas/SVG rendering or custom DOM positioning
|
|
for (const line of lines) {
|
|
const span = document.createElement('span')
|
|
span.textContent = line.text
|
|
span.style.position = 'absolute'
|
|
span.style.left = `${line.x}px`
|
|
span.style.top = `${line.y}px`
|
|
container.appendChild(span)
|
|
}
|
|
```
|
|
|
|
### Pretext API Reference
|
|
|
|
```
|
|
PRETEXT API CHEATSHEET:
|
|
|
|
prepare(text, font) → handle
|
|
One-time text measurement. Call after document.fonts.ready.
|
|
Font: CSS shorthand like '16px Inter' or 'bold 24px Georgia'.
|
|
|
|
layout(prepared, maxWidth, lineHeight) → { height, lineCount }
|
|
Fast layout computation. Call on every resize. Sub-millisecond.
|
|
|
|
prepareWithSegments(text, font) → handle
|
|
Like prepare() but enables line-level APIs below.
|
|
|
|
layoutWithLines(segs, maxWidth, lineHeight) → { lines: [{text, width, x, y}...], height }
|
|
Full line-by-line breakdown. For Canvas/SVG rendering.
|
|
|
|
walkLineRanges(segs, maxWidth, onLine) → void
|
|
Calls onLine(lineCount, startIdx, endIdx) for each possible layout.
|
|
Find minimum width for N lines. For tight-fit containers.
|
|
|
|
layoutNextLine(segs, state, maxWidth, lineHeight) → { text, width, state } | null
|
|
Iterator. Different maxWidth per line = text around obstacles.
|
|
Pass null as initial state. Returns null when text is exhausted.
|
|
|
|
clearCache() → void
|
|
Clears internal measurement caches. Use when cycling many fonts.
|
|
|
|
setLocale(locale?) → void
|
|
Retargets word segmenter for future prepare() calls.
|
|
```
|
|
|
|
---
|
|
|
|
## Step 3.5: Live Reload Server
|
|
|
|
After writing the HTML file, start a simple HTTP server for live preview:
|
|
|
|
```bash
|
|
# Start a simple HTTP server in the output directory
|
|
_OUTPUT_DIR=$(dirname <path-to-finalized.html>)
|
|
cd "$_OUTPUT_DIR"
|
|
python3 -m http.server 0 --bind 127.0.0.1 &
|
|
_SERVER_PID=$!
|
|
_PORT=$(lsof -i -P -n | grep "$_SERVER_PID" | grep LISTEN | awk '{print $9}' | cut -d: -f2 | head -1)
|
|
echo "SERVER: http://localhost:$_PORT/finalized.html"
|
|
echo "PID: $_SERVER_PID"
|
|
```
|
|
|
|
If python3 is not available, fall back to:
|
|
```bash
|
|
open <path-to-finalized.html>
|
|
```
|
|
|
|
Tell the user: "Live preview running at http://localhost:$_PORT/finalized.html.
|
|
After each edit, just refresh the browser (Cmd+R) to see changes."
|
|
|
|
When the refinement loop ends (Step 4 exits), kill the server:
|
|
```bash
|
|
kill $_SERVER_PID 2>/dev/null || true
|
|
```
|
|
|
|
---
|
|
|
|
## Step 4: Preview + Refinement Loop
|
|
|
|
### Verification Screenshots
|
|
|
|
If `$B` is available (browse binary), take verification screenshots at 3 viewports:
|
|
|
|
```bash
|
|
$B goto "file://<path-to-finalized.html>"
|
|
$B screenshot /tmp/gstack-verify-mobile.png --width 375
|
|
$B screenshot /tmp/gstack-verify-tablet.png --width 768
|
|
$B screenshot /tmp/gstack-verify-desktop.png --width 1440
|
|
```
|
|
|
|
Show all three screenshots inline using the Read tool. Check for:
|
|
- Text overflow (text cut off or extending beyond containers)
|
|
- Layout collapse (elements overlapping or missing)
|
|
- Responsive breakage (content not adapting to viewport)
|
|
|
|
If issues are found, note them and fix before presenting to the user.
|
|
|
|
If `$B` is not available, skip verification and note:
|
|
"Browse binary not available. Skipping automated viewport verification."
|
|
|
|
### Refinement Loop
|
|
|
|
```
|
|
LOOP:
|
|
1. If server is running, tell user to open http://localhost:PORT/finalized.html
|
|
Otherwise: open <path>/finalized.html
|
|
|
|
2. If an approved mockup PNG exists, show it inline (Read tool) for visual comparison.
|
|
If in plan-driven or freeform mode, skip this step.
|
|
|
|
3. AskUserQuestion (adjust wording based on mode):
|
|
With mockup: "The HTML is live in your browser. Here's the approved mockup for comparison.
|
|
Try: resize the window (text should reflow dynamically),
|
|
click any text (it's editable, layout recomputes instantly).
|
|
What needs to change? Say 'done' when satisfied."
|
|
Without mockup: "The HTML is live in your browser. Try: resize the window
|
|
(text should reflow dynamically), click any text (it's editable, layout
|
|
recomputes instantly). What needs to change? Say 'done' when satisfied."
|
|
|
|
4. If "done" / "ship it" / "looks good" / "perfect" → exit loop, go to Step 5
|
|
|
|
5. Apply feedback using targeted Edit tool changes on the HTML file
|
|
(do NOT regenerate the entire file — surgical edits only)
|
|
|
|
6. Brief summary of what changed (2-3 lines max)
|
|
|
|
7. If verification screenshots are available, re-take them to confirm the fix
|
|
|
|
8. Go to LOOP
|
|
```
|
|
|
|
Maximum 10 iterations. If the user hasn't said "done" after 10, use AskUserQuestion:
|
|
"We've done 10 rounds of refinement. Want to continue iterating or call it done?"
|
|
|
|
---
|
|
|
|
## Step 5: Save & Next Steps
|
|
|
|
### Design Token Extraction
|
|
|
|
If no `DESIGN.md` exists in the repo root, offer to create one from the generated HTML:
|
|
|
|
Extract from the HTML:
|
|
- CSS custom properties (colors, spacing, font sizes)
|
|
- Font families and weights used
|
|
- Color palette (primary, secondary, accent, neutral)
|
|
- Spacing scale
|
|
- Border radius values
|
|
- Shadow values
|
|
|
|
Use AskUserQuestion:
|
|
> No DESIGN.md found. I can extract the design tokens from the HTML we just built
|
|
> and create a DESIGN.md for your project. This means future /design-shotgun and
|
|
> /design-html runs will be style-consistent automatically.
|
|
> A) Create DESIGN.md from these tokens
|
|
> B) Skip — I'll handle the design system later
|
|
|
|
If A: write `DESIGN.md` to the repo root with the extracted tokens.
|
|
|
|
### Save Metadata
|
|
|
|
Write `finalized.json` alongside the HTML:
|
|
```json
|
|
{
|
|
"source_mockup": "<approved variant PNG path or null>",
|
|
"source_plan": "<CEO plan path or null>",
|
|
"mode": "<approved-mockup|plan-driven|freeform|evolve>",
|
|
"html_file": "<path to finalized.html or component file>",
|
|
"pretext_tier": "<selected tier>",
|
|
"framework": "<vanilla|react|svelte|vue>",
|
|
"iterations": <number of refinement iterations>,
|
|
"date": "<ISO 8601>",
|
|
"screen": "<screen name>",
|
|
"branch": "<current branch>"
|
|
}
|
|
```
|
|
|
|
### Next Steps
|
|
|
|
Use AskUserQuestion:
|
|
> Design finalized with Pretext-native layout. What's next?
|
|
> A) Copy to project — copy the HTML/component into your codebase
|
|
> B) Iterate more — keep refining
|
|
> C) Done — I'll use this as a reference
|
|
|
|
---
|
|
|
|
## Important Rules
|
|
|
|
- **Source of truth fidelity over code elegance.** When an approved mockup exists,
|
|
pixel-match it. If that requires `width: 312px` instead of a CSS grid class, that's
|
|
correct. When in plan-driven or freeform mode, the user's feedback during the
|
|
refinement loop is the source of truth. Code cleanup happens later during
|
|
component extraction.
|
|
|
|
- **Always use Pretext for text layout.** Even if the design looks simple, Pretext
|
|
ensures correct height computation on resize. The overhead is 30KB. Every page benefits.
|
|
|
|
- **Surgical edits in the refinement loop.** Use the Edit tool to make targeted changes,
|
|
not the Write tool to regenerate the entire file. The user may have made manual edits
|
|
via contenteditable that should be preserved.
|
|
|
|
- **Real content only.** When a mockup exists, extract text from it. In plan-driven mode,
|
|
use content from the plan. In freeform mode, generate realistic content based on the
|
|
user's description. Never use "Lorem ipsum", "Your text here", or placeholder content.
|
|
|
|
- **One page per invocation.** For multi-page designs, run /design-html once per page.
|
|
Each run produces one HTML file.
|