Skip to content
playhtml

Capabilities

Every capability in playhtml is a single HTML attribute. Slap it on an element, give the element an id, and the library handles sync, persistence, and event wiring. This page catalogs the eight built-ins plus can-play, in order from most-commonly used to most-specialized.

Every section below has the same shape:

  • A short description of what the capability does and the kind of state it owns.
  • A live demo you can interact with — open the page in a second tab and watch both update.
  • The vanilla HTML form, and the React form. Pick the framework you’re shipping in; the reader’s choice syncs across every snippet on the page.

This page also runs a few can-play demos directly in its margins — a doodle strip below, a guestbook further down, a prize wheel at the bottom, and a shared tally on every code block you copy from. They’re discussed properly in the can-play section.

CAN-PLAY · doodle strip

Draw a tiny 64×64 face. It joins the strip for everyone reading this page. Capped at 20.

What it does. Drag anywhere. Position persists and syncs.

Use it for: drag-and-drop affordances — fridge magnets, game pieces, arrangeable stickers, a puzzle.

<div id="arena" style="position: relative; height: 320px;">
  <img can-move can-move-bounds="arena" id="hat" src="/yankees-hat.png" alt="" />
  <img can-move can-move-bounds="arena" id="cat" src="/long-cat.png" alt="" />
</div>

