Reactivity

Domphy uses listener-based reactivity. Any value can be a function that receives a listener. When a subscribed state changes, Domphy re-runs only that reactive part.

Reactivity
const count = toState(0)

const counter = {
  button: (listener) => `Count: ${count.get(listener)}`,
  onClick: () => count.set(count.get() + 1),
}

count.get(listener) does two things:

  • returns the current value
  • subscribes that reactive function to future changes

Subscriptions are released automatically when the node is removed.

import { type DomphyElement, toState } from "@domphy/core";

// Create a State instance
const count = toState(0);

const text: DomphyElement<"p"> = {
  // Reactive values can be reactive functions.
  // Reading state with `count.get(listener)` also add listener to state.
  // State change => call listener => re render property
  p: (listener) => `Count: ${count.get(listener)}`,
};

const button: DomphyElement<"button"> = {
  button: "Increment",
  onClick: () => count.set(count.get() + 1),

  // Standard Nested CSS nesting
  style: {
    padding: "4px 16px",
    backgroundColor: "#0f62fe",
    borderRadius: "6px",
    color: "#ffffff",
    "&:hover": {
      backgroundColor: "#4589ff",
    },
  },
};

const App: DomphyElement<"div"> = {
  div: [text, button],
};

export default App;

Attributes

Reactive attributes are already fine-grained. When the state changes, Domphy updates only that attribute.

const open = toState(false)

const button = {
  button: "Toggle",
  ariaExpanded: (listener) => open.get(listener),
  disabled: (listener) => !open.get(listener),
}

This does not re-create the node. It only updates the affected DOM attributes.

Use reactive attributes for:

  • disabled
  • hidden
  • value
  • aria-*
  • data-*
  • any attribute whose value should track state directly

CSS Props

Reactive CSS properties are also fine-grained. Domphy updates only the specific CSS declaration that changed.

const active = toState(false)

const box = {
  div: "Hello",
  style: {
    color: (listener) => active.get(listener) ? "red" : "gray",
    opacity: (listener) => active.get(listener) ? 1 : 0.5,
  },
}

This is different from re-rendering the whole node. The existing style rule stays mounted; only the changed CSS properties are updated.

Use reactive style props when:

  • the element itself stays the same
  • only visual state changes
  • you want the smallest possible DOM/CSS update

Children Update

Reactive children are more complex than attributes or CSS props. When the child function runs again, Domphy calls children.update(...) and reconciles the child list.

const items = toState([
  { id: 1, name: "A" },
  { id: 2, name: "B" },
])

const list = {
  ul: (listener) => items.get(listener).map(item => ({
    li: item.name,
    _key: item.id,
  })),
}

Default Rerender

For light children such as text or simple unkeyed content, the default reactive child update is usually enough.

const count = toState(0)

const app = {
  p: (listener) => `Count: ${count.get(listener)}`,
}

This is the simplest form and should be the default choice for simple text children or lightweight child trees.

Fine-Grain With _key

When children are dynamic lists, _key gives Domphy a reconciliation identity.

const list = {
  ul: (listener) => items.get(listener).map(item => ({
    li: item.name,
    _key: item.id,
  })),
}

_key is used only for child diffing. If the key matches, Domphy reuses the existing node instance and DOM node instead of creating a new one.

Use _key when:

  • items can reorder
  • items can insert in the middle
  • items can be removed from the middle
  • child instances carry important runtime behavior

Without _key, child diffing is more positional.

Fine-Grain With Low-Level API

For the most control, update the child list imperatively through the ElementList API instead of relying on a reactive child function to rebuild the array.

const app = {
  div: [
    {
      button: "Add child",
      _onInit: (node) => {
        node.addEvent("click", () => {
          node.parent!.children.insert({ span: "New child" })
        })
      },
    },
  ],
}

Or inside a normal event handler:

{
  button: "Add child",
  onClick: (_, node) => {
    node.parent!.children.insert({ span: "New child" })
  },
}

