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 🙏

© 2025 – Pkg Stats / Ryan Hefner

@defx/dynamo

v0.3.7

Published

A JavaScript library for progressive enhancing HTML

Downloads

68

Readme

@defx/dynamo

🚧 Please kindly note that this project is a work in progress 🚧

npm gzip size

The small yet powerful JavaScript library for progressively enhancing HTML.

If your only tool is a hammer then every problem looks like a nail

It's likely that 80-100% of your JavaScript interaction/reactivity requirements can be solved in a very simple, performant, and easily maintainable way without the use of a browser-side templating solution. Dynamo is designed specifically to support this approach, whilst also giving you the flexibility to provide the bare minimum amount of templating required as and when it's really needed.

Two types of reactive updates

Reactive DOM updates can broadly be divided into two categories:

  • updating attributes and properties of an existing DOM tree
  • changing the structure of the DOM tree itself

Making this distinction is useful to consider in terms of the associated costs, because structural updates require a browser templating solution, whereas updating attributes and properties of existing nodes does not, and the cost of templating in the browser can be quite expensive. Consider that, in order to update server-rendered content, your website or application must ship your templates to the browser so that your library understands how to re-render those parts. This makes perfect sense when your entire DOM tree needs to be structurally modifed at any point in time, however most websites and applications don't fall into that category. A common e-commerce website, for example, has very little requirement (if any) for structural DOM updates, and yet many such websites are being built with tools that are designed specifically for that purpose.

Install

cdn

import { Dynamo } from "https://unpkg.com/@defx/dynamo"

npm

> npm i @defx/dynamo

Usage

Dynamo is instantiated with a DOM node (referred to as the root node) and a configuration object. Everything that happens next is a combination of two things:

  • the configuration object you provide during instantiation
  • the special x-attributes that exist on the DOM tree attached to your root node

Syntax

Dynamo(rootNode, {
  state: {
    /* the initial state */
  },
  action: {
    /* a dictionary of functions that synchronously update state */
  },
  node: {
    /* a dictionary of functions that update individual node attributes 
    and properties as a side-effect of every state transition */
  },
})

Parameters

state

Specifies the initial state.

action

Action handlers are synchronous functions that receive the current state as their first argument, an Action object as their second argument, and return the next state.

Dynamo(rootNode, {
  action: {
    toggleMenu: (state) => ({
      ...state,
      menuIsOpen: !state.menuIsOpen,
    }),
  },
})

node

node functions accept the current state and return an object that is used to update the attributes and/or properties of each node that references it via the [x-node] attribute.

Dynamo(rootNode, {
  action: {
    toggle: (state) => ({
      ...state,
      isOpen: !state.isOpen,
    }),
  },
  node: {
    accordionTrigger: (state) => ({
      ariaExpanded: state.isOpen,
      ariaControls: "accordionPanel",
    }),
    accordionPanel: (state) => ({
      id: "accordionPanel",
      hidden: !state.isOpen,
    }),
  },
})
<h3>
  <button x-node="accordionTrigger" x-on="click:toggle">
    <!-- content title -->
  </button>
</h3>
<div x-node="accordionPanel"><!-- content detail --></div>

[x-*] attributes

Attributes prefixed with an "x-" tell Dynamo about the parts of the DOM tree that need to be updated and/or react to, and there are only 4 such attributes.

x-node

The [x-node] attribute is used to identify a named function that can be used to derive that nodes attributes and/or properties from state.

x-on

Used to bind an event to an action handler. Accepts two arguments separated by a colon x-on="eventType:actionName" where eventType is the type of event that you want to listen for and actionName is the name of the action handler you wish to invoke.

<button x-on="click:toggleMenu">[=]</button>
<nav x-node="navMenu">...</nav>
Dynamo(rootNode, {
  action: {
    toggleMenu: (state) => ({
      ...state,
      menuIsOpen: !state.menuIsOpen,
    }),
  },
  node: {
    navMenu: (state) => ({
      class: {
        open: state.menuIsOpen,
      },
    }),
  },
})

If you use [x-on] with a <form> submit event then Dynamo will automatically provide the forms FormData as the action payload

x-control

Used to bind any user input control element (e.g., <input>, <select>, <textarea>) to a property in state. The x-control attribute is a boolean attribute that doesn't expect any value; the name of the property reflected in state will taken from the elements [name] attribute, so this must also be provided in all cases.

Applying the [name] + [x-control] pattern encourages parity between form data and request data for the purpose of progressive enhancement

<select name="sortBy" x-control>
  <option value="bestsellers">Bestsellers</option>
  <option value="priceLowToHigh">Price (low - high)</option>
  <option value="priceHighToLow">Price (high - low)</option>
  <option value="rating">Rating</option>
</select>

x-each

Used to declare an element as an item within an ordered collection.

The state for each item in the collection is derived from each elements dataset, so be sure to declare any values you need to work with as data attributes. For example, if you want to allow a user to sort a list, make sure that you declare any values you wish to sort on in each elements dataset:

<li x-each="products" id="f7g649f9" data-price="19.99" data-rating="4.2">
  <p>19.99</p>
</li>

The only hard requirement for x-each item nodes is that they must include an id attribute so that the list can be reliably re-ordered.

If you need the ability to dynamically add more items into your collection at any point, then you can configure a template using the same name as the collection and Dynamo will use it:

Dynamo(rootNode, {
  template: {
    products: ({ id, price, rating }) => `
    <li x-each="products" id="${id}" data-price="${price}" data-rating="${rating}">
      <p>${price}</p>
    </li>`,
  },
})

If you're collection doesn't need to have items added at any point then there's no need to provide a template as re-ordering and removals can all be achieved without it.