Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/joewalker/domtemplate
An Experimental DOM Template Engine
https://github.com/joewalker/domtemplate
Last synced: 3 months ago
JSON representation
An Experimental DOM Template Engine
- Host: GitHub
- URL: https://github.com/joewalker/domtemplate
- Owner: joewalker
- License: other
- Created: 2010-09-20T17:12:29.000Z (over 14 years ago)
- Default Branch: master
- Last Pushed: 2013-02-26T15:33:07.000Z (almost 12 years ago)
- Last Synced: 2024-10-02T07:23:22.790Z (4 months ago)
- Language: JavaScript
- Homepage:
- Size: 559 KB
- Stars: 21
- Watchers: 1
- Forks: 11
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE.txt
Awesome Lists containing this project
README
About DomTemplate
=================DomTemplate is yet another template engine.
Rather than doing templating using string manipulation, it uses the DOM
directly for several reasons:* It allows you to register event handers as part of the template process
without needing an extra lookup step
* It can inform you of references to created nodes to save on lookup steps
* It's more secure - it's like using a SQL query API rather than string
manipulation
* Since the templates are as close to HTML as possible, they are manipulable
along with other parts of your website. This makes test/preview easy.
* It allows for asynchronous templating.This implementation is also small (around a couple of hundred lines without
comments) and it is used in Mozilla Bespin/Skywriter and in some of the Firefox
developer tools, so it should be well supported.See below for full usage. However if all you need is a taste:
${contents}
+
template('ex1', { contents:'world' });
↓
world
[Try it...](http://jsfiddle.net/dcCK5/)DomTemplate can be run on the server on the server in NodeJS using [jsdom]
(https://github.com/tmpvar/jsdom).DomTemplate engine has a number of features to help applying arbitrary data to
your page:* Nested data and arbitrary Javascript (``${a.b.c}``)
* Registration of event handlers (``onclick="${function}"``)
* Conditional evaluation (``if="${condition}"``)
* Looping (```` and ``foreach="page in ${pages}"``)
* Getting references to cloned nodes (``save="${element}"``)
* Hidden attributes (``_src="${...}"``)
* Asynchronous Templates
* Grabbing the current node (``${__element}``)Things to be wary of:
* So far DomTemplate has had good exercise on modern browsers, but not much
exposure to older browsers, particularly IE6. This will probably change if
people find it useful.
* We're doing something technically nasty in using custom attribute names which
could have future meaning to a browser. We could consider an alternate
implementation that uses HTML5 data-attributes.Using DomTemplate
-----------------The signature of the template() function is as follows:
var t = template(node, data, options);
Where:
* _node_ Either a string which points to an element with that id, or the
element itself.
* _data_ A JavaScript object containing the properties to be used as the
'global' object for ``${...}`` blocks.
* _options_ A JavaScript object containing options which customize how the
templates are processed. Options include:
* _allowEval_ Allow arbitrary JavaScript inside ``${...}`` blocks in addition
to property paths. Allowing use of eval() can significantly slow down
JavaScript processing.
* _blankNullUndefined_ Output null or undefined values as empty strings
rather than the values 'null' or 'undefined'.
* _stack_ domtemplate maintains a 'stack' of the nodes it is processing which
can help in debugging. The stack option allows you to 'pre-load' the stack.
It can be specified either as a string for a single frame, or an array of
strings for multiple frames. This is most useful to specify the template
name, or usage location.The return value is the Templater object used to process the template.
Nested data and arbitrary Javascript (``${a.b.c}``)
---------------------------------------------------``${...}`` blocks can be in attribute values and in HTML content. When used in
an attribute value, the retrieved data is converted to a string before being
added to the DOM.
When used in element content, ``${...}`` blocks can return either strings
(which will be added to the DOM inside a TextNode (i.e. with HTML escaped) or
they can return DOM elements, in which case the DOM element will be added to
the tree.The data passed to the template can contain nested data, which is then accessed
using a familiar property path:
${a.b.c}
+
template('ex2', { a: { b: { c: 42 } } } );
↓
42
[Try it...](http://jsfiddle.net/dcCK5/1/)Normally any ${} element will be processed as a property path (that is a set of
properties separated by '.') However DomTemplate also supports using arbitrary
JavaScript inside ``${}``, when the ``{ allowEval:true }`` option is used.As an example, this is possible
${console.log('hi'); document.createTextNode('BANG!')}
+
template('ex3', null, { allowEval:true });
↓
BANG!
[Try it...](http://jsfiddle.net/dcCK5/2/)In the real world doing this kind of thing often leads to pain down the road,
however it can be a useful get-out-of-jail-free card.Normally array subscripting is performed using code like ``redBalloons[99]``,
however this is not a property script. To subscript an array in a property
script, you can also do ``redBalloons.99`.Registration of event handlers (``onclick="${function}"``)
----------------------------------------------------------Events are registered using event handlers in a way that is similar to normal
HTML event registration. All you need is the ${...} clause to point to a
function.Example:
Hello
+
template('ex4', {
clickHandler:function(ev) {
console.log('div clicked');
}
});
↓
Hello
[Try it...](http://jsfiddle.net/dcCK5/3/)Here we are registering an onClick handler for the div. Any type of event
handler can be registered.This is particularly handy when `this` is used as the data to the template
engine. We make sure that the context of the function is the object that called
it, so you have access to all your data:
${name}
+
function Person(name) {
this.name = name;
}
Person.prototype = {
display: function() {
template('ex5', this);
},
clickHandler: function(ev) {
console.log('You clicked on ' + this.name);
}
};
var joe = new Person('Joe');
joe.display();
↓
Joe
[Try it...](http://jsfiddle.net/dcCK5/4/)i.e. DomTemplate automatically binds function calls in the way we wish
JavaScript had done from day one.If you wish to use the capture phase of an event, you can use the following
syntax:
...Something to be aware of:
* Although it looks like we are using DOM level 0 event registration (i.e.
element.onfoo = somefunc) we are actually using DOM level 2, by stripping
off the 'on' prefix and then using addEventListener('foo', ...).Conditional evaluation (``if="${condition}"``)
----------------------------------------------If an element contains an 'if' attribute, then its value will be evaluated and
if the result is 'falsey', then the entire element will be removed from the
tree. This allows simple if statements.Example:
Hi, ${name}
+
template('ex6', { name: 'Fred' });
↓
Hi, Fred
[Try it...](http://jsfiddle.net/dcCK5/5/)However:
Hi, ${name}
+
template('ex6', { name: null });
↓
[Try it...](http://jsfiddle.net/dcCK5/6/)In the second example, the entire 'p' element has been removed by processing
the if attribute.Looping (```` and ``foreach="page in ${pages}"``)
-------------------------------------------------------If an element contains a `foreach` attribute, then that element will be repeated
in the final document once for each member of the array returned by the
attribute value.Example:
${index}
+
template('ex7', { array: [ 1, 2, 3 ] });
↓
123
[Try it...](http://jsfiddle.net/dcCK5/7/)If you wish to create a number of elements for each member of the array, then
you can use a special element. This will be removed from the resulting
tree.Or a more complex example:
${person.name}
${person.address}
+
var data = {
people: [
{ name: 'Miss Marple', address: 'St Mary Mead' },
{ name: 'Sherlock Holmes', address: '221B Baker St' },
{ name: 'Hercule Poirot', address: 'Whitehaven Mansions' }
]
};
template('ex8', data);
↓
Miss Marple
St Mary Mead
Sherlock Holmes
221B Baker St
Hercule Poirot
Whitehaven Mansions
[Try it...](http://jsfiddle.net/dcCK5/8/)The foreach element can be used with arrays or objects. If an object is used
then we will iterate over the enumerable property names.2 things to be aware of when using ``foreach``, one obvious, the other not so:
* Any ``id`` attributes in the duplicated data, will be duplicated, this is
probably not what you want. Avoid using id inside an element with a foreach.
* ```` element are very picky about what they contain, specifically,
````s in places the HTML parser doesn't expect can be pushed to after
the table.Getting references to cloned nodes (``save="${element}"``)
----------------------------------------------------------The save attribute is special. It takes the current node at sets it into the
pointed to structure. In this case ${} is not arbitrary Javascript but a dot
path to an element to set.This is useful whenever you need to work with the created nodes.
+
var data = { name: 'Fred' };
template('ex9', data);
console.log(data.element.innerHTML); // "Fred"
↓
Fred
[Try it...](http://jsfiddle.net/dcCK5/9/)Hidden attributes (``_src="${...}"``)
-------------------------------------Since DomTemplate uses pre-existing DOM elements, there could be attributes that
the browser will try to use before templating, and will discover invalid values.The solution is to prefix the attribute name with an underscore. The templating
process will remove the _.For example:
This will the processed, and (assuming 'path' is correctly set) the right image
will be displayed, however you may notice your browser giving a 404 message from
${path}/thing.png as it has attempted to retrieve the image before the template
process had a chance to substitute the correct path.To solve the problem, and have the browser only attempt to fetch the image when
the correct path has been specified, do the following
Should you wish to have an attribute in the resulting document prefixed with an
underscore, simply begin your attribute name with 2 underscores. (Is this a
common scenario? If you know of another scenario where attribute names are
prefixed with _, please contact me.)Asynchronous Templates
----------------------What if the data for the template isn't available when the template is run?
DomTemplate checks returned data for a 'then' function, and assumes that it's
a promise which needs resolving before use.For example:
${name}
+
var p = new Promise();
template('ex10', { name: p });
↓
↓
p.resolve("Joe");
↓
Joe
[Try it...](http://jsfiddle.net/dcCK5/10/)Grabbing the current node (``${__element}``)
--------------------------------------------During templating you may need to get access to the current element.
Sometimes there's just no nice way to describe what you need to do, so the
__element tracks the element that is under examination.For example:
+
template('ex11', null, { allowEval: true });
// logs 'bar' to the console.
↓
[Try it...](http://jsfiddle.net/dcCK5/11/)This technique had been largely superseded by asynchronous templates, and may
be removed in future. If you have a strong use for it please tell us to prevent
it from being removed.