Permission message rules: Each rule must have >= 1 required permissions
[chromium-blink-merge.git] / third_party / jstemplate / jstemplate_test.js
blob5bbd521a2aa3d70a629960fe77fd5def444da49e
1 // Copyright 2006 Google Inc.
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 // http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
12 // implied. See the License for the specific language governing
13 // permissions and limitations under the License.
14 /**
15 * @author Steffen Meschkat (mesch@google.com)
16 * @fileoverview Unittest and examples for jstemplates.
19 function jstWrap(data, template) {
20 return jstProcess(new JsEvalContext(data), template);
23 function testJstSelect() {
24 // Template cardinality from jsselect.
25 var t = document.getElementById('t1');
26 var d = {
27 items: [ 'A', 'B', 'C', '' ]
29 jstWrap(d, t);
31 var h = t.innerHTML;
32 var clone = domCloneNode(t);
33 assertTrue(/>A<\/div>/.test(h));
34 assertTrue(/>B<\/div>/.test(h));
35 assertTrue(/>C<\/div>/.test(h));
36 assertTrue(/><\/div>/.test(h));
38 // Reprocessing with identical data.
39 jstWrap(d, t);
40 assertAttributesMatch(t, clone);
42 // Reprocessing with changed data.
43 d.items[1] = 'BB';
44 jstWrap(d, t);
46 h = t.innerHTML;
47 assertTrue(/>A<\/div>/.test(h));
48 assertFalse(/>B<\/div>/.test(h));
49 assertTrue(/>BB<\/div>/.test(h));
50 assertTrue(/>C<\/div>/.test(h));
52 // Reprocessing with dropped data.
53 d.items.pop();
54 d.items.pop();
55 jstWrap(d, t);
56 h = t.innerHTML;
57 assertTrue(/>A<\/div>/.test(h));
58 assertTrue(/>BB<\/div>/.test(h));
59 assertFalse(/>C<\/div>/.test(h));
60 assertFalse(/><\/div>/.test(h));
62 // Reprocessing with dropped data, once more.
63 d.items.pop();
64 jstWrap(d, t);
65 h = t.innerHTML;
66 assertTrue(/>A<\/div>/.test(h));
67 assertFalse(/>BB<\/div>/.test(h));
68 assertFalse(/>C<\/div>/.test(h));
70 // Reprocessing with empty data -- the last template instance is
71 // preserved, and only hidden.
72 d.items.pop();
73 jstWrap(d, t);
75 assertTrue(/>A<\/div>/.test(h));
76 assertFalse(/>BB<\/div>/.test(h));
77 assertFalse(/>C<\/div>/.test(h));
79 // Reprocessing with added data.
80 d.items.push('D');
81 jstWrap(d, t);
82 h = t.innerHTML;
83 assertFalse(/>A<\/div>/.test(h));
84 assertTrue(/>D<\/div>/.test(h));
87 function testJstDisplay() {
88 var t = document.getElementById('t2');
89 var d = {
90 display: true
92 jstWrap(d, t);
94 var h = t.innerHTML;
95 assertFalse(/display:\s*none/.test(h));
97 d.display = false;
98 jstWrap(d, t);
100 h = t.innerHTML;
101 assertTrue(/display:\s*none/.test(h));
103 // Check that 'this' within js expressions is the template node
104 t = document.getElementById('t2a');
105 d = {
106 showId: 'x'
108 jstWrap(d, t);
110 h = t.innerHTML;
111 assertFalse(/display:\s*none/.test(h));
113 d.showId = 'y';
114 jstWrap(d, t);
116 h = t.innerHTML;
117 assertTrue(/display:\s*none/.test(h));
120 function stringContains(str, sub) {
121 return str.indexOf(sub) != -1;
124 function testJseval() {
125 var data = {};
127 var counter = 0;
128 var ctx = new JsEvalContext(data);
129 ctx.setVariable("callback1", function() {
130 ++counter;
132 ctx.setVariable("callback2", function() {
133 counter *= 2;
136 jstProcess(ctx, document.getElementById('testJseval1'));
137 assertEquals("testJseval1", 1, counter);
139 jstProcess(ctx, document.getElementById('testJseval2'));
140 assertEquals("testJseval2", 4, counter);
143 function testJstValues() {
144 var t = document.getElementById('t3');
145 var d = {};
146 jstWrap(d, t);
147 var h = t.innerHTML;
148 assertTrue(stringContains(h, 'http://maps.google.com/'));
149 var t3a = document.getElementById('t3a');
150 assertEquals('http://maps.google.com/', t3a.foo.bar.baz);
151 assertEquals('http://maps.google.com/', t3a.bar);
152 assertEquals('red', t3a.style.backgroundColor);
155 function testJstTransclude() {
156 var t = document.getElementById('t4');
157 var p = document.getElementById('parent');
158 var d = {};
159 jstWrap(d, t);
160 var h = p.innerHTML;
161 assertTrue(h, stringContains(h, 'http://maps.google.com/'));
164 function assertAttributesMatch(first, second) {
165 assertEquals('assertAttributesMatch: number of child nodes',
166 jsLength(first.childNodes), jsLength(second.childNodes));
167 var b = second.firstChild;
168 for (var a = first.firstChild; a; a = a.nextSibling) {
169 var att = a.attributes;
170 if (att) {
171 assertTrue(b.attributes != null);
172 assertEquals('assertAttributesMatch: number of attribute nodes',
173 att.length, b.attributes.length);
174 for (var i = 0; i < jsLength(att); i++) {
175 var a = att[i];
176 assertEquals('assertAttributesMatch: value of attribute ' + a.name,
177 a.value, b.getAttribute(a.name));
179 } else {
180 assertNull(b.attributes);
182 b = b.nextSibling;
186 function testJsskip() {
187 var div = domCreateElement(document, "DIV");
188 div.innerHTML = [
189 '<div jseval="outercallback()" jsskip="1">',
190 '<div jseval="innercallback()">',
191 '</div>',
192 '</div>'
193 ].join('');
195 var data = {};
196 var ctx = new JsEvalContext(data);
197 var outerCalled = false;
198 ctx.setVariable("outercallback", function() {
199 outerCalled = true;
201 var innerCalled = false;
202 ctx.setVariable("innercallback", function() {
203 innerCalled = true;
205 jstProcess(ctx, div);
207 assertTrue(outerCalled);
208 assertFalse(innerCalled);
211 function testScalarContext() {
212 var t = document.getElementById('testScalarContext');
214 jstWrap(true, t);
215 assertTrue(/>true</.test(t.innerHTML));
217 jstWrap(false, t);
218 assertTrue(/>false</.test(t.innerHTML));
220 jstWrap(0, t);
221 assertTrue(/>0</.test(t.innerHTML));
223 jstWrap("foo", t);
224 assertTrue(/>foo</.test(t.innerHTML));
226 jstWrap(undefined, t);
227 assertTrue(/>undefined</.test(t.innerHTML));
229 jstWrap(null, t);
230 assertTrue(/>null</.test(t.innerHTML));
233 function testJstLoadTemplate() {
234 var wrapperId = 'testJstLoadTemplateWrapper';
235 var id = 'testJstLoadTemplate';
236 jstLoadTemplate_(document, '<div id="' + id + '">content</div>', wrapperId);
237 var wrapperElem = document.getElementById(wrapperId);
238 assertTrue('Expected wrapper element to be in document',
239 !!wrapperElem);
240 var newTemplate = document.getElementById(id);
241 assertTrue('Expected newly loaded template to be in document',
242 !!newTemplate);
243 assertTrue('Expected wrapper to be grandparent of template',
244 newTemplate.parentNode.parentNode == wrapperElem);
246 // Make sure the next template loaded with the same wrapper id re-uses the
247 // wrapper element.
248 var id2 = 'testJstLoadTemplate2';
249 jstLoadTemplate_(document, '<div id="' + id2 + '">content</div>', wrapperId);
250 var newTemplate2 = document.getElementById(id2);
251 assertTrue('Expected newly loaded template to be in document',
252 !!newTemplate2);
253 assertTrue('Expected wrapper to be grandparent of template',
254 newTemplate2.parentNode.parentNode == wrapperElem);
257 function testJstGetTemplateFromDom() {
258 var element;
259 // Get by id a template in the document
260 // Success
261 element = jstGetTemplate('t1');
262 assertTrue("Asserted jstGetTemplate('t1') to return a dom element",
263 !!element);
264 // Failure
265 element = jstGetTemplate('asdf');
266 assertFalse("Asserted jstGetTemplate('asdf') to return null",
267 !!element);
270 function testJstGetTemplateFromFunction() {
271 var element;
272 // Fetch a jstemplate by id from within a html string, passed via a function.
273 function returnHtmlWithId(id) {
274 var html =
275 '<div>' +
276 '<div id="' + id + '">Here is the template</div>' +
277 '</div>';
278 return html;
280 // Success
281 element = jstGetTemplate('template',
282 partial(returnHtmlWithId, 'template'));
283 assertTrue("Expected jstGetTemplate('template') to return a dom element",
284 !!element);
286 // Failure
287 element = jstGetTemplate('asdf',
288 partial(returnHtmlWithId, 'zxcv'));
289 assertFalse("Expected jstGetTemplate('zxcv') to return null",
290 !!element);
293 function testPrepareNode() {
294 var id, node;
295 // Reset the cache so we're testing from a known state.
296 JstProcessor.jstCache_ = {};
297 JstProcessor.jstCache_[0] = {};
299 // Skip pre-processed nodes. Preprocessed nodes are those with a
300 // PROP_jstcache property.
301 var t = document.getElementById('t1');
302 var caches = [];
303 caches.push(JstProcessor.prepareNode_(t));
304 caches.push(JstProcessor.prepareNode_(t));
305 assertEquals('The same cache should be returned on each call to prepareNode',
306 caches[0], caches[1]);
308 // Preprocessing a node with a jst attribute should return a valid struct
309 id = 'testPrepareNodeWithAttributes';
310 jstLoadTemplate_(document, '<div id="' + id + '" jsskip="1"></div>');
311 node = document.getElementById(id);
312 var cache = JstProcessor.prepareNode_(node);
313 try {
314 var jsskip = cache['jsskip']({}, {});
315 } catch (e) {
316 fail('Exception when evaluating jsskip from cache');
318 assertEquals(1, jsskip);
322 function testPrepareNodeWithNoAttributes() {
323 // Preprocessing a node with no jst attributes should return null
324 var id = 'testPrepareNodeNoAttributes';
325 jstLoadTemplate_(document, '<div id="' + id + '"></div>');
326 var node = document.getElementById(id);
327 assertEquals('prepareNode with no jst attributes should return default',
328 JstProcessor.jstcache_[0], JstProcessor.prepareNode_(node));
332 function testJsVars() {
333 var template = document.createElement('div');
334 document.body.appendChild(template);
335 template.innerHTML = '<div jsvars="foo:\'foo\';bar:true;$baz:1"></div>';
337 var context = new JsEvalContext;
338 jstProcess(context, template);
340 assertEquals('foo', context.getVariable('foo'));
341 assertEquals(1, context.getVariable('$baz'));
342 assertTrue(context.getVariable('bar'));
343 assertUndefined(context.getVariable('foobar'));
347 function testCacheReuse() {
348 var template = document.createElement('div');
349 document.body.appendChild(template);
350 template.innerHTML =
351 '<div jsvars="foo:\'foo\';bar:true;$baz:1"></div>' +
352 '<span jsvars="foo:\'foo\';bar:true;$baz:1"></span>';
353 JstProcessor.prepareTemplate_(template);
354 assertEquals(template.firstChild.getAttribute(ATT_jstcache),
355 template.lastChild.getAttribute(ATT_jstcache));