OLD | NEW |
1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file |
2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
4 | 4 |
5 library template_element_test; | 5 library template_element_test; |
6 | 6 |
7 import 'dart:async'; | 7 import 'dart:async'; |
8 import 'dart:collection'; | 8 import 'dart:collection'; |
9 import 'dart:html'; | 9 import 'dart:html'; |
10 import 'dart:math' as math; | 10 import 'dart:math' as math; |
11 import 'package:mdv/mdv.dart' as mdv; | 11 import 'package:mdv/mdv.dart' as mdv; |
12 import 'package:observe/observe.dart'; | 12 import 'package:observe/observe.dart'; |
13 import 'package:unittest/html_config.dart'; | 13 import 'package:unittest/html_config.dart'; |
14 import 'package:unittest/unittest.dart'; | 14 import 'package:unittest/unittest.dart'; |
15 import 'observe_utils.dart'; | 15 import 'mdv_test_utils.dart'; |
16 | 16 |
17 // Note: this file ported from | 17 // Note: this file ported from |
18 // https://github.com/toolkitchen/mdv/blob/master/tests/template_element.js | 18 // https://github.com/toolkitchen/mdv/blob/master/tests/template_element.js |
19 // TODO(jmesserly): submit a small cleanup patch to original. I fixed some | 19 // TODO(jmesserly): submit a small cleanup patch to original. I fixed some |
20 // cases where "div" and "t" were unintentionally using the JS global scope; | 20 // cases where "div" and "t" were unintentionally using the JS global scope; |
21 // look for "assertNodesAre". | 21 // look for "assertNodesAre". |
22 | 22 |
23 main() { | 23 main() { |
24 mdv.initialize(); | 24 mdv.initialize(); |
25 useHtmlConfiguration(); | 25 useHtmlConfiguration(); |
(...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
66 recursivelySetTemplateModel(element, model) { | 66 recursivelySetTemplateModel(element, model) { |
67 for (var node in element.queryAll('*')) { | 67 for (var node in element.queryAll('*')) { |
68 if (node.isTemplate) node.model = model; | 68 if (node.isTemplate) node.model = model; |
69 } | 69 } |
70 } | 70 } |
71 | 71 |
72 dispatchEvent(type, target) { | 72 dispatchEvent(type, target) { |
73 target.dispatchEvent(new Event(type, cancelable: false)); | 73 target.dispatchEvent(new Event(type, cancelable: false)); |
74 } | 74 } |
75 | 75 |
76 var expando = new Expando('test'); | 76 var expando = new Expando('observeTest'); |
77 void addExpandos(node) { | 77 void addExpandos(node) { |
78 while (node != null) { | 78 while (node != null) { |
79 expando[node] = node.text; | 79 expando[node] = node.text; |
80 node = node.nextNode; | 80 node = node.nextNode; |
81 } | 81 } |
82 } | 82 } |
83 | 83 |
84 void checkExpandos(node) { | 84 void checkExpandos(node) { |
85 expect(node, isNotNull); | 85 expect(node, isNotNull); |
86 while (node != null) { | 86 while (node != null) { |
87 expect(expando[node], node.text); | 87 expect(expando[node], node.text); |
88 node = node.nextNode; | 88 node = node.nextNode; |
89 } | 89 } |
90 } | 90 } |
91 | 91 |
92 test('Template', () { | 92 observeTest('Template', () { |
93 var div = createTestHtml('<template bind={{}}>text</template>'); | 93 var div = createTestHtml('<template bind={{}}>text</template>'); |
94 recursivelySetTemplateModel(div, null); | 94 recursivelySetTemplateModel(div, null); |
95 deliverChangeRecords(); | 95 performMicrotaskCheckpoint(); |
96 expect(div.nodes.length, 2); | 96 expect(div.nodes.length, 2); |
97 expect(div.nodes.last.text, 'text'); | 97 expect(div.nodes.last.text, 'text'); |
98 }); | 98 }); |
99 | 99 |
100 test('Template bind, no parent', () { | 100 observeTest('Template bind, no parent', () { |
101 var div = createTestHtml('<template bind>text</template>'); | 101 var div = createTestHtml('<template bind>text</template>'); |
102 var template = div.nodes[0]; | 102 var template = div.nodes[0]; |
103 template.remove(); | 103 template.remove(); |
104 | 104 |
105 recursivelySetTemplateModel(template, toSymbolMap({})); | 105 recursivelySetTemplateModel(template, toSymbolMap({})); |
106 deliverChangeRecords(); | 106 performMicrotaskCheckpoint(); |
107 expect(template.nodes.length, 0); | 107 expect(template.nodes.length, 0); |
108 expect(template.nextNode, null); | 108 expect(template.nextNode, null); |
109 // TODO(jmesserly): the JS tests assert that observer callbacks had no | 109 // TODO(jmesserly): the JS tests assert that observer callbacks had no |
110 // exceptions. How do we replicate this? | 110 // exceptions. How do we replicate this? |
111 }); | 111 }); |
112 | 112 |
113 test('Template bind, no defaultView', () { | 113 observeTest('Template bind, no defaultView', () { |
114 var div = createTestHtml('<template bind>text</template>'); | 114 var div = createTestHtml('<template bind>text</template>'); |
115 var template = div.nodes[0]; | 115 var template = div.nodes[0]; |
116 var doc = document.implementation.createHtmlDocument(''); | 116 var doc = document.implementation.createHtmlDocument(''); |
117 doc.adoptNode(div); | 117 doc.adoptNode(div); |
118 recursivelySetTemplateModel(template, toSymbolMap({})); | 118 recursivelySetTemplateModel(template, toSymbolMap({})); |
119 deliverChangeRecords(); | 119 performMicrotaskCheckpoint(); |
120 expect(div.nodes.length, 1); | 120 expect(div.nodes.length, 1); |
121 // TODO(jmesserly): the JS tests assert that observer callbacks had no | 121 // TODO(jmesserly): the JS tests assert that observer callbacks had no |
122 // exceptions. How do we replicate this? | 122 // exceptions. How do we replicate this? |
123 }); | 123 }); |
124 | 124 |
125 test('Template-Empty Bind', () { | 125 observeTest('Template-Empty Bind', () { |
126 var div = createTestHtml('<template bind>text</template>'); | 126 var div = createTestHtml('<template bind>text</template>'); |
127 recursivelySetTemplateModel(div, null); | 127 recursivelySetTemplateModel(div, null); |
128 deliverChangeRecords(); | 128 performMicrotaskCheckpoint(); |
129 expect(div.nodes.length, 2); | 129 expect(div.nodes.length, 2); |
130 expect(div.nodes.last.text, 'text'); | 130 expect(div.nodes.last.text, 'text'); |
131 }); | 131 }); |
132 | 132 |
133 test('TextTemplateWithNullStringBinding', () { | 133 observeTest('TextTemplateWithNullStringBinding', () { |
134 var div = createTestHtml('<template bind={{}}>a{{b}}c</template>'); | 134 var div = createTestHtml('<template bind={{}}>a{{b}}c</template>'); |
135 var model = toSymbolMap({'b': 'B'}); | 135 var model = toSymbolMap({'b': 'B'}); |
136 recursivelySetTemplateModel(div, model); | 136 recursivelySetTemplateModel(div, model); |
137 | 137 |
138 deliverChanges(model); | 138 deliverChanges(model); |
139 expect(div.nodes.length, 2); | 139 expect(div.nodes.length, 2); |
140 expect(div.nodes.last.text, 'aBc'); | 140 expect(div.nodes.last.text, 'aBc'); |
141 | 141 |
142 model[sym('b')] = 'b'; | 142 model[sym('b')] = 'b'; |
143 deliverChanges(model); | 143 deliverChanges(model); |
144 expect(div.nodes.last.text, 'abc'); | 144 expect(div.nodes.last.text, 'abc'); |
145 | 145 |
146 model[sym('b')] = null; | 146 model[sym('b')] = null; |
147 deliverChanges(model); | 147 deliverChanges(model); |
148 expect(div.nodes.last.text, 'ac'); | 148 expect(div.nodes.last.text, 'ac'); |
149 | 149 |
150 model = null; | 150 model = null; |
151 deliverChanges(model); | 151 deliverChanges(model); |
152 // setting model isn't observable. | 152 // setting model isn't observable. |
153 expect(div.nodes.last.text, 'ac'); | 153 expect(div.nodes.last.text, 'ac'); |
154 }); | 154 }); |
155 | 155 |
156 test('TextTemplateWithBindingPath', () { | 156 observeTest('TextTemplateWithBindingPath', () { |
157 var div = createTestHtml( | 157 var div = createTestHtml( |
158 '<template bind="{{ data }}">a{{b}}c</template>'); | 158 '<template bind="{{ data }}">a{{b}}c</template>'); |
159 var model = toSymbolMap({ 'data': {'b': 'B'} }); | 159 var model = toSymbolMap({ 'data': {'b': 'B'} }); |
160 recursivelySetTemplateModel(div, model); | 160 recursivelySetTemplateModel(div, model); |
161 | 161 |
162 deliverChanges(model); | 162 deliverChanges(model); |
163 expect(div.nodes.length, 2); | 163 expect(div.nodes.length, 2); |
164 expect(div.nodes.last.text, 'aBc'); | 164 expect(div.nodes.last.text, 'aBc'); |
165 | 165 |
166 model[sym('data')][sym('b')] = 'b'; | 166 model[sym('data')][sym('b')] = 'b'; |
167 deliverChanges(model); | 167 deliverChanges(model); |
168 expect(div.nodes.last.text, 'abc'); | 168 expect(div.nodes.last.text, 'abc'); |
169 | 169 |
170 model[sym('data')] = toSymbols({'b': 'X'}); | 170 model[sym('data')] = toSymbols({'b': 'X'}); |
171 deliverChanges(model); | 171 deliverChanges(model); |
172 expect(div.nodes.last.text, 'aXc'); | 172 expect(div.nodes.last.text, 'aXc'); |
173 | 173 |
174 model[sym('data')] = null; | 174 model[sym('data')] = null; |
175 deliverChanges(model); | 175 deliverChanges(model); |
176 expect(div.nodes.last.text, 'ac'); | 176 expect(div.nodes.last.text, 'ac'); |
177 }); | 177 }); |
178 | 178 |
179 test('TextTemplateWithBindingAndConditional', () { | 179 observeTest('TextTemplateWithBindingAndConditional', () { |
180 var div = createTestHtml( | 180 var div = createTestHtml( |
181 '<template bind="{{}}" if="{{ d }}">a{{b}}c</template>'); | 181 '<template bind="{{}}" if="{{ d }}">a{{b}}c</template>'); |
182 var model = toSymbolMap({'b': 'B', 'd': 1}); | 182 var model = toSymbolMap({'b': 'B', 'd': 1}); |
183 recursivelySetTemplateModel(div, model); | 183 recursivelySetTemplateModel(div, model); |
184 | 184 |
185 deliverChanges(model); | 185 deliverChanges(model); |
186 expect(div.nodes.length, 2); | 186 expect(div.nodes.length, 2); |
187 expect(div.nodes.last.text, 'aBc'); | 187 expect(div.nodes.last.text, 'aBc'); |
188 | 188 |
189 model[sym('b')] = 'b'; | 189 model[sym('b')] = 'b'; |
190 deliverChanges(model); | 190 deliverChanges(model); |
191 expect(div.nodes.last.text, 'abc'); | 191 expect(div.nodes.last.text, 'abc'); |
192 | 192 |
193 // TODO(jmesserly): MDV set this to empty string and relies on JS conversion | 193 // TODO(jmesserly): MDV set this to empty string and relies on JS conversion |
194 // rules. Is that intended? | 194 // rules. Is that intended? |
195 // See https://github.com/toolkitchen/mdv/issues/59 | 195 // See https://github.com/toolkitchen/mdv/issues/59 |
196 model[sym('d')] = null; | 196 model[sym('d')] = null; |
197 deliverChanges(model); | 197 deliverChanges(model); |
198 expect(div.nodes.length, 1); | 198 expect(div.nodes.length, 1); |
199 | 199 |
200 model[sym('d')] = 'here'; | 200 model[sym('d')] = 'here'; |
201 model[sym('b')] = 'd'; | 201 model[sym('b')] = 'd'; |
202 | 202 |
203 deliverChanges(model); | 203 deliverChanges(model); |
204 expect(div.nodes.length, 2); | 204 expect(div.nodes.length, 2); |
205 expect(div.nodes.last.text, 'adc'); | 205 expect(div.nodes.last.text, 'adc'); |
206 }); | 206 }); |
207 | 207 |
208 test('TemplateWithTextBinding2', () { | 208 observeTest('TemplateWithTextBinding2', () { |
209 var div = createTestHtml( | 209 var div = createTestHtml( |
210 '<template bind="{{ b }}">a{{value}}c</template>'); | 210 '<template bind="{{ b }}">a{{value}}c</template>'); |
211 expect(div.nodes.length, 1); | 211 expect(div.nodes.length, 1); |
212 var model = toSymbolMap({'b': {'value': 'B'}}); | 212 var model = toSymbolMap({'b': {'value': 'B'}}); |
213 recursivelySetTemplateModel(div, model); | 213 recursivelySetTemplateModel(div, model); |
214 | 214 |
215 deliverChanges(model); | 215 deliverChanges(model); |
216 expect(div.nodes.length, 2); | 216 expect(div.nodes.length, 2); |
217 expect(div.nodes.last.text, 'aBc'); | 217 expect(div.nodes.last.text, 'aBc'); |
218 | 218 |
219 model[sym('b')] = toSymbols({'value': 'b'}); | 219 model[sym('b')] = toSymbols({'value': 'b'}); |
220 deliverChanges(model); | 220 deliverChanges(model); |
221 expect(div.nodes.last.text, 'abc'); | 221 expect(div.nodes.last.text, 'abc'); |
222 }); | 222 }); |
223 | 223 |
224 test('TemplateWithAttributeBinding', () { | 224 observeTest('TemplateWithAttributeBinding', () { |
225 var div = createTestHtml( | 225 var div = createTestHtml( |
226 '<template bind="{{}}">' | 226 '<template bind="{{}}">' |
227 '<div foo="a{{b}}c"></div>' | 227 '<div foo="a{{b}}c"></div>' |
228 '</template>'); | 228 '</template>'); |
229 var model = toSymbolMap({'b': 'B'}); | 229 var model = toSymbolMap({'b': 'B'}); |
230 recursivelySetTemplateModel(div, model); | 230 recursivelySetTemplateModel(div, model); |
231 | 231 |
232 deliverChanges(model); | 232 deliverChanges(model); |
233 expect(div.nodes.length, 2); | 233 expect(div.nodes.length, 2); |
234 expect(div.nodes.last.attributes['foo'], 'aBc'); | 234 expect(div.nodes.last.attributes['foo'], 'aBc'); |
235 | 235 |
236 model[sym('b')] = 'b'; | 236 model[sym('b')] = 'b'; |
237 deliverChanges(model); | 237 deliverChanges(model); |
238 expect(div.nodes.last.attributes['foo'], 'abc'); | 238 expect(div.nodes.last.attributes['foo'], 'abc'); |
239 | 239 |
240 model[sym('b')] = 'X'; | 240 model[sym('b')] = 'X'; |
241 deliverChanges(model); | 241 deliverChanges(model); |
242 expect(div.nodes.last.attributes['foo'], 'aXc'); | 242 expect(div.nodes.last.attributes['foo'], 'aXc'); |
243 }); | 243 }); |
244 | 244 |
245 test('TemplateWithConditionalBinding', () { | 245 observeTest('TemplateWithConditionalBinding', () { |
246 var div = createTestHtml( | 246 var div = createTestHtml( |
247 '<template bind="{{}}">' | 247 '<template bind="{{}}">' |
248 '<div foo?="{{b}}"></div>' | 248 '<div foo?="{{b}}"></div>' |
249 '</template>'); | 249 '</template>'); |
250 var model = toSymbolMap({'b': 'b'}); | 250 var model = toSymbolMap({'b': 'b'}); |
251 recursivelySetTemplateModel(div, model); | 251 recursivelySetTemplateModel(div, model); |
252 | 252 |
253 deliverChanges(model); | 253 deliverChanges(model); |
254 expect(div.nodes.length, 2); | 254 expect(div.nodes.length, 2); |
255 expect(div.nodes.last.attributes['foo'], ''); | 255 expect(div.nodes.last.attributes['foo'], ''); |
256 expect(div.nodes.last.attributes, isNot(contains('foo?'))); | 256 expect(div.nodes.last.attributes, isNot(contains('foo?'))); |
257 | 257 |
258 model[sym('b')] = null; | 258 model[sym('b')] = null; |
259 deliverChanges(model); | 259 deliverChanges(model); |
260 expect(div.nodes.last.attributes, isNot(contains('foo'))); | 260 expect(div.nodes.last.attributes, isNot(contains('foo'))); |
261 }); | 261 }); |
262 | 262 |
263 test('Repeat', () { | 263 observeTest('Repeat', () { |
264 var div = createTestHtml( | 264 var div = createTestHtml( |
265 '<template repeat="{{}}"">text</template>'); | 265 '<template repeat="{{}}"">text</template>'); |
266 | 266 |
267 var model = toSymbols([0, 1, 2]); | 267 var model = toSymbols([0, 1, 2]); |
268 recursivelySetTemplateModel(div, model); | 268 recursivelySetTemplateModel(div, model); |
269 | 269 |
270 deliverChanges(model); | 270 deliverChanges(model); |
271 expect(div.nodes.length, 4); | 271 expect(div.nodes.length, 4); |
272 | 272 |
273 model.length = 1; | 273 model.length = 1; |
274 deliverChanges(model); | 274 deliverChanges(model); |
275 expect(div.nodes.length, 2); | 275 expect(div.nodes.length, 2); |
276 | 276 |
277 model.addAll(toSymbols([3, 4])); | 277 model.addAll(toSymbols([3, 4])); |
278 deliverChanges(model); | 278 deliverChanges(model); |
279 expect(div.nodes.length, 4); | 279 expect(div.nodes.length, 4); |
280 | 280 |
281 model.removeRange(1, 2); | 281 model.removeRange(1, 2); |
282 deliverChanges(model); | 282 deliverChanges(model); |
283 expect(div.nodes.length, 3); | 283 expect(div.nodes.length, 3); |
284 }); | 284 }); |
285 | 285 |
286 test('Repeat - Reuse Instances', () { | 286 observeTest('Repeat - Reuse Instances', () { |
287 var div = createTestHtml('<template repeat>{{ val }}</template>'); | 287 var div = createTestHtml('<template repeat>{{ val }}</template>'); |
288 | 288 |
289 var model = toSymbols([ | 289 var model = toSymbols([ |
290 {'val': 10}, | 290 {'val': 10}, |
291 {'val': 5}, | 291 {'val': 5}, |
292 {'val': 2}, | 292 {'val': 2}, |
293 {'val': 8}, | 293 {'val': 8}, |
294 {'val': 1} | 294 {'val': 1} |
295 ]); | 295 ]); |
296 recursivelySetTemplateModel(div, model); | 296 recursivelySetTemplateModel(div, model); |
(...skipping 20 matching lines...) Expand all Loading... |
317 } | 317 } |
318 | 318 |
319 deliverChanges(model); | 319 deliverChanges(model); |
320 expect(div.nodes[1].text, "11"); | 320 expect(div.nodes[1].text, "11"); |
321 expect(div.nodes[2].text, "9"); | 321 expect(div.nodes[2].text, "9"); |
322 expect(div.nodes[3].text, "6"); | 322 expect(div.nodes[3].text, "6"); |
323 expect(div.nodes[4].text, "3"); | 323 expect(div.nodes[4].text, "3"); |
324 expect(div.nodes[5].text, "2"); | 324 expect(div.nodes[5].text, "2"); |
325 }); | 325 }); |
326 | 326 |
327 test('Bind - Reuse Instance', () { | 327 observeTest('Bind - Reuse Instance', () { |
328 var div = createTestHtml( | 328 var div = createTestHtml( |
329 '<template bind="{{ foo }}">{{ bar }}</template>'); | 329 '<template bind="{{ foo }}">{{ bar }}</template>'); |
330 | 330 |
331 var model = toObservable({ 'foo': { 'bar': 5 }}); | 331 var model = toObservable({ 'foo': { 'bar': 5 }}); |
332 recursivelySetTemplateModel(div, model); | 332 recursivelySetTemplateModel(div, model); |
333 | 333 |
334 deliverChanges(model); | 334 deliverChanges(model); |
335 expect(div.nodes.length, 2); | 335 expect(div.nodes.length, 2); |
336 var template = div.firstChild; | 336 var template = div.firstChild; |
337 | 337 |
338 addExpandos(template.nextNode); | 338 addExpandos(template.nextNode); |
339 checkExpandos(template.nextNode); | 339 checkExpandos(template.nextNode); |
340 | 340 |
341 model = toObservable({'foo': model['foo']}); | 341 model = toObservable({'foo': model['foo']}); |
342 recursivelySetTemplateModel(div, model); | 342 recursivelySetTemplateModel(div, model); |
343 deliverChanges(model); | 343 deliverChanges(model); |
344 checkExpandos(template.nextNode); | 344 checkExpandos(template.nextNode); |
345 }); | 345 }); |
346 | 346 |
347 test('Repeat-Empty', () { | 347 observeTest('Repeat-Empty', () { |
348 var div = createTestHtml( | 348 var div = createTestHtml( |
349 '<template repeat>text</template>'); | 349 '<template repeat>text</template>'); |
350 | 350 |
351 var model = toSymbols([0, 1, 2]); | 351 var model = toSymbols([0, 1, 2]); |
352 recursivelySetTemplateModel(div, model); | 352 recursivelySetTemplateModel(div, model); |
353 | 353 |
354 deliverChanges(model); | 354 deliverChanges(model); |
355 expect(div.nodes.length, 4); | 355 expect(div.nodes.length, 4); |
356 | 356 |
357 model.length = 1; | 357 model.length = 1; |
358 deliverChanges(model); | 358 deliverChanges(model); |
359 expect(div.nodes.length, 2); | 359 expect(div.nodes.length, 2); |
360 | 360 |
361 model.addAll(toSymbols([3, 4])); | 361 model.addAll(toSymbols([3, 4])); |
362 deliverChanges(model); | 362 deliverChanges(model); |
363 expect(div.nodes.length, 4); | 363 expect(div.nodes.length, 4); |
364 | 364 |
365 model.removeRange(1, 2); | 365 model.removeRange(1, 2); |
366 deliverChanges(model); | 366 deliverChanges(model); |
367 expect(div.nodes.length, 3); | 367 expect(div.nodes.length, 3); |
368 }); | 368 }); |
369 | 369 |
370 test('Removal from iteration needs to unbind', () { | 370 observeTest('Removal from iteration needs to unbind', () { |
371 var div = createTestHtml( | 371 var div = createTestHtml( |
372 '<template repeat="{{}}"><a>{{v}}</a></template>'); | 372 '<template repeat="{{}}"><a>{{v}}</a></template>'); |
373 var model = toSymbols([{'v': 0}, {'v': 1}, {'v': 2}, {'v': 3}, {'v': 4}]); | 373 var model = toSymbols([{'v': 0}, {'v': 1}, {'v': 2}, {'v': 3}, {'v': 4}]); |
374 recursivelySetTemplateModel(div, model); | 374 recursivelySetTemplateModel(div, model); |
375 deliverChanges(model); | 375 deliverChanges(model); |
376 | 376 |
377 var nodes = div.nodes.skip(1).toList(); | 377 var nodes = div.nodes.skip(1).toList(); |
378 var vs = model.toList(); | 378 var vs = model.toList(); |
379 | 379 |
380 for (var i = 0; i < 5; i++) { | 380 for (var i = 0; i < 5; i++) { |
381 expect(nodes[i].text, '$i'); | 381 expect(nodes[i].text, '$i'); |
382 } | 382 } |
383 | 383 |
384 model.length = 3; | 384 model.length = 3; |
385 deliverChanges(model); | 385 deliverChanges(model); |
386 for (var i = 0; i < 5; i++) { | 386 for (var i = 0; i < 5; i++) { |
387 expect(nodes[i].text, '$i'); | 387 expect(nodes[i].text, '$i'); |
388 } | 388 } |
389 | 389 |
390 vs[3][sym('v')] = 33; | 390 vs[3][sym('v')] = 33; |
391 vs[4][sym('v')] = 44; | 391 vs[4][sym('v')] = 44; |
392 deliverChanges(model); | 392 deliverChanges(model); |
393 for (var i = 0; i < 5; i++) { | 393 for (var i = 0; i < 5; i++) { |
394 expect(nodes[i].text, '$i'); | 394 expect(nodes[i].text, '$i'); |
395 } | 395 } |
396 }); | 396 }); |
397 | 397 |
398 test('DOM Stability on Iteration', () { | 398 observeTest('DOM Stability on Iteration', () { |
399 var div = createTestHtml( | 399 var div = createTestHtml( |
400 '<template repeat="{{}}">{{}}</template>'); | 400 '<template repeat="{{}}">{{}}</template>'); |
401 var model = toSymbols([1, 2, 3, 4, 5]); | 401 var model = toSymbols([1, 2, 3, 4, 5]); |
402 recursivelySetTemplateModel(div, model); | 402 recursivelySetTemplateModel(div, model); |
403 | 403 |
404 deliverChanges(model); | 404 deliverChanges(model); |
405 | 405 |
406 // Note: the node at index 0 is the <template>. | 406 // Note: the node at index 0 is the <template>. |
407 var nodes = div.nodes.toList(); | 407 var nodes = div.nodes.toList(); |
408 expect(nodes.length, 6, reason: 'list has 5 items'); | 408 expect(nodes.length, 6, reason: 'list has 5 items'); |
(...skipping 28 matching lines...) Expand all Loading... |
437 | 437 |
438 expect(div.nodes.length, 7, reason: 'list has 6 items'); | 438 expect(div.nodes.length, 7, reason: 'list has 6 items'); |
439 expect(identical(div.nodes[1], nodes[1]), true); | 439 expect(identical(div.nodes[1], nodes[1]), true); |
440 expect(identical(div.nodes[2], nodes[2]), true); | 440 expect(identical(div.nodes[2], nodes[2]), true); |
441 expect(nodes.contains(div.nodes[3]), false, reason: '8 is a new node'); | 441 expect(nodes.contains(div.nodes[3]), false, reason: '8 is a new node'); |
442 expect(identical(div.nodes[4], nodes[3]), true); | 442 expect(identical(div.nodes[4], nodes[3]), true); |
443 expect(identical(div.nodes[5], nodes[4]), true); | 443 expect(identical(div.nodes[5], nodes[4]), true); |
444 expect(identical(div.nodes[6], nodes[5]), true); | 444 expect(identical(div.nodes[6], nodes[5]), true); |
445 }); | 445 }); |
446 | 446 |
447 test('Repeat2', () { | 447 observeTest('Repeat2', () { |
448 var div = createTestHtml( | 448 var div = createTestHtml( |
449 '<template repeat="{{}}">{{value}}</template>'); | 449 '<template repeat="{{}}">{{value}}</template>'); |
450 expect(div.nodes.length, 1); | 450 expect(div.nodes.length, 1); |
451 | 451 |
452 var model = toSymbols([ | 452 var model = toSymbols([ |
453 {'value': 0}, | 453 {'value': 0}, |
454 {'value': 1}, | 454 {'value': 1}, |
455 {'value': 2} | 455 {'value': 2} |
456 ]); | 456 ]); |
457 recursivelySetTemplateModel(div, model); | 457 recursivelySetTemplateModel(div, model); |
(...skipping 12 matching lines...) Expand all Loading... |
470 expect(div.nodes[3].text, '2'); | 470 expect(div.nodes[3].text, '2'); |
471 | 471 |
472 model.replaceRange(0, 1, toSymbols([{'value': 'Zero'}])); | 472 model.replaceRange(0, 1, toSymbols([{'value': 'Zero'}])); |
473 deliverChanges(model); | 473 deliverChanges(model); |
474 expect(div.nodes.length, 4); | 474 expect(div.nodes.length, 4); |
475 expect(div.nodes[1].text, 'Zero'); | 475 expect(div.nodes[1].text, 'Zero'); |
476 expect(div.nodes[2].text, 'One'); | 476 expect(div.nodes[2].text, 'One'); |
477 expect(div.nodes[3].text, '2'); | 477 expect(div.nodes[3].text, '2'); |
478 }); | 478 }); |
479 | 479 |
480 test('TemplateWithInputValue', () { | 480 observeTest('TemplateWithInputValue', () { |
481 var div = createTestHtml( | 481 var div = createTestHtml( |
482 '<template bind="{{}}">' | 482 '<template bind="{{}}">' |
483 '<input value="{{x}}">' | 483 '<input value="{{x}}">' |
484 '</template>'); | 484 '</template>'); |
485 var model = toSymbolMap({'x': 'hi'}); | 485 var model = toSymbolMap({'x': 'hi'}); |
486 recursivelySetTemplateModel(div, model); | 486 recursivelySetTemplateModel(div, model); |
487 | 487 |
488 deliverChanges(model); | 488 deliverChanges(model); |
489 expect(div.nodes.length, 2); | 489 expect(div.nodes.length, 2); |
490 expect(div.nodes.last.value, 'hi'); | 490 expect(div.nodes.last.value, 'hi'); |
491 | 491 |
492 model[sym('x')] = 'bye'; | 492 model[sym('x')] = 'bye'; |
493 expect(div.nodes.last.value, 'hi'); | 493 expect(div.nodes.last.value, 'hi'); |
494 deliverChanges(model); | 494 deliverChanges(model); |
495 expect(div.nodes.last.value, 'bye'); | 495 expect(div.nodes.last.value, 'bye'); |
496 | 496 |
497 div.nodes.last.value = 'hello'; | 497 div.nodes.last.value = 'hello'; |
498 dispatchEvent('input', div.nodes.last); | 498 dispatchEvent('input', div.nodes.last); |
499 expect(model[sym('x')], 'hello'); | 499 expect(model[sym('x')], 'hello'); |
500 deliverChanges(model); | 500 deliverChanges(model); |
501 expect(div.nodes.last.value, 'hello'); | 501 expect(div.nodes.last.value, 'hello'); |
502 }); | 502 }); |
503 | 503 |
504 ////////////////////////////////////////////////////////////////////////////// | 504 ////////////////////////////////////////////////////////////////////////////// |
505 | 505 |
506 test('Decorated', () { | 506 observeTest('Decorated', () { |
507 var div = createTestHtml( | 507 var div = createTestHtml( |
508 '<template bind="{{ XX }}" id="t1">' | 508 '<template bind="{{ XX }}" id="t1">' |
509 '<p>Crew member: {{name}}, Job title: {{title}}</p>' | 509 '<p>Crew member: {{name}}, Job title: {{title}}</p>' |
510 '</template>' | 510 '</template>' |
511 '<template bind="{{ XY }}" id="t2" ref="t1"></template>'); | 511 '<template bind="{{ XY }}" id="t2" ref="t1"></template>'); |
512 | 512 |
513 var model = toSymbolMap({ | 513 var model = toSymbolMap({ |
514 'XX': {'name': 'Leela', 'title': 'Captain'}, | 514 'XX': {'name': 'Leela', 'title': 'Captain'}, |
515 'XY': {'name': 'Fry', 'title': 'Delivery boy'}, | 515 'XY': {'name': 'Fry', 'title': 'Delivery boy'}, |
516 'XZ': {'name': 'Zoidberg', 'title': 'Doctor'} | 516 'XZ': {'name': 'Zoidberg', 'title': 'Doctor'} |
(...skipping 10 matching lines...) Expand all Loading... |
527 instance = t2.nextElementSibling; | 527 instance = t2.nextElementSibling; |
528 expect(instance.text, 'Crew member: Fry, Job title: Delivery boy'); | 528 expect(instance.text, 'Crew member: Fry, Job title: Delivery boy'); |
529 | 529 |
530 expect(div.children.length, 4); | 530 expect(div.children.length, 4); |
531 expect(div.nodes.length, 4); | 531 expect(div.nodes.length, 4); |
532 | 532 |
533 expect(div.nodes[1].tagName, 'P'); | 533 expect(div.nodes[1].tagName, 'P'); |
534 expect(div.nodes[3].tagName, 'P'); | 534 expect(div.nodes[3].tagName, 'P'); |
535 }); | 535 }); |
536 | 536 |
537 test('DefaultStyles', () { | 537 observeTest('DefaultStyles', () { |
538 var t = new Element.tag('template'); | 538 var t = new Element.tag('template'); |
539 TemplateElement.decorate(t); | 539 TemplateElement.decorate(t); |
540 | 540 |
541 document.body.append(t); | 541 document.body.append(t); |
542 expect(t.getComputedStyle().display, 'none'); | 542 expect(t.getComputedStyle().display, 'none'); |
543 | 543 |
544 t.remove(); | 544 t.remove(); |
545 }); | 545 }); |
546 | 546 |
547 | 547 |
548 test('Bind', () { | 548 observeTest('Bind', () { |
549 var div = createTestHtml('<template bind="{{}}">Hi {{ name }}</template>'); | 549 var div = createTestHtml('<template bind="{{}}">Hi {{ name }}</template>'); |
550 var model = toSymbolMap({'name': 'Leela'}); | 550 var model = toSymbolMap({'name': 'Leela'}); |
551 recursivelySetTemplateModel(div, model); | 551 recursivelySetTemplateModel(div, model); |
552 | 552 |
553 deliverChanges(model); | 553 deliverChanges(model); |
554 expect(div.nodes[1].text, 'Hi Leela'); | 554 expect(div.nodes[1].text, 'Hi Leela'); |
555 }); | 555 }); |
556 | 556 |
557 test('BindImperative', () { | 557 observeTest('BindImperative', () { |
558 var div = createTestHtml( | 558 var div = createTestHtml( |
559 '<template>' | 559 '<template>' |
560 'Hi {{ name }}' | 560 'Hi {{ name }}' |
561 '</template>'); | 561 '</template>'); |
562 var t = div.nodes.first; | 562 var t = div.nodes.first; |
563 | 563 |
564 var model = toSymbolMap({'name': 'Leela'}); | 564 var model = toSymbolMap({'name': 'Leela'}); |
565 t.bind('bind', model, ''); | 565 t.bind('bind', model, ''); |
566 | 566 |
567 deliverChanges(model); | 567 deliverChanges(model); |
568 expect(div.nodes[1].text, 'Hi Leela'); | 568 expect(div.nodes[1].text, 'Hi Leela'); |
569 }); | 569 }); |
570 | 570 |
571 test('BindPlaceHolderHasNewLine', () { | 571 observeTest('BindPlaceHolderHasNewLine', () { |
572 var div = createTestHtml('<template bind="{{}}">Hi {{\nname\n}}</template>')
; | 572 var div = createTestHtml('<template bind="{{}}">Hi {{\nname\n}}</template>')
; |
573 var model = toSymbolMap({'name': 'Leela'}); | 573 var model = toSymbolMap({'name': 'Leela'}); |
574 recursivelySetTemplateModel(div, model); | 574 recursivelySetTemplateModel(div, model); |
575 | 575 |
576 deliverChanges(model); | 576 deliverChanges(model); |
577 expect(div.nodes[1].text, 'Hi Leela'); | 577 expect(div.nodes[1].text, 'Hi Leela'); |
578 }); | 578 }); |
579 | 579 |
580 test('BindWithRef', () { | 580 observeTest('BindWithRef', () { |
581 var id = 't${new math.Random().nextDouble()}'; | 581 var id = 't${new math.Random().nextDouble()}'; |
582 var div = createTestHtml( | 582 var div = createTestHtml( |
583 '<template id="$id">' | 583 '<template id="$id">' |
584 'Hi {{ name }}' | 584 'Hi {{ name }}' |
585 '</template>' | 585 '</template>' |
586 '<template ref="$id" bind="{{}}"></template>'); | 586 '<template ref="$id" bind="{{}}"></template>'); |
587 | 587 |
588 var t1 = div.nodes.first; | 588 var t1 = div.nodes.first; |
589 var t2 = div.nodes[1]; | 589 var t2 = div.nodes[1]; |
590 | 590 |
591 expect(t2.ref, t1); | 591 expect(t2.ref, t1); |
592 | 592 |
593 var model = toSymbolMap({'name': 'Fry'}); | 593 var model = toSymbolMap({'name': 'Fry'}); |
594 recursivelySetTemplateModel(div, model); | 594 recursivelySetTemplateModel(div, model); |
595 | 595 |
596 deliverChanges(model); | 596 deliverChanges(model); |
597 expect(t2.nextNode.text, 'Hi Fry'); | 597 expect(t2.nextNode.text, 'Hi Fry'); |
598 }); | 598 }); |
599 | 599 |
600 test('BindChanged', () { | 600 observeTest('BindChanged', () { |
601 var model = toSymbolMap({ | 601 var model = toSymbolMap({ |
602 'XX': {'name': 'Leela', 'title': 'Captain'}, | 602 'XX': {'name': 'Leela', 'title': 'Captain'}, |
603 'XY': {'name': 'Fry', 'title': 'Delivery boy'}, | 603 'XY': {'name': 'Fry', 'title': 'Delivery boy'}, |
604 'XZ': {'name': 'Zoidberg', 'title': 'Doctor'} | 604 'XZ': {'name': 'Zoidberg', 'title': 'Doctor'} |
605 }); | 605 }); |
606 | 606 |
607 var div = createTestHtml( | 607 var div = createTestHtml( |
608 '<template bind="{{ XX }}">Hi {{ name }}</template>'); | 608 '<template bind="{{ XX }}">Hi {{ name }}</template>'); |
609 | 609 |
610 recursivelySetTemplateModel(div, model); | 610 recursivelySetTemplateModel(div, model); |
(...skipping 14 matching lines...) Expand all Loading... |
625 assertNodesAre(div, [arguments]) { | 625 assertNodesAre(div, [arguments]) { |
626 var expectedLength = arguments.length; | 626 var expectedLength = arguments.length; |
627 expect(div.nodes.length, expectedLength + 1); | 627 expect(div.nodes.length, expectedLength + 1); |
628 | 628 |
629 for (var i = 0; i < arguments.length; i++) { | 629 for (var i = 0; i < arguments.length; i++) { |
630 var targetNode = div.nodes[i + 1]; | 630 var targetNode = div.nodes[i + 1]; |
631 expect(targetNode.text, arguments[i]); | 631 expect(targetNode.text, arguments[i]); |
632 } | 632 } |
633 } | 633 } |
634 | 634 |
635 test('Repeat3', () { | 635 observeTest('Repeat3', () { |
636 var div = createTestHtml( | 636 var div = createTestHtml( |
637 '<template repeat="{{ contacts }}">Hi {{ name }}</template>'); | 637 '<template repeat="{{ contacts }}">Hi {{ name }}</template>'); |
638 var t = div.nodes.first; | 638 var t = div.nodes.first; |
639 | 639 |
640 var m = toSymbols({ | 640 var m = toSymbols({ |
641 'contacts': [ | 641 'contacts': [ |
642 {'name': 'Raf'}, | 642 {'name': 'Raf'}, |
643 {'name': 'Arv'}, | 643 {'name': 'Arv'}, |
644 {'name': 'Neal'} | 644 {'name': 'Neal'} |
645 ] | 645 ] |
(...skipping 29 matching lines...) Expand all Loading... |
675 | 675 |
676 m[sym('contacts')] = toSymbols([{'name': 'Alex'}]); | 676 m[sym('contacts')] = toSymbols([{'name': 'Alex'}]); |
677 deliverChanges(m); | 677 deliverChanges(m); |
678 assertNodesAre(div, ['Hi Alex']); | 678 assertNodesAre(div, ['Hi Alex']); |
679 | 679 |
680 m[sym('contacts')].length = 0; | 680 m[sym('contacts')].length = 0; |
681 deliverChanges(m); | 681 deliverChanges(m); |
682 assertNodesAre(div, []); | 682 assertNodesAre(div, []); |
683 }); | 683 }); |
684 | 684 |
685 test('RepeatModelSet', () { | 685 observeTest('RepeatModelSet', () { |
686 var div = createTestHtml( | 686 var div = createTestHtml( |
687 '<template repeat="{{ contacts }}">' | 687 '<template repeat="{{ contacts }}">' |
688 'Hi {{ name }}' | 688 'Hi {{ name }}' |
689 '</template>'); | 689 '</template>'); |
690 var m = toSymbols({ | 690 var m = toSymbols({ |
691 'contacts': [ | 691 'contacts': [ |
692 {'name': 'Raf'}, | 692 {'name': 'Raf'}, |
693 {'name': 'Arv'}, | 693 {'name': 'Arv'}, |
694 {'name': 'Neal'} | 694 {'name': 'Neal'} |
695 ] | 695 ] |
696 }); | 696 }); |
697 recursivelySetTemplateModel(div, m); | 697 recursivelySetTemplateModel(div, m); |
698 | 698 |
699 deliverChanges(m); | 699 deliverChanges(m); |
700 var t = div.nodes.first; | 700 var t = div.nodes.first; |
701 | 701 |
702 assertNodesAre(div, ['Hi Raf', 'Hi Arv', 'Hi Neal']); | 702 assertNodesAre(div, ['Hi Raf', 'Hi Arv', 'Hi Neal']); |
703 }); | 703 }); |
704 | 704 |
705 test('RepeatEmptyPath', () { | 705 observeTest('RepeatEmptyPath', () { |
706 var div = createTestHtml('<template repeat="{{}}">Hi {{ name }}</template>')
; | 706 var div = createTestHtml('<template repeat="{{}}">Hi {{ name }}</template>')
; |
707 var t = div.nodes.first; | 707 var t = div.nodes.first; |
708 | 708 |
709 var m = toSymbols([ | 709 var m = toSymbols([ |
710 {'name': 'Raf'}, | 710 {'name': 'Raf'}, |
711 {'name': 'Arv'}, | 711 {'name': 'Arv'}, |
712 {'name': 'Neal'} | 712 {'name': 'Neal'} |
713 ]); | 713 ]); |
714 recursivelySetTemplateModel(div, m); | 714 recursivelySetTemplateModel(div, m); |
715 | 715 |
(...skipping 20 matching lines...) Expand all Loading... |
736 m.replaceRange(0, 1, toSymbols([{'name': 'Tab'}, {'name': 'Neal'}])); | 736 m.replaceRange(0, 1, toSymbols([{'name': 'Tab'}, {'name': 'Neal'}])); |
737 deliverChanges(m); | 737 deliverChanges(m); |
738 assertNodesAre(div, ['Hi Tab', 'Hi Neal', 'Hi Erik', 'Hi Dimitri', 'Hi Alex'
]); | 738 assertNodesAre(div, ['Hi Tab', 'Hi Neal', 'Hi Erik', 'Hi Dimitri', 'Hi Alex'
]); |
739 | 739 |
740 m.length = 0; | 740 m.length = 0; |
741 m.add(toSymbols({'name': 'Alex'})); | 741 m.add(toSymbols({'name': 'Alex'})); |
742 deliverChanges(m); | 742 deliverChanges(m); |
743 assertNodesAre(div, ['Hi Alex']); | 743 assertNodesAre(div, ['Hi Alex']); |
744 }); | 744 }); |
745 | 745 |
746 test('RepeatNullModel', () { | 746 observeTest('RepeatNullModel', () { |
747 var div = createTestHtml('<template repeat="{{}}">Hi {{ name }}</template>')
; | 747 var div = createTestHtml('<template repeat="{{}}">Hi {{ name }}</template>')
; |
748 var t = div.nodes.first; | 748 var t = div.nodes.first; |
749 | 749 |
750 var m = null; | 750 var m = null; |
751 recursivelySetTemplateModel(div, m); | 751 recursivelySetTemplateModel(div, m); |
752 | 752 |
753 expect(div.nodes.length, 1); | 753 expect(div.nodes.length, 1); |
754 | 754 |
755 t.attributes['iterate'] = ''; | 755 t.attributes['iterate'] = ''; |
756 m = toSymbols({}); | 756 m = toSymbols({}); |
757 recursivelySetTemplateModel(div, m); | 757 recursivelySetTemplateModel(div, m); |
758 | 758 |
759 deliverChanges(m); | 759 deliverChanges(m); |
760 expect(div.nodes.length, 1); | 760 expect(div.nodes.length, 1); |
761 }); | 761 }); |
762 | 762 |
763 test('RepeatReuse', () { | 763 observeTest('RepeatReuse', () { |
764 var div = createTestHtml('<template repeat="{{}}">Hi {{ name }}</template>')
; | 764 var div = createTestHtml('<template repeat="{{}}">Hi {{ name }}</template>')
; |
765 var t = div.nodes.first; | 765 var t = div.nodes.first; |
766 | 766 |
767 var m = toSymbols([ | 767 var m = toSymbols([ |
768 {'name': 'Raf'}, | 768 {'name': 'Raf'}, |
769 {'name': 'Arv'}, | 769 {'name': 'Arv'}, |
770 {'name': 'Neal'} | 770 {'name': 'Neal'} |
771 ]); | 771 ]); |
772 recursivelySetTemplateModel(div, m); | 772 recursivelySetTemplateModel(div, m); |
773 deliverChanges(m); | 773 deliverChanges(m); |
(...skipping 12 matching lines...) Expand all Loading... |
786 reason: 'Should not reuse when replacing'); | 786 reason: 'Should not reuse when replacing'); |
787 expect(div.nodes[3], node3, | 787 expect(div.nodes[3], node3, |
788 reason: 'model[2] did not change so the node should not have changed'); | 788 reason: 'model[2] did not change so the node should not have changed'); |
789 | 789 |
790 node2 = div.nodes[2]; | 790 node2 = div.nodes[2]; |
791 m.insert(0, toSymbols({'name': 'Alex'})); | 791 m.insert(0, toSymbols({'name': 'Alex'})); |
792 deliverChanges(m); | 792 deliverChanges(m); |
793 assertNodesAre(div, ['Hi Alex', 'Hi Raf', 'Hi Erik', 'Hi Neal']); | 793 assertNodesAre(div, ['Hi Alex', 'Hi Raf', 'Hi Erik', 'Hi Neal']); |
794 }); | 794 }); |
795 | 795 |
796 test('TwoLevelsDeepBug', () { | 796 observeTest('TwoLevelsDeepBug', () { |
797 var div = createTestHtml( | 797 var div = createTestHtml( |
798 '<template bind="{{}}"><span><span>{{ foo }}</span></span></template>'); | 798 '<template bind="{{}}"><span><span>{{ foo }}</span></span></template>'); |
799 | 799 |
800 var model = toSymbolMap({'foo': 'bar'}); | 800 var model = toSymbolMap({'foo': 'bar'}); |
801 recursivelySetTemplateModel(div, model); | 801 recursivelySetTemplateModel(div, model); |
802 deliverChanges(model); | 802 deliverChanges(model); |
803 | 803 |
804 expect(div.nodes[1].nodes[0].nodes[0].text, 'bar'); | 804 expect(div.nodes[1].nodes[0].nodes[0].text, 'bar'); |
805 }); | 805 }); |
806 | 806 |
807 test('Checked', () { | 807 observeTest('Checked', () { |
808 var div = createTestHtml( | 808 var div = createTestHtml( |
809 '<template>' | 809 '<template>' |
810 '<input type="checkbox" checked="{{a}}">' | 810 '<input type="checkbox" checked="{{a}}">' |
811 '</template>'); | 811 '</template>'); |
812 var t = div.nodes.first; | 812 var t = div.nodes.first; |
813 var m = toSymbols({ | 813 var m = toSymbols({ |
814 'a': true | 814 'a': true |
815 }); | 815 }); |
816 t.bind('bind', m, ''); | 816 t.bind('bind', m, ''); |
817 deliverChanges(m); | 817 deliverChanges(m); |
(...skipping 28 matching lines...) Expand all Loading... |
846 | 846 |
847 m[sym('a')][sym('b')] = 11; | 847 m[sym('a')][sym('b')] = 11; |
848 deliverChanges(m); | 848 deliverChanges(m); |
849 expect(div.nodes[start].text, '11'); | 849 expect(div.nodes[start].text, '11'); |
850 | 850 |
851 m[sym('a')][sym('c')] = toSymbols({'d': 22}); | 851 m[sym('a')][sym('c')] = toSymbols({'d': 22}); |
852 deliverChanges(m); | 852 deliverChanges(m); |
853 expect(div.nodes[start + 2].text, '22'); | 853 expect(div.nodes[start + 2].text, '22'); |
854 } | 854 } |
855 | 855 |
856 test('Nested', () { | 856 observeTest('Nested', () { |
857 nestedHelper( | 857 nestedHelper( |
858 '<template bind="{{a}}">' | 858 '<template bind="{{a}}">' |
859 '{{b}}' | 859 '{{b}}' |
860 '<template bind="{{c}}">' | 860 '<template bind="{{c}}">' |
861 '{{d}}' | 861 '{{d}}' |
862 '</template>' | 862 '</template>' |
863 '</template>', 1); | 863 '</template>', 1); |
864 }); | 864 }); |
865 | 865 |
866 test('NestedWithRef', () { | 866 observeTest('NestedWithRef', () { |
867 nestedHelper( | 867 nestedHelper( |
868 '<template id="inner">{{d}}</template>' | 868 '<template id="inner">{{d}}</template>' |
869 '<template id="outer" bind="{{a}}">' | 869 '<template id="outer" bind="{{a}}">' |
870 '{{b}}' | 870 '{{b}}' |
871 '<template ref="inner" bind="{{c}}"></template>' | 871 '<template ref="inner" bind="{{c}}"></template>' |
872 '</template>', 2); | 872 '</template>', 2); |
873 }); | 873 }); |
874 | 874 |
875 nestedIterateInstantiateHelper(s, start) { | 875 nestedIterateInstantiateHelper(s, start) { |
876 var div = createTestHtml(s); | 876 var div = createTestHtml(s); |
(...skipping 25 matching lines...) Expand all Loading... |
902 m[sym('a')][1] = toSymbols({ | 902 m[sym('a')][1] = toSymbols({ |
903 'b': 3, | 903 'b': 3, |
904 'c': {'d': 33} | 904 'c': {'d': 33} |
905 }); | 905 }); |
906 | 906 |
907 deliverChanges(m); | 907 deliverChanges(m); |
908 expect(div.nodes[start + 3].text, '3'); | 908 expect(div.nodes[start + 3].text, '3'); |
909 expect(div.nodes[start + 5].text, '33'); | 909 expect(div.nodes[start + 5].text, '33'); |
910 } | 910 } |
911 | 911 |
912 test('NestedRepeatBind', () { | 912 observeTest('NestedRepeatBind', () { |
913 nestedIterateInstantiateHelper( | 913 nestedIterateInstantiateHelper( |
914 '<template repeat="{{a}}">' | 914 '<template repeat="{{a}}">' |
915 '{{b}}' | 915 '{{b}}' |
916 '<template bind="{{c}}">' | 916 '<template bind="{{c}}">' |
917 '{{d}}' | 917 '{{d}}' |
918 '</template>' | 918 '</template>' |
919 '</template>', 1); | 919 '</template>', 1); |
920 }); | 920 }); |
921 | 921 |
922 test('NestedRepeatBindWithRef', () { | 922 observeTest('NestedRepeatBindWithRef', () { |
923 nestedIterateInstantiateHelper( | 923 nestedIterateInstantiateHelper( |
924 '<template id="inner">' | 924 '<template id="inner">' |
925 '{{d}}' | 925 '{{d}}' |
926 '</template>' | 926 '</template>' |
927 '<template repeat="{{a}}">' | 927 '<template repeat="{{a}}">' |
928 '{{b}}' | 928 '{{b}}' |
929 '<template ref="inner" bind="{{c}}"></template>' | 929 '<template ref="inner" bind="{{c}}"></template>' |
930 '</template>', 2); | 930 '</template>', 2); |
931 }); | 931 }); |
932 | 932 |
(...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
965 }); | 965 }); |
966 | 966 |
967 i = start + 4; | 967 i = start + 4; |
968 deliverChanges(m); | 968 deliverChanges(m); |
969 expect(div.nodes[start + 4].text, '3'); | 969 expect(div.nodes[start + 4].text, '3'); |
970 expect(div.nodes[start + 6].text, '31'); | 970 expect(div.nodes[start + 6].text, '31'); |
971 expect(div.nodes[start + 7].text, '32'); | 971 expect(div.nodes[start + 7].text, '32'); |
972 expect(div.nodes[start + 8].text, '33'); | 972 expect(div.nodes[start + 8].text, '33'); |
973 } | 973 } |
974 | 974 |
975 test('NestedRepeatBind', () { | 975 observeTest('NestedRepeatBind', () { |
976 nestedIterateIterateHelper( | 976 nestedIterateIterateHelper( |
977 '<template repeat="{{a}}">' | 977 '<template repeat="{{a}}">' |
978 '{{b}}' | 978 '{{b}}' |
979 '<template repeat="{{c}}">' | 979 '<template repeat="{{c}}">' |
980 '{{d}}' | 980 '{{d}}' |
981 '</template>' | 981 '</template>' |
982 '</template>', 1); | 982 '</template>', 1); |
983 }); | 983 }); |
984 | 984 |
985 test('NestedRepeatRepeatWithRef', () { | 985 observeTest('NestedRepeatRepeatWithRef', () { |
986 nestedIterateIterateHelper( | 986 nestedIterateIterateHelper( |
987 '<template id="inner">' | 987 '<template id="inner">' |
988 '{{d}}' | 988 '{{d}}' |
989 '</template>' | 989 '</template>' |
990 '<template repeat="{{a}}">' | 990 '<template repeat="{{a}}">' |
991 '{{b}}' | 991 '{{b}}' |
992 '<template ref="inner" repeat="{{c}}"></template>' | 992 '<template ref="inner" repeat="{{c}}"></template>' |
993 '</template>', 2); | 993 '</template>', 2); |
994 }); | 994 }); |
995 | 995 |
996 test('NestedRepeatSelfRef', () { | 996 observeTest('NestedRepeatSelfRef', () { |
997 var div = createTestHtml( | 997 var div = createTestHtml( |
998 '<template id="t" repeat="{{}}">' | 998 '<template id="t" repeat="{{}}">' |
999 '{{name}}' | 999 '{{name}}' |
1000 '<template ref="t" repeat="{{items}}"></template>' | 1000 '<template ref="t" repeat="{{items}}"></template>' |
1001 '</template>'); | 1001 '</template>'); |
1002 | 1002 |
1003 var m = toSymbols([ | 1003 var m = toSymbols([ |
1004 { | 1004 { |
1005 'name': 'Item 1', | 1005 'name': 'Item 1', |
1006 'items': [ | 1006 'items': [ |
(...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1040 | 1040 |
1041 m[0] = toSymbols({'name': 'Item 1 changed'}); | 1041 m[0] = toSymbols({'name': 'Item 1 changed'}); |
1042 | 1042 |
1043 i = 1; | 1043 i = 1; |
1044 deliverChanges(m); | 1044 deliverChanges(m); |
1045 expect(div.nodes[i++].text, 'Item 1 changed'); | 1045 expect(div.nodes[i++].text, 'Item 1 changed'); |
1046 expect(div.nodes[i++].tagName, 'TEMPLATE'); | 1046 expect(div.nodes[i++].tagName, 'TEMPLATE'); |
1047 expect(div.nodes[i++].text, 'Item 2'); | 1047 expect(div.nodes[i++].text, 'Item 2'); |
1048 }); | 1048 }); |
1049 | 1049 |
1050 test('Attribute Template Option/Optgroup', () { | 1050 observeTest('Attribute Template Option/Optgroup', () { |
1051 var div = createTestHtml( | 1051 var div = createTestHtml( |
1052 '<template bind>' | 1052 '<template bind>' |
1053 '<select selectedIndex="{{ selected }}">' | 1053 '<select selectedIndex="{{ selected }}">' |
1054 '<optgroup template repeat="{{ groups }}" label="{{ name }}">' | 1054 '<optgroup template repeat="{{ groups }}" label="{{ name }}">' |
1055 '<option template repeat="{{ items }}">{{ val }}</option>' | 1055 '<option template repeat="{{ items }}">{{ val }}</option>' |
1056 '</optgroup>' | 1056 '</optgroup>' |
1057 '</select>' | 1057 '</select>' |
1058 '</template>'); | 1058 '</template>'); |
1059 | 1059 |
1060 var m = toSymbols({ | 1060 var m = toSymbols({ |
(...skipping 19 matching lines...) Expand all Loading... |
1080 expect(select.nodes[0].ref.content.nodes[0].tagName, 'OPTGROUP'); | 1080 expect(select.nodes[0].ref.content.nodes[0].tagName, 'OPTGROUP'); |
1081 | 1081 |
1082 var optgroup = select.nodes[1]; | 1082 var optgroup = select.nodes[1]; |
1083 expect(optgroup.nodes[0].tagName, 'TEMPLATE'); | 1083 expect(optgroup.nodes[0].tagName, 'TEMPLATE'); |
1084 expect(optgroup.nodes[1].tagName, 'OPTION'); | 1084 expect(optgroup.nodes[1].tagName, 'OPTION'); |
1085 expect(optgroup.nodes[1].text, '0'); | 1085 expect(optgroup.nodes[1].text, '0'); |
1086 expect(optgroup.nodes[2].tagName, 'OPTION'); | 1086 expect(optgroup.nodes[2].tagName, 'OPTION'); |
1087 expect(optgroup.nodes[2].text, '1'); | 1087 expect(optgroup.nodes[2].text, '1'); |
1088 }); | 1088 }); |
1089 | 1089 |
1090 test('NestedIterateTableMixedSemanticNative', () { | 1090 observeTest('NestedIterateTableMixedSemanticNative', () { |
1091 if (!TemplateElement.supported) return; | 1091 if (!TemplateElement.supported) return; |
1092 | 1092 |
1093 var div = createTestHtml( | 1093 var div = createTestHtml( |
1094 '<table><tbody>' | 1094 '<table><tbody>' |
1095 '<template repeat="{{}}">' | 1095 '<template repeat="{{}}">' |
1096 '<tr>' | 1096 '<tr>' |
1097 '<td template repeat="{{}}" class="{{ val }}">{{ val }}</td>' | 1097 '<td template repeat="{{}}" class="{{ val }}">{{ val }}</td>' |
1098 '</tr>' | 1098 '</tr>' |
1099 '</template>' | 1099 '</template>' |
1100 '</tbody></table>'); | 1100 '</tbody></table>'); |
(...skipping 20 matching lines...) Expand all Loading... |
1121 // 1 for the <td template>, 2 * (1 td) | 1121 // 1 for the <td template>, 2 * (1 td) |
1122 expect(tbody.nodes[2].nodes.length, 3); | 1122 expect(tbody.nodes[2].nodes.length, 3); |
1123 expect(tbody.nodes[2].nodes[1].text, '2'); | 1123 expect(tbody.nodes[2].nodes[1].text, '2'); |
1124 expect(tbody.nodes[2].nodes[2].text, '3'); | 1124 expect(tbody.nodes[2].nodes[2].text, '3'); |
1125 | 1125 |
1126 // Asset the 'class' binding is retained on the semantic template (just | 1126 // Asset the 'class' binding is retained on the semantic template (just |
1127 // check the last one). | 1127 // check the last one). |
1128 expect(tbody.nodes[2].nodes[2].attributes["class"], '3'); | 1128 expect(tbody.nodes[2].nodes[2].attributes["class"], '3'); |
1129 }); | 1129 }); |
1130 | 1130 |
1131 test('NestedIterateTable', () { | 1131 observeTest('NestedIterateTable', () { |
1132 var div = createTestHtml( | 1132 var div = createTestHtml( |
1133 '<table><tbody>' | 1133 '<table><tbody>' |
1134 '<tr template repeat="{{}}">' | 1134 '<tr template repeat="{{}}">' |
1135 '<td template repeat="{{}}" class="{{ val }}">{{ val }}</td>' | 1135 '<td template repeat="{{}}" class="{{ val }}">{{ val }}</td>' |
1136 '</tr>' | 1136 '</tr>' |
1137 '</tbody></table>'); | 1137 '</tbody></table>'); |
1138 | 1138 |
1139 var m = toSymbols([ | 1139 var m = toSymbols([ |
1140 [{ 'val': 0 }, { 'val': 1 }], | 1140 [{ 'val': 0 }, { 'val': 1 }], |
1141 [{ 'val': 2 }, { 'val': 3 }] | 1141 [{ 'val': 2 }, { 'val': 3 }] |
(...skipping 16 matching lines...) Expand all Loading... |
1158 // 1 for the <td template>, 2 * (1 td) | 1158 // 1 for the <td template>, 2 * (1 td) |
1159 expect(tbody.nodes[2].nodes.length, 3); | 1159 expect(tbody.nodes[2].nodes.length, 3); |
1160 expect(tbody.nodes[2].nodes[1].text, '2'); | 1160 expect(tbody.nodes[2].nodes[1].text, '2'); |
1161 expect(tbody.nodes[2].nodes[2].text, '3'); | 1161 expect(tbody.nodes[2].nodes[2].text, '3'); |
1162 | 1162 |
1163 // Asset the 'class' binding is retained on the semantic template (just chec
k | 1163 // Asset the 'class' binding is retained on the semantic template (just chec
k |
1164 // the last one). | 1164 // the last one). |
1165 expect(tbody.nodes[2].nodes[2].attributes['class'], '3'); | 1165 expect(tbody.nodes[2].nodes[2].attributes['class'], '3'); |
1166 }); | 1166 }); |
1167 | 1167 |
1168 test('NestedRepeatDeletionOfMultipleSubTemplates', () { | 1168 observeTest('NestedRepeatDeletionOfMultipleSubTemplates', () { |
1169 var div = createTestHtml( | 1169 var div = createTestHtml( |
1170 '<ul>' | 1170 '<ul>' |
1171 '<template repeat="{{}}" id=t1>' | 1171 '<template repeat="{{}}" id=t1>' |
1172 '<li>{{name}}' | 1172 '<li>{{name}}' |
1173 '<ul>' | 1173 '<ul>' |
1174 '<template ref=t1 repaet="{{items}}"></template>' | 1174 '<template ref=t1 repaet="{{items}}"></template>' |
1175 '</ul>' | 1175 '</ul>' |
1176 '</li>' | 1176 '</li>' |
1177 '</template>' | 1177 '</template>' |
1178 '</ul>'); | 1178 '</ul>'); |
1179 | 1179 |
1180 var m = toSymbols([ | 1180 var m = toSymbols([ |
1181 { | 1181 { |
1182 'name': 'Item 1', | 1182 'name': 'Item 1', |
1183 'items': [ | 1183 'items': [ |
1184 { | 1184 { |
1185 'name': 'Item 1.1' | 1185 'name': 'Item 1.1' |
1186 } | 1186 } |
1187 ] | 1187 ] |
1188 } | 1188 } |
1189 ]); | 1189 ]); |
1190 | 1190 |
1191 recursivelySetTemplateModel(div, m); | 1191 recursivelySetTemplateModel(div, m); |
1192 | 1192 |
1193 deliverChanges(m); | 1193 deliverChanges(m); |
1194 m.removeAt(0); | 1194 m.removeAt(0); |
1195 deliverChanges(m); | 1195 deliverChanges(m); |
1196 }); | 1196 }); |
1197 | 1197 |
1198 test('DeepNested', () { | 1198 observeTest('DeepNested', () { |
1199 var div = createTestHtml( | 1199 var div = createTestHtml( |
1200 '<template bind="{{a}}">' | 1200 '<template bind="{{a}}">' |
1201 '<p>' | 1201 '<p>' |
1202 '<template bind="{{b}}">' | 1202 '<template bind="{{b}}">' |
1203 '{{ c }}' | 1203 '{{ c }}' |
1204 '</template>' | 1204 '</template>' |
1205 '</p>' | 1205 '</p>' |
1206 '</template>'); | 1206 '</template>'); |
1207 | 1207 |
1208 var m = toSymbols({ | 1208 var m = toSymbols({ |
1209 'a': { | 1209 'a': { |
1210 'b': { | 1210 'b': { |
1211 'c': 42 | 1211 'c': 42 |
1212 } | 1212 } |
1213 } | 1213 } |
1214 }); | 1214 }); |
1215 recursivelySetTemplateModel(div, m); | 1215 recursivelySetTemplateModel(div, m); |
1216 deliverChanges(m); | 1216 deliverChanges(m); |
1217 | 1217 |
1218 expect(div.nodes[1].tagName, 'P'); | 1218 expect(div.nodes[1].tagName, 'P'); |
1219 expect(div.nodes[1].nodes.first.tagName, 'TEMPLATE'); | 1219 expect(div.nodes[1].nodes.first.tagName, 'TEMPLATE'); |
1220 expect(div.nodes[1].nodes[1].text, '42'); | 1220 expect(div.nodes[1].nodes[1].text, '42'); |
1221 }); | 1221 }); |
1222 | 1222 |
1223 test('TemplateContentRemoved', () { | 1223 observeTest('TemplateContentRemoved', () { |
1224 var div = createTestHtml('<template bind="{{}}">{{ }}</template>'); | 1224 var div = createTestHtml('<template bind="{{}}">{{ }}</template>'); |
1225 var model = 42; | 1225 var model = 42; |
1226 | 1226 |
1227 recursivelySetTemplateModel(div, model); | 1227 recursivelySetTemplateModel(div, model); |
1228 deliverChanges(model); | 1228 deliverChanges(model); |
1229 expect(div.nodes[1].text, '42'); | 1229 expect(div.nodes[1].text, '42'); |
1230 expect(div.nodes[0].text, ''); | 1230 expect(div.nodes[0].text, ''); |
1231 }); | 1231 }); |
1232 | 1232 |
1233 test('TemplateContentRemovedEmptyArray', () { | 1233 observeTest('TemplateContentRemovedEmptyArray', () { |
1234 var div = createTestHtml('<template iterate>Remove me</template>'); | 1234 var div = createTestHtml('<template iterate>Remove me</template>'); |
1235 var model = toSymbols([]); | 1235 var model = toSymbols([]); |
1236 | 1236 |
1237 recursivelySetTemplateModel(div, model); | 1237 recursivelySetTemplateModel(div, model); |
1238 deliverChanges(model); | 1238 deliverChanges(model); |
1239 expect(div.nodes.length, 1); | 1239 expect(div.nodes.length, 1); |
1240 expect(div.nodes[0].text, ''); | 1240 expect(div.nodes[0].text, ''); |
1241 }); | 1241 }); |
1242 | 1242 |
1243 test('TemplateContentRemovedNested', () { | 1243 observeTest('TemplateContentRemovedNested', () { |
1244 var div = createTestHtml( | 1244 var div = createTestHtml( |
1245 '<template bind="{{}}">' | 1245 '<template bind="{{}}">' |
1246 '{{ a }}' | 1246 '{{ a }}' |
1247 '<template bind="{{}}">' | 1247 '<template bind="{{}}">' |
1248 '{{ b }}' | 1248 '{{ b }}' |
1249 '</template>' | 1249 '</template>' |
1250 '</template>'); | 1250 '</template>'); |
1251 | 1251 |
1252 var model = toSymbolMap({ | 1252 var model = toSymbolMap({ |
1253 'a': 1, | 1253 'a': 1, |
1254 'b': 2 | 1254 'b': 2 |
1255 }); | 1255 }); |
1256 recursivelySetTemplateModel(div, model); | 1256 recursivelySetTemplateModel(div, model); |
1257 deliverChanges(model); | 1257 deliverChanges(model); |
1258 | 1258 |
1259 expect(div.nodes[0].text, ''); | 1259 expect(div.nodes[0].text, ''); |
1260 expect(div.nodes[1].text, '1'); | 1260 expect(div.nodes[1].text, '1'); |
1261 expect(div.nodes[2].text, ''); | 1261 expect(div.nodes[2].text, ''); |
1262 expect(div.nodes[3].text, '2'); | 1262 expect(div.nodes[3].text, '2'); |
1263 }); | 1263 }); |
1264 | 1264 |
1265 test('BindWithUndefinedModel', () { | 1265 observeTest('BindWithUndefinedModel', () { |
1266 var div = createTestHtml( | 1266 var div = createTestHtml( |
1267 '<template bind="{{}}" if="{{}}">{{ a }}</template>'); | 1267 '<template bind="{{}}" if="{{}}">{{ a }}</template>'); |
1268 | 1268 |
1269 var model = toSymbolMap({'a': 42}); | 1269 var model = toSymbolMap({'a': 42}); |
1270 recursivelySetTemplateModel(div, model); | 1270 recursivelySetTemplateModel(div, model); |
1271 deliverChanges(model); | 1271 deliverChanges(model); |
1272 expect(div.nodes[1].text, '42'); | 1272 expect(div.nodes[1].text, '42'); |
1273 | 1273 |
1274 model = null; | 1274 model = null; |
1275 recursivelySetTemplateModel(div, model); | 1275 recursivelySetTemplateModel(div, model); |
1276 deliverChanges(model); | 1276 deliverChanges(model); |
1277 expect(div.nodes.length, 1); | 1277 expect(div.nodes.length, 1); |
1278 | 1278 |
1279 model = toSymbols({'a': 42}); | 1279 model = toSymbols({'a': 42}); |
1280 recursivelySetTemplateModel(div, model); | 1280 recursivelySetTemplateModel(div, model); |
1281 deliverChanges(model); | 1281 deliverChanges(model); |
1282 expect(div.nodes[1].text, '42'); | 1282 expect(div.nodes[1].text, '42'); |
1283 }); | 1283 }); |
1284 | 1284 |
1285 test('BindNested', () { | 1285 observeTest('BindNested', () { |
1286 var div = createTestHtml( | 1286 var div = createTestHtml( |
1287 '<template bind="{{}}">' | 1287 '<template bind="{{}}">' |
1288 'Name: {{ name }}' | 1288 'Name: {{ name }}' |
1289 '<template bind="{{wife}}" if="{{wife}}">' | 1289 '<template bind="{{wife}}" if="{{wife}}">' |
1290 'Wife: {{ name }}' | 1290 'Wife: {{ name }}' |
1291 '</template>' | 1291 '</template>' |
1292 '<template bind="{{child}}" if="{{child}}">' | 1292 '<template bind="{{child}}" if="{{child}}">' |
1293 'Child: {{ name }}' | 1293 'Child: {{ name }}' |
1294 '</template>' | 1294 '</template>' |
1295 '</template>'); | 1295 '</template>'); |
(...skipping 15 matching lines...) Expand all Loading... |
1311 deliverChanges(m); | 1311 deliverChanges(m); |
1312 expect(div.nodes.length, 6); | 1312 expect(div.nodes.length, 6); |
1313 expect(div.nodes[5].text, 'Child: Dwight'); | 1313 expect(div.nodes[5].text, 'Child: Dwight'); |
1314 | 1314 |
1315 m.remove(sym('wife')); | 1315 m.remove(sym('wife')); |
1316 deliverChanges(m); | 1316 deliverChanges(m); |
1317 expect(div.nodes.length, 5); | 1317 expect(div.nodes.length, 5); |
1318 expect(div.nodes[4].text, 'Child: Dwight'); | 1318 expect(div.nodes[4].text, 'Child: Dwight'); |
1319 }); | 1319 }); |
1320 | 1320 |
1321 test('BindRecursive', () { | 1321 observeTest('BindRecursive', () { |
1322 var div = createTestHtml( | 1322 var div = createTestHtml( |
1323 '<template bind="{{}}" if="{{}}" id="t">' | 1323 '<template bind="{{}}" if="{{}}" id="t">' |
1324 'Name: {{ name }}' | 1324 'Name: {{ name }}' |
1325 '<template bind="{{friend}}" if="{{friend}}" ref="t"></template>' | 1325 '<template bind="{{friend}}" if="{{friend}}" ref="t"></template>' |
1326 '</template>'); | 1326 '</template>'); |
1327 | 1327 |
1328 var m = toSymbols({ | 1328 var m = toSymbols({ |
1329 'name': 'Fry', | 1329 'name': 'Fry', |
1330 'friend': { | 1330 'friend': { |
1331 'name': 'Bender' | 1331 'name': 'Bender' |
(...skipping 10 matching lines...) Expand all Loading... |
1342 deliverChanges(m); | 1342 deliverChanges(m); |
1343 expect(div.nodes.length, 7); | 1343 expect(div.nodes.length, 7); |
1344 expect(div.nodes[5].text, 'Name: Leela'); | 1344 expect(div.nodes[5].text, 'Name: Leela'); |
1345 | 1345 |
1346 m[sym('friend')] = toSymbols({'name': 'Leela'}); | 1346 m[sym('friend')] = toSymbols({'name': 'Leela'}); |
1347 deliverChanges(m); | 1347 deliverChanges(m); |
1348 expect(div.nodes.length, 5); | 1348 expect(div.nodes.length, 5); |
1349 expect(div.nodes[3].text, 'Name: Leela'); | 1349 expect(div.nodes[3].text, 'Name: Leela'); |
1350 }); | 1350 }); |
1351 | 1351 |
1352 test('ChangeFromBindToRepeat', () { | 1352 observeTest('ChangeFromBindToRepeat', () { |
1353 var div = createTestHtml( | 1353 var div = createTestHtml( |
1354 '<template bind="{{a}}">' | 1354 '<template bind="{{a}}">' |
1355 '{{ length }}' | 1355 '{{ length }}' |
1356 '</template>'); | 1356 '</template>'); |
1357 var template = div.nodes.first; | 1357 var template = div.nodes.first; |
1358 | 1358 |
1359 // Note: this test data is a little different from the JS version, because | 1359 // Note: this observeTest data is a little different from the JS version, be
cause |
1360 // we allow binding to the "length" field of the Map in preference to | 1360 // we allow binding to the "length" field of the Map in preference to |
1361 // binding keys. | 1361 // binding keys. |
1362 var m = toSymbols({ | 1362 var m = toSymbols({ |
1363 'a': [ | 1363 'a': [ |
1364 [], | 1364 [], |
1365 { 'b': [1,2,3,4] }, | 1365 { 'b': [1,2,3,4] }, |
1366 // Note: this will use the Map "length" property, not the "length" key. | 1366 // Note: this will use the Map "length" property, not the "length" key. |
1367 {'length': 42, 'c': 123} | 1367 {'length': 42, 'c': 123} |
1368 ] | 1368 ] |
1369 }); | 1369 }); |
(...skipping 12 matching lines...) Expand all Loading... |
1382 expect(div.nodes[3].text, '2'); | 1382 expect(div.nodes[3].text, '2'); |
1383 | 1383 |
1384 template.unbind('repeat'); | 1384 template.unbind('repeat'); |
1385 template.bind('bind', m, 'a.1.b'); | 1385 template.bind('bind', m, 'a.1.b'); |
1386 | 1386 |
1387 deliverChanges(m); | 1387 deliverChanges(m); |
1388 expect(div.nodes.length, 2); | 1388 expect(div.nodes.length, 2); |
1389 expect(div.nodes[1].text, '4'); | 1389 expect(div.nodes[1].text, '4'); |
1390 }); | 1390 }); |
1391 | 1391 |
1392 test('ChangeRefId', () { | 1392 observeTest('ChangeRefId', () { |
1393 var div = createTestHtml( | 1393 var div = createTestHtml( |
1394 '<template id="a">a:{{ }}</template>' | 1394 '<template id="a">a:{{ }}</template>' |
1395 '<template id="b">b:{{ }}</template>' | 1395 '<template id="b">b:{{ }}</template>' |
1396 '<template repeat="{{}}">' | 1396 '<template repeat="{{}}">' |
1397 '<template ref="a" bind="{{}}"></template>' | 1397 '<template ref="a" bind="{{}}"></template>' |
1398 '</template>'); | 1398 '</template>'); |
1399 var model = toSymbols([]); | 1399 var model = toSymbols([]); |
1400 recursivelySetTemplateModel(div, model); | 1400 recursivelySetTemplateModel(div, model); |
1401 deliverChanges(model); | 1401 deliverChanges(model); |
1402 | 1402 |
1403 expect(div.nodes.length, 3); | 1403 expect(div.nodes.length, 3); |
1404 | 1404 |
1405 document.getElementById('a').id = 'old-a'; | 1405 document.getElementById('a').id = 'old-a'; |
1406 document.getElementById('b').id = 'a'; | 1406 document.getElementById('b').id = 'a'; |
1407 | 1407 |
1408 model..add(1)..add(2); | 1408 model..add(1)..add(2); |
1409 deliverChanges(model); | 1409 deliverChanges(model); |
1410 | 1410 |
1411 expect(div.nodes.length, 7); | 1411 expect(div.nodes.length, 7); |
1412 expect(div.nodes[4].text, 'b:1'); | 1412 expect(div.nodes[4].text, 'b:1'); |
1413 expect(div.nodes[6].text, 'b:2'); | 1413 expect(div.nodes[6].text, 'b:2'); |
1414 }); | 1414 }); |
1415 | 1415 |
1416 test('Content', () { | 1416 observeTest('Content', () { |
1417 var div = createTestHtml( | 1417 var div = createTestHtml( |
1418 '<template><a></a></template>' | 1418 '<template><a></a></template>' |
1419 '<template><b></b></template>'); | 1419 '<template><b></b></template>'); |
1420 var templateA = div.nodes.first; | 1420 var templateA = div.nodes.first; |
1421 var templateB = div.nodes.last; | 1421 var templateB = div.nodes.last; |
1422 var contentA = templateA.content; | 1422 var contentA = templateA.content; |
1423 var contentB = templateB.content; | 1423 var contentB = templateB.content; |
1424 expect(contentA, isNotNull); | 1424 expect(contentA, isNotNull); |
1425 | 1425 |
1426 expect(templateA.document, isNot(equals(contentA.document))); | 1426 expect(templateA.document, isNot(equals(contentA.document))); |
1427 expect(templateB.document, isNot(equals(contentB.document))); | 1427 expect(templateB.document, isNot(equals(contentB.document))); |
1428 | 1428 |
1429 expect(templateB.document, templateA.document); | 1429 expect(templateB.document, templateA.document); |
1430 expect(contentB.document, contentA.document); | 1430 expect(contentB.document, contentA.document); |
1431 | 1431 |
1432 expect(templateA.document.window, window); | 1432 expect(templateA.document.window, window); |
1433 expect(templateB.document.window, window); | 1433 expect(templateB.document.window, window); |
1434 | 1434 |
1435 expect(contentA.document.window, null); | 1435 expect(contentA.document.window, null); |
1436 expect(contentB.document.window, null); | 1436 expect(contentB.document.window, null); |
1437 | 1437 |
1438 expect(contentA.nodes.last, contentA.nodes.first); | 1438 expect(contentA.nodes.last, contentA.nodes.first); |
1439 expect(contentA.nodes.first.tagName, 'A'); | 1439 expect(contentA.nodes.first.tagName, 'A'); |
1440 | 1440 |
1441 expect(contentB.nodes.last, contentB.nodes.first); | 1441 expect(contentB.nodes.last, contentB.nodes.first); |
1442 expect(contentB.nodes.first.tagName, 'B'); | 1442 expect(contentB.nodes.first.tagName, 'B'); |
1443 }); | 1443 }); |
1444 | 1444 |
1445 test('NestedContent', () { | 1445 observeTest('NestedContent', () { |
1446 var div = createTestHtml( | 1446 var div = createTestHtml( |
1447 '<template>' | 1447 '<template>' |
1448 '<template></template>' | 1448 '<template></template>' |
1449 '</template>'); | 1449 '</template>'); |
1450 var templateA = div.nodes.first; | 1450 var templateA = div.nodes.first; |
1451 var templateB = templateA.content.nodes.first; | 1451 var templateB = templateA.content.nodes.first; |
1452 | 1452 |
1453 expect(templateB.document, templateA.content.document); | 1453 expect(templateB.document, templateA.content.document); |
1454 expect(templateB.content.document, templateA.content.document); | 1454 expect(templateB.content.document, templateA.content.document); |
1455 }); | 1455 }); |
1456 | 1456 |
1457 test('BindShadowDOM', () { | 1457 observeTest('BindShadowDOM', () { |
1458 if (ShadowRoot.supported) { | 1458 if (ShadowRoot.supported) { |
1459 var root = createShadowTestHtml( | 1459 var root = createShadowTestHtml( |
1460 '<template bind="{{}}">Hi {{ name }}</template>'); | 1460 '<template bind="{{}}">Hi {{ name }}</template>'); |
1461 var model = toSymbolMap({'name': 'Leela'}); | 1461 var model = toSymbolMap({'name': 'Leela'}); |
1462 recursivelySetTemplateModel(root, model); | 1462 recursivelySetTemplateModel(root, model); |
1463 deliverChanges(model); | 1463 deliverChanges(model); |
1464 expect(root.nodes[1].text, 'Hi Leela'); | 1464 expect(root.nodes[1].text, 'Hi Leela'); |
1465 } | 1465 } |
1466 }); | 1466 }); |
1467 | 1467 |
1468 test('BindShadowDOM bindModel', () { | 1468 observeTest('BindShadowDOM bindModel', () { |
1469 if (ShadowRoot.supported) { | 1469 if (ShadowRoot.supported) { |
1470 var root = createShadowTestHtml('Hi {{ name }}'); | 1470 var root = createShadowTestHtml('Hi {{ name }}'); |
1471 var model = toSymbolMap({'name': 'Leela'}); | 1471 var model = toSymbolMap({'name': 'Leela'}); |
1472 mdv.bindModel(root, model); | 1472 mdv.bindModel(root, model); |
1473 deliverChangeRecords(); | 1473 performMicrotaskCheckpoint(); |
1474 expect(root.text, 'Hi Leela'); | 1474 expect(root.text, 'Hi Leela'); |
1475 } | 1475 } |
1476 }); | 1476 }); |
1477 | 1477 |
1478 test('bindModel to polyfilled shadow root', () { | 1478 observeTest('bindModel to polyfilled shadow root', () { |
1479 var root = createTestHtml('Hi {{ name }}'); | 1479 var root = createTestHtml('Hi {{ name }}'); |
1480 var model = toSymbolMap({'name': 'Leela'}); | 1480 var model = toSymbolMap({'name': 'Leela'}); |
1481 mdv.bindModel(root, model); | 1481 mdv.bindModel(root, model); |
1482 deliverChangeRecords(); | 1482 performMicrotaskCheckpoint(); |
1483 expect(root.text, 'Hi Leela'); | 1483 expect(root.text, 'Hi Leela'); |
1484 }); | 1484 }); |
1485 | 1485 |
1486 test('BindShadowDOM Template Ref', () { | 1486 observeTest('BindShadowDOM Template Ref', () { |
1487 if (ShadowRoot.supported) { | 1487 if (ShadowRoot.supported) { |
1488 var root = createShadowTestHtml( | 1488 var root = createShadowTestHtml( |
1489 '<template id=foo>Hi</template><template bind ref=foo></template>'); | 1489 '<template id=foo>Hi</template><template bind ref=foo></template>'); |
1490 recursivelySetTemplateModel(root, toSymbolMap({})); | 1490 recursivelySetTemplateModel(root, toSymbolMap({})); |
1491 deliverChangeRecords(); | 1491 performMicrotaskCheckpoint(); |
1492 expect(root.nodes.length, 3); | 1492 expect(root.nodes.length, 3); |
1493 } | 1493 } |
1494 }); | 1494 }); |
1495 | 1495 |
1496 // https://github.com/toolkitchen/mdv/issues/8 | 1496 // https://github.com/toolkitchen/mdv/issues/8 |
1497 test('UnbindingInNestedBind', () { | 1497 observeTest('UnbindingInNestedBind', () { |
1498 var div = createTestHtml( | 1498 var div = createTestHtml( |
1499 '<template bind="{{outer}}" if="{{outer}}" syntax="testHelper">' | 1499 '<template bind="{{outer}}" if="{{outer}}" syntax="testHelper">' |
1500 '<template bind="{{inner}}" if="{{inner}}">' | 1500 '<template bind="{{inner}}" if="{{inner}}">' |
1501 '{{ age }}' | 1501 '{{ age }}' |
1502 '</template>' | 1502 '</template>' |
1503 '</template>'); | 1503 '</template>'); |
1504 | 1504 |
1505 var syntax = new UnbindingInNestedBindSyntax(); | 1505 var syntax = new UnbindingInNestedBindSyntax(); |
1506 TemplateElement.syntax['testHelper'] = syntax; | 1506 TemplateElement.syntax['testHelper'] = syntax; |
1507 try { | 1507 try { |
(...skipping 20 matching lines...) Expand all Loading... |
1528 syntax.expectedAge = 2; | 1528 syntax.expectedAge = 2; |
1529 | 1529 |
1530 deliverChanges(model); | 1530 deliverChanges(model); |
1531 expect(syntax.count, 2); | 1531 expect(syntax.count, 2); |
1532 } finally { | 1532 } finally { |
1533 TemplateElement.syntax.remove('testHelper'); | 1533 TemplateElement.syntax.remove('testHelper'); |
1534 } | 1534 } |
1535 }); | 1535 }); |
1536 | 1536 |
1537 // https://github.com/toolkitchen/mdv/issues/8 | 1537 // https://github.com/toolkitchen/mdv/issues/8 |
1538 test('DontCreateInstancesForAbandonedIterators', () { | 1538 observeTest('DontCreateInstancesForAbandonedIterators', () { |
1539 var div = createTestHtml( | 1539 var div = createTestHtml( |
1540 '<template bind="{{}} {{}}">' | 1540 '<template bind="{{}} {{}}">' |
1541 '<template bind="{{}}">Foo' | 1541 '<template bind="{{}}">Foo' |
1542 '</template>' | 1542 '</template>' |
1543 '</template>'); | 1543 '</template>'); |
1544 recursivelySetTemplateModel(div, null); | 1544 recursivelySetTemplateModel(div, null); |
1545 // TODO(jmesserly): how to fix this test? | 1545 // TODO(jmesserly): how to fix this observeTest? |
1546 // Perhaps revive the original? | 1546 // Perhaps revive the original? |
1547 // https://github.com/toolkitchen/mdv/commit/8bc1e3466aeb6930150c0d3148f0e83
0184bf599#L3R1278 | 1547 // https://github.com/toolkitchen/mdv/commit/8bc1e3466aeb6930150c0d3148f0e83
0184bf599#L3R1278 |
1548 //expect(!!ChangeSummary._errorThrownDuringCallback, false); | 1548 //expect(!!ChangeSummary._errorThrownDuringCallback, false); |
1549 }); | 1549 }); |
1550 | 1550 |
1551 test('CreateInstance', () { | 1551 observeTest('CreateInstance', () { |
1552 TemplateElement.syntax['Test'] = new TestBindingSyntax(); | 1552 TemplateElement.syntax['Test'] = new TestBindingSyntax(); |
1553 try { | 1553 try { |
1554 var div = createTestHtml( | 1554 var div = createTestHtml( |
1555 '<template bind="{{a}}">' | 1555 '<template bind="{{a}}">' |
1556 '<template bind="{{b}}">' | 1556 '<template bind="{{b}}">' |
1557 '{{ foo }}:{{ replaceme }}' | 1557 '{{ foo }}:{{ replaceme }}' |
1558 '</template>' | 1558 '</template>' |
1559 '</template>'); | 1559 '</template>'); |
1560 var outer = div.nodes.first; | 1560 var outer = div.nodes.first; |
1561 var model = toSymbolMap({'b': {'foo': 'bar'}}); | 1561 var model = toSymbolMap({'b': {'foo': 'bar'}}); |
1562 | 1562 |
1563 var host = new DivElement(); | 1563 var host = new DivElement(); |
1564 var instance = outer.createInstance(model, 'Test'); | 1564 var instance = outer.createInstance(model, 'Test'); |
1565 expect(outer.content.nodes.first, instance.nodes.first.ref); | 1565 expect(outer.content.nodes.first, instance.nodes.first.ref); |
1566 expect(instance.firstChild.attributes['syntax'], 'Test'); | 1566 expect(instance.firstChild.attributes['syntax'], 'Test'); |
1567 | 1567 |
1568 host.append(instance); | 1568 host.append(instance); |
1569 deliverChangeRecords(); | 1569 performMicrotaskCheckpoint(); |
1570 expect(host.firstChild.nextNode.text, 'bar:replaced'); | 1570 expect(host.firstChild.nextNode.text, 'bar:replaced'); |
1571 } finally { | 1571 } finally { |
1572 TemplateElement.syntax.remove('Test'); | 1572 TemplateElement.syntax.remove('Test'); |
1573 } | 1573 } |
1574 }); | 1574 }); |
1575 | 1575 |
1576 test('Bootstrap', () { | 1576 observeTest('Bootstrap', () { |
1577 var div = new DivElement(); | 1577 var div = new DivElement(); |
1578 div.innerHtml = | 1578 div.innerHtml = |
1579 '<template>' | 1579 '<template>' |
1580 '<div></div>' | 1580 '<div></div>' |
1581 '<template>' | 1581 '<template>' |
1582 'Hello' | 1582 'Hello' |
1583 '</template>' | 1583 '</template>' |
1584 '</template>'; | 1584 '</template>'; |
1585 | 1585 |
1586 TemplateElement.bootstrap(div); | 1586 TemplateElement.bootstrap(div); |
(...skipping 13 matching lines...) Expand all Loading... |
1600 '</template>'; | 1600 '</template>'; |
1601 | 1601 |
1602 TemplateElement.bootstrap(template); | 1602 TemplateElement.bootstrap(template); |
1603 template2 = template.content.nodes.first; | 1603 template2 = template.content.nodes.first; |
1604 expect(template2.content.nodes.length, 2); | 1604 expect(template2.content.nodes.length, 2); |
1605 var template3 = template2.content.nodes.first.nextNode; | 1605 var template3 = template2.content.nodes.first.nextNode; |
1606 expect(template3.content.nodes.length, 1); | 1606 expect(template3.content.nodes.length, 1); |
1607 expect(template3.content.nodes.first.text, 'Hello'); | 1607 expect(template3.content.nodes.first.text, 'Hello'); |
1608 }); | 1608 }); |
1609 | 1609 |
1610 test('instanceCreated hack', () { | 1610 observeTest('instanceCreated hack', () { |
1611 var called = false; | 1611 var called = false; |
1612 var sub = mdv.instanceCreated.listen((node) { | 1612 var sub = mdv.instanceCreated.listen((node) { |
1613 called = true; | 1613 called = true; |
1614 expect(node.nodeType, Node.DOCUMENT_FRAGMENT_NODE); | 1614 expect(node.nodeType, Node.DOCUMENT_FRAGMENT_NODE); |
1615 }); | 1615 }); |
1616 | 1616 |
1617 var div = createTestHtml('<template bind="{{}}">Foo</template>'); | 1617 var div = createTestHtml('<template bind="{{}}">Foo</template>'); |
1618 expect(called, false); | 1618 expect(called, false); |
1619 | 1619 |
1620 recursivelySetTemplateModel(div, null); | 1620 recursivelySetTemplateModel(div, null); |
1621 deliverChangeRecords(); | 1621 performMicrotaskCheckpoint(); |
1622 expect(called, true); | 1622 expect(called, true); |
1623 | 1623 |
1624 sub.cancel(); | 1624 sub.cancel(); |
1625 }); | 1625 }); |
1626 } | 1626 } |
1627 | 1627 |
1628 class TestBindingSyntax extends CustomBindingSyntax { | 1628 class TestBindingSyntax extends CustomBindingSyntax { |
1629 getBinding(model, String path, name, node) { | 1629 getBinding(model, String path, name, node) { |
1630 if (path.trim() == 'replaceme') return new ObservableBox('replaced'); | 1630 if (path.trim() == 'replaceme') return new ObservableBox('replaced'); |
1631 return null; | 1631 return null; |
1632 } | 1632 } |
1633 } | 1633 } |
1634 | 1634 |
1635 class UnbindingInNestedBindSyntax extends CustomBindingSyntax { | 1635 class UnbindingInNestedBindSyntax extends CustomBindingSyntax { |
1636 int expectedAge = 42; | 1636 int expectedAge = 42; |
1637 int count = 0; | 1637 int count = 0; |
1638 | 1638 |
1639 getBinding(model, path, name, node) { | 1639 getBinding(model, path, name, node) { |
1640 if (name != 'text' || path != 'age') | 1640 if (name != 'text' || path != 'age') |
1641 return; | 1641 return; |
1642 | 1642 |
1643 expect(model[sym('age')], expectedAge); | 1643 expect(model[sym('age')], expectedAge); |
1644 count++; | 1644 count++; |
1645 } | 1645 } |
1646 } | 1646 } |
1647 | 1647 |
1648 /** Verifies that the model is Observable, then calls [deliverChangeRecords]. */ | 1648 /** Verifies that the model is Observable, then calls [performMicrotaskCheckpoin
t]. */ |
1649 void deliverChanges(model) { | 1649 void deliverChanges(model) { |
1650 expectObservable(model); | 1650 expectObservable(model); |
1651 deliverChangeRecords(); | 1651 performMicrotaskCheckpoint(); |
1652 } | 1652 } |
1653 | 1653 |
1654 void expectObservable(model) { | 1654 void expectObservable(model) { |
1655 if (model is! Observable) { | 1655 if (model is! Observable) { |
1656 // This is here to eagerly catch a bug in the test; it means the test | 1656 // This is here to eagerly catch a bug in the observeTest; it means the obse
rveTest |
1657 // forgot a toSymbols somewhere. | 1657 // forgot a toSymbols somewhere. |
1658 expect(identical(toSymbols(model), model), true, | 1658 expect(identical(toSymbols(model), model), true, |
1659 reason: 'model type "${model.runtimeType}" should be observable'); | 1659 reason: 'model type "${model.runtimeType}" should be observable'); |
1660 return; | 1660 return; |
1661 } | 1661 } |
1662 if (model is ObservableList) { | 1662 if (model is ObservableList) { |
1663 for (var item in model) { | 1663 for (var item in model) { |
1664 expectObservable(item); | 1664 expectObservable(item); |
1665 } | 1665 } |
1666 } else if (model is ObservableMap) { | 1666 } else if (model is ObservableMap) { |
(...skipping 15 matching lines...) Expand all Loading... |
1682 k = k is String ? sym(k) : _deepToSymbol(k); | 1682 k = k is String ? sym(k) : _deepToSymbol(k); |
1683 result[k] = _deepToSymbol(v); | 1683 result[k] = _deepToSymbol(v); |
1684 }); | 1684 }); |
1685 return result; | 1685 return result; |
1686 } | 1686 } |
1687 if (value is Iterable) { | 1687 if (value is Iterable) { |
1688 return value.map(_deepToSymbol).toList(); | 1688 return value.map(_deepToSymbol).toList(); |
1689 } | 1689 } |
1690 return value; | 1690 return value; |
1691 } | 1691 } |
OLD | NEW |