@liquid-dom/layout
v0.1.0
Published
A renderer-agnostic JavaScript layout engine inspired by SwiftUI layout.
Readme
@liquid-dom/layout
Description
@liquid-dom/layout is a renderer-agnostic TypeScript layout engine inspired by SwiftUI's two-step layout model: parents propose a size, children report a size, then parents place children into rectangles.
It does not render UI. It gives you mutable layout nodes with stable ids, layout properties, measurement caching, and calculated geometry written directly to each node. Your DOM, Canvas, SVG, WebGL, native, or custom renderer owns the visual objects and reads node.layout.
Install
pnpm add @liquid-dom/layoutQuick Start
import { createLayoutEngine, frame, hstack, leaf, spacer } from '@liquid-dom/layout'
const label = leaf({
measure: () => ({ width: 82, height: 28 }),
})
const button = leaf({
measure: () => ({ width: 68, height: 34 }),
})
const row = hstack({ spacing: 12, alignment: 'center' }, label, spacer(), button)
const root = frame(row, { width: 320, height: 56 })
const engine = createLayoutEngine({ root })
const stats = engine.layout({ width: 320, height: 56 })
console.log(stats.measureCalls)
console.log(label.layout?.rect)
row.spacing = 20
engine.layout({ width: 320, height: 56 })API Overview
Layout Engine
import { createLayoutEngine } from '@liquid-dom/layout'
const engine = createLayoutEngine({
root,
onInvalidate: () => requestAnimationFrame(render),
maxCachedMeasurements: 50_000,
})
engine.root = root
const stats = engine.layout({ width: 800 })
engine.dispose()layout(proposal) throws until root is assigned. It mutates reachable nodes by writing node.layout, then returns debug stats. Set maxCachedMeasurements: 0 to disable measurement caching while profiling.
createLayoutEngine(options) accepts LayoutEngineOptions:
| Option | Type | Description |
| --- | --- | --- |
| root | LayoutNode | Optional initial root. |
| onInvalidate | (invalidation: LayoutInvalidation) => void | Called when a node mutation invalidates measurement. |
| dev | boolean | Enables development-oriented runtime checks. |
| maxCachedMeasurements | number | Maximum measurement cache entries. Use 0 to disable caching. |
LayoutEngine exposes root, layout(proposal), getDebugStats(), and dispose(). LayoutDebugStats includes measureCalls, cacheHits, cacheMisses, invalidations, activeSubscriptions, and nodes.
Layout Nodes
Nodes are mutable objects. Builders such as hstack, vstack, frame, padding, and leaf return node instances with stable generated ids and property setters.
const row = hstack({ spacing: 8 })
row.append(title, spacer(), button)
row.spacing = 16
engine.layout({ width: 800 })A node has one parent. Appending it to another parent automatically detaches it from the old parent, matching DOM parenting.
Every node exposes:
id,kind,parent,children, andlayoutappend,prepend,insertBefore,replaceChildren,remove, anddispose
layout.rect is relative to the parent layout node. Detached nodes keep their last layout until they are laid out again.
Core geometry and layout types:
| Type | Shape |
| --- | --- |
| ProposedSize | { width?: number; height?: number } |
| Size | { width: number; height: number } |
| Rect | { x: number; y: number; width: number; height: number } |
| NodeLayout | { rect: Rect } |
| Length | number \| 'infinity' |
| Insets | { top: number; right: number; bottom: number; left: number } |
| InsetsInput | number, partial edge insets, or { horizontal?: number; vertical?: number } |
| ChildInput | A node, an array of optional nodes, null, false, or undefined. |
Alignment types:
| Type | Values |
| --- | --- |
| Axis | 'horizontal', 'vertical' |
| StackAlignment | 'start', 'center', 'end', 'leading', 'trailing', 'top', 'bottom' |
| Alignment | 'center', edge/corner strings such as 'topLeading', or { x?: 'start' \| 'center' \| 'end'; y?: 'start' \| 'center' \| 'end' } |
Leaves
leaf(spec) creates a renderer-owned leaf. Leaves define their own measurement behavior.
const title = leaf({
measure: (proposal) => ({
width: proposal.width ?? 180,
height: 32,
}),
subscribe: (notify) => {
const unsubscribe = model.onChange(() => notify('model'))
return unsubscribe
},
measureKey: model.version,
})
title.invalidateMeasure('manual')Use measureKey, replace measure, or call invalidateMeasure() when measurement behavior changes outside a subscription.
LeafSpec fields:
| Field | Type | Description |
| --- | --- | --- |
| measure | LeafMeasure | Required measurement callback. |
| subscribe | LeafSubscribe | Optional subscription that calls notify(cause) when measurement changes. |
| measureKey | unknown | Cache identity for measurement behavior. |
LeafNode extends LayoutNode with measure, subscribe, measureKey, and invalidateMeasure(cause?).
Built-In Layouts
import {
background,
defineLayout,
frame,
hstack,
leaf,
noop,
overlay,
padding,
spacer,
vstack,
zstack,
} from '@liquid-dom/layout'hstackandvstackplace children along one axis with fixed spacing and cross-axis alignment.zstacksizes to the maximum child width and height, then aligns each child inside that shared bounds.frameproposes constraints to its child, clamps the reported size, and aligns the child inside the frame.paddingsubtracts insets before measuring the child and adds them back to its own size.spacerexpands in finite proposals.noopforwards the same proposal to its single child.backgroundandoverlayplace decorations in the content bounds without affecting parent layout.defineLayoutcreates custom containers with explicitmeasureandplacefunctions.
Built-in option types:
| Builder | Option type | Options | Node properties |
| --- | --- | --- | --- |
| hstack, vstack | StackOptions | spacing?: number, alignment?: StackAlignment | StackNode.spacing, StackNode.alignment |
| zstack | ZStackOptions | alignment?: Alignment | ZStackNode.alignment |
| frame | FrameOptions | width, height, minWidth, minHeight, idealWidth, idealHeight, maxWidth, maxHeight, alignment | Matching FrameNode properties |
| padding | PaddingOptions | insets?: InsetsInput | PaddingNode.insets |
| background, overlay | DecorationOptions | alignment?: Alignment | DecorationNode.alignment |
| spacer | SpacerOptions | minLength?: number | SpacerNode.minLength |
| defineLayout | DefineLayoutOptions | kind, props, measure, place | CustomLayoutNode.props |
frame(node, options) and padding(node, options) attach an initial child. Calling them with only options creates an empty node. noop(child?) creates a NoopNode.
Custom Layouts
const flow = defineLayout({
kind: 'flow',
measure: ({ children, proposal }) => {
const sizes = children.map((child) => child.measure(proposal))
return {
width: sizes.reduce((sum, size) => sum + size.width, 0),
height: Math.max(0, ...sizes.map((size) => size.height)),
}
},
place: ({ bounds, children, proposal }) => {
let x = bounds.x
for (const child of children) {
const size = child.measure(proposal)
child.place({ x, y: bounds.y, width: size.width, height: size.height }, size)
x += size.width
}
},
})place is command-style: call child.place(...) directly. bounds is in the layout node's own local coordinate space, so direct child placements are parent-local rects.
LayoutMeasureInput contains proposal, children, and node. LayoutPlaceInput contains bounds, proposal, children, and node. Each LayoutChild exposes node, id, kind, isSpacer, measure(proposal), and place(bounds, proposal?).
DOM Helpers
The @liquid-dom/layout/dom subpath provides domLeaf, measureDomElement, and subscribeDomElement for HTML element measurement.
import { domLeaf } from '@liquid-dom/layout/dom'
const node = domLeaf({
element,
sizing: 'intrinsic',
})DOM helper API:
| API | Description |
| --- | --- |
| domLeaf({ element, sizing, measureKey }) | Creates a LeafNode measured from an HTMLElement. |
| measureDomElement(element, proposal?, options?) | Measures an element clone using the selected sizing mode. |
| subscribeDomElement(element, notify) | Subscribes to resize, class/style/content mutation, image load/error, and font-load changes. |
| DomLeafSizing | 'intrinsic', 'constrained-width', or 'fill'. |
| DomMeasureOptions | { sizing?: DomLeafSizing } |
| DomLeafOptions | { element: HTMLElement; sizing?: DomLeafSizing; measureKey?: unknown } |
Integration Notes
Keep render metadata outside layout nodes. A UI object can hold a reference to its layout node:
type View = {
element: HTMLElement
layoutNode: import('@liquid-dom/layout').LayoutNode
}
function applyViewLayout(view: View) {
const layout = view.layoutNode.layout
if (!layout) return
Object.assign(view.element.style, {
position: 'absolute',
left: '0px',
top: '0px',
transform: `translate3d(${layout.rect.x}px, ${layout.rect.y}px, 0)`,
width: `${layout.rect.width}px`,
height: `${layout.rect.height}px`,
})
}If your renderer skips layout-only intermediary nodes, accumulate ancestor offsets in userland or attach render groups to the intermediary layout nodes that own those coordinate boundaries.
Local Development
pnpm --filter @liquid-dom/layout build
pnpm --filter @liquid-dom/layout test
pnpm --filter @liquid-dom/layout typecheck