A light-weight html
template tag function for writing declarative-reactive web apps.
const colors = ['red', 'green', 'blue']
const feeling = 'simplicity'
document.body.append(
...html`
<ul .onclick=${() => console.log(feeling)}>
${colors.map(c => html`<li>${c}</li>`)}
</ul>
`(),
)
- Zero dependencies: Lightweight implementation using standard DOM APIs, in a single small file
- Framework-agnostic: Works with any JavaScript framework or no framework at all
- Declarative and reactive: Update DOM by re-invoking templates with new values
- buildless: No build step required, works in any modern browser as-is
- Template caching: Templates are cached based on source location for optimal performance
- Instance management: Unique keys create unique DOM instances that can be updated in place
- Attribute and property binding: Supports regular attributes, boolean attributes, JS properties, and event handlers
- Type-safe: Written in plain JavaScript with JSDoc types for TypeScript compatibility
- Lit-compatible syntax: Uses familiar
html
template syntax for easy adoption
import {html} from 'nimble-html'
let value = 'Hello World'
const key = Symbol() // or any unique value
// Value interpolation
const template = () => html`<div>Message: ${value}</div>`(key)
const [div] = template() // Creates and returns DOM
console.log(div instanceof HTMLDivElement) // true
console.log(div.textContent) // "Message: Hello World"
// Update values any time.
let value = 'Hello Webdev'
template() // updates the DOM
console.log(div.textContent) // "Message: Hello Webdev"
Simply copy html.js
to your project, or import it directly
from GitHub into your JS code (f.e. using the raw.githack.com proxy):
<script type="module">
import {html} from 'https://rawcdn.githack.com/lume/nimble-html/v0.1.0/html.js'
const feeling = 'wonderfulness'
const [div] = html`<div>${feeling}</div>`()
document.body.append(div)
</script>
If you want to keep your app import statements clean, define an importmap:
<script type="importmap">
{
"imports": {
"nimble-html": "https://rawcdn.githack.com/lume/nimble-html/v0.1.0/html.js"
}
}
</script>
<script type="module">
import {html} from 'nimble-html'
const feeling = 'wonderfulness'
const [div] = html`<div>${feeling}</div>`()
document.body.append(div)
</script>
If you have Node.js tooling, you can also install it via npm
:
npm install nimble-html
Template instances are formed by two things:
- Source location - The unique template strings array from your source code
- Key - A unique reference you provide for instance identity when calling the template
Each template instance has a unique DOM tree that can be updated in place by repeatedly calling the same template with the same key.
const key = Symbol() // A key to represent a single template instance.
function render(key, value) {
return html`<div>Value: ${value}</div>`(key)
}
const [div1] = render(key, 'first')
document.body.append(div1)
console.log(div1.textContent) // "Value: first" - initial content
const [div2] = render(key, 'second')
console.log(div1 === div2) // true - same DOM instance!
console.log(div1.textContent) // "Value: second" - content updated
const newKey = Symbol() // A different key for a new template instance.
const [div3] = render(newKey, 'third')
console.log(div2 !== div3) // true - new DOM instance
console.log(div3.textContent) // "Value: third" - new content
This key feature (pun intended) enables higher-level frameworks to build different ways of managing template instances, while being simple out of the box.
The library supports different attribute binding syntaxes:
const value = 'dynamic'
const isEnabled = true
const clickHandler = () => console.log('clicked')
const elements = html`
<!-- Regular attributes -->
<div class=${value}></div>
<div class="${value} static-class"></div>
<!-- Boolean attributes (with ? prefix) -->
<input ?disabled=${!isEnabled} />
<!-- JS properties (with . prefix) -->
<input .value=${value} />
<!-- Event handlers (with @ prefix) -->
<button @click=${clickHandler}>Click me</button>
`(key)
An html
template returns an array of nodes for convenient access:
const elements = html`
<div>First element</div>
<p>Second element</p>
`(key)
console.log(elements) // [div, p]
A useful syntax is array destructuring to grab elements or text nodes that are visually visible at the top level of a template:
const [div, button, textNode] = html`
<div>harmony</div>
<button>peace</button>
${'compassion'}
`
console.log(div.textContent) // harmony
console.log(button.textContent) // peace
console.log(textNode.data) // compassion
The template syntax is standard HTML, including all whitespace handling, except
for whitespace at the top level of a template, for convenience. Top level
whitespace is trimmed out, and the list of returned nodes are those that you
visually see. That's why the return value in the previous example is three
items ([div, button, textNode]
), and not seven.
The only text nodes returned are those that are visible via an interpolation site.
If you need white space preservation, use explicit text nodes at the top level, or wrap text in elements:
const [pre, whitespace, span, p] = html`
<pre> preserved whitespace here </pre>
${' ' /* explicit top-level whitespace */}
<span> preserved whitespace here </span>
<p>
preserved whitespace here
<span> preserved whitespace here </span>
</p>
`
You see four nodes, you get four nodes.
Warning
Whitespace handling may be subject to change. Perhaps it is better to preserve all whitespace as-is, for consistency, while still allowing the return value to return visible nodes only. Feedback welcome!
html
templates can be nested inside other templates:
function itemTemplate(name) {
return html`<li>${name}</li>`
}
function listTemplate(key, items) {
return html`
<ul>
${items.map(item => itemTemplate(item))}
</ul>
`(key)
}
let items = ['apple', 'banana', 'cherry']
const myListKey = Symbol()
document.body.append(...listTemplate(myListKey, items))
It is not required to pass a key to nested templates, and in that case the nested template instance will be unique per parent template instance. In other words, for each parent template instance, there will be a single instance of a child template at a given interpolation site unless a different key is specified for the child template.
With that in mind, this will work as expected and will update the list items in
place when the parent listTemplate
is called again with the same myListKey
:
items = ['avocado', 'blueberry', 'clementine']
listTemplate(myListKey, items)
Event handlers can be functions or strings:
const handleClick = event => {
console.log('Button clicked!', event.target)
}
const button = html`
<!-- Function handler -->
<button @click=${handleClick}>Click me</button>
<!-- String handler (evaluated as code) -->
<button @click="alert('Hello!')">Alert</button>
`(key)
The @event=
syntax uses addEventListener
under the hood. This is especially
useful for custom elements that don't implement event handlers via JS properties
like native elements do, f.e. el.oninput = () => {...}
.
With property-based event handlers such as .onclick
, setting up an event
handler using html
, like this,
const [el] = html`<button .onclick=${() => {...}}></button>`
is the same as setting it up with the JS property directly:
const el = document.createElement('button')
el.onclick = () => {...}
The .onevent=
syntax is not doing anything special with event handlers, it is
literally just setting a JS property exactly as specified (some elements just so
happen to accept function values for various properties, f.e. event handlers).
With that in mind, the @event=
syntax is special in that it is not setting a
JS property, but setting up event listeners with add/removeEventListener()
,
regardless if the element has JS properties that accept event handler functions
(most custom elements in the wild do not have such event handler JS properties).
It's really simple with this nimble html
tag!
This example is so small, it's not really a framework, but it shows how
straightforward it is to build custom elements with declarative-reactive html
templates:
class MyElement extends HTMLElement {
#value = 123
get value() {
return this.#value
}
set value(val) {
this.#value = val
// Re-running the template with same key updates existing DOM. How easy!
this.template()
}
template() {
return html` <div>Current value: ${this.value}</div> `(this) // Use 'this' as the key
}
constructor() {
super()
this.attachShadow({mode: 'open'})
this.shadowRoot.append(...this.template())
}
}
customElements.define('my-element', MyElement)
const key = Symbol()
const app = value => html`<my-element .value=${value}></my-element>`(key)
document.body.append(...app(123))
// Renders "Current value: 123"
app(456)
// Renders "Current value: 456"
The html
tagged template string function. It is not meant to be called as a regular
function, but as a template string tag function (calling it as a plain function
will destroy its internal template caching and optimization).
const template = html`<div>${value}</div>`
Parameters:
strings
: TemplateStringsArray - The template literal stringsvalues
: Array of interpolation valueskey
: Any unique value used to identify the DOM instance
Returns:
- Array of the template's top-level Nodes.
Supported Interpolations:
- Text content:
${value}
- Attributes:
attr=${value}
orattr="${value}"
- Boolean attributes:
?attr=${boolean}
- Properties:
.prop=${value}
- Events:
@event=${handler}
This project uses:
- Plain JavaScript with type definitions via JSDoc types. The source code runs as-is in any browser.
- TypeScript for type checking and producing type declaration files to enable type checking in downstream projects.
@web/test-runner
for browser-based testing
# This is not required for plain JS usage. It generates type declaration files only, while performing a type check.
npm run build
# Type check only
npm run typecheck
# Type check in watch mode
npm run typecheck:watch
# Run tests (includes build)
npm test
# Watch tests (no build)
npm run test:watch
MIT