domorphic

pure js DOM templating

Usage no npm install needed!

<script type="module">
  import domorphic from 'https://cdn.skypack.dev/domorphic';
</script>

README

domorphic

is a plain javascript program to interact with the DOM, in a functional, reactive and keep-it-simple philosophy.

<script src="https://cdn.jsdelivr.net/npm/domorphic@0.0.2/dist/dom.min.js"></script> 

Inspired by the power of d3 and the beauty of elm, this library attempts to breed their many respective qualities, to make shaping DOM interfaces within js enjoyable, smooth and pure.

Foreword

There are two layers to this library:

  • the first one is really just convenience for templating: pure js code which outputs DOM subtrees, regardless of anything else, so that you may very well be content with it.

  • the second one, much more subtle, is about designing stateful and reactive applications with pure monadic code.

Whether you've ever heard about monads or not, they're intuitive and powerful, though there's no need to worship their metaphysical nature to use the package :) For more documentation and examples, visit the project's website 🌍 !

Usage

Everything happens in the polymorphic type a -> Node of functions returning DOM nodes, which in scientific terms, is also called the category of types above Node. What?!

Above the DOM. A domorphic instance is just a function returning a DOM node. That's it!

let node = dom('h1').html('Hello World:!')();

At first glance, the library hence only helps you conveniently parametrise functions returning DOM nodes. And you may very well do document.body.appendChild or whatever you like with them.

Syntax. All view-related code is meant to be as light and readable as any other templating language: except it's javascript!

The dom constructor parses arguments looking for the following pattern:

dom('tag#id.class', ?{...attrs}, m ?-> [...branches])

Each branch may be given either as a dom instance, or as an array of arguments following the same pattern, so that you may nest arrays just as you would nest html tags.

Attributes. All node attributes are interpreted either as values or as data dependent functions.

You may supply them by d3-like chainable accessors:

let a = dom('a')
    .html("internet")
    .attr('href', m => m.href)
    .on('hover', () => alert("wooo"))

although the dom constructor also interprets the equivalent syntax:

let a = dom('a', {
    html: "internet",
    href: m => m.href,
    onhover: () => alert("wooo")),
});

Branches. Nodes are essentially trees of DOM elements, i.e. Node = (Element, [Node]).

The branching attribute itself may be given in a -> [Node], as a node array returning function:

let m = ['cats', 'are', 'cute'],

//  p : Str -> Node
let p = dom('p')
    .html(m => m);

//  div : [Str] -> Node
let div = dom('div')
    .branch(m => m.map(p));

document.body.appendChild(div(m));

and the above produces the same output as:

//  div : () -> Node
let div = dom('div', [
    ['p', {html: "cats"}],
    ['p', {html: "are"}],
    ['p', {html: "cute"}]
]);

document.body.appendChild(div());

Functors

In the category of types, a functor T:

  • assigns to every type a a type T a,
  • transforms any map a -> b to a map T a -> T b.

Because dom instances are pure functions, it is perfectly safe to pipe them into functorial transformations.

Pullbacks. Given a function a -> b, precompose a b -> Node instance to get an a -> Node map:

dom.pull : (a -> b) -> (b -> Node) -> (a -> Node)  

You're looking at the contravariant hom-functor Hom(-, Node) in the cartesian category of types 🔬 :)

Arrays. Functions of type a -> Node are naturally transformed to [a] -> [Node] maps:

dom.map : (a -> Node) -> [a] -> [Node] 

As far as dom instances are pure functions, this is rigorously equivalent to calling Array.map.

Using dom.map will be mostly useful to associate index-specific DOM actions in the IO monad.

Records. Similarly, we map a -> Node to a type of functions on records {a} -> {Node}:

dom.rmap : (a -> Node) -> {a} -> {Node} 

This is not yet implemented, but there's little more to it than its array counterpart.

Monads

The main originality of domorphic is the monadic approach it takes to describe interactions between an internal state and the DOM state.

KISS: no automatic diff refreshes -- only handmade, pure pipelines.

IO. The IO monad describes input/output operations with the DOM.

IO(e) : IO operations, eventually triggering an event of type e.

State. The State monad describes stateful computations.

St(s, a) = s -> (a, s) : Computations with state in s, return value in a.

Updates. The State and IO monads yield a composed monadic type St(s, IO(e)).

Upon an event of type e, this monad describes how to update the internal state and which IO operations should take place:

update : e -> St(s, IO(e)) : Upon event, update state and trigger IO actions.

Binding then the update function to its return value 🔄 just defines the main loop!

let main = (e0, s0) => {
    let [io, s1] = update(e0).run(s0);
    return io.bind(e1 => main(e1, s1));
}

let start = s0 => main('start', s0);