hyper-dom-expressions
v0.39.4
Published
A Fine-Grained Rendering Runtime API using HyperScript
Downloads
1,563
Readme
Hyper DOM Expressions
This package is a Runtime API built for DOM Expressions to provide HyperScript DSL for reactive libraries that do fine grained change detection. While the JSX plugin Babel Plugin JSX DOM Expressions is more optimized with precompilation, smaller size, and cleaner syntax, this HyperScript solution has the flexibility of not being precompiled. However, Tagged Template Literals are likely a better choice in terms of performance in non-compiled environments Lit DOM Expressions.
Compatible Libraries
- Solid: A declarative JavaScript library for building user interfaces.
- ko-jsx: Knockout JS with JSX rendering.
- mobx-jsx: Ever wondered how much more performant MobX is without React? A lot.
Getting Started
Install alongside the DOM Expressions and the fine grained library of your choice. For example with S.js:
> npm install s-js dom-expressions hyper-dom-expressions
Create your configuration and run dom-expressions command to generate runtime.js. More info here.
Use it to initialize your Hyper function
import { createHyperScript } from 'hyper-dom-expressions';
import * as r from './runtime';
const h = createHyperScript(r);
Profit:
const view = h('table.table.table-hover.table-striped.test-data',
h('tbody', mapSample(() => state.data, row =>
h('tr', [
h('td.col-md-1', row.id),
h('td.col-md-4', h('a', {onClick: [select, row.id]}, () => row.label)),
h('td.col-md-1', h('a', {onClick: [remove, row.id]}, h('span.glyphicon.glyphicon-remove'))),
h('td.col-md-6')
])
))
));
S.root(() => r.insert(document.getElementById('main'), view());)
Libraries may expose access to h in different ways. For example Solid has it's own entry point 'solid-js/h'.
Differences from JSX
There are also several small differences but generally follows HyperScript conventions. Ref work by passing a function. Keep in mind you need to wrap expressions in functions if you want them to be observed. For attributes since wrapping in a function is the only indicator of reactivity, passing a non-event function as a value requires wrapping it in a function.
Fragments are just arrays. Components are handled by passing a Function to the first argument of the h function. Ie:
const view = <>
<Component prop={value} />
{someValue()}
</>
// is equivalent to:
const view = [
h(Component, {prop: value}),
() => someValue()
]
Status
I'm still working out API details and performance profiling.