When generating HTML with nyooom, you never stop writing vanilla JavaScript that runs directly in your browser.
+
+
+ What this means is that there's no new syntax for things you already know how to do.
+ Functions like filter
or map
can be applied directly to your data,
+ nodes can be assigned to variables right as you create them,
+ common structures can be extracted into function and even passed around as arguments,
+ and the syntax is just the boring old javascript that most tools already understand.
+
+
+
+ State changes in Nyooom
+
+ import {ObservableObject} from 'nyooom/observable.js'
+
+ nyooom.js
+
+
+ Nyooom considers object with a truthy observable
property to be Observables.
+ These objects are assumed to have a value
property representing their current
+ value, and to emit a "change"
event when their value changes.
+
+
+
+ Passing an observable to a generator function where an attribute would be expected
+ will bind the attribute to the observable.
+ The attribute value will be set to the initial value of the observable,
+ and future changes of the observable will update the value of the attribute.
+
+
+
+ Passing an observable where a child element would be expected
+ will insert a reactive child element by inserting the current value
+ of the observable and replacing it whenever the value gets updated.
+ For primitives, this means they will first be converted to strings and then into a text node.
+
+
+
+ Special attributes are also supported. For example, setting an attribute to
+ an observable with a function value, the function will be registered as an
+ event listener. Changing the value of the observable to a different function
+ will unregister the current listener and register the new one instead.
+
+
+ observable.js
+
+
+ This module exports several classes that store state and emit events whenever the state changes.
+
+
+
+ The most generic one of these would be ObservableObject
,
+ which in its values
property exposes a Proxy to a plain JavaScript object.
+ Changing any value on this proxy will emit an event on the associated ObservableObject
.
+ By default, changes on all Observables are enqueued and an event is dispatched in a microtask.
+
+
+
+
+const counter = new State({value: 0})
+
+counter.valueChanged = newValue =>
+ console.log(`Value: ${newValue}`)
+
+return html.flexColumn(
+ {gap: 1},
+ html.button(
+ "Click Me! ",
+ html.span(counter),
+ { click: () => {
+ counter.value += 1
+ }}
+ ),
+ html.button(
+ "Clear",
+ { click: () => {
+ counter.value = 0
+ }}
+ )
+)
+
+
+
+
+
+ The basic State
object is backed by a plain JS object, so their attributes are unique and do not persist page reload.
+ By contrast, the StoredState
class, which extends the core State
class, is backed by a Storage
object like
+ window.localStorage
or window.sessionStorage
, meaning that they persist page reloads.
+ Additionally, they detect changes both from the current as well as from other browser tabs/windows, so any updates of the state
+ get propagated automatically to all states backed by the same storage.
+
+
+
+
+
+ A simple Todo list
+
+
+ A simple, but not completely bare-bones todo list application,
+ using nothing more than Nyooom and the CSS already present
+ on this page to save some code.
+
+
+
+
+let todo, input
+const task = value =>
+html.flexRow (
+ {class: ["todo"], gap: 1},
+ value,
+ html.span("[x]", {
+ style: {
+ color: "var(--primary-6)",
+ cursor: "pointer"
+ },
+ click: event => {
+ event
+ .target
+ .closest(".todo")
+ .remove()
+ }
+ })
+)
+return todo =
+html.flexColumn(
+ {gap: 1},
+ input=html.input({
+ type: "text",
+ placeholder:
+ "Do some stuff",
+ }),
+ html.button("Add Task",
+ {
+ click: event => {
+ todo.append(task(
+ input.value ||
+ input.placeholder
+ ))
+ }
+ })
+)
+
+
+
+
+