Guided Demo Pattern
Add a self-narrating walkthrough to any HTML/web application. A declarative step array drives a setTimeout loop that toggles CSS classes on DOM elements and writes text character-by-character into a fixed panel. The engine overlays the existing page without modifying its code. When the demo stops, all state resets.
About 100 lines of JS and 30 lines of CSS for the core. Interstitials, speed controls, keyboard shortcuts, and progress bar are optional layering.
Before implementing
Ask the user these questions (skip any already answered in conversation):
- What are the sections? Tabs, routes, scroll positions, slide indices? Determines how section switching works.
- Framework? Vanilla HTML works directly. React/Vue/Svelte need the engine as a conditional overlay component, and
querySelectormust run after render. - Presenter talking over it, or self-narrating? Talking over: shorter text, longer pauses. Self-narrating: detailed text, moderate pauses.
- Interactive elements to trigger? Map to step actions. Confirm CSS selectors are stable (not framework-generated hashes).
- Offline requirement? Keep everything in one file if so. No CDN dependencies without fallbacks.
- How many steps? Under 15 for a pitch, 20-30 for a detailed walkthrough. Over 40 loses attention.
- Touch device support needed? Most guided demos target desktop/laptop presentation contexts. The narrator panel has on-screen prev/play/next buttons that work on touch, but swipe gestures or mobile-specific layout are rarely needed.
Implementation
Read references/implementation.md for all code snippets. The four required pieces are:
- Narrator panel - fixed-position bar with text container and progress bar
- Highlight class -
outline(notborder) with animated glow, no layout shift - Typewriter function - recursive
setTimeout, narration text set viatextContent(never set user-authored narration viainnerHTML), blinking cursor via CSS. TheinnerHTML = ''clearing before each tick is intentional and safe (no user content involved) - Playback loop -
playStep(idx)that switches sections, runs actions, highlights, types, and auto-advances
Step array
The single source of truth. Each step is a plain object:
const DEMO_SCRIPT = [
{ section: 0, target: '.card', text: "Narration text." },
{ section: 1, target: null, text: "Transition.", transition: true },
{ section: 1, target: '#el', text: "Detail.", action: 'open', actionTarget: 'panel-id' },
];
Properties: section (which view), target (CSS selector or null), text (narrator copy), transition (show interstitial), action/actionTarget (trigger UI change).
Timing defaults
| Constant | Default | When to adjust |
|---|---|---|
| TYPE_SPEED | 5ms/char | 3ms for long text, 15ms for dramatic short statements |
| PAUSE_MS | 3000ms | 4-5s if audience reads rather than listens |
| Speed range | 0.5x-2x | Both constants divided by multiplier |
Writing narrator copy
The typewriter effect means each word lands individually, so writing style matters:
- Short declarative sentences. Each step should make exactly one point.
- Conversational tone, present tense. Address the audience directly.
- Describe what the element means, not what the UI shows: "This column quantifies the cost impact" not "The cost is shown in this column".
- No jargon the audience wouldn't know. Match terminology to the domain, not the implementation.
- Under 30 words per step for presenter-led demos, up to 50 for self-narrating.
Keyboard controls
Gate all keyboard capture behind an isActive flag so it does not interfere with normal page interaction. Space = play/pause, arrows = step, Escape = exit and reset.
Optional: transition interstitials
Full-screen overlay with cycling status messages between sections. Simulates processing time. Define messages per section in a 2D array. Fade each message, then dismiss overlay via callback.
Optional: step actions
String-matched in executeStep(). Actions run before highlighting because elements inside collapsed panels can't be found by querySelector until the panel is open. Adding a new action is one if block. Keep it simple. Common patterns beyond expand/collapse: expandOne (open one panel, close all siblings - accordion style), call (trigger a named function like requestBriefing()), addClass/removeClass (toggle a CSS class on document.body for global state changes).
Gotchas
These are the failure points that come up repeatedly:
- Layout shift: Use
outlinenotborderfor highlighting. Outline does not affect box model. - Hidden elements: If target is inside a collapsed container, the action must open it first. This is why actions execute before highlighting.
- Dynamic selectors: Framework-generated class names (
.css-1a2b3c) break between builds. Usedata-*attributes or IDs. - Scroll conflicts:
scrollIntoView({ block: 'center' })conflicts with fixed headers/panels. Setscroll-padding-bottomon the scroll container to account for the narrator panel height. - Z-index: Narrator panel at 500+, interstitials at 490, highlighted elements at 2+. Check for conflicts with existing modals or dropdowns.
- Cleanup on exit: Reset every piece of state the demo touched: close opened panels, remove highlights, clear timers. Missing cleanup leaves confusing UI state.
- ES modules: If using
<script type="module">, all demo functions called fromonclickmust be onwindow.*. - Print: Hide demo panel and interstitial in
@media print. - Accessibility: Narrator panel should have
role="status"andaria-live="polite". Highlight outlines must meet contrast requirements.
Applicability
Works for: prototypes, PoCs, HTML slide decks, data storytelling dashboards, product demos, workshop facilitation, investor pitches, onboarding walkthroughs.
Does not replace: user testing tools, screen recorders, production onboarding tours (use a tour library with persistence and analytics for those).
For framework apps, mount the demo engine as a conditional overlay component and pass the script array as a prop. For single-file demos, inline everything for offline/USB-stick distribution.
