skills/meta/html-artifact/SKILL.md
Generate rich self-contained HTML artifacts instead of markdown. Auto-detects artifact shape (spec, code-review, prototype, report, editor, data-viz, diagram, deck) and loads shape-specific patterns. Bundles Birchline design system with 4 theme presets. Use for "make HTML", "as HTML", "HTML artifact", or auto-injected by router when output benefits from rich visualization.
npx skillsauth add notque/claude-code-toolkit html-artifactInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Generate single self-contained .html files that replace markdown when the output needs color, interactivity, layout, or visualization. Auto-detect artifact shape from the request, load shape-specific patterns, generate, validate, deliver.
Core constraint: Every artifact is ONE .html file. All CSS in <style>, all JS in <script>. No CDN links, no frameworks, no build steps, no external dependencies. Works offline, opens in any browser.
5-phase pipeline: DETECT SHAPE, LOAD CONTEXT, GENERATE, VALIDATE, DELIVER. Phase 1 classifies the request into one of 8 shapes via deterministic script. Phase 2 loads the Birchline design system plus shape-specific reference. Phase 3 dispatches a subagent to generate the HTML. Phase 4 validates structure. Phase 5 delivers the file path and offers browser preview. Phase 6 EXPORT (optional) renders to PDF when the user asks for one.
Classify the user's request into one of 8 artifact shapes.
Run: python3 skills/meta/html-artifact/scripts/detect-shape.py --request "{user_request}"
The script outputs a shape name and confidence score.
| Shape | Trigger Signals | What It Produces | |---|---|---| | spec | plan, explore options, compare N approaches, brainstorm | Side-by-side grids, Pro/Con badges, SVG data-flow diagrams, risk tables | | code-review | review PR, explain diff, annotate code, understand module | Diff rendering, severity colors, margin annotations, jump links | | prototype | prototype, animation, tune, try options, component variants | Sliders, CSS var live update, animation sandbox, contact sheets | | report | report, summarize, status update, explain how X works, incident | TL;DR box, collapsible sections, timeline, metric callouts, SVG diagrams | | editor | reorder, triage, edit config, tune prompt, pick values | Drag-drop, kanban, toggle switches, split-pane, export buttons | | data-viz | visualize, chart, dashboard, show data, trends | SVG charts, canvas, interactive tooltips, filter controls | | diagram | diagram, flowchart, architecture, sequence, SVG, illustrate, figure | Inline SVG diagrams, annotated flowcharts, figure sheets, interactive node details | | deck | slides, presentation, deck, talk, pitch | Arrow-key navigable slide deck, 16:9 aspect ratio, slide types, progress bar |
Gate: Shape detected with medium+ confidence. -- because low-confidence classification produces artifacts that mix concerns and satisfy no shape well. Fallback to "report" (safest general-purpose shape) if confidence is low or ambiguous.
Real content often combines two shapes — a report with embedded diagrams, a spec with data-viz charts. When detect-shape.py returns a primary shape with medium/high confidence but the request also contains signals for a secondary shape, use the hybrid pattern:
| Primary Shape | + Secondary | Result | |---|---|---| | report | + diagram | Report layout (TL;DR, collapsibles, TOC) with inline SVG diagrams between sections | | report | + data-viz | Report layout with embedded SVG charts illustrating key metrics | | spec | + diagram | Comparison grid with SVG flow diagrams showing each option's architecture | | spec | + data-viz | Comparison grid with charts showing performance/cost per option | | diagram | + report | Figure sheet with explanatory text sections between diagram groups |
Detection: After running detect-shape.py, check if the secondary_shape field is non-null. If so, load BOTH shape references in Phase 2.
Generation rule: Primary shape controls page layout (outer structure). Secondary shape provides embedded components (inner elements). The html-builder agent receives both shape patterns and uses primary for structure, secondary for visual elements within sections.
Example: "create a visual companion for my pipelines article with diagrams and explanations" → primary: report (explain, article), secondary: diagram (visual, diagrams). Load shape-report-research.md AND shape-diagram-illustration.md.
Two parallel steps: (A) run the template assembler to produce a pre-filled HTML skeleton, and (B) load principle-focused reference files for the builder agent.
Step A -- Assemble template (deterministic):
Run: python3 skills/meta/html-artifact/scripts/assemble-template.py --shape {shape} --title "{title}" --components {components}
The script reads CSS/JS from templates/ and injects:
templates/base-reset.css)templates/themes/{theme}.css)templates/shapes/{shape}.css)templates/components/{name}.{css,js})The assembler also emits a self-describing stamp as the FIRST CSS comment, so a later run can re-audit the build statelessly (recover shape/theme from output):
/* vexjoy-artifact: shape=<shape> theme=<name> contrast=<pass|fail|n/a> */
shape and theme come from this build's decisions. contrast=n/a at assembly time because the assembler runs no WCAG check; the stamp is a claim, not proof — Phase 4's slop scan verifies the rendered CSS independently rather than trusting it.
Select components based on shape needs:
| Shape | Typical Components |
|---|---|
| spec | tabs,copy-button,theme-toggle |
| code-review | collapsible,filter,keyboard-nav,theme-toggle |
| prototype | slider,copy-button,theme-toggle |
| report | collapsible,theme-toggle,copy-button |
| editor | drag-drop,filter,copy-button |
| data-viz | filter,theme-toggle |
| diagram | copy-button,theme-toggle |
| deck | keyboard-nav,theme-toggle |
Step B -- Load reference files (principles + guidance):
Always load:
references/design-system.md -- Theme selection, token architecture, accessibility checklist, SVG conventions, common mistakesreferences/interaction-patterns.md -- Component descriptions, when-to-use guidance, accessibility rules, composition guideLoad per detected shape:
| Shape | Reference File | Key Content |
|---|---|---|
| spec | references/shape-spec-exploration.md | Layout descriptions, composition guide, common mistakes |
| code-review | references/shape-code-review.md | Severity system, interaction patterns, section ordering |
| prototype | references/shape-design-prototype.md | Control types, export requirements, layout patterns |
| report | references/shape-report-research.md | Section ordering, TL;DR placement, metric patterns |
| editor | references/shape-custom-editor.md | Editor types, export bar rules, common mistakes |
| data-viz | references/shape-data-visualization.md | Chart types, coordinate system, color scales |
| diagram | references/shape-diagram-illustration.md | SVG construction rules, diagram types, interaction patterns |
| deck | references/shape-slide-deck.md | Slide types, navigation, print styles |
Gate: Template assembled + required references loaded. -- because the template provides deterministic CSS/JS injection, and references provide the judgment guidance the builder needs.
Dispatch the html-builder subagent with the pre-assembled template.
agents/html-builder.md for the subagent prompt.html file to the project directory (or /tmp/html-artifacts/ if no project context)Self-contained file constraints (inline here because they govern generation):
| Constraint | Reason |
|---|---|
| All CSS in <style> tag | No external stylesheets -- file must work offline |
| All JS in <script> tag | No CDN imports -- no React, Vue, Tailwind CDN, Bootstrap CDN |
| Vanilla JS only | Single file, no build step, no transpilation |
| Must include <title> | Browser tab identification, validation requirement |
| Must include <meta charset="utf-8"> | Consistent rendering across platforms |
| Must include <meta name="viewport"> | Responsive on mobile/tablet |
| Semantic HTML sections | <header>, <main>, <section>, <footer> for structure |
| SVG inline, not <img src> | No external file references |
| Max 500KB file size | Keeps generation time reasonable, prevents bloated inline assets |
Constraint: No framework boilerplate. -- because React/Vue/Svelte require build steps and external imports that violate the single-file self-contained requirement. Vanilla JS handles all 8 shapes adequately.
Constraint: Generate HTML directly, never generate markdown then convert. -- because markdown-to-HTML conversion loses the shape-specific layout, interactivity, and visual structure that justifies using HTML in the first place.
Gate: .html file exists on disk.
-- because Phase 4 validation reads the file; a missing file means generation failed silently.
Run deterministic validation on the generated file.
Run: python3 skills/meta/html-artifact/scripts/validate-artifact.py {html_file_path}
The script checks:
| Check | Fails When |
|---|---|
| Valid HTML structure | Missing <html>, <head>, or <body> |
| No external dependencies | Any src= or href= pointing to external URLs |
| Has <title> | Missing or empty <title> tag |
| Has charset meta | Missing <meta charset> |
| Has viewport meta | Missing viewport meta tag |
| File size under 500KB | Excessive inline assets or animation keyframes |
| No broken internal refs | href="#id" pointing to nonexistent id attributes |
| Rendered-CSS slop scan | css_slop_rules.scan_css over the file content (warnings only, non-blocking) |
The slop scan (vendored css_slop_rules.py) flags 7 rendered-CSS patterns — transition-all, universal-hover-scale, gradient-text-headline, focus-ring-fade, emoji-feature-icon, two-line-cta, contrast-canary. It is shape-agnostic (checks CSS/markup, not page structure), so it applies to all 8 shapes including hero-less ones. Findings surface as warnings and do not fail the build yet; promote a rule to error in css_slop_rules.py to make it blocking.
Gate: All validation checks pass. -- because an HTML file with external dependencies fails offline, missing meta tags render inconsistently across browsers, and missing structure breaks accessibility.
If validation fails: Read the specific failures from script output, fix the identified issues in the HTML file, re-run validation. Maximum 3 fix attempts before showing the user the remaining issues and asking for guidance.
open {file} (macOS) or xdg-open {file} (Linux)Constraint: Detect headless/SSH environments before offering browser open.
-- because xdg-open fails without a display server, producing confusing errors. Check $DISPLAY on Linux or $SSH_TTY presence. If headless, print path only and skip the open offer.
Render the generated HTML to PDF. Opt-in only — HTML stays the default deliverable.
Fires when the user message contains any of: "PDF", "export PDF", "make a PDF", "as PDF", "send as PDF", "save as PDF", "PDF version", "PDF export". Without one of those signals, this phase stays dormant.
Runs:
python3 skills/meta/html-artifact/scripts/to-pdf.py \
--input <generated.html> \
--output <generated.pdf> \
--json
The script auto-detects shape from <body data-shape="..."> (the assembler adds it in Phase 2). Page size, orientation, and margins come from a per-shape map: deck renders 13.333in × 7.5in landscape with no margin; spec, code-review, prototype, data-viz, and diagram render Letter landscape; report and editor render Letter portrait. Falls back to Letter portrait when shape is unknown.
Delivers both file paths to the user. The JSON output reports {"output", "page_count", "shape", "bytes"} — page_count reflects slide count for decks, 0 otherwise.
If Playwright is unavailable, exit code 2 surfaces install instructions: pip install -e ".[pdf]" && playwright install chromium. Pass that hint along to the user instead of failing silently.
See references/pdf-export.md for the full page-size table, troubleshooting (font fallback, image timing, install failures), and per-shape print stylesheet inventory.
Render the generated HTML deck to an editable Microsoft PowerPoint .pptx. Opt-in only — HTML stays the default deliverable. Mirrors Phase 6 EXPORT-PDF in shape: signal-triggered, deterministic, runs after the HTML is built.
Fires when the user message contains any of: "pptx", ".pptx", "powerpoint", "editable deck", "editable", "as pptx", "export pptx", "hand-off", "corporate template". Without one of those signals, this phase stays dormant.
Only valid when the detected shape is deck. Other shapes (report, spec, diagram, etc.) cannot be exported to PPTX — fall back to Phase 6 PDF if the user asks.
Runs:
python3 skills/meta/html-artifact/scripts/pptx-bridge/run-unified.py \
--input <generated.html> \
--format pptx \
--out <generated.pptx> \
--no-render
The bridge re-authors slides natively via python-pptx because no general HTML→PPTX converter preserves CSS-rich layout. Output is 13.333 × 7.5 in (16:9), dark navy theme, Aptos body / Cascadia Code mono. Each <section class="slide"> becomes one editable slide; up to 12 layout types (title, content, metric_grid, layer_rows, pipeline, code_block, compare_table_2col/3col, outcome_grid, split_narrow, closing) map 1:1 to native python-pptx builders.
--out accepts either a .pptx file path (single-file mode) or a directory (writes the .pptx plus slides.json, report.md, optional render/ siblings). --no-render skips the optional LibreOffice QA step; required on hosts without soffice.
If python-pptx is unavailable, exit code 1 surfaces install instructions: pip install python-pptx. Pass that hint along to the user instead of failing silently.
See references/pptx-export.md for the full layout table, THEME dict, CLI reference, validation criteria, and failure modes.
| Error | Cause | Solution |
|---|---|---|
| detect-shape.py returns low confidence | Ambiguous request mapping to multiple shapes | Fall back to "report" shape -- safest general-purpose format |
| Generated HTML has external dependencies | Builder included CDN links or external src refs | Regenerate with explicit constraint: "no external deps, all CSS/JS inline" |
| File exceeds 500KB | Excessive inline SVGs or animation keyframes | Simplify SVG paths, reduce keyframe count, compress data |
| Browser won't open | No display server (headless, SSH, WSL without WSLg) | Print path only, suggest scp or a local preview: python3 -m http.server --bind 127.0.0.1. For public access use nginx, not http.server; see the public-web-deploy skill. |
| Validation fails repeatedly (3+ attempts) | Structural issue the builder cannot self-correct | Show validation output to user, ask for guidance |
| Shape misclassified | Auto-detection picked wrong shape for request | User overrides with /html --shape=<name> <request> |
What it looks like: <link href="https://cdn.jsdelivr.net/..."> or <script src="https://unpkg.com/react@18/..."> in the generated HTML.
Why wrong: Breaks the self-contained contract. File fails offline, introduces version drift, adds weight the user didn't ask for.
Do instead: Inline all CSS in <style>. Write vanilla JS in <script>. The Birchline design system in references/design-system.md provides the full token set.
What it looks like: Generating a markdown document first, then running it through a converter or wrapping it in <pre> tags.
Why wrong: Loses shape-specific layout, interactivity, SVG diagrams, and responsive grid structures. Produces "markdown in a browser" instead of a native HTML artifact.
Do instead: Generate HTML directly using shape-specific patterns from references. The HTML structure IS the output format, not a rendering layer on top of text.
What it looks like: One giant <div> with inline styles on every element, no semantic structure, no comments.
Why wrong: Unreadable source, hard to debug, impossible for the user to modify. Accessibility tools cannot navigate it.
Do instead: Use semantic HTML (<header>, <main>, <section>, <footer>). Define CSS classes in <style>. Add section comments. Group related elements logically.
What it looks like: Generating a full interactive dashboard when the user asked for a simple comparison table.
Why wrong: 2-4x generation time for features the user didn't request. Complexity without value.
Do instead: Match artifact complexity to request complexity. A comparison of 3 options needs a grid with cards, not a filterable dashboard with animations.
| Rationalization | Why Wrong | Required Action | |---|---|---| | "Markdown is fine for this" | If shape detection triggered, the request has visual/interactive needs markdown can't serve | Generate HTML; user opts out with "as markdown" | | "I'll add Tailwind CDN for faster styling" | Breaks self-contained requirement, fails offline | Use Birchline tokens from design-system.md | | "The HTML looks right, skip validation" | Visual inspection misses missing meta tags, broken internal links, external deps | Run validate-artifact.py every time | | "Report shape works for everything" | Each shape has distinct layout and interaction patterns; report is a fallback, not a default | Use the detected shape; report only when confidence is genuinely low |
| Signal | Load These Files | Why |
|---|---|---|
| Any html-artifact invocation | references/design-system.md | Theme selection, token architecture, accessibility, common mistakes |
| Any html-artifact invocation | references/interaction-patterns.md | Component descriptions, when-to-use, accessibility rules |
| Shape = spec | references/shape-spec-exploration.md | Layout descriptions, composition guide, common mistakes |
| Shape = code-review | references/shape-code-review.md | Severity system, interaction patterns, section ordering |
| Shape = prototype | references/shape-design-prototype.md | Control types, export requirements, layout patterns |
| Shape = report | references/shape-report-research.md | Section ordering, TL;DR placement, metric patterns |
| Shape = editor | references/shape-custom-editor.md | Editor types, export bar rules, common mistakes |
| Shape = data-viz | references/shape-data-visualization.md | Chart types, coordinate system, color scales |
| Shape = diagram | references/shape-diagram-illustration.md | SVG construction rules, diagram types, interaction patterns |
| Shape = deck | references/shape-slide-deck.md | Slide types, navigation, print styles |
| Request mentions scroll, reveal, animate on scroll, progressive | references/scrollytelling-patterns.md | IntersectionObserver scroll animations, stagger, counters, progress bar |
| Request mentions PDF, export PDF, as PDF, PDF version | references/pdf-export.md | Phase 6 trigger conditions, page-size table, troubleshooting, install instructions |
| Request mentions pptx, .pptx, powerpoint, editable deck, hand-off, corporate template | references/pptx-export.md | Phase 7 trigger conditions, layout types, THEME dict, CLI reference, failure modes |
| Shape = diagram OR request contains "SVG", "architecture diagram", "flowchart", "sequence diagram" | references/diagram-layering.md | SVG layer order, masking rect technique, semantic color system for dark-theme diagrams |
| Shape = data-viz OR request contains "infographic", "layout", "visualize data", "chart type" | references/infographic-layouts.md | 21 layout types with content-type pairings and 22 visual styles |
This skill uses:
references/design-system.md: Theme selection, token architecture, accessibility checklist, SVG conventions, common mistakesreferences/interaction-patterns.md: Component descriptions, when-to-use guidance, accessibility rules, composition guidereferences/shape-spec-exploration.md: Spec shape -- layout, composition guide, common mistakesreferences/shape-code-review.md: Code review shape -- severity system, interaction patterns, section orderingreferences/shape-design-prototype.md: Prototype shape -- control types, export requirements, layout patternsreferences/shape-report-research.md: Report shape -- section ordering, TL;DR placement, metric patternsreferences/shape-custom-editor.md: Editor shape -- editor types, export bar rules, common mistakesreferences/shape-data-visualization.md: Data viz shape -- chart types, coordinate system, color scalesreferences/shape-diagram-illustration.md: Diagram shape -- SVG construction rules, diagram types, interaction patternsreferences/shape-slide-deck.md: Deck shape -- slide types, navigation, print stylesagents/html-builder.md: Subagent prompt for HTML generationreferences/scrollytelling-patterns.md: IntersectionObserver scroll animation patternsreferences/pdf-export.md: Phase 6 EXPORT — trigger conditions, page-size table, print stylesheet inventory, troubleshootingreferences/pptx-export.md: Phase 7 EXPORT-PPTX — trigger conditions, layout types, THEME dict, CLI reference, failure modesreferences/diagram-layering.md: SVG layer order, masking rect technique, dark design system constants, semantic color palettereferences/infographic-layouts.md: 21 layout types with structure and use guidance, 22 visual styles, content-type pairingsscripts/detect-shape.py: Deterministic shape classification from user requestscripts/assemble-template.py: Template assembly with theme, shape, and component CSS/JS injectionscripts/validate-artifact.py: HTML structure, self-containment, and rendered-CSS slop validationscripts/css_slop_rules.py: vendored slop scanner (scan_css) — 7 rendered-CSS rules, dependency-free. Keep in sync with distinctive-frontend-design.scripts/to-pdf.py: Playwright-based PDF rendering with per-shape page sizingscripts/pptx-bridge/: HTML deck → editable PPTX (extract_slides.py, _pptx_engine.py, render_pptx.py, run-unified.py)templates/: CSS/JS template files organized by themes/, shapes/, components/, print/documentation
Document translation: quick/normal/refined modes with chunked parallel subagents and glossary support.
development
AI image generation: Gemini and Nano Banana backends; single/series/batch workflows with prompt-to-disk.
testing
Unified voice content generation pipeline with mandatory validation and joy-check. 13-phase pipeline: LOAD, GROUND, STATS-CHECKPOINT, GENERATE, HOOK-GATE, VALIDATE, REFINE, VARIETY-GATE, JOY-CHECK, ANTI-AI, CLOSE-GATE, OUTPUT, CLEANUP. Use when writing articles, blog posts, or any content that uses a voice profile. Use for "write article", "blog post", "write in voice", "generate content", "draft article", "write about".
documentation
Critique-and-rewrite loop for voice fidelity validation.