lavish-axi
For when a rich editor is not rich enough.
HTML is the new markdown. Lavish is the new editor for your HTML artifacts.
Agents are good at producing rich HTML artifacts, but the human-agent collaboration loop on such artifacts is lacking and falls back into screenshots and long responses for “tell me what to change.” That loses the thing HTML is best at: interactivity.
Lavish Editor opens agent-generated HTML files in a local browser, lets you pinpoint elements or selected text and send feedback to the agent to address.
- Local-first - Review local HTML artifacts with a local CLI and no cloud dependency in the core feedback loop; hosted sharing through third-party ht-ml.app is explicit and opt-in.
- Human-AI collaboration - Annotate elements, selected text ranges, and send messages to the agent without leaving Lavish Editor.
- Battery included - Lavish Editor teaches your agent good visualization for common use cases such as product or technical plans, design explorations and more out of the box.
Lavish Editor is an AXI, which means -
- It's just a CLI any capable agent can run without setup.
- It's optimized for agent ergonomics. TOON output, long polling, and contextual disclosure making it highly token efficient.
- The skill and hooks below only handle discovery; agents learn to use the AXI by using it.
Quick Start
Install the Lavish skill in the Agent Skills format with npx skills:
npx skills add kunchenguid/lavish-axi --skill lavish
That is the entire setup - no npm install needed.
The skill teaches your agent to run Lavish through npx -y lavish-axi, so the CLI comes along on demand.
Its frontmatter also includes Hermes Agent metadata, so Hermes-compatible harnesses can categorize and surface it as a first-class productivity skill.
Then, in agents that expose skills as slash commands (Claude Code, for example), invoke it directly:
/lavish let's discuss our plan here
Or just ask for anything that is easier to grasp visually - a plan, comparison, diagram, table, code view, or report - and the agent loads the skill on its own when it recognizes the task.
By default the skill lands in the current project's skills directory (.claude/skills/, for example); add -g to install it for all projects (~/.claude/skills/).
Other Ways to Use Lavish
The skill is the recommended path, but it is not the only one.
Zero setup
Lavish is an AXI, so any capable agent can run the CLI directly with nothing installed at all. Just tell your agent:
Use `npx lavish-axi` to write a product or technical plan for what we discussed.
Session hook
Want Lavish's ambient context - including your live open sessions - fed into every agent session instead of loading on demand? Install the CLI globally and opt into the hook:
npm install -g lavish-axi
lavish-axi setup hooks
This installs a SessionStart hook for Claude Code, Codex, OpenCode, and GitHub Copilot CLI that surfaces open sessions, visualization playbooks, and usage guidance at the start of each session.
Unlike the skill, the hook also shows your live open sessions, so a fresh agent session can resume an in-flight review.
Restart your agent session after running this so the new hook takes effect.
From source
git clone https://github.com/kunchenguid/lavish-axi.git
cd lavish-axi
pnpm install --frozen-lockfile
pnpm run build
pnpm link
How It Works
┌───────────────┐
│ Agent writes │
│ artifact.html │
└───────┬───────┘
▼
┌────────────────────────┐
│ lavish-axi <file_path> │
│ opens local browser UI │
└───────┬────────────────┘
▼
┌────────────────────────┐
│ Human annotates text │
│ or elements, sends │
│ chat, or browser audit │
│ reports layout issues │
└───────┬────────────────┘
▼
┌────────────────────────┐
│ lavish-axi poll waits │
│ and returns prompts │
│ or layout warnings │
└────────────────────────┘
- File-path identity - Sessions are keyed by the canonical HTML file path, so agents do not need opaque IDs.
- Portable artifacts - The artifact runs in an iframe while Lavish injects a small SDK for annotations, snapshots, feedback controls, and render-time layout checks.
Lavish does not inject any design system, so the saved HTML file renders identically whether you open it through
lavish-axior directly in a browser. Before writing HTML, choose a design system in strict priority order: follow a user-requested look first; otherwise inspect the project the artifact is about - the subject or product whose content or UI it represents, which may differ from your current working directory - and match that project's Tailwind or theme config, CSS variables or design tokens, component library, brand assets, or existing styled pages. If the artifact previews, proposes, or mocks a specific app's UI, render it in that app's own design system so it faithfully shows the product, even when you are running in a different repo. Only when both come up empty, runlavish-axi designfor a copy-pasteable Tailwind CSS v4 + DaisyUI v5 CDN fallback, a content-to-playbook router, and Mermaid diagram tooling. That fallback guidance recommends DaisyUI'sluxurytheme by default, warns not to@applyDaisyUI classes inside Tailwind browser-runtime style blocks, includes an optional layout safety CSS snippet for dense nested grid/flex layouts, and provides a pinned Mermaid CDN snippet with initialization for flows, architecture, state, and sequence diagrams. - Open-time layout gate - The browser chrome masks each artifact until the real in-iframe layout audit reports no error-severity findings.
Warning-only artifacts reveal normally; error findings notify the agent through the same
layout_warningspoll path and keep the curtain up until a clean reload. The user can click Show anyway, and a bounded safety timeout reveals with a persistent layout-issues banner so review is never blocked indefinitely. - Layout warnings - After fonts load and layout settles, the injected SDK audits the real browser render for page horizontal overflow, element overflow, clipped or visibly spilling text, and overlapping text.
Intentional horizontal scrollers using
overflow-x: autoorscrollare excluded from horizontal checks, andoverflow-y: autoorscrollis treated as intentional for vertical overflow. Current findings are returned fromlavish-axi pollaslayout_warningswithselector,kind,overflowPx,viewportWidth,severity, andpersistent. Fresh error-severity findings should be fixed and rechecked before asking the human to review; repeated or warning-only findings can be surfaced to the human with a note when the cause is not obvious. - Local assets - Copy local images, CSS, fonts, and scripts next to the HTML artifact and reference them with relative paths from that directory; root-prefixed paths such as
/assets/logo.pngwill not resolve through Lavish's artifact route. - Export and sharing -
lavish-axi exportwrites<name>.export.htmlby inlining local assets only, stripping the annotation SDK, and leaving remote CDN/font references as links that still need network access.lavish-axi sharepublishes the same local-inlined HTML to ht-ml.app, a third-party hosting service not part of Lavish. Publishing sends the artifact to ht-ml.app's servers, public by default, or private and password-protected with--password; the response includes a secretupdate_keyshown once for later management. Bundling never fetches remote URLs, Lavish itself does not set a CSP, local reads stay confined and size-capped, and absolutefile://paths outside safe inlined asset references are redacted before output. Per-asset and per-bundle inline caps default to 10 MB and 25 MB, overridable withLAVISH_AXI_EXPORT_MAX_ASSET_BYTESandLAVISH_AXI_EXPORT_MAX_BUNDLE_BYTES. Unresolved local assets or export notices such as author-set CSP meta tags and redacted file URLs are surfaced in command or browser output. Use--tokenorLAVISH_AXI_HTML_APP_TOKENfor an optional bearer token; setLAVISH_AXI_HTML_APP_API_URLonly when overriding the ht-ml.app API base. - Live reload - Lavish watches the HTML artifact file by default and preserves the artifact iframe scroll position across reloads. To also reload on sibling asset changes, add
data-lavish-live-reload-rootto the root element or<meta name="lavish-live-reload" content="root">. - Feedback controls - Native controls (radios, checkboxes, inputs, selects, buttons, labels, disclosure summaries, contenteditable) are interactive automatically, so they do not need
data-lavish-action. For reversible choices, let option clicks update local state, then queue exactly one final answer from a per-question submit or Queue answer button withwindow.lavish.queuePrompt(). Mark only custom (non-native) clickable elements withdata-lavish-actionso Lavish does not annotate them, and usedata-lavish-questionorqueueKeywhen pre-send updates for the same question should replace each other. The browser chrome keeps editing actions in the overflow menu (copy path, reload artifact, copy DOM snapshot, export standalone HTML, publish link, end session) and can submit queued prompts with Send & end session, which sends the prompts and user-ended attribution together. - Keyboard shortcuts - In the chrome composer, Enter sends queued prompts and Shift+Enter inserts a newline. In the annotation card, Enter queues the annotation, Shift+Enter inserts a newline, and Ctrl+Enter (Cmd+Enter on macOS) queues it and sends all queued prompts immediately.
- Agent presence - The browser shows when no agent is listening, keeps queued feedback and fresh layout warnings for the next successful
lavish-axi pollsend even across reloads, and only blocks human sends while the agent is working on delivered feedback. The no-timeout poll writes an immediate stderr banner and periodic stderr heartbeats while stdout stays reserved for the final response; if the poll is interrupted or times out, re-run it because queued feedback is never lost. - Session end etiquette - Lavish tracks who ended a session: a human clicking End session (or Send & end session) in the browser is a user-initiated end, while
lavish-axi end <html-file>is agent-initiated. A plainlavish-axi <html-file>after a user-initiated end refuses to reopen the browser and returns guidance instead; pass--reopenonly when the user asks for further review or something important needs their visual attention. Agent-initiated ends keep reopening normally, same as before.lavish-axi poll'sendedresponse and thefeedbackresponse for the final batch before an end both carrynext_stepguidance telling the agent to stop polling and deliver remaining updates in chat instead of reopening. - Precise targets - Text annotations include selected text plus range anchors, so agents are not limited to whole-element selectors.
- Server cleanup - The detached server stops after the last session ends when nothing is connected, or after
LAVISH_AXI_IDLE_TIMEOUT_MS(default 30 minutes) with no browser or poll connections. SetLAVISH_AXI_IDLE_TIMEOUT_MS=0oroffto disable idle self-shutdown. - Local-first state - Session state stays under
~/.lavish-axi/by default, orLAVISH_AXI_STATE_DIRwhen set. - Network binding - The server binds to loopback (
127.0.0.1) by default. SetLAVISH_AXI_HOSTto bind elsewhere; a wildcard (0.0.0.0or::) binds every interface. Binding beyond loopback exposes an unauthenticated server that can read and serve arbitrary local files to anything that can reach it, so only do so on a trusted network. SetLAVISH_AXI_LINK_HOSTto control the hostname written into generated session links (defaults to the bind address, or loopback when bound to a wildcard).
CLI Reference
| Command | Description |
|---|---|
lavish-axi |
Show current sessions and usage guidance. |
lavish-axi update |
Check for or apply the latest npm release through the AXI SDK self-updater. |
lavish-axi <html-file> |
Open or resume a Lavish Editor session, with the open-time layout gate enabled by default. Refuses to reopen a session the user explicitly ended from the browser unless --reopen is passed. |
lavish-axi poll <html-file> |
Long-poll until the user sends feedback, ends the session, or the browser reports fresh layout_warnings; leave no-timeout polls running, or re-run them if interrupted. On status: ended, stop polling and do not reopen uninvited. |
lavish-axi end <html-file> |
End a session as the agent; unlike a user-initiated end from the browser, this still allows a plain reopen later. |
lavish-axi export <html-file> |
Write a portable copy of the artifact: one HTML file with its local assets inlined, so it opens with no server and no sibling files. Remote CDN/font references are left as links. |
lavish-axi share <html-file> |
Publish the artifact (local assets inlined) to ht-ml.app, a third-party host not part of Lavish, and print a visitable URL plus a secret update key; shares are public by default, and --password makes viewers enter the password before viewing. |
lavish-axi stop |
Shut down the background server. |
lavish-axi playbook [id] |
List focused artifact guidance or show one playbook; agents must open each matching playbook before writing HTML. |
lavish-axi design |
Show the Tailwind + DaisyUI CDN fallback, content-to-playbook router, Mermaid diagram tooling, luxury default theme, DaisyUI @apply warning, and layout safety snippet. |
lavish-axi setup hooks |
Install or repair optional SessionStart hooks for Claude Code, Codex, OpenCode, and GitHub Copilot CLI; restart the agent session afterward. |
lavish-axi server |
Run the local Lavish Editor server. |
Known playbook IDs: diagram, table, comparison, plan, code, input, slides.
One artifact often combines several playbooks, such as a plan that includes a comparison and a diagram, so agents must match against each use_when trigger and open every matching playbook before writing HTML.
For flows, architecture, state, or sequence diagrams, open the diagram playbook and use the Mermaid tooling from lavish-axi design unless SVG is needed for richly annotated nodes; avoid hand-built div/flexbox boxes-and-arrows.
Flags
| Command | Flag | Description |
|---|---|---|
lavish-axi <html-file> |
--no-open |
Ensure the server/session exists without opening another browser window. |
lavish-axi <html-file> |
--no-gate |
Skip the open-time layout curtain for this browser open. |
lavish-axi <html-file> |
--reopen |
Reopen a session the user explicitly ended from the browser; without it, a plain open refuses and explains why instead of reopening uninvited. |
lavish-axi update |
--check |
Report current vs latest npm version without installing an update. |
lavish-axi export |
--out <path> |
Write the export to a specific path instead of <name>.export.html next to the source. |
lavish-axi share |
--password <pw> |
Make the third-party ht-ml.app page private; viewers must supply the password. |
lavish-axi share |
--token <t> |
Attach an optional bearer token (LAVISH_AXI_HTML_APP_TOKEN); never required to publish. |
lavish-axi poll |
--agent-reply "..." |
Show the agent's reply in the existing browser chat before polling again. |
lavish-axi poll |
--timeout-ms <ms> |
Test/debug escape hatch only; agents should normally omit it and leave the long poll running. |
lavish-axi stop |
--port <port> |
Shut down a server running on a non-default port. |
lavish-axi server |
--verbose |
Log session and watcher events to stderr; can also be enabled with LAVISH_AXI_DEBUG=1. Detached server output is appended to ~/.lavish-axi/server.log (or LAVISH_AXI_STATE_DIR/server.log) for startup and crash diagnostics. |
Development
pnpm run check # Run all verification commands
pnpm run build # Bundle the publishable CLI, chrome, and design assets
pnpm run build:skill # Regenerate the installable lavish skill
pnpm test # Run node:test tests
pnpm run lint # Run ESLint
pnpm run format:check # Check Prettier formatting
pnpm run typecheck # Run TypeScript checkJs validation