Emerj is a tiny JavaScript library to render live HTML/DOM updates efficiently and non-destructively, by merging an updated DOM tree into the live DOM, element-by-element, and only changing those elements that differ. It's a lightweight, powerful answer to the some of the same problems addressed by Facebook's React.
Specifically, Emerj solves the problem of rendering on-the-fly changes to your HTML/DOM webpage or UI. In most user-facing software, you want to keep the UI-rendering logic quite separate from the business logic, and not clutter your logic with all kinds of extra code to update the UI when the data is modified.
In traditional HTML/JavaScript, this can be tricky to do tidily, which is the main driver for the recentish proliferation of tools like Angular, React, Ractive, etc. However, these tools all require a moderately significant paradigm shift, and your code needs to adopt the framework from the ground up. They also come with a fair amount of hidden (and not-so-hidden) complexity.
Emerj aims to cut through all this complexity by doing just one thing, and using the native browser DOM APIs for all the heavy lifting. You provide an HTML string (or an out-of-document DOM tree) and Emerj compares this to the live DOM and updates just those elements and attributes that differ. To do this, Emerj uses a similar method used by React described in its reconciliation algorithm.
For an extended explanation of the thinking behind Emerj, and the design decisions I made along the way, see the introductory blog post.
$ npm install emerj
This exposes CommonJS (index.js
), UMD (index.umd.js
), and ESM (index.mjs
) versions e.g.:
const emerj = require('emerj');
// or
import emerj from 'emerj';
Alternatively, a CDN can be used e.g.:
<script type='text/javascript' src='https://unpkg.com/[email protected]'>
Feel free to post issues or questions as they come up.
There are four basic steps:
- Set up your document, include the necessary scripts, and an empty element ready to fill with some content.
- Create your data model. This could be as simple as
data = {todo: []}
- Write some code to generate HTML based on that data. You could try Nunjucks or anything you like.
- Whenever you update your data, regenerate your HTML, and use Emerj to update the live document.
I learn best by example, so here you are:
<!doctype html>
<html>
<head>
<script type=text/javascript src='nunjucks.js'>
<script type=text/javascript src='emerj.umd.js'>
</head>
<body>
<div id=root><!-- The live document will go here --></div>
<script type=text/template name='todo.html'>
<!-- Your favourite template language. I like Nunjucks. -->
<ul class='todo'>
{% for item in todo %}
<li><input type=checkbox {% if item.done %}checked{% endif %}> {{ item.text }}</li>
{% else %}
Add an item below
{% endfor %}
<li><input type=text name=new></li>
</ul>
</script>
<script type=text/javascript>
var data = {todo: []}
var template = nunjucks.compile(document.querySelector('[name="todo.html"]').innerHTML);
addEventListener('data:updated', function () { // Use your preferred event system.
// You could use requestAnimationFrame() here for better performance.
var html = template.render(data);
emerj.merge(document.querySelector('#root'), html);
})
document.querySelector('[name=new]').addEventListener('change', function() {
data.todo.append(this.value);
dispatchEvent(new Event('data:updated'));
})
dispatchEvent(new Event('data:updated'));
</script>
</body>
</html>
If you need to create dynamic HTML on-the-fly in the browser, based on some data, and you want something lightweight but powerful and adaptable, and/or Preact/React isn't your cup of tea, then (particularly if you like generating HTML from moustachioed templates, but can't accept the brokenness of overwriting your entire document every render), Emerj might be your answer.
For more, see the introductory blog post. But here's the short version:
After becoming frustrated with the horribleness of ad-hoc DOM manipulation, and the problems with plain template rendering, I grew attracted to React's clever idea of comparing two virtual DOMs and just updating the differences in the live document. But I couldn't abide JSX, and any solution I adopt needs to retrofit into sizeable existing applications, without requiring a complete overhaul of the UI templates.
A comment on Hacker News made me realise how useful it would be to have the virtual-DOM concept of React, but without being locked into JSX. A few things fell nicely into place, and Emerj was the result.
Emerj is based on three primary concepts:
- Be unopinionated about how the HTML or DOM is generated. Just use any plain HTML or DOM that is passed in, and let the user create it however they like.
- Realise that every browser has a perfectly functional virtual DOM built in. Any out-of-document element will do. The browser also provides a functional API to parse an HTML string into a virtual DOM:
element.innerHTML = html
. So use it. - Using this virtual DOM, merge it with the live DOM in the approximate manner described in React's reconciliation algorithm.
There's not really much more to say. It's that simple. The code is simple enough to refer you to it for any more details.
No. See the blog post.
Parsing HTML is more slow. But not nearly enough to justify the slow sad shake of your head you're doing right now. Don't forget your real bottleneck is probably the server round-trip for your API calls.
Yep. The best code is no code.
But, in case you completely missed the whole point, Nunjucks on its own (or whatever template language you use) will blow away and rerender your DOM on every update, losing all document state (scroll position, input state, etc) and sucking your CPU.
See the performance section in the introductory blog post. In short, Emerj+Nunjucks is within cooee of React for a typical modern web app (what's that?!). But test it out for your own needs, and if it's not performant enough for you, either build a component system on top of it, or use something else.
1.0.0
Copyright © 2017-2018 by Bryan Hoyt.
This is free software; you can redistribute it and/or modify it under the terms of the MIT License.