This is also fine-grained:

  • insert() creates only the new child
  • remove() removes only that child
  • move() reorders existing children
  • swap() swaps existing children

Use the low-level API when updates are event-driven and local, and when you want explicit control over exactly which child changes.

Derived Reactivity

The (listener) => state.get(listener) form is the foundation: an explicit listener subscribes a reactive part to a state. On top of it, Domphy ships derived primitives — computed, effect, effectScope, batch, and untrack — for computations that depend on other reactive values. They build on the same Notifier machinery, so they participate in the same flush and cycle detection as a plain state.get.

These primitives auto-track: a reactive read with no explicit listener inside a computed or effect subscribes automatically. The explicit (l) => state.get(l) path used in elements is unchanged — both work, and they compose.

import { toState } from "@domphy/core"

const a = toState(1)
const b = toState(2)

computed

computed(fn) is a lazy, cached derived value. fn runs on first read and the result is cached; it re-evaluates only after a tracked dependency changes — never on every read. A computed is read like a state: c.get() for the current value, c.get(listener) to subscribe, and (l) => c.get(l) to bind it in an element.

import { computed } from "@domphy/core"

const sum = computed(() => a.get() + b.get()) // auto-tracks a and b

sum.get() // 3 — computes and caches

const view = {
  p: (listener) => `Sum: ${sum.get(listener)}`, // re-runs only when sum changes
}

When a dependency changes, the computed recomputes and notifies its own downstream listeners only if the new value differs by === from the cached one. An identical value short-circuits, so unchanged derivations cause no downstream churn.

effect

effect(fn) runs fn immediately, auto-tracking every reactive read inside it, and re-runs it whenever any tracked dependency changes. It returns a dispose() that releases all subscriptions.

import { effect } from "@domphy/core"

const stop = effect(() => {
  console.log("a + b =", a.get() + b.get())
})
// logs immediately, then re-runs whenever a or b changes

stop() // unsubscribe

Each run re-collects dependencies, so reads no longer reached — for example behind a branch that is now false — are dropped automatically.

effectScope

effectScope() groups reactive resources so they can be disposed together. Anything created inside scope.run(fn) — effects, computeds, listeners, and nested scopes — is owned by the scope, and scope.stop() tears the whole group down in one call.

import { effectScope } from "@domphy/core"

const scope = effectScope()

scope.run(() => {
  effect(() => console.log(a.get()))
  effect(() => console.log(b.get()))
})

scope.stop() // disposes both effects (and any nested scope) at once

batch

batch(fn) coalesces every state write inside fn into a single downstream flush, so dependents react once instead of once per write.

import { batch } from "@domphy/core"

batch(() => {
  a.set(10)
  b.set(20)
})
// effects / computeds depending on a and b re-run a single time

untrack

untrack(fn) runs fn and returns its result without registering its reads into the currently active collector. Use it to read a state inside an effect or computed without making it a dependency.

import { untrack } from "@domphy/core"

effect(() => {
  // Re-runs when `a` changes, but NOT when `b` changes.
  console.log(a.get(), untrack(() => b.get()))
})

External State Systems

Domphy does not enforce a state architecture. Any system that can call a function works:

store.subscribe(() => listener())   // Zustand
atom.subscribe(() => listener())    // Nanostores
count$.subscribe(() => listener())  // RxJS

State API

Not To Do

  • Do not create reactive update loops where one reactive read immediately feeds an event that writes the same source again without a clear boundary.
const text = toState("")

const field = {
  input: null,
  value: (listener) => text.get(listener),
  onChange: (event) => text.set((event.target as HTMLInputElement).value),
}
  • Do not think of this as two-way binding; treat it as one-way data flow instead, where state drives the view and events explicitly write the next state.
const text = toState("")

const field = {
  input: null,
  value: (listener) => text.get(listener),
  onInput: (event) => {
    text.set((event.target as HTMLInputElement).value)
  },
}
  • Do not move ordinary form synchronization into hooks; keep it in flat event handlers such as onInput, onChange, or onClick so the read path and write path stay visible.