npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2026 – Pkg Stats / Ryan Hefner

@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/layout

Quick 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, and layout
  • append, prepend, insertBefore, replaceChildren, remove, and dispose

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'
  • hstack and vstack place children along one axis with fixed spacing and cross-axis alignment.
  • zstack sizes to the maximum child width and height, then aligns each child inside that shared bounds.
  • frame proposes constraints to its child, clamps the reported size, and aligns the child inside the frame.
  • padding subtracts insets before measuring the child and adds them back to its own size.
  • spacer expands in finite proposals.
  • noop forwards the same proposal to its single child.
  • background and overlay place decorations in the content bounds without affecting parent layout.
  • defineLayout creates custom containers with explicit measure and place functions.

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