1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
|
/*
* Template code
*
* A template is just a javascript structure. An element is represented as:
*
* [tag_name, {attr_name:attr_value}, child1, child2]
*
* the children can either be strings (which act like text nodes), other templates or
* functions (see below)
*
* A text node is represented as
*
* ["{text}", value]
*
* String values have a simple substitution syntax; ${foo} represents a variable foo.
*
* It is possible to embed logic in templates by using a function in a place where a
* node would usually go. The function must either return part of a template or null.
*
* In cases where a set of nodes are required as output rather than a single node
* with children it is possible to just use a list
* [node1, node2, node3]
*
* Usage:
*
* render(template, substitutions) - take a template and an object mapping
* variable names to parameters and return either a DOM node or a list of DOM nodes
*
* substitute(template, substitutions) - take a template and variable mapping object,
* make the variable substitutions and return the substituted template
*
*/
function is_single_node(template)
{
return typeof template[0] === "string";
}
function substitute(template, substitutions)
{
if (typeof template === "function") {
var replacement = template(substitutions);
if (replacement)
{
var rv = substitute(replacement, substitutions);
return rv;
}
else
{
return null;
}
}
else if (is_single_node(template))
{
return substitute_single(template, substitutions);
}
else
{
return filter(map(template, function(x) {
return substitute(x, substitutions);
}), function(x) {return x !== null;});
}
}
expose(substitute, "template.substitute");
function substitute_single(template, substitutions)
{
var substitution_re = /\${([^ }]*)}/g;
function do_substitution(input) {
var components = input.split(substitution_re);
var rv = [];
for (var i=0; i<components.length; i+=2)
{
rv.push(components[i]);
if (components[i+1])
{
rv.push(substitutions[components[i+1]]);
}
}
return rv;
}
var rv = [];
rv.push(do_substitution(String(template[0])).join(""));
if (template[0] === "{text}") {
substitute_children(template.slice(1), rv);
} else {
substitute_attrs(template[1], rv);
substitute_children(template.slice(2), rv);
}
function substitute_attrs(attrs, rv)
{
rv[1] = {};
for (name in template[1])
{
if (attrs.hasOwnProperty(name))
{
var new_name = do_substitution(name).join("");
var new_value = do_substitution(attrs[name]).join("");
rv[1][new_name] = new_value;
};
}
}
function substitute_children(children, rv)
{
for (var i=0; i<children.length; i++)
{
if (children[i] instanceof Object) {
var replacement = substitute(children[i], substitutions);
if (replacement !== null)
{
if (is_single_node(replacement))
{
rv.push(replacement);
}
else
{
extend(rv, replacement);
}
}
}
else
{
extend(rv, do_substitution(String(children[i])));
}
}
return rv;
}
return rv;
}
function make_dom_single(template)
{
if (template[0] === "{text}")
{
var element = document.createTextNode("");
for (var i=1; i<template.length; i++)
{
element.data += template[i];
}
}
else
{
var element = document.createElement(template[0]);
for (name in template[1]) {
if (template[1].hasOwnProperty(name))
{
element.setAttribute(name, template[1][name]);
}
}
for (var i=2; i<template.length; i++)
{
if (template[i] instanceof Object)
{
var sub_element = make_dom(template[i]);
element.appendChild(sub_element);
}
else
{
var text_node = document.createTextNode(template[i]);
element.appendChild(text_node);
}
}
}
return element;
}
function make_dom(template, substitutions)
{
if (is_single_node(template))
{
return make_dom_single(template);
}
else
{
return map(template, function(x) {
return make_dom_single(x);
});
}
}
function render(template, substitutions)
{
return make_dom(substitute(template, substitutions));
}
expose(render, "template.render");
function expose(object, name)
{
var components = name.split(".");
var target = window;
for (var i=0; i<components.length - 1; i++)
{
if (!(components[i] in target))
{
target[components[i]] = {};
}
target = target[components[i]];
}
target[components[components.length - 1]] = object;
}
function extend(array, items)
{
Array.prototype.push.apply(array, items);
}
|