Webreel
Create and record scripted browser demo videos with webreel. Generates MP4, GIF, or WebM recordings with cursor animation, keystroke overlays, and sound…
Install
Quick install
npx skills add https://github.com/vercel-labs/webreel/tree/HEAD/skills/webreelnpx skills add vercel-labs/webreel --skill webreel --agent claude-codenpx skills add vercel-labs/webreel --skill webreel --agent cursornpx skills add vercel-labs/webreel --skill webreel --agent codexnpx skills add vercel-labs/webreel --skill webreel --agent opencodenpx skills add vercel-labs/webreel --skill webreel --agent github-copilotnpx skills add vercel-labs/webreel --skill webreel --agent windsurfMore install options
Shorthand — useful for multi-skill repos:
npx skills add vercel-labs/webreel --skill webreelManual — clone the repo and drop the folder into your agent's skills directory:
git clone https://github.com/vercel-labs/webreel.gitcp -r webreel/skills/webreel ~/.claude/skills/webreel
Create and record scripted browser demo videos with webreel. Generates MP4, GIF, or WebM recordings with cursor animation, keystroke overlays, and sound…
webreelby vercel
Create and record scripted browser demo videos with webreel. Generates MP4, GIF, or WebM recordings with cursor animation, keystroke overlays, and sound…npx skills add https://github.com/vercel-labs/webreel --skill webreelDownload ZIPGitHub
webreel
webreel records scripted browser demos as MP4, GIF, or WebM with cursor animation, keystroke overlays, and sound effects. You define steps in a JSON config, and webreel drives headless Chrome, captures frames, and encodes with ffmpeg.
Installation
Install webreel as a project dependency so the version is pinned in the lockfile. This ensures deterministic recordings across machines and CI.
`npm install webreel
`
If the project already has webreel in its dependencies, skip this step.
Prerequisites
webreel requires Chrome and ffmpeg, but you do NOT need to install them manually. Both are automatically downloaded to ~/.webreel on first run if not already present. Do not install Chrome or Chromium via puppeteer, playwright, or any other tool. webreel manages its own browser.
To download dependencies explicitly, or to fix corrupted/broken binaries:
`npx webreel install
npx webreel install --force # delete cached binaries and re-download
`
To override the auto-downloaded binaries, set these environment variables:
CHROME_PATH- path to a Chrome or Chromium binary (used for preview)
CHROME_HEADLESS_PATH- path to a chrome-headless-shell binary (used for recording)
FFMPEG_PATH- path to an ffmpeg binary
If a recording fails with "No inspectable targets" or similar browser errors, the issue is almost certainly in the webreel config (wrong waitFor, missing element, timing), not a missing browser. Check the config and use --verbose to debug.
.gitignore
The .webreel directory is created at the project root during recording (frames, intermediate files). Add it to .gitignore:
`.webreel
`
Quick start
`# Scaffold a config
npx webreel init --name my-demo --url https://example.com
# Edit webreel.config.json with your steps
# Preview in a visible browser (no recording)
npx webreel preview my-demo
# Record the video
npx webreel record my-demo
`
npx resolves to the locally installed version when webreel is in devDependencies. Output lands in videos/ by default (configurable via outDir).
CLI commands
init
Scaffold a new webreel.config.json.
`webreel init
webreel init --name login-flow --url https://myapp.com
webreel init --name hero -o hero.config.json
`
Flags: --name (video name), --url (starting URL), -o, --output (output file path).
record
Record one or more videos.
`webreel record # all videos in config
webreel record hero login # specific videos by name
webreel record -c custom.config.json # custom config path
webreel record --watch # re-record on config change
webreel record --verbose # log each step
webreel record --dry-run # print resolved config only
webreel record --frames # save raw JPEGs to .webreel/frames/
`
preview
Run steps in a visible browser without recording.
`webreel preview
webreel preview hero --verbose
`
composite
Re-apply overlays (cursor, HUD, sfx) to existing raw video without re-recording. Useful for tweaking theme settings.
`webreel composite
webreel composite hero
`
install
Download Chrome and ffmpeg to ~/.webreel. Also use this to fix corrupted or broken binaries.
`webreel install
webreel install --force # delete cached binaries and re-download
`
validate
Check config for errors without running anything.
`webreel validate
webreel validate -c custom.config.json
`
Config structure
Config files are auto-discovered as webreel.config.json (or .ts, .mts, .js, .mjs). Use -c to specify a custom path.
Top-level fields
FieldDefaultDescription$schema-"https://webreel.dev/schema/v1.json"outDir"videos/"Output directory for rendered videosbaseUrl""Base URL prepended to relative video URLsviewport1080x1080Default viewport { width, height }theme-Cursor and HUD overlay themesfx-Sound effect settingsinclude-Array of step file paths prepended to all videosdefaultDelay-Default delay (ms) appended after each stepclickDwell-Cursor dwell time (ms) before a click
Per-video fields
Each entry in the videos map supports:
FieldDefaultDescriptionurlrequiredURL to open (absolute or relative to baseUrl)viewportinheritedOverride viewport { width, height }zoom-CSS zoom factorwaitFor-Selector or text to wait for before starting stepsoutput"<name>.mp4"Output path (.mp4, .gif, .webm)thumbnail{ time: 0 }Thumbnail config, or { enabled: false }includeinheritedStep files to prependthemeinheritedOverride themesfxinheritedOverride sound effectsdefaultDelayinheritedOverride default delayclickDwellinheritedOverride click dwellfps60Frame ratequality80Encoding quality (1-100)stepsrequiredArray of step objects
Videos map
Videos are keyed by name in the config:
`{
"videos": {
"hero": { "url": "...", "steps": [...] },
"login": { "url": "...", "steps": [...] }
}
}
`
Record specific videos by name: webreel record hero login.
Step types
Each step has an action field. Most steps accept optional label, delay (ms after step), and description fields.
ActionKey fieldsPurposepausemsWait for a durationclicktext or selector, within, modifiersClick an elementtypetext, selector, within, charDelayType text into an inputkeykey, targetPress a key combo (e.g. "cmd+s")dragfrom, to (element targets)Drag between two elementsscrollx, y, selectorScroll the page or an elementwaitselector or text, timeoutWait for an element to appearmoveTotext or selector, withinMove cursor to an elementnavigateurlNavigate to a new URLhovertext or selector, withinHover over an elementselectselector, valueSelect a dropdown valuescreenshotoutputCapture a PNG screenshot
For full field details on every step type, see steps-reference.md.
Element targeting
Many steps target elements using these fields:
text- match by visible text content
selector- match by CSS selector
within- narrow the search to a parent matching this CSS selector
You can use text or selector (not both). within is optional and scopes the search.
`{ "action": "click", "text": "Submit" }
{ "action": "click", "selector": "#submit-btn" }
{ "action": "click", "text": "Submit", "within": ".modal" }
`
Viewport presets
Use preset names as string values for viewport, or specify { width, height }:
desktop (1920x1080), desktop-hd (2560x1440), laptop (1366x768), macbook-air (1440x900), macbook-pro (1512x982), ipad (1024x1366), ipad-pro (834x1194), ipad-mini (768x1024), iphone-15 (393x852), iphone-15-pro-max (430x932), iphone-se (375x667), pixel-8 (412x915), galaxy-s24 (360x780).
Theme
Customize cursor appearance and keystroke HUD:
`{
"theme": {
"cursor": {
"image": "./cursor.svg",
"size": 32,
"hotspot": "center"
},
"hud": {
"background": "rgba(30, 41, 59, 0.85)",
"color": "#e2e8f0",
"fontSize": 48,
"fontFamily": "\"SF Mono\", monospace",
"borderRadius": 12,
"position": "top"
}
}
}
`
cursor.image- path to a custom cursor SVG or PNG
cursor.size- cursor size in pixels
cursor.hotspot-"top-left"(default) or"center"
hud.position-"top"or"bottom"
Common patterns
Shared steps via include
Factor out reusable step sequences (e.g. dismissing a cookie banner) into JSON files:
`// steps/dismiss-banner.json
{
"steps": [
{ "action": "wait", "selector": ".cookie-banner", "timeout": 5000 },
{ "action": "click", "selector": ".accept-btn", "delay": 300 }
]
}
`
Reference them in the config:
`{
"include": ["./steps/dismiss-banner.json"],
"videos": { ... }
}
`
Multiple videos in one config
Define several videos in the videos map. Shared settings (viewport, theme, defaultDelay) are inherited from the top level.
Environment variables
Config values support $VAR and ${VAR} substitution from the environment.
Output formats
Set the output extension to control format: .mp4 (default), .gif, .webm.
`{ "output": "demo.gif" }
`
Tips
- Always set
waitForon a video to ensure the page is ready before steps run.
- Use
delayon individual steps to control pacing between actions.
- Use
--watchduring development for automatic re-recording on config changes.
- Use
compositeto iterate on theme/overlay settings without re-recording.
- Use
--verboseto debug step execution.
- Use
--dry-runto inspect the fully resolved config (includes, env vars, defaults).
- Use
zoomto scale up small UIs for readability in the recording.
- Start with
previewto verify steps work before committing to a full recording.
Reference files
- steps-reference.md - detailed docs for all 12 step types
- examples.md - annotated config examples for common use cases
More skills from vercel
agent-friendly-apisby vercelCompanion skill for the Agent-Friendly APIs course on Vercel Academy. Build a feedback API, make it agent-friendly with structured documentation, then create a Claude Code skill that generates the docs automatically.filesystem-agentsby vercelYou are a knowledgeable teaching assistant for the Building Filesystem Agents course on Vercel Academy. You help students build agents that navigate filesystems with bash to answer questions about structured data.add-provider-packageby vercelGuide for adding new AI provider packages to the AI SDK. Use when creating a new @ai-sdk/<provider> package to integrate an AI service into the SDK.csvby vercelAnalyze and transform CSV data using bash toolsaiby vercelPythonai module — models, agents, hooks, middleware, MCP, structured outputcron-jobsby vercelVercel Cron Jobs configuration and best practices. Use when adding, editing, or debugging scheduled tasks in vercel.json.frontend-designby vercelCreate distinctive, production-grade frontend interfaces with high design quality. Use this skill when the user asks to build web components, pages, artifacts,…vercel-react-best-practicesby vercelReact and Next.js performance optimization guidelines from Vercel Engineering. This skill should be used when writing, reviewing, or refactoring React/Next.js…
---
Source: https://github.com/vercel-labs/webreel/tree/HEAD/skills/webreel
Author: vercel
Discovered via: mcpservers.org
SKILL.md source
---
name: webreel
description: Create and record scripted browser demo videos with webreel. Generates MP4, GIF, or WebM recordings with cursor animation, keystroke overlays, and sound…
---
# webreel
Create and record scripted browser demo videos with webreel. Generates MP4, GIF, or WebM recordings with cursor animation, keystroke overlays, and sound…
# webreelby vercel
Create and record scripted browser demo videos with webreel. Generates MP4, GIF, or WebM recordings with cursor animation, keystroke overlays, and sound…
`npx skills add https://github.com/vercel-labs/webreel --skill webreel`Download ZIPGitHub
## webreel
webreel records scripted browser demos as MP4, GIF, or WebM with cursor animation, keystroke overlays, and sound effects. You define steps in a JSON config, and webreel drives headless Chrome, captures frames, and encodes with ffmpeg.
## Installation
Install webreel as a project dependency so the version is pinned in the lockfile. This ensures deterministic recordings across machines and CI.
```
`npm install webreel
`
```
If the project already has webreel in its dependencies, skip this step.
## Prerequisites
webreel requires Chrome and ffmpeg, but you do NOT need to install them manually. Both are automatically downloaded to `~/.webreel` on first run if not already present. Do not install Chrome or Chromium via puppeteer, playwright, or any other tool. webreel manages its own browser.
To download dependencies explicitly, or to fix corrupted/broken binaries:
```
`npx webreel install
npx webreel install --force # delete cached binaries and re-download
`
```
To override the auto-downloaded binaries, set these environment variables:
* `CHROME_PATH` - path to a Chrome or Chromium binary (used for preview)
* `CHROME_HEADLESS_PATH` - path to a chrome-headless-shell binary (used for recording)
* `FFMPEG_PATH` - path to an ffmpeg binary
If a recording fails with "No inspectable targets" or similar browser errors, the issue is almost certainly in the webreel config (wrong `waitFor`, missing element, timing), not a missing browser. Check the config and use `--verbose` to debug.
## .gitignore
The `.webreel` directory is created at the project root during recording (frames, intermediate files). Add it to `.gitignore`:
```
`.webreel
`
```
## Quick start
```
`# Scaffold a config
npx webreel init --name my-demo --url https://example.com
# Edit webreel.config.json with your steps
# Preview in a visible browser (no recording)
npx webreel preview my-demo
# Record the video
npx webreel record my-demo
`
```
`npx` resolves to the locally installed version when webreel is in `devDependencies`. Output lands in `videos/` by default (configurable via `outDir`).
## CLI commands
### init
Scaffold a new `webreel.config.json`.
```
`webreel init
webreel init --name login-flow --url https://myapp.com
webreel init --name hero -o hero.config.json
`
```
Flags: `--name` (video name), `--url` (starting URL), `-o, --output` (output file path).
### record
Record one or more videos.
```
`webreel record # all videos in config
webreel record hero login # specific videos by name
webreel record -c custom.config.json # custom config path
webreel record --watch # re-record on config change
webreel record --verbose # log each step
webreel record --dry-run # print resolved config only
webreel record --frames # save raw JPEGs to .webreel/frames/
`
```
### preview
Run steps in a visible browser without recording.
```
`webreel preview
webreel preview hero --verbose
`
```
### composite
Re-apply overlays (cursor, HUD, sfx) to existing raw video without re-recording. Useful for tweaking theme settings.
```
`webreel composite
webreel composite hero
`
```
### install
Download Chrome and ffmpeg to `~/.webreel`. Also use this to fix corrupted or broken binaries.
```
`webreel install
webreel install --force # delete cached binaries and re-download
`
```
### validate
Check config for errors without running anything.
```
`webreel validate
webreel validate -c custom.config.json
`
```
## Config structure
Config files are auto-discovered as `webreel.config.json` (or `.ts`, `.mts`, `.js`, `.mjs`). Use `-c` to specify a custom path.
### Top-level fields
FieldDefaultDescription`$schema`-`"https://webreel.dev/schema/v1.json"``outDir``"videos/"`Output directory for rendered videos`baseUrl``""`Base URL prepended to relative video URLs`viewport``1080x1080`Default viewport `{ width, height }``theme`-Cursor and HUD overlay theme`sfx`-Sound effect settings`include`-Array of step file paths prepended to all videos`defaultDelay`-Default delay (ms) appended after each step`clickDwell`-Cursor dwell time (ms) before a click
### Per-video fields
Each entry in the `videos` map supports:
FieldDefaultDescription`url`requiredURL to open (absolute or relative to `baseUrl`)`viewport`inheritedOverride viewport `{ width, height }``zoom`-CSS zoom factor`waitFor`-Selector or text to wait for before starting steps`output``"<name>.mp4"`Output path (`.mp4`, `.gif`, `.webm`)`thumbnail``{ time: 0 }`Thumbnail config, or `{ enabled: false }``include`inheritedStep files to prepend`theme`inheritedOverride theme`sfx`inheritedOverride sound effects`defaultDelay`inheritedOverride default delay`clickDwell`inheritedOverride click dwell`fps``60`Frame rate`quality``80`Encoding quality (1-100)`steps`requiredArray of step objects
### Videos map
Videos are keyed by name in the config:
```
`{
"videos": {
"hero": { "url": "...", "steps": [...] },
"login": { "url": "...", "steps": [...] }
}
}
`
```
Record specific videos by name: `webreel record hero login`.
## Step types
Each step has an `action` field. Most steps accept optional `label`, `delay` (ms after step), and `description` fields.
ActionKey fieldsPurpose`pause``ms`Wait for a duration`click``text` or `selector`, `within`, `modifiers`Click an element`type``text`, `selector`, `within`, `charDelay`Type text into an input`key``key`, `target`Press a key combo (e.g. `"cmd+s"`)`drag``from`, `to` (element targets)Drag between two elements`scroll``x`, `y`, `selector`Scroll the page or an element`wait``selector` or `text`, `timeout`Wait for an element to appear`moveTo``text` or `selector`, `within`Move cursor to an element`navigate``url`Navigate to a new URL`hover``text` or `selector`, `within`Hover over an element`select``selector`, `value`Select a dropdown value`screenshot``output`Capture a PNG screenshot
For full field details on every step type, see steps-reference.md.
## Element targeting
Many steps target elements using these fields:
* `text` - match by visible text content
* `selector` - match by CSS selector
* `within` - narrow the search to a parent matching this CSS selector
You can use `text` or `selector` (not both). `within` is optional and scopes the search.
```
`{ "action": "click", "text": "Submit" }
{ "action": "click", "selector": "#submit-btn" }
{ "action": "click", "text": "Submit", "within": ".modal" }
`
```
## Viewport presets
Use preset names as string values for `viewport`, or specify `{ width, height }`:
`desktop` (1920x1080), `desktop-hd` (2560x1440), `laptop` (1366x768), `macbook-air` (1440x900), `macbook-pro` (1512x982), `ipad` (1024x1366), `ipad-pro` (834x1194), `ipad-mini` (768x1024), `iphone-15` (393x852), `iphone-15-pro-max` (430x932), `iphone-se` (375x667), `pixel-8` (412x915), `galaxy-s24` (360x780).
## Theme
Customize cursor appearance and keystroke HUD:
```
`{
"theme": {
"cursor": {
"image": "./cursor.svg",
"size": 32,
"hotspot": "center"
},
"hud": {
"background": "rgba(30, 41, 59, 0.85)",
"color": "#e2e8f0",
"fontSize": 48,
"fontFamily": "\"SF Mono\", monospace",
"borderRadius": 12,
"position": "top"
}
}
}
`
```
* `cursor.image` - path to a custom cursor SVG or PNG
* `cursor.size` - cursor size in pixels
* `cursor.hotspot` - `"top-left"` (default) or `"center"`
* `hud.position` - `"top"` or `"bottom"`
## Common patterns
### Shared steps via include
Factor out reusable step sequences (e.g. dismissing a cookie banner) into JSON files:
```
`// steps/dismiss-banner.json
{
"steps": [
{ "action": "wait", "selector": ".cookie-banner", "timeout": 5000 },
{ "action": "click", "selector": ".accept-btn", "delay": 300 }
]
}
`
```
Reference them in the config:
```
`{
"include": ["./steps/dismiss-banner.json"],
"videos": { ... }
}
`
```
### Multiple videos in one config
Define several videos in the `videos` map. Shared settings (`viewport`, `theme`, `defaultDelay`) are inherited from the top level.
### Environment variables
Config values support `$VAR` and `${VAR}` substitution from the environment.
### Output formats
Set the `output` extension to control format: `.mp4` (default), `.gif`, `.webm`.
```
`{ "output": "demo.gif" }
`
```
## Tips
* Always set `waitFor` on a video to ensure the page is ready before steps run.
* Use `delay` on individual steps to control pacing between actions.
* Use `--watch` during development for automatic re-recording on config changes.
* Use `composite` to iterate on theme/overlay settings without re-recording.
* Use `--verbose` to debug step execution.
* Use `--dry-run` to inspect the fully resolved config (includes, env vars, defaults).
* Use `zoom` to scale up small UIs for readability in the recording.
* Start with `preview` to verify steps work before committing to a full recording.
## Reference files
* steps-reference.md - detailed docs for all 12 step types
* examples.md - annotated config examples for common use cases
## More skills from vercel
agent-friendly-apisby vercelCompanion skill for the Agent-Friendly APIs course on Vercel Academy. Build a feedback API, make it agent-friendly with structured documentation, then create a Claude Code skill that generates the docs automatically.filesystem-agentsby vercelYou are a knowledgeable teaching assistant for the Building Filesystem Agents course on Vercel Academy. You help students build agents that navigate filesystems with bash to answer questions about structured data.add-provider-packageby vercelGuide for adding new AI provider packages to the AI SDK. Use when creating a new @ai-sdk/<provider> package to integrate an AI service into the SDK.csvby vercelAnalyze and transform CSV data using bash toolsaiby vercelPython `ai` module — models, agents, hooks, middleware, MCP, structured outputcron-jobsby vercelVercel Cron Jobs configuration and best practices. Use when adding, editing, or debugging scheduled tasks in vercel.json.frontend-designby vercelCreate distinctive, production-grade frontend interfaces with high design quality. Use this skill when the user asks to build web components, pages, artifacts,…vercel-react-best-practicesby vercelReact and Next.js performance optimization guidelines from Vercel Engineering. This skill should be used when writing, reviewing, or refactoring React/Next.js…
---
**Source**: https://github.com/vercel-labs/webreel/tree/HEAD/skills/webreel
**Author**: vercel
**Discovered via**: mcpservers.org
Related skills 6
caveman
Ultra-compressed communication mode. Cuts token usage ~75% by speaking like caveman while keeping full technical accuracy. Supports intensity levels: lite, full (default), ultra, wenyan-lite, wenyan-full, wenyan-ultra. Use when user says "caveman mode", "talk like caveman", "use caveman", "less tokens", "be brief", or invokes /caveman. Also auto-triggers when token efficiency is requested.
secure-linux-web-hosting
Use when setting up, hardening, or reviewing a cloud server for self-hosting, including DNS, SSH, firewalls, Nginx, static-site hosting, reverse-proxying an app, HTTPS with Let's Encrypt or ACME clients, safe HTTP-to-HTTPS redirects, or optional post-launch network tuning such as BBR.
readme-i18n
Use when the user wants to translate a repository README, make a repo multilingual, localize docs, add a language switcher, internationalize the README, or update localized README variants in a GitHub-style repository.
lark-shared
Use when first setting up lark-cli, running auth login, switching user/bot identity (--as), handling permission denied or scope errors, needing to update lark-cli, or seeing _notice in JSON output.
improve-codebase-architecture
Find deepening opportunities in a codebase, informed by the domain language in CONTEXT.md and the decisions in docs/adr/. Use when the user wants to improve architecture, find refactoring opportunities, consolidate tightly-coupled modules, or make a codebase more testable and AI-navigable.
paper-context-resolver
Optional RigorPilot helper for README-first deep learning repo reproduction. Use only when the README and repository files leave a narrow reproduction-critical gap and the task is to resolve a specific paper detail such as dataset split, preprocessing, evaluation protocol, checkpoint mapping, or runtime assumption from primary paper sources while recording conflicts. Do not use for general paper summary, repo scanning, environment setup, command execution, title-only paper lookup, or replacin...