Use can-move-bounds (vanilla) or the bounds prop (React) to keep the element inside a container. The value is an element id (with or without the leading #) or any valid CSS selector.

<div id="fridge" style="position: relative; height: 400px;">
  <div can-move can-move-bounds="fridge" id="magnet-a">🍎</div>
  <div can-move can-move-bounds="#fridge" id="magnet-b">🥐</div>
</div>

The cursor can go past the container edge — only the element’s position is clamped. By default, the keep-visible slice is max(25% of element size, 60px) on every edge, so readers always have something to grab. Two knobs let you tune that:

  • can-move-bounds-min-visible / boundsMinVisible — fraction (0–1) of the element to keep inside. 1 pins fully inside (strict clamp); 0 drops the fraction constraint entirely.
  • can-move-bounds-min-visible-px / boundsMinVisiblePx — absolute pixel floor. Useful when an image has transparent padding around its paint; a pure fraction of the layout bbox can let the visible pixels clip into invisible border.

The effective slice on each axis is max(fraction × size, pxFloor). Set both to 0 to opt fully out of the keep-visible guarantee and let the element slip entirely out of view.

<!-- Keep 50% of the magnet inside the fridge, ignoring the px floor -->
<div can-move can-move-bounds="fridge"
     can-move-bounds-min-visible="0.5"
     can-move-bounds-min-visible-px="0"
     id="magnet">
  🧲
</div>

<!-- Small magnet: fraction of 0.25 × 40px = 10px is too small to grab.
     The 60px default floor keeps the whole magnet visible regardless. -->
<div can-move can-move-bounds="fridge" id="tiny-magnet" style="width: 40px;">
  🌶️
</div>

What it does. Click to flip an on/off boolean. Persists and syncs.

Use it for: shared switches, lamps, “is-this-thing-open” signs, per-element read/unread state. The playhtml homepage uses this on the wordmark letters and the hanging lamp — click either to flip it for everyone.

<button id="my-switch" class="switch" can-toggle>off</button>

<style>
  .switch.clicked { background: #6cd97e; }
  .switch.clicked::after { content: "on"; }
</style>

What it does. Click to scale up. Alt-click (or equivalent modifier) to scale down. Persists and syncs.

Use it for: zoomable images, inflating a balloon or sticker, growing a banner over time.

<img can-grow id="balloon" src="/water-balloon.png" alt="" />

What it does. Drag to rotate. Persists and syncs.

Use it for: wheels, dials, gauges, spinnable stickers or album covers, “what’s your answer” wheels.

<img can-spin id="wheel" src="/bike-wheel.webp" alt="" />

What it does. Tracks which cursors are currently hovering an element. Presence-based, not persistent — when a user leaves, their hover clears.

Use it for: “who’s looking at this right now” affordances, social read-receipts, zero-latency shared hover effects.

<div can-hover id="hover-pad">hover me</div>

What it does. Click a trigger to clone an existing element into a new one. Every clone is independent shared state.

Use it for: user-generated galleries, seed-from-template UIs, stamps, “leave your mark” patterns.

<img id="bunny-template" src="/pixel-bunny.png" alt="" />
<button can-duplicate="#bunny-template" id="clone-btn">clone a bunny</button>

<script>
  // The "reset" button sweeps every spawned rabbit from shared state.
  document.getElementById("reset-btn").addEventListener("click", () => {
    document.querySelectorAll("[data-duplicate-source='#bunny-template']").forEach((el) => {
      playhtml.deleteElementData("can-duplicate", el.id);
      el.remove();
    });
  });
</script>
<button id="reset-btn">reset</button>

What it does. A meta-capability: can-mirror auto-syncs every attribute, child-node change, and form-state change on the element. You don’t write defaultData, setData, or an updateElement callback — the library observes the DOM for you.

Use it for: when the state you want to share is the DOM: a shared textarea, a form, a growing list of children.

A textarea that filters to emoji characters on input. The filtered value mirrors to everyone.

A wide emoji-only textarea. Type anything; only emojis stick. Everyone composes the message together.

<textarea can-mirror id="emoji-pad" rows="4" placeholder="emojis only..."></textarea>

<script>
  const emojiOnly = /\p{Extended_Pictographic}/gu;
  document.getElementById("emoji-pad").addEventListener("input", (e) => {
    const match = e.target.value.match(emojiOnly);
    e.target.value = match ? match.join("") : "";
  });
</script>

Vignette B — a list you can add children to

Section titled “Vignette B — a list you can add children to”

Click ”+” to append a new <li>. The child addition itself mirrors — new readers see the whole list, not just the latest.

A shared guestbook list. Append your signature; the full list mirrors to everyone else.

<ul can-mirror id="guestbook">
  <li>first</li>
</ul>
<button onclick="
  document.getElementById('guestbook').appendChild(
    Object.assign(document.createElement('li'), { textContent: new Date().toLocaleTimeString() })
  );
">add entry</button>
PlaygroundSee every can-mirror edge case →

Hover, focus, every form input, contenteditable, programmatic attribute changes — one page with a live demo for each. Open in two tabs to watch it sync.

CAN-PLAY · guestbook

Drop a note about what you’re learning or building. The list is shared, capped at 20, oldest entries fall off.

What it does. The “build your own capability” escape hatch. You define defaultData, onClick / onDrag / onMount handlers, and an updateElement callback — playhtml syncs the data and fires your handlers.

Use it for: anything the built-ins don’t already cover. Custom counters, guestbooks, chat, games, reactions, per-user state, event-based broadcasts.

This page is itself a can-play showcase. Four examples live in the margins:

  • Doodle strip — freehand canvas input encoded as a tiny PNG data URL. The shared array caps at 20 via splice inside the setData mutator.
  • Guestbook — a growing list of short text entries. Same “cap the array” pattern, different payload.
  • Prize wheel — one shared piece of data (the spin seed + target) animates identically on every tab. Every visitor sees the wheel land on the same label.
  • Tally on every code block — every <pre> on this page is a vanilla can-play element. Click copy on any snippet and watch a tally tick appear next to the button — readers all over the site share that counter, so popular snippets accumulate marks like a well-loved library book.

That’s four wildly different shapes — short array, blob payload, animation seed, single integer — built on the same primitive. The minimum viable shape is much smaller, though:

<button can-play id="cheer" data-count="0">❤️ <span>0</span></button>

<script>
  const el = document.getElementById("cheer");
  el.defaultData = { count: 0 };
  el.onClick = (_e, { data, setData }) => setData({ count: data.count + 1 });
  el.updateElement = ({ element, data }) => {
    element.querySelector("span").textContent = data.count;
  };
</script>

Deeper material on custom capabilities lives in Data essentials and Dynamic elements.


These capabilities are the building blocks. The best way to see them in concert is to visit the live experiments room, where readers leave permanent marks in a shared space.

These are the same three experiments the homepage slot machine surfaces — the fastest way to feel playhtml at room-scale.

CAN-PLAY · prize wheel

Spin to pick a docs page to read next. The wheel’s spin seed is shared, so everyone here sees it land on the same label.