| 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_binding.test.template_binding_test; | 5 library template_binding.test.template_binding_test; |
| 6 | 6 |
| 7 import 'dart:async'; | 7 import 'dart:async'; |
| 8 import 'dart:collection'; | |
| 9 import 'dart:html'; | 8 import 'dart:html'; |
| 10 import 'dart:math' as math; | 9 import 'dart:math' as math; |
| 11 import 'package:observe/observe.dart'; | 10 import 'package:observe/observe.dart'; |
| 12 import 'package:template_binding/template_binding.dart'; | 11 import 'package:template_binding/template_binding.dart'; |
| 13 import 'package:unittest/html_config.dart'; | 12 import 'package:unittest/html_config.dart'; |
| 14 import 'package:unittest/unittest.dart'; | 13 import 'package:unittest/unittest.dart'; |
| 15 | 14 |
| 16 // TODO(jmesserly): merge this file? | 15 // TODO(jmesserly): merge this file? |
| 17 import 'binding_syntax.dart' show syntaxTests; | 16 import 'binding_syntax.dart' show syntaxTests; |
| 18 import 'utils.dart'; | 17 import 'utils.dart'; |
| 19 | 18 |
| 20 // Note: this file ported from | 19 // Note: this file ported from |
| 21 // https://github.com/Polymer/TemplateBinding/blob/ed3266266e751b5ab1f75f8e0509d
0d5f0ef35d8/tests/tests.js | 20 // https://github.com/Polymer/TemplateBinding/blob/fcb7a502794f19544f2d4b77c96ee
bb70830591d/tests/tests.js |
| 22 | 21 |
| 23 // TODO(jmesserly): submit a small cleanup patch to original. I fixed some | 22 // TODO(jmesserly): submit a small cleanup patch to original. I fixed some |
| 24 // cases where "div" and "t" were unintentionally using the JS global scope; | 23 // cases where "div" and "t" were unintentionally using the JS global scope; |
| 25 // look for "assertNodesAre". | 24 // look for "assertNodesAre". |
| 26 | 25 |
| 27 main() { | 26 main() => dirtyCheckZone().run(() { |
| 28 useHtmlConfiguration(); | 27 useHtmlConfiguration(); |
| 29 | 28 |
| 30 // Load MutationObserver polyfill in case IE needs it. | 29 // Load MutationObserver polyfill in case IE needs it. |
| 31 var script = new ScriptElement() | 30 var script = new ScriptElement() |
| 32 ..src = '/root_dart/pkg/mutation_observer/lib/mutation_observer.min.js'; | 31 ..src = '/root_dart/pkg/mutation_observer/lib/mutation_observer.min.js'; |
| 33 var polyfillLoaded = script.onLoad.first; | 32 var polyfillLoaded = script.onLoad.first; |
| 34 document.head.append(script); | 33 document.head.append(script); |
| 35 | 34 |
| 36 setUp(() => polyfillLoaded.then((_) { | 35 setUp(() => polyfillLoaded.then((_) { |
| 37 document.body.append(testDiv = new DivElement()); | 36 document.body.append(testDiv = new DivElement()); |
| 38 })); | 37 })); |
| 39 | 38 |
| 40 tearDown(() { | 39 tearDown(() { |
| 41 testDiv.remove(); | 40 testDiv.remove(); |
| 42 testDiv = null; | 41 testDiv = null; |
| 43 }); | 42 }); |
| 44 | 43 |
| 45 test('MutationObserver is supported', () { | 44 test('MutationObserver is supported', () { |
| 46 expect(MutationObserver.supported, true, reason: 'polyfill was loaded.'); | 45 expect(MutationObserver.supported, true, reason: 'polyfill was loaded.'); |
| 47 }); | 46 }); |
| 48 | 47 |
| 49 group('Template Instantiation', templateInstantiationTests); | 48 group('Template', templateInstantiationTests); |
| 50 | 49 |
| 51 group('Binding Delegate API', () { | 50 group('Binding Delegate API', () { |
| 52 group('with Observable', () { | 51 group('with Observable', () { |
| 53 syntaxTests(([f, b]) => new FooBarModel(f, b)); | 52 syntaxTests(([f, b]) => new FooBarModel(f, b)); |
| 54 }); | 53 }); |
| 55 | 54 |
| 56 group('with ChangeNotifier', () { | 55 group('with ChangeNotifier', () { |
| 57 syntaxTests(([f, b]) => new FooBarNotifyModel(f, b)); | 56 syntaxTests(([f, b]) => new FooBarNotifyModel(f, b)); |
| 58 }); | 57 }); |
| 59 }); | 58 }); |
| 60 | 59 |
| 61 group('Compat', compatTests); | 60 group('Compat', compatTests); |
| 62 } | 61 }); |
| 63 | 62 |
| 64 var expando = new Expando('test'); | 63 var expando = new Expando('test'); |
| 65 void addExpandos(node) { | 64 void addExpandos(node) { |
| 66 while (node != null) { | 65 while (node != null) { |
| 67 expando[node] = node.text; | 66 expando[node] = node.text; |
| 68 node = node.nextNode; | 67 node = node.nextNode; |
| 69 } | 68 } |
| 70 } | 69 } |
| 71 | 70 |
| 72 void checkExpandos(node) { | 71 void checkExpandos(node) { |
| 73 expect(node, isNotNull); | 72 expect(node, isNotNull); |
| 74 while (node != null) { | 73 while (node != null) { |
| 75 expect(expando[node], node.text); | 74 expect(expando[node], node.text); |
| 76 node = node.nextNode; | 75 node = node.nextNode; |
| 77 } | 76 } |
| 78 } | 77 } |
| 79 | 78 |
| 80 templateInstantiationTests() { | 79 templateInstantiationTests() { |
| 81 | 80 // Dart note: renamed some of these tests to have unique names |
| 82 observeTest('Template', () { | 81 |
| 82 test('Bind (simple)', () { |
| 83 var div = createTestHtml('<template bind={{}}>text</template>'); | 83 var div = createTestHtml('<template bind={{}}>text</template>'); |
| 84 templateBind(div.firstChild).model = {}; | 84 templateBind(div.firstChild).model = {}; |
| 85 performMicrotaskCheckpoint(); | 85 return new Future(() { |
| 86 expect(div.nodes.length, 2); | 86 expect(div.nodes.length, 2); |
| 87 expect(div.nodes.last.text, 'text'); | 87 expect(div.nodes.last.text, 'text'); |
| 88 | 88 |
| 89 templateBind(div.firstChild).model = null; | 89 // Dart note: null is used instead of undefined to clear the template. |
| 90 performMicrotaskCheckpoint(); | 90 templateBind(div.firstChild).model = null; |
| 91 expect(div.nodes.length, 1); | 91 |
| 92 }); | 92 }).then(endOfMicrotask).then((_) { |
| 93 | 93 expect(div.nodes.length, 1); |
| 94 observeTest('Template bind, no parent', () { | 94 templateBind(div.firstChild).model = 123; |
| 95 |
| 96 }).then(endOfMicrotask).then((_) { |
| 97 expect(div.nodes.length, 2); |
| 98 expect(div.nodes.last.text, 'text'); |
| 99 }); |
| 100 }); |
| 101 |
| 102 test('oneTime-Bind', () { |
| 103 var div = createTestHtml('<template bind="[[ bound ]]">text</template>'); |
| 104 var model = toObservable({'bound': 1}); |
| 105 templateBind(div.firstChild).model = model; |
| 106 return new Future(() { |
| 107 expect(div.nodes.length, 2); |
| 108 expect(div.nodes.last.text, 'text'); |
| 109 |
| 110 model['bound'] = false; |
| 111 |
| 112 }).then(endOfMicrotask).then((_) { |
| 113 expect(div.nodes.length, 2); |
| 114 expect(div.nodes.last.text, 'text'); |
| 115 }); |
| 116 }); |
| 117 |
| 118 test('Bind - no parent', () { |
| 95 var div = createTestHtml('<template bind>text</template>'); | 119 var div = createTestHtml('<template bind>text</template>'); |
| 96 var template = div.firstChild; | 120 var template = div.firstChild; |
| 97 template.remove(); | 121 template.remove(); |
| 98 | 122 |
| 99 templateBind(template).model = {}; | 123 templateBind(template).model = {}; |
| 100 performMicrotaskCheckpoint(); | 124 return new Future(() { |
| 101 expect(template.nodes.length, 0); | 125 expect(template.nodes.length, 0); |
| 102 expect(template.nextNode, null); | 126 expect(template.nextNode, null); |
| 103 }); | 127 }); |
| 104 | 128 }); |
| 105 observeTest('Template bind, no defaultView', () { | 129 |
| 130 test('Bind - no defaultView', () { |
| 106 var div = createTestHtml('<template bind>text</template>'); | 131 var div = createTestHtml('<template bind>text</template>'); |
| 107 var template = div.firstChild; | 132 var template = div.firstChild; |
| 108 var doc = document.implementation.createHtmlDocument(''); | 133 var doc = document.implementation.createHtmlDocument(''); |
| 109 doc.adoptNode(div); | 134 doc.adoptNode(div); |
| 110 recursivelySetTemplateModel(template, {}); | 135 recursivelySetTemplateModel(template, {}); |
| 111 performMicrotaskCheckpoint(); | 136 return new Future(() => expect(div.nodes.length, 1)); |
| 112 expect(div.nodes.length, 1); | 137 }); |
| 113 }); | 138 |
| 114 | 139 test('Empty Bind', () { |
| 115 observeTest('Template-Empty Bind', () { | |
| 116 var div = createTestHtml('<template bind>text</template>'); | 140 var div = createTestHtml('<template bind>text</template>'); |
| 117 var template = div.firstChild; | 141 var template = div.firstChild; |
| 118 templateBind(template).model = {}; | 142 templateBind(template).model = {}; |
| 119 performMicrotaskCheckpoint(); | 143 return new Future(() { |
| 120 expect(div.nodes.length, 2); | 144 expect(div.nodes.length, 2); |
| 121 expect(div.nodes.last.text, 'text'); | 145 expect(div.nodes.last.text, 'text'); |
| 122 }); | 146 }); |
| 123 | 147 }); |
| 124 observeTest('Template Bind If', () { | 148 |
| 125 var div = createTestHtml('<template bind if="{{ foo }}">text</template>'); | 149 test('Bind If', () { |
| 126 // Note: changed this value from 0->null because zero is not falsey in Dart. | 150 var div = createTestHtml( |
| 151 '<template bind="{{ bound }}" if="{{ predicate }}">' |
| 152 'value:{{ value }}' |
| 153 '</template>'); |
| 154 // Dart note: predicate changed from 0->null because 0 isn't falsey in Dart. |
| 127 // See https://code.google.com/p/dart/issues/detail?id=11956 | 155 // See https://code.google.com/p/dart/issues/detail?id=11956 |
| 128 var m = toObservable({ 'foo': null }); | 156 // Changed bound from null->1 since null is equivalent to JS undefined, |
| 129 var template = div.firstChild; | 157 // and would cause the template to not be expanded. |
| 130 templateBind(template).model = m; | 158 var m = toObservable({ 'predicate': null, 'bound': 1 }); |
| 131 performMicrotaskCheckpoint(); | 159 var template = div.firstChild; |
| 132 expect(div.nodes.length, 1); | 160 templateBind(template).model = m; |
| 133 | 161 return new Future(() { |
| 134 m['foo'] = 1; | 162 expect(div.nodes.length, 1); |
| 135 performMicrotaskCheckpoint(); | 163 |
| 136 expect(div.nodes.length, 2); | 164 m['predicate'] = 1; |
| 137 expect(div.lastChild.text, 'text'); | 165 |
| 138 | 166 }).then(endOfMicrotask).then((_) { |
| 139 templateBind(template).model = null; | 167 expect(div.nodes.length, 2); |
| 140 performMicrotaskCheckpoint(); | 168 expect(div.lastChild.text, 'value:'); |
| 141 expect(div.nodes.length, 1); | 169 |
| 142 }); | 170 m['bound'] = toObservable({ 'value': 2 }); |
| 143 | 171 |
| 144 observeTest('Template Bind If, 2', () { | 172 }).then(endOfMicrotask).then((_) { |
| 173 expect(div.nodes.length, 2); |
| 174 expect(div.lastChild.text, 'value:2'); |
| 175 |
| 176 m['bound']['value'] = 3; |
| 177 |
| 178 }).then(endOfMicrotask).then((_) { |
| 179 expect(div.nodes.length, 2); |
| 180 expect(div.lastChild.text, 'value:3'); |
| 181 |
| 182 templateBind(template).model = null; |
| 183 |
| 184 }).then(endOfMicrotask).then((_) { |
| 185 expect(div.nodes.length, 1); |
| 186 }); |
| 187 }); |
| 188 |
| 189 test('Bind oneTime-If - predicate false', () { |
| 190 var div = createTestHtml( |
| 191 '<template bind="{{ bound }}" if="[[ predicate ]]">' |
| 192 'value:{{ value }}' |
| 193 '</template>'); |
| 194 // Dart note: predicate changed from 0->null because 0 isn't falsey in Dart. |
| 195 // See https://code.google.com/p/dart/issues/detail?id=11956 |
| 196 // Changed bound from null->1 since null is equivalent to JS undefined, |
| 197 // and would cause the template to not be expanded. |
| 198 var m = toObservable({ 'predicate': null, 'bound': 1 }); |
| 199 var template = div.firstChild; |
| 200 templateBind(template).model = m; |
| 201 |
| 202 return new Future(() { |
| 203 expect(div.nodes.length, 1); |
| 204 |
| 205 m['predicate'] = 1; |
| 206 |
| 207 }).then(endOfMicrotask).then((_) { |
| 208 expect(div.nodes.length, 1); |
| 209 |
| 210 m['bound'] = toObservable({ 'value': 2 }); |
| 211 |
| 212 }).then(endOfMicrotask).then((_) { |
| 213 expect(div.nodes.length, 1); |
| 214 |
| 215 m['bound']['value'] = 3; |
| 216 |
| 217 }).then(endOfMicrotask).then((_) { |
| 218 expect(div.nodes.length, 1); |
| 219 |
| 220 templateBind(template).model = null; |
| 221 |
| 222 }).then(endOfMicrotask).then((_) { |
| 223 expect(div.nodes.length, 1); |
| 224 }); |
| 225 }); |
| 226 |
| 227 test('Bind oneTime-If - predicate true', () { |
| 228 var div = createTestHtml( |
| 229 '<template bind="{{ bound }}" if="[[ predicate ]]">' |
| 230 'value:{{ value }}' |
| 231 '</template>'); |
| 232 |
| 233 // Dart note: changed bound from null->1 since null is equivalent to JS |
| 234 // undefined, and would cause the template to not be expanded. |
| 235 var m = toObservable({ 'predicate': 1, 'bound': 1 }); |
| 236 var template = div.firstChild; |
| 237 templateBind(template).model = m; |
| 238 |
| 239 return new Future(() { |
| 240 expect(div.nodes.length, 2); |
| 241 expect(div.lastChild.text, 'value:'); |
| 242 |
| 243 m['bound'] = toObservable({ 'value': 2 }); |
| 244 |
| 245 }).then(endOfMicrotask).then((_) { |
| 246 expect(div.nodes.length, 2); |
| 247 expect(div.lastChild.text, 'value:2'); |
| 248 |
| 249 m['bound']['value'] = 3; |
| 250 |
| 251 }).then(endOfMicrotask).then((_) { |
| 252 expect(div.nodes.length, 2); |
| 253 expect(div.lastChild.text, 'value:3'); |
| 254 |
| 255 m['predicate'] = null; // will have no effect |
| 256 |
| 257 }).then(endOfMicrotask).then((_) { |
| 258 expect(div.nodes.length, 2); |
| 259 expect(div.lastChild.text, 'value:3'); |
| 260 |
| 261 templateBind(template).model = null; |
| 262 |
| 263 }).then(endOfMicrotask).then((_) { |
| 264 expect(div.nodes.length, 1); |
| 265 }); |
| 266 }); |
| 267 |
| 268 test('oneTime-Bind If', () { |
| 269 var div = createTestHtml( |
| 270 '<template bind="[[ bound ]]" if="{{ predicate }}">' |
| 271 'value:{{ value }}' |
| 272 '</template>'); |
| 273 |
| 274 var m = toObservable({'predicate': null, 'bound': {'value': 2}}); |
| 275 var template = div.firstChild; |
| 276 templateBind(template).model = m; |
| 277 |
| 278 return new Future(() { |
| 279 expect(div.nodes.length, 1); |
| 280 |
| 281 m['predicate'] = 1; |
| 282 |
| 283 }).then(endOfMicrotask).then((_) { |
| 284 expect(div.nodes.length, 2); |
| 285 expect(div.lastChild.text, 'value:2'); |
| 286 |
| 287 m['bound']['value'] = 3; |
| 288 |
| 289 }).then(endOfMicrotask).then((_) { |
| 290 expect(div.nodes.length, 2); |
| 291 expect(div.lastChild.text, 'value:3'); |
| 292 |
| 293 m['bound'] = toObservable({'value': 4 }); |
| 294 |
| 295 }).then(endOfMicrotask).then((_) { |
| 296 expect(div.nodes.length, 2); |
| 297 expect(div.lastChild.text, 'value:3'); |
| 298 |
| 299 templateBind(template).model = null; |
| 300 |
| 301 }).then(endOfMicrotask).then((_) { |
| 302 expect(div.nodes.length, 1); |
| 303 }); |
| 304 }); |
| 305 |
| 306 test('oneTime-Bind oneTime-If', () { |
| 307 var div = createTestHtml( |
| 308 '<template bind="[[ bound ]]" if="[[ predicate ]]">' |
| 309 'value:{{ value }}' |
| 310 '</template>'); |
| 311 |
| 312 var m = toObservable({'predicate': 1, 'bound': {'value': 2}}); |
| 313 var template = div.firstChild; |
| 314 templateBind(template).model = m; |
| 315 |
| 316 return new Future(() { |
| 317 expect(div.nodes.length, 2); |
| 318 expect(div.lastChild.text, 'value:2'); |
| 319 |
| 320 m['bound']['value'] = 3; |
| 321 |
| 322 }).then(endOfMicrotask).then((_) { |
| 323 expect(div.nodes.length, 2); |
| 324 expect(div.lastChild.text, 'value:3'); |
| 325 |
| 326 m['bound'] = toObservable({'value': 4 }); |
| 327 |
| 328 }).then(endOfMicrotask).then((_) { |
| 329 expect(div.nodes.length, 2); |
| 330 expect(div.lastChild.text, 'value:3'); |
| 331 |
| 332 m['predicate'] = false; |
| 333 |
| 334 }).then(endOfMicrotask).then((_) { |
| 335 expect(div.nodes.length, 2); |
| 336 expect(div.lastChild.text, 'value:3'); |
| 337 |
| 338 templateBind(template).model = null; |
| 339 |
| 340 }).then(endOfMicrotask).then((_) { |
| 341 expect(div.nodes.length, 1); |
| 342 }); |
| 343 }); |
| 344 |
| 345 test('Bind If, 2', () { |
| 145 var div = createTestHtml( | 346 var div = createTestHtml( |
| 146 '<template bind="{{ foo }}" if="{{ bar }}">{{ bat }}</template>'); | 347 '<template bind="{{ foo }}" if="{{ bar }}">{{ bat }}</template>'); |
| 147 var m = toObservable({ 'bar': null, 'foo': { 'bat': 'baz' } }); | 348 var m = toObservable({ 'bar': null, 'foo': { 'bat': 'baz' } }); |
| 148 recursivelySetTemplateModel(div, m); | 349 recursivelySetTemplateModel(div, m); |
| 149 performMicrotaskCheckpoint(); | 350 return new Future(() { |
| 150 expect(div.nodes.length, 1); | 351 expect(div.nodes.length, 1); |
| 151 | 352 |
| 152 m['bar'] = 1; | 353 m['bar'] = 1; |
| 153 performMicrotaskCheckpoint(); | 354 }).then(endOfMicrotask).then((_) { |
| 154 expect(div.nodes.length, 2); | 355 expect(div.nodes.length, 2); |
| 155 expect(div.lastChild.text, 'baz'); | 356 expect(div.lastChild.text, 'baz'); |
| 156 }); | 357 }); |
| 157 | 358 }); |
| 158 observeTest('Template If', () { | 359 |
| 360 test('If', () { |
| 159 var div = createTestHtml('<template if="{{ foo }}">{{ value }}</template>'); | 361 var div = createTestHtml('<template if="{{ foo }}">{{ value }}</template>'); |
| 160 // Note: changed this value from 0->null because zero is not falsey in | 362 // Dart note: foo changed from 0->null because 0 isn't falsey in Dart. |
| 161 // Dart. See https://code.google.com/p/dart/issues/detail?id=11956 | 363 // See https://code.google.com/p/dart/issues/detail?id=11956 |
| 162 var m = toObservable({ 'foo': null, 'value': 'foo' }); | 364 var m = toObservable({ 'foo': null, 'value': 'foo' }); |
| 163 var template = div.firstChild; | 365 var template = div.firstChild; |
| 164 templateBind(template).model = m; | 366 templateBind(template).model = m; |
| 165 performMicrotaskCheckpoint(); | 367 return new Future(() { |
| 166 expect(div.nodes.length, 1); | 368 expect(div.nodes.length, 1); |
| 167 | 369 |
| 168 m['foo'] = 1; | 370 m['foo'] = 1; |
| 169 performMicrotaskCheckpoint(); | 371 }).then(endOfMicrotask).then((_) { |
| 170 expect(div.nodes.length, 2); | 372 expect(div.nodes.length, 2); |
| 171 expect(div.lastChild.text, 'foo'); | 373 expect(div.lastChild.text, 'foo'); |
| 172 | 374 |
| 173 templateBind(template).model = null; | 375 templateBind(template).model = null; |
| 174 performMicrotaskCheckpoint(); | 376 }).then(endOfMicrotask).then((_) { |
| 175 expect(div.nodes.length, 1); | 377 expect(div.nodes.length, 1); |
| 176 }); | 378 }); |
| 177 | 379 }); |
| 178 observeTest('Template Empty-If', () { | 380 |
| 381 test('Empty-If', () { |
| 179 var div = createTestHtml('<template if>{{ value }}</template>'); | 382 var div = createTestHtml('<template if>{{ value }}</template>'); |
| 180 var m = toObservable({ 'value': 'foo' }); | 383 var m = toObservable({ 'value': 'foo' }); |
| 181 recursivelySetTemplateModel(div, null); | 384 recursivelySetTemplateModel(div, null); |
| 182 performMicrotaskCheckpoint(); | 385 return new Future(() { |
| 183 expect(div.nodes.length, 1); | 386 expect(div.nodes.length, 1); |
| 184 | 387 |
| 388 recursivelySetTemplateModel(div, m); |
| 389 }).then(endOfMicrotask).then((_) { |
| 390 expect(div.nodes.length, 2); |
| 391 expect(div.lastChild.text, 'foo'); |
| 392 }); |
| 393 }); |
| 394 |
| 395 test('OneTime - simple text', () { |
| 396 var div = createTestHtml('<template bind>[[ value ]]</template>'); |
| 397 var m = toObservable({ 'value': 'foo' }); |
| 185 recursivelySetTemplateModel(div, m); | 398 recursivelySetTemplateModel(div, m); |
| 186 performMicrotaskCheckpoint(); | 399 return new Future(() { |
| 187 expect(div.nodes.length, 2); | 400 expect(div.nodes.length, 2); |
| 188 expect(div.lastChild.text, 'foo'); | 401 expect(div.lastChild.text, 'foo'); |
| 189 }); | 402 |
| 190 | 403 m['value'] = 'bar'; |
| 191 observeTest('Template Repeat If', () { | 404 |
| 192 var div = createTestHtml( | 405 }).then(endOfMicrotask).then((_) { |
| 193 '<template repeat="{{ foo }}" if="{{ bar }}">{{ }}</template>'); | 406 // unchanged. |
| 194 // Note: changed this value from 0->null because zero is not falsey in Dart. | 407 expect(div.lastChild.text, 'foo'); |
| 408 }); |
| 409 }); |
| 410 |
| 411 test('OneTime - compound text', () { |
| 412 var div = createTestHtml( |
| 413 '<template bind>[[ foo ]] bar [[ baz ]]</template>'); |
| 414 var m = toObservable({ 'foo': 'FOO', 'baz': 'BAZ' }); |
| 415 recursivelySetTemplateModel(div, m); |
| 416 return new Future(() { |
| 417 expect(div.nodes.length, 2); |
| 418 expect(div.lastChild.text, 'FOO bar BAZ'); |
| 419 |
| 420 m['foo'] = 'FI'; |
| 421 m['baz'] = 'BA'; |
| 422 |
| 423 }).then(endOfMicrotask).then((_) { |
| 424 // unchanged. |
| 425 expect(div.nodes.length, 2); |
| 426 expect(div.lastChild.text, 'FOO bar BAZ'); |
| 427 }); |
| 428 }); |
| 429 |
| 430 test('OneTime/Dynamic Mixed - compound text', () { |
| 431 var div = createTestHtml( |
| 432 '<template bind>[[ foo ]] bar {{ baz }}</template>'); |
| 433 var m = toObservable({ 'foo': 'FOO', 'baz': 'BAZ' }); |
| 434 recursivelySetTemplateModel(div, m); |
| 435 return new Future(() { |
| 436 expect(div.nodes.length, 2); |
| 437 expect(div.lastChild.text, 'FOO bar BAZ'); |
| 438 |
| 439 m['foo'] = 'FI'; |
| 440 m['baz'] = 'BA'; |
| 441 |
| 442 }).then(endOfMicrotask).then((_) { |
| 443 // unchanged [[ foo ]]. |
| 444 expect(div.nodes.length, 2); |
| 445 expect(div.lastChild.text, 'FOO bar BA'); |
| 446 }); |
| 447 }); |
| 448 |
| 449 test('OneTime - simple attribute', () { |
| 450 var div = createTestHtml( |
| 451 '<template bind><div foo="[[ value ]]"></div></template>'); |
| 452 var m = toObservable({ 'value': 'foo' }); |
| 453 recursivelySetTemplateModel(div, m); |
| 454 return new Future(() { |
| 455 expect(div.nodes.length, 2); |
| 456 expect(div.lastChild.attributes['foo'], 'foo'); |
| 457 |
| 458 m['value'] = 'bar'; |
| 459 |
| 460 }).then(endOfMicrotask).then((_) { |
| 461 // unchanged. |
| 462 expect(div.nodes.length, 2); |
| 463 expect(div.lastChild.attributes['foo'], 'foo'); |
| 464 }); |
| 465 }); |
| 466 |
| 467 test('OneTime - compound attribute', () { |
| 468 var div = createTestHtml( |
| 469 '<template bind>' |
| 470 '<div foo="[[ value ]]:[[ otherValue ]]"></div>' |
| 471 '</template>'); |
| 472 var m = toObservable({ 'value': 'foo', 'otherValue': 'bar' }); |
| 473 recursivelySetTemplateModel(div, m); |
| 474 return new Future(() { |
| 475 expect(div.nodes.length, 2); |
| 476 expect(div.lastChild.attributes['foo'], 'foo:bar'); |
| 477 |
| 478 m['value'] = 'baz'; |
| 479 m['otherValue'] = 'bot'; |
| 480 |
| 481 }).then(endOfMicrotask).then((_) { |
| 482 // unchanged. |
| 483 expect(div.lastChild.attributes['foo'], 'foo:bar'); |
| 484 }); |
| 485 }); |
| 486 |
| 487 test('OneTime/Dynamic mixed - compound attribute', () { |
| 488 var div = createTestHtml( |
| 489 '<template bind>' |
| 490 '<div foo="{{ value }}:[[ otherValue ]]"></div>' |
| 491 '</template>'); |
| 492 var m = toObservable({ 'value': 'foo', 'otherValue': 'bar' }); |
| 493 recursivelySetTemplateModel(div, m); |
| 494 return new Future(() { |
| 495 expect(div.nodes.length, 2); |
| 496 expect(div.lastChild.attributes['foo'], 'foo:bar'); |
| 497 |
| 498 m['value'] = 'baz'; |
| 499 m['otherValue'] = 'bot'; |
| 500 |
| 501 }).then(endOfMicrotask).then((_) { |
| 502 // unchanged [[ otherValue ]]. |
| 503 expect(div.lastChild.attributes['foo'], 'baz:bar'); |
| 504 }); |
| 505 }); |
| 506 |
| 507 test('Repeat If', () { |
| 508 var div = createTestHtml( |
| 509 '<template repeat="{{ items }}" if="{{ predicate }}">{{}}</template>'); |
| 510 // Dart note: predicate changed from 0->null because 0 isn't falsey in Dart. |
| 195 // See https://code.google.com/p/dart/issues/detail?id=11956 | 511 // See https://code.google.com/p/dart/issues/detail?id=11956 |
| 196 var m = toObservable({ 'bar': null, 'foo': [1, 2, 3] }); | 512 var m = toObservable({ 'predicate': null, 'items': [1] }); |
| 197 var template = div.firstChild; | 513 var template = div.firstChild; |
| 198 templateBind(template).model = m; | 514 templateBind(template).model = m; |
| 199 performMicrotaskCheckpoint(); | 515 return new Future(() { |
| 200 expect(div.nodes.length, 1); | 516 expect(div.nodes.length, 1); |
| 201 | 517 |
| 202 m['bar'] = 1; | 518 m['predicate'] = 1; |
| 203 performMicrotaskCheckpoint(); | 519 |
| 204 expect(div.nodes.length, 4); | 520 }).then(endOfMicrotask).then((_) { |
| 205 expect(div.nodes[1].text, '1'); | 521 expect(div.nodes.length, 2); |
| 206 expect(div.nodes[2].text, '2'); | 522 expect(div.nodes[1].text, '1'); |
| 207 expect(div.nodes[3].text, '3'); | 523 |
| 208 | 524 m['items']..add(2)..add(3); |
| 209 templateBind(template).model = null; | 525 |
| 210 performMicrotaskCheckpoint(); | 526 }).then(endOfMicrotask).then((_) { |
| 211 expect(div.nodes.length, 1); | 527 expect(div.nodes.length, 4); |
| 212 }); | 528 expect(div.nodes[1].text, '1'); |
| 213 | 529 expect(div.nodes[2].text, '2'); |
| 214 observeTest('TextTemplateWithNullStringBinding', () { | 530 expect(div.nodes[3].text, '3'); |
| 531 |
| 532 m['items'] = [4]; |
| 533 |
| 534 }).then(endOfMicrotask).then((_) { |
| 535 expect(div.nodes.length, 2); |
| 536 expect(div.nodes[1].text, '4'); |
| 537 |
| 538 templateBind(template).model = null; |
| 539 }).then(endOfMicrotask).then((_) { |
| 540 expect(div.nodes.length, 1); |
| 541 }); |
| 542 }); |
| 543 |
| 544 test('Repeat oneTime-If (predicate false)', () { |
| 545 var div = createTestHtml( |
| 546 '<template repeat="{{ items }}" if="[[ predicate ]]">{{}}</template>'); |
| 547 // Dart note: predicate changed from 0->null because 0 isn't falsey in Dart. |
| 548 // See https://code.google.com/p/dart/issues/detail?id=11956 |
| 549 var m = toObservable({ 'predicate': null, 'items': [1] }); |
| 550 var template = div.firstChild; |
| 551 templateBind(template).model = m; |
| 552 return new Future(() { |
| 553 expect(div.nodes.length, 1); |
| 554 |
| 555 m['predicate'] = 1; |
| 556 |
| 557 }).then(endOfMicrotask).then((_) { |
| 558 expect(div.nodes.length, 1, reason: 'unchanged'); |
| 559 |
| 560 m['items']..add(2)..add(3); |
| 561 |
| 562 }).then(endOfMicrotask).then((_) { |
| 563 expect(div.nodes.length, 1, reason: 'unchanged'); |
| 564 |
| 565 m['items'] = [4]; |
| 566 |
| 567 }).then(endOfMicrotask).then((_) { |
| 568 expect(div.nodes.length, 1, reason: 'unchanged'); |
| 569 |
| 570 templateBind(template).model = null; |
| 571 }).then(endOfMicrotask).then((_) { |
| 572 expect(div.nodes.length, 1); |
| 573 }); |
| 574 }); |
| 575 |
| 576 test('Repeat oneTime-If (predicate true)', () { |
| 577 var div = createTestHtml( |
| 578 '<template repeat="{{ items }}" if="[[ predicate ]]">{{}}</template>'); |
| 579 |
| 580 var m = toObservable({ 'predicate': true, 'items': [1] }); |
| 581 var template = div.firstChild; |
| 582 templateBind(template).model = m; |
| 583 return new Future(() { |
| 584 expect(div.nodes.length, 2); |
| 585 expect(div.nodes[1].text, '1'); |
| 586 |
| 587 m['items']..add(2)..add(3); |
| 588 |
| 589 }).then(endOfMicrotask).then((_) { |
| 590 expect(div.nodes.length, 4); |
| 591 expect(div.nodes[1].text, '1'); |
| 592 expect(div.nodes[2].text, '2'); |
| 593 expect(div.nodes[3].text, '3'); |
| 594 |
| 595 m['items'] = [4]; |
| 596 |
| 597 }).then(endOfMicrotask).then((_) { |
| 598 expect(div.nodes.length, 2); |
| 599 expect(div.nodes[1].text, '4'); |
| 600 |
| 601 m['predicate'] = false; |
| 602 |
| 603 }).then(endOfMicrotask).then((_) { |
| 604 expect(div.nodes.length, 2, reason: 'unchanged'); |
| 605 expect(div.nodes[1].text, '4', reason: 'unchanged'); |
| 606 |
| 607 templateBind(template).model = null; |
| 608 }).then(endOfMicrotask).then((_) { |
| 609 expect(div.nodes.length, 1); |
| 610 }); |
| 611 }); |
| 612 |
| 613 test('oneTime-Repeat If', () { |
| 614 var div = createTestHtml( |
| 615 '<template repeat="[[ items ]]" if="{{ predicate }}">{{}}</template>'); |
| 616 |
| 617 var m = toObservable({ 'predicate': false, 'items': [1] }); |
| 618 var template = div.firstChild; |
| 619 templateBind(template).model = m; |
| 620 return new Future(() { |
| 621 expect(div.nodes.length, 1); |
| 622 |
| 623 m['predicate'] = true; |
| 624 |
| 625 }).then(endOfMicrotask).then((_) { |
| 626 expect(div.nodes.length, 2); |
| 627 expect(div.nodes[1].text, '1'); |
| 628 |
| 629 m['items']..add(2)..add(3); |
| 630 |
| 631 }).then(endOfMicrotask).then((_) { |
| 632 expect(div.nodes.length, 2); |
| 633 expect(div.nodes[1].text, '1'); |
| 634 |
| 635 m['items'] = [4]; |
| 636 |
| 637 }).then(endOfMicrotask).then((_) { |
| 638 expect(div.nodes.length, 2); |
| 639 expect(div.nodes[1].text, '1'); |
| 640 |
| 641 templateBind(template).model = null; |
| 642 }).then(endOfMicrotask).then((_) { |
| 643 expect(div.nodes.length, 1); |
| 644 }); |
| 645 }); |
| 646 |
| 647 test('oneTime-Repeat oneTime-If', () { |
| 648 var div = createTestHtml( |
| 649 '<template repeat="[[ items ]]" if="[[ predicate ]]">{{}}</template>'); |
| 650 |
| 651 var m = toObservable({ 'predicate': true, 'items': [1] }); |
| 652 var template = div.firstChild; |
| 653 templateBind(template).model = m; |
| 654 return new Future(() { |
| 655 expect(div.nodes.length, 2); |
| 656 expect(div.nodes[1].text, '1'); |
| 657 |
| 658 m['items']..add(2)..add(3); |
| 659 |
| 660 }).then(endOfMicrotask).then((_) { |
| 661 expect(div.nodes.length, 2); |
| 662 expect(div.nodes[1].text, '1'); |
| 663 |
| 664 m['items'] = [4]; |
| 665 |
| 666 }).then(endOfMicrotask).then((_) { |
| 667 expect(div.nodes.length, 2); |
| 668 expect(div.nodes[1].text, '1'); |
| 669 |
| 670 m['predicate'] = false; |
| 671 |
| 672 }).then(endOfMicrotask).then((_) { |
| 673 expect(div.nodes.length, 2); |
| 674 expect(div.nodes[1].text, '1'); |
| 675 |
| 676 templateBind(template).model = null; |
| 677 }).then(endOfMicrotask).then((_) { |
| 678 expect(div.nodes.length, 1); |
| 679 }); |
| 680 }); |
| 681 |
| 682 test('TextTemplateWithNullStringBinding', () { |
| 215 var div = createTestHtml('<template bind={{}}>a{{b}}c</template>'); | 683 var div = createTestHtml('<template bind={{}}>a{{b}}c</template>'); |
| 216 var model = toObservable({'b': 'B'}); | 684 var model = toObservable({'b': 'B'}); |
| 217 recursivelySetTemplateModel(div, model); | 685 recursivelySetTemplateModel(div, model); |
| 218 | 686 |
| 219 performMicrotaskCheckpoint(); | 687 return new Future(() { |
| 220 expect(div.nodes.length, 2); | 688 expect(div.nodes.length, 2); |
| 221 expect(div.nodes.last.text, 'aBc'); | 689 expect(div.nodes.last.text, 'aBc'); |
| 222 | 690 |
| 223 model['b'] = 'b'; | 691 model['b'] = 'b'; |
| 224 performMicrotaskCheckpoint(); | 692 }).then(endOfMicrotask).then((_) { |
| 225 expect(div.nodes.last.text, 'abc'); | 693 expect(div.nodes.last.text, 'abc'); |
| 226 | 694 |
| 227 model['b'] = null; | 695 model['b'] = null; |
| 228 performMicrotaskCheckpoint(); | 696 }).then(endOfMicrotask).then((_) { |
| 229 expect(div.nodes.last.text, 'ac'); | 697 expect(div.nodes.last.text, 'ac'); |
| 230 | 698 |
| 231 model = null; | 699 model = null; |
| 232 performMicrotaskCheckpoint(); | 700 }).then(endOfMicrotask).then((_) { |
| 233 // setting model isn't observable. | 701 // setting model isn't bindable. |
| 234 expect(div.nodes.last.text, 'ac'); | 702 expect(div.nodes.last.text, 'ac'); |
| 235 }); | 703 }); |
| 236 | 704 }); |
| 237 observeTest('TextTemplateWithBindingPath', () { | 705 |
| 706 test('TextTemplateWithBindingPath', () { |
| 238 var div = createTestHtml( | 707 var div = createTestHtml( |
| 239 '<template bind="{{ data }}">a{{b}}c</template>'); | 708 '<template bind="{{ data }}">a{{b}}c</template>'); |
| 240 var model = toObservable({ 'data': {'b': 'B'} }); | 709 var model = toObservable({ 'data': {'b': 'B'} }); |
| 241 var template = div.firstChild; | 710 var template = div.firstChild; |
| 242 templateBind(template).model = model; | 711 templateBind(template).model = model; |
| 243 | 712 |
| 244 performMicrotaskCheckpoint(); | 713 return new Future(() { |
| 245 expect(div.nodes.length, 2); | 714 expect(div.nodes.length, 2); |
| 246 expect(div.nodes.last.text, 'aBc'); | 715 expect(div.nodes.last.text, 'aBc'); |
| 247 | 716 |
| 248 model['data']['b'] = 'b'; | 717 model['data']['b'] = 'b'; |
| 249 performMicrotaskCheckpoint(); | 718 }).then(endOfMicrotask).then((_) { |
| 250 expect(div.nodes.last.text, 'abc'); | 719 expect(div.nodes.last.text, 'abc'); |
| 251 | 720 |
| 252 model['data'] = toObservable({'b': 'X'}); | 721 model['data'] = toObservable({'b': 'X'}); |
| 253 performMicrotaskCheckpoint(); | 722 }).then(endOfMicrotask).then((_) { |
| 254 expect(div.nodes.last.text, 'aXc'); | 723 expect(div.nodes.last.text, 'aXc'); |
| 255 | 724 |
| 256 // Dart note: changed from `null` since our null means don't render a model. | 725 // Dart note: changed from `null` since our null means don't render a mode
l. |
| 257 model['data'] = toObservable({}); | 726 model['data'] = toObservable({}); |
| 258 performMicrotaskCheckpoint(); | 727 }).then(endOfMicrotask).then((_) { |
| 259 expect(div.nodes.last.text, 'ac'); | 728 expect(div.nodes.last.text, 'ac'); |
| 260 | 729 |
| 261 model['data'] = null; | 730 model['data'] = null; |
| 262 performMicrotaskCheckpoint(); | 731 }).then(endOfMicrotask).then((_) { |
| 263 expect(div.nodes.length, 1); | 732 expect(div.nodes.length, 1); |
| 264 }); | 733 }); |
| 265 | 734 }); |
| 266 observeTest('TextTemplateWithBindingAndConditional', () { | 735 |
| 736 test('TextTemplateWithBindingAndConditional', () { |
| 267 var div = createTestHtml( | 737 var div = createTestHtml( |
| 268 '<template bind="{{}}" if="{{ d }}">a{{b}}c</template>'); | 738 '<template bind="{{}}" if="{{ d }}">a{{b}}c</template>'); |
| 269 var model = toObservable({'b': 'B', 'd': 1}); | 739 var model = toObservable({'b': 'B', 'd': 1}); |
| 270 recursivelySetTemplateModel(div, model); | 740 recursivelySetTemplateModel(div, model); |
| 271 | 741 |
| 272 performMicrotaskCheckpoint(); | 742 return new Future(() { |
| 273 expect(div.nodes.length, 2); | 743 expect(div.nodes.length, 2); |
| 274 expect(div.nodes.last.text, 'aBc'); | 744 expect(div.nodes.last.text, 'aBc'); |
| 275 | 745 |
| 276 model['b'] = 'b'; | 746 model['b'] = 'b'; |
| 277 performMicrotaskCheckpoint(); | 747 }).then(endOfMicrotask).then((_) { |
| 278 expect(div.nodes.last.text, 'abc'); | 748 expect(div.nodes.last.text, 'abc'); |
| 279 | 749 |
| 280 // TODO(jmesserly): MDV set this to empty string and relies on JS conversion | 750 // TODO(jmesserly): MDV set this to empty string and relies on JS conversi
on |
| 281 // rules. Is that intended? | 751 // rules. Is that intended? |
| 282 // See https://github.com/toolkitchen/mdv/issues/59 | 752 // See https://github.com/Polymer/TemplateBinding/issues/59 |
| 283 model['d'] = null; | 753 model['d'] = null; |
| 284 performMicrotaskCheckpoint(); | 754 }).then(endOfMicrotask).then((_) { |
| 285 expect(div.nodes.length, 1); | 755 expect(div.nodes.length, 1); |
| 286 | 756 |
| 287 model['d'] = 'here'; | 757 model['d'] = 'here'; |
| 288 model['b'] = 'd'; | 758 model['b'] = 'd'; |
| 289 | 759 |
| 290 performMicrotaskCheckpoint(); | 760 }).then(endOfMicrotask).then((_) { |
| 291 expect(div.nodes.length, 2); | 761 expect(div.nodes.length, 2); |
| 292 expect(div.nodes.last.text, 'adc'); | 762 expect(div.nodes.last.text, 'adc'); |
| 293 }); | 763 }); |
| 294 | 764 }); |
| 295 observeTest('TemplateWithTextBinding2', () { | 765 |
| 766 test('TemplateWithTextBinding2', () { |
| 296 var div = createTestHtml( | 767 var div = createTestHtml( |
| 297 '<template bind="{{ b }}">a{{value}}c</template>'); | 768 '<template bind="{{ b }}">a{{value}}c</template>'); |
| 298 expect(div.nodes.length, 1); | 769 expect(div.nodes.length, 1); |
| 299 var model = toObservable({'b': {'value': 'B'}}); | 770 var model = toObservable({'b': {'value': 'B'}}); |
| 300 recursivelySetTemplateModel(div, model); | 771 recursivelySetTemplateModel(div, model); |
| 301 | 772 |
| 302 performMicrotaskCheckpoint(); | 773 return new Future(() { |
| 303 expect(div.nodes.length, 2); | 774 expect(div.nodes.length, 2); |
| 304 expect(div.nodes.last.text, 'aBc'); | 775 expect(div.nodes.last.text, 'aBc'); |
| 305 | 776 |
| 306 model['b'] = toObservable({'value': 'b'}); | 777 model['b'] = toObservable({'value': 'b'}); |
| 307 performMicrotaskCheckpoint(); | 778 }).then(endOfMicrotask).then((_) { |
| 308 expect(div.nodes.last.text, 'abc'); | 779 expect(div.nodes.last.text, 'abc'); |
| 309 }); | 780 }); |
| 310 | 781 }); |
| 311 observeTest('TemplateWithAttributeBinding', () { | 782 |
| 783 test('TemplateWithAttributeBinding', () { |
| 312 var div = createTestHtml( | 784 var div = createTestHtml( |
| 313 '<template bind="{{}}">' | 785 '<template bind="{{}}">' |
| 314 '<div foo="a{{b}}c"></div>' | 786 '<div foo="a{{b}}c"></div>' |
| 315 '</template>'); | 787 '</template>'); |
| 316 var model = toObservable({'b': 'B'}); | 788 var model = toObservable({'b': 'B'}); |
| 317 recursivelySetTemplateModel(div, model); | 789 recursivelySetTemplateModel(div, model); |
| 318 | 790 |
| 319 performMicrotaskCheckpoint(); | 791 return new Future(() { |
| 320 expect(div.nodes.length, 2); | 792 expect(div.nodes.length, 2); |
| 321 expect(div.nodes.last.attributes['foo'], 'aBc'); | 793 expect(div.nodes.last.attributes['foo'], 'aBc'); |
| 322 | 794 |
| 323 model['b'] = 'b'; | 795 model['b'] = 'b'; |
| 324 performMicrotaskCheckpoint(); | 796 }).then(endOfMicrotask).then((_) { |
| 325 expect(div.nodes.last.attributes['foo'], 'abc'); | 797 expect(div.nodes.last.attributes['foo'], 'abc'); |
| 326 | 798 |
| 327 model['b'] = 'X'; | 799 model['b'] = 'X'; |
| 328 performMicrotaskCheckpoint(); | 800 }).then(endOfMicrotask).then((_) { |
| 329 expect(div.nodes.last.attributes['foo'], 'aXc'); | 801 expect(div.nodes.last.attributes['foo'], 'aXc'); |
| 802 }); |
| 330 }); | 803 }); |
| 331 | 804 |
| 332 observeTest('TemplateWithConditionalBinding', () { | 805 test('TemplateWithConditionalBinding', () { |
| 333 var div = createTestHtml( | 806 var div = createTestHtml( |
| 334 '<template bind="{{}}">' | 807 '<template bind="{{}}">' |
| 335 '<div foo?="{{b}}"></div>' | 808 '<div foo?="{{b}}"></div>' |
| 336 '</template>'); | 809 '</template>'); |
| 337 var model = toObservable({'b': 'b'}); | 810 var model = toObservable({'b': 'b'}); |
| 338 recursivelySetTemplateModel(div, model); | 811 recursivelySetTemplateModel(div, model); |
| 339 | 812 |
| 340 performMicrotaskCheckpoint(); | 813 return new Future(() { |
| 341 expect(div.nodes.length, 2); | 814 expect(div.nodes.length, 2); |
| 342 expect(div.nodes.last.attributes['foo'], ''); | 815 expect(div.nodes.last.attributes['foo'], ''); |
| 343 expect(div.nodes.last.attributes, isNot(contains('foo?'))); | 816 expect(div.nodes.last.attributes, isNot(contains('foo?'))); |
| 344 | 817 |
| 345 model['b'] = null; | 818 model['b'] = null; |
| 346 performMicrotaskCheckpoint(); | 819 }).then(endOfMicrotask).then((_) { |
| 347 expect(div.nodes.last.attributes, isNot(contains('foo'))); | 820 expect(div.nodes.last.attributes, isNot(contains('foo'))); |
| 821 }); |
| 348 }); | 822 }); |
| 349 | 823 |
| 350 observeTest('Repeat', () { | 824 test('Repeat', () { |
| 351 var div = createTestHtml( | 825 var div = createTestHtml( |
| 352 '<template repeat="{{}}"">text</template>'); | 826 '<template repeat="{{ array }}">{{}},</template>'); |
| 827 |
| 828 var model = toObservable({'array': [0, 1, 2]}); |
| 829 var template = templateBind(div.firstChild); |
| 830 template.model = model; |
| 831 |
| 832 return new Future(() { |
| 833 expect(div.nodes.length, 4); |
| 834 expect(div.text, '0,1,2,'); |
| 835 |
| 836 model['array'].length = 1; |
| 837 |
| 838 }).then(endOfMicrotask).then((_) { |
| 839 expect(div.nodes.length, 2); |
| 840 expect(div.text, '0,'); |
| 841 |
| 842 model['array'].addAll([3, 4]); |
| 843 |
| 844 }).then(endOfMicrotask).then((_) { |
| 845 expect(div.nodes.length, 4); |
| 846 expect(div.text, '0,3,4,'); |
| 847 |
| 848 model['array'].removeRange(1, 2); |
| 849 |
| 850 }).then(endOfMicrotask).then((_) { |
| 851 expect(div.nodes.length, 3); |
| 852 expect(div.text, '0,4,'); |
| 853 |
| 854 model['array'].addAll([5, 6]); |
| 855 model['array'] = toObservable(['x', 'y']); |
| 856 |
| 857 }).then(endOfMicrotask).then((_) { |
| 858 expect(div.nodes.length, 3); |
| 859 expect(div.text, 'x,y,'); |
| 860 |
| 861 template.model = null; |
| 862 |
| 863 }).then(endOfMicrotask).then((_) { |
| 864 expect(div.nodes.length, 1); |
| 865 expect(div.text, ''); |
| 866 }); |
| 867 }); |
| 868 |
| 869 test('Repeat - oneTime', () { |
| 870 var div = createTestHtml('<template repeat="[[]]">text</template>'); |
| 353 | 871 |
| 354 var model = toObservable([0, 1, 2]); | 872 var model = toObservable([0, 1, 2]); |
| 355 recursivelySetTemplateModel(div, model); | 873 var template = templateBind(div.firstChild); |
| 874 template.model = model; |
| 356 | 875 |
| 357 performMicrotaskCheckpoint(); | 876 return new Future(() { |
| 358 expect(div.nodes.length, 4); | 877 expect(div.nodes.length, 4); |
| 359 | 878 |
| 360 model.length = 1; | 879 model.length = 1; |
| 361 performMicrotaskCheckpoint(); | 880 }).then(endOfMicrotask).then((_) { |
| 362 expect(div.nodes.length, 2); | 881 expect(div.nodes.length, 4); |
| 363 | 882 |
| 364 model.addAll(toObservable([3, 4])); | 883 model.addAll([3, 4]); |
| 365 performMicrotaskCheckpoint(); | 884 }).then(endOfMicrotask).then((_) { |
| 366 expect(div.nodes.length, 4); | 885 expect(div.nodes.length, 4); |
| 367 | 886 |
| 368 model.removeRange(1, 2); | 887 model.removeRange(1, 2); |
| 369 performMicrotaskCheckpoint(); | 888 }).then(endOfMicrotask).then((_) { |
| 370 expect(div.nodes.length, 3); | 889 expect(div.nodes.length, 4); |
| 890 |
| 891 template.model = null; |
| 892 }).then(endOfMicrotask).then((_) { |
| 893 expect(div.nodes.length, 1); |
| 894 }); |
| 371 }); | 895 }); |
| 372 | 896 |
| 373 observeTest('Repeat - Reuse Instances', () { | 897 test('Repeat - Reuse Instances', () { |
| 374 var div = createTestHtml('<template repeat>{{ val }}</template>'); | 898 var div = createTestHtml('<template repeat>{{ val }}</template>'); |
| 375 | 899 |
| 376 var model = toObservable([ | 900 var model = toObservable([ |
| 377 {'val': 10}, | 901 {'val': 10}, |
| 378 {'val': 5}, | 902 {'val': 5}, |
| 379 {'val': 2}, | 903 {'val': 2}, |
| 380 {'val': 8}, | 904 {'val': 8}, |
| 381 {'val': 1} | 905 {'val': 1} |
| 382 ]); | 906 ]); |
| 383 recursivelySetTemplateModel(div, model); | 907 recursivelySetTemplateModel(div, model); |
| 384 | |
| 385 performMicrotaskCheckpoint(); | |
| 386 expect(div.nodes.length, 6); | |
| 387 var template = div.firstChild; | 908 var template = div.firstChild; |
| 388 | 909 |
| 389 addExpandos(template.nextNode); | 910 return new Future(() { |
| 390 checkExpandos(template.nextNode); | 911 expect(div.nodes.length, 6); |
| 391 | 912 |
| 392 model.sort((a, b) => a['val'] - b['val']); | 913 addExpandos(template.nextNode); |
| 393 performMicrotaskCheckpoint(); | 914 checkExpandos(template.nextNode); |
| 394 checkExpandos(template.nextNode); | |
| 395 | 915 |
| 396 model = toObservable(model.reversed); | 916 model.sort((a, b) => a['val'] - b['val']); |
| 397 recursivelySetTemplateModel(div, model); | 917 }).then(endOfMicrotask).then((_) { |
| 398 performMicrotaskCheckpoint(); | 918 checkExpandos(template.nextNode); |
| 399 checkExpandos(template.nextNode); | |
| 400 | 919 |
| 401 for (var item in model) { | 920 model = toObservable(model.reversed); |
| 402 item['val'] += 1; | 921 recursivelySetTemplateModel(div, model); |
| 403 } | 922 }).then(endOfMicrotask).then((_) { |
| 923 checkExpandos(template.nextNode); |
| 404 | 924 |
| 405 performMicrotaskCheckpoint(); | 925 for (var item in model) { |
| 406 expect(div.nodes[1].text, "11"); | 926 item['val'] += 1; |
| 407 expect(div.nodes[2].text, "9"); | 927 } |
| 408 expect(div.nodes[3].text, "6"); | 928 |
| 409 expect(div.nodes[4].text, "3"); | 929 }).then(endOfMicrotask).then((_) { |
| 410 expect(div.nodes[5].text, "2"); | 930 expect(div.nodes[1].text, "11"); |
| 931 expect(div.nodes[2].text, "9"); |
| 932 expect(div.nodes[3].text, "6"); |
| 933 expect(div.nodes[4].text, "3"); |
| 934 expect(div.nodes[5].text, "2"); |
| 935 }); |
| 411 }); | 936 }); |
| 412 | 937 |
| 413 observeTest('Bind - Reuse Instance', () { | 938 test('Bind - Reuse Instance', () { |
| 414 var div = createTestHtml( | 939 var div = createTestHtml( |
| 415 '<template bind="{{ foo }}">{{ bar }}</template>'); | 940 '<template bind="{{ foo }}">{{ bar }}</template>'); |
| 416 | 941 |
| 417 var model = toObservable({ 'foo': { 'bar': 5 }}); | 942 var model = toObservable({ 'foo': { 'bar': 5 }}); |
| 418 recursivelySetTemplateModel(div, model); | 943 recursivelySetTemplateModel(div, model); |
| 419 | |
| 420 performMicrotaskCheckpoint(); | |
| 421 expect(div.nodes.length, 2); | |
| 422 var template = div.firstChild; | 944 var template = div.firstChild; |
| 423 | 945 |
| 424 addExpandos(template.nextNode); | 946 return new Future(() { |
| 425 checkExpandos(template.nextNode); | 947 expect(div.nodes.length, 2); |
| 426 | 948 |
| 427 model = toObservable({'foo': model['foo']}); | 949 addExpandos(template.nextNode); |
| 428 recursivelySetTemplateModel(div, model); | 950 checkExpandos(template.nextNode); |
| 429 performMicrotaskCheckpoint(); | 951 |
| 430 checkExpandos(template.nextNode); | 952 model = toObservable({'foo': model['foo']}); |
| 953 recursivelySetTemplateModel(div, model); |
| 954 }).then(endOfMicrotask).then((_) { |
| 955 checkExpandos(template.nextNode); |
| 956 }); |
| 431 }); | 957 }); |
| 432 | 958 |
| 433 observeTest('Repeat-Empty', () { | 959 test('Repeat-Empty', () { |
| 434 var div = createTestHtml( | 960 var div = createTestHtml( |
| 435 '<template repeat>text</template>'); | 961 '<template repeat>text</template>'); |
| 436 | 962 |
| 437 var model = toObservable([0, 1, 2]); | 963 var model = toObservable([0, 1, 2]); |
| 438 recursivelySetTemplateModel(div, model); | 964 recursivelySetTemplateModel(div, model); |
| 439 | 965 |
| 440 performMicrotaskCheckpoint(); | 966 return new Future(() { |
| 441 expect(div.nodes.length, 4); | 967 expect(div.nodes.length, 4); |
| 442 | 968 |
| 443 model.length = 1; | 969 model.length = 1; |
| 444 performMicrotaskCheckpoint(); | 970 }).then(endOfMicrotask).then((_) { |
| 445 expect(div.nodes.length, 2); | 971 expect(div.nodes.length, 2); |
| 446 | 972 |
| 447 model.addAll(toObservable([3, 4])); | 973 model.addAll(toObservable([3, 4])); |
| 448 performMicrotaskCheckpoint(); | 974 }).then(endOfMicrotask).then((_) { |
| 449 expect(div.nodes.length, 4); | 975 expect(div.nodes.length, 4); |
| 450 | 976 |
| 451 model.removeRange(1, 2); | 977 model.removeRange(1, 2); |
| 452 performMicrotaskCheckpoint(); | 978 }).then(endOfMicrotask).then((_) { |
| 453 expect(div.nodes.length, 3); | 979 expect(div.nodes.length, 3); |
| 980 }); |
| 454 }); | 981 }); |
| 455 | 982 |
| 456 observeTest('Removal from iteration needs to unbind', () { | 983 test('Removal from iteration needs to unbind', () { |
| 457 var div = createTestHtml( | 984 var div = createTestHtml( |
| 458 '<template repeat="{{}}"><a>{{v}}</a></template>'); | 985 '<template repeat="{{}}"><a>{{v}}</a></template>'); |
| 459 var model = toObservable([{'v': 0}, {'v': 1}, {'v': 2}, {'v': 3}, | 986 var model = toObservable([{'v': 0}, {'v': 1}, {'v': 2}, {'v': 3}, |
| 460 {'v': 4}]); | 987 {'v': 4}]); |
| 461 recursivelySetTemplateModel(div, model); | 988 recursivelySetTemplateModel(div, model); |
| 462 performMicrotaskCheckpoint(); | |
| 463 | 989 |
| 464 var nodes = div.nodes.skip(1).toList(); | 990 var nodes, vs; |
| 465 var vs = model.toList(); | 991 return new Future(() { |
| 466 | 992 |
| 467 for (var i = 0; i < 5; i++) { | 993 nodes = div.nodes.skip(1).toList(); |
| 468 expect(nodes[i].text, '$i'); | 994 vs = model.toList(); |
| 469 } | |
| 470 | 995 |
| 471 model.length = 3; | 996 for (var i = 0; i < 5; i++) { |
| 472 performMicrotaskCheckpoint(); | 997 expect(nodes[i].text, '$i'); |
| 473 for (var i = 0; i < 5; i++) { | 998 } |
| 474 expect(nodes[i].text, '$i'); | |
| 475 } | |
| 476 | 999 |
| 477 vs[3]['v'] = 33; | 1000 model.length = 3; |
| 478 vs[4]['v'] = 44; | 1001 }).then(endOfMicrotask).then((_) { |
| 479 performMicrotaskCheckpoint(); | 1002 for (var i = 0; i < 5; i++) { |
| 480 for (var i = 0; i < 5; i++) { | 1003 expect(nodes[i].text, '$i'); |
| 481 expect(nodes[i].text, '$i'); | 1004 } |
| 482 } | 1005 |
| 1006 vs[3]['v'] = 33; |
| 1007 vs[4]['v'] = 44; |
| 1008 }).then(endOfMicrotask).then((_) { |
| 1009 for (var i = 0; i < 5; i++) { |
| 1010 expect(nodes[i].text, '$i'); |
| 1011 } |
| 1012 }); |
| 483 }); | 1013 }); |
| 484 | 1014 |
| 485 observeTest('DOM Stability on Iteration', () { | 1015 test('DOM Stability on Iteration', () { |
| 486 var div = createTestHtml( | 1016 var div = createTestHtml( |
| 487 '<template repeat="{{}}">{{}}</template>'); | 1017 '<template repeat="{{}}">{{}}</template>'); |
| 488 var model = toObservable([1, 2, 3, 4, 5]); | 1018 var model = toObservable([1, 2, 3, 4, 5]); |
| 489 recursivelySetTemplateModel(div, model); | 1019 recursivelySetTemplateModel(div, model); |
| 490 | 1020 |
| 491 performMicrotaskCheckpoint(); | 1021 var nodes; |
| 1022 return new Future(() { |
| 1023 // Note: the node at index 0 is the <template>. |
| 1024 nodes = div.nodes.toList(); |
| 1025 expect(nodes.length, 6, reason: 'list has 5 items'); |
| 492 | 1026 |
| 493 // Note: the node at index 0 is the <template>. | 1027 model.removeAt(0); |
| 494 var nodes = div.nodes.toList(); | 1028 model.removeLast(); |
| 495 expect(nodes.length, 6, reason: 'list has 5 items'); | |
| 496 | 1029 |
| 497 model.removeAt(0); | 1030 }).then(endOfMicrotask).then((_) { |
| 498 model.removeLast(); | 1031 expect(div.nodes.length, 4, reason: 'list has 3 items'); |
| 1032 expect(identical(div.nodes[1], nodes[2]), true, reason: '2 not removed'); |
| 1033 expect(identical(div.nodes[2], nodes[3]), true, reason: '3 not removed'); |
| 1034 expect(identical(div.nodes[3], nodes[4]), true, reason: '4 not removed'); |
| 499 | 1035 |
| 500 performMicrotaskCheckpoint(); | 1036 model.insert(0, 5); |
| 501 expect(div.nodes.length, 4, reason: 'list has 3 items'); | 1037 model[2] = 6; |
| 502 expect(identical(div.nodes[1], nodes[2]), true, reason: '2 not removed'); | 1038 model.add(7); |
| 503 expect(identical(div.nodes[2], nodes[3]), true, reason: '3 not removed'); | |
| 504 expect(identical(div.nodes[3], nodes[4]), true, reason: '4 not removed'); | |
| 505 | 1039 |
| 506 model.insert(0, 5); | 1040 }).then(endOfMicrotask).then((_) { |
| 507 model[2] = 6; | |
| 508 model.add(7); | |
| 509 | 1041 |
| 510 performMicrotaskCheckpoint(); | 1042 expect(div.nodes.length, 6, reason: 'list has 5 items'); |
| 1043 expect(nodes.contains(div.nodes[1]), false, reason: '5 is a new node'); |
| 1044 expect(identical(div.nodes[2], nodes[2]), true); |
| 1045 expect(nodes.contains(div.nodes[3]), false, reason: '6 is a new node'); |
| 1046 expect(identical(div.nodes[4], nodes[4]), true); |
| 1047 expect(nodes.contains(div.nodes[5]), false, reason: '7 is a new node'); |
| 511 | 1048 |
| 512 expect(div.nodes.length, 6, reason: 'list has 5 items'); | 1049 nodes = div.nodes.toList(); |
| 513 expect(nodes.contains(div.nodes[1]), false, reason: '5 is a new node'); | |
| 514 expect(identical(div.nodes[2], nodes[2]), true); | |
| 515 expect(nodes.contains(div.nodes[3]), false, reason: '6 is a new node'); | |
| 516 expect(identical(div.nodes[4], nodes[4]), true); | |
| 517 expect(nodes.contains(div.nodes[5]), false, reason: '7 is a new node'); | |
| 518 | 1050 |
| 519 nodes = div.nodes.toList(); | 1051 model.insert(2, 8); |
| 520 | 1052 |
| 521 model.insert(2, 8); | 1053 }).then(endOfMicrotask).then((_) { |
| 522 | 1054 |
| 523 performMicrotaskCheckpoint(); | 1055 expect(div.nodes.length, 7, reason: 'list has 6 items'); |
| 524 | 1056 expect(identical(div.nodes[1], nodes[1]), true); |
| 525 expect(div.nodes.length, 7, reason: 'list has 6 items'); | 1057 expect(identical(div.nodes[2], nodes[2]), true); |
| 526 expect(identical(div.nodes[1], nodes[1]), true); | 1058 expect(nodes.contains(div.nodes[3]), false, reason: '8 is a new node'); |
| 527 expect(identical(div.nodes[2], nodes[2]), true); | 1059 expect(identical(div.nodes[4], nodes[3]), true); |
| 528 expect(nodes.contains(div.nodes[3]), false, reason: '8 is a new node'); | 1060 expect(identical(div.nodes[5], nodes[4]), true); |
| 529 expect(identical(div.nodes[4], nodes[3]), true); | 1061 expect(identical(div.nodes[6], nodes[5]), true); |
| 530 expect(identical(div.nodes[5], nodes[4]), true); | 1062 }); |
| 531 expect(identical(div.nodes[6], nodes[5]), true); | |
| 532 }); | 1063 }); |
| 533 | 1064 |
| 534 observeTest('Repeat2', () { | 1065 test('Repeat2', () { |
| 535 var div = createTestHtml( | 1066 var div = createTestHtml( |
| 536 '<template repeat="{{}}">{{value}}</template>'); | 1067 '<template repeat="{{}}">{{value}}</template>'); |
| 537 expect(div.nodes.length, 1); | 1068 expect(div.nodes.length, 1); |
| 538 | 1069 |
| 539 var model = toObservable([ | 1070 var model = toObservable([ |
| 540 {'value': 0}, | 1071 {'value': 0}, |
| 541 {'value': 1}, | 1072 {'value': 1}, |
| 542 {'value': 2} | 1073 {'value': 2} |
| 543 ]); | 1074 ]); |
| 544 recursivelySetTemplateModel(div, model); | 1075 recursivelySetTemplateModel(div, model); |
| 545 | 1076 |
| 546 performMicrotaskCheckpoint(); | 1077 return new Future(() { |
| 547 expect(div.nodes.length, 4); | 1078 expect(div.nodes.length, 4); |
| 548 expect(div.nodes[1].text, '0'); | 1079 expect(div.nodes[1].text, '0'); |
| 549 expect(div.nodes[2].text, '1'); | 1080 expect(div.nodes[2].text, '1'); |
| 550 expect(div.nodes[3].text, '2'); | 1081 expect(div.nodes[3].text, '2'); |
| 551 | 1082 |
| 552 model[1]['value'] = 'One'; | 1083 model[1]['value'] = 'One'; |
| 553 performMicrotaskCheckpoint(); | 1084 }).then(endOfMicrotask).then((_) { |
| 554 expect(div.nodes.length, 4); | 1085 expect(div.nodes.length, 4); |
| 555 expect(div.nodes[1].text, '0'); | 1086 expect(div.nodes[1].text, '0'); |
| 556 expect(div.nodes[2].text, 'One'); | 1087 expect(div.nodes[2].text, 'One'); |
| 557 expect(div.nodes[3].text, '2'); | 1088 expect(div.nodes[3].text, '2'); |
| 558 | 1089 |
| 559 model.replaceRange(0, 1, toObservable([{'value': 'Zero'}])); | 1090 model.replaceRange(0, 1, toObservable([{'value': 'Zero'}])); |
| 560 performMicrotaskCheckpoint(); | 1091 }).then(endOfMicrotask).then((_) { |
| 561 expect(div.nodes.length, 4); | 1092 expect(div.nodes.length, 4); |
| 562 expect(div.nodes[1].text, 'Zero'); | 1093 expect(div.nodes[1].text, 'Zero'); |
| 563 expect(div.nodes[2].text, 'One'); | 1094 expect(div.nodes[2].text, 'One'); |
| 564 expect(div.nodes[3].text, '2'); | 1095 expect(div.nodes[3].text, '2'); |
| 1096 }); |
| 565 }); | 1097 }); |
| 566 | 1098 |
| 567 observeTest('TemplateWithInputValue', () { | 1099 test('TemplateWithInputValue', () { |
| 568 var div = createTestHtml( | 1100 var div = createTestHtml( |
| 569 '<template bind="{{}}">' | 1101 '<template bind="{{}}">' |
| 570 '<input value="{{x}}">' | 1102 '<input value="{{x}}">' |
| 571 '</template>'); | 1103 '</template>'); |
| 572 var model = toObservable({'x': 'hi'}); | 1104 var model = toObservable({'x': 'hi'}); |
| 573 recursivelySetTemplateModel(div, model); | 1105 recursivelySetTemplateModel(div, model); |
| 574 | 1106 |
| 575 performMicrotaskCheckpoint(); | 1107 return new Future(() { |
| 576 expect(div.nodes.length, 2); | 1108 expect(div.nodes.length, 2); |
| 577 expect(div.nodes.last.value, 'hi'); | 1109 expect(div.nodes.last.value, 'hi'); |
| 578 | 1110 |
| 579 model['x'] = 'bye'; | 1111 model['x'] = 'bye'; |
| 580 expect(div.nodes.last.value, 'hi'); | 1112 expect(div.nodes.last.value, 'hi'); |
| 581 performMicrotaskCheckpoint(); | 1113 }).then(endOfMicrotask).then((_) { |
| 582 expect(div.nodes.last.value, 'bye'); | 1114 expect(div.nodes.last.value, 'bye'); |
| 583 | 1115 |
| 584 div.nodes.last.value = 'hello'; | 1116 div.nodes.last.value = 'hello'; |
| 585 dispatchEvent('input', div.nodes.last); | 1117 dispatchEvent('input', div.nodes.last); |
| 586 expect(model['x'], 'hello'); | 1118 expect(model['x'], 'hello'); |
| 587 performMicrotaskCheckpoint(); | 1119 }).then(endOfMicrotask).then((_) { |
| 588 expect(div.nodes.last.value, 'hello'); | 1120 expect(div.nodes.last.value, 'hello'); |
| 1121 }); |
| 589 }); | 1122 }); |
| 590 | 1123 |
| 591 ////////////////////////////////////////////////////////////////////////////// | 1124 ////////////////////////////////////////////////////////////////////////////// |
| 592 | 1125 |
| 593 observeTest('Decorated', () { | 1126 test('Decorated', () { |
| 594 var div = createTestHtml( | 1127 var div = createTestHtml( |
| 595 '<template bind="{{ XX }}" id="t1">' | 1128 '<template bind="{{ XX }}" id="t1">' |
| 596 '<p>Crew member: {{name}}, Job title: {{title}}</p>' | 1129 '<p>Crew member: {{name}}, Job title: {{title}}</p>' |
| 597 '</template>' | 1130 '</template>' |
| 598 '<template bind="{{ XY }}" id="t2" ref="t1"></template>'); | 1131 '<template bind="{{ XY }}" id="t2" ref="t1"></template>'); |
| 599 | 1132 |
| 600 var model = toObservable({ | 1133 var model = toObservable({ |
| 601 'XX': {'name': 'Leela', 'title': 'Captain'}, | 1134 'XX': {'name': 'Leela', 'title': 'Captain'}, |
| 602 'XY': {'name': 'Fry', 'title': 'Delivery boy'}, | 1135 'XY': {'name': 'Fry', 'title': 'Delivery boy'}, |
| 603 'XZ': {'name': 'Zoidberg', 'title': 'Doctor'} | 1136 'XZ': {'name': 'Zoidberg', 'title': 'Doctor'} |
| 604 }); | 1137 }); |
| 605 recursivelySetTemplateModel(div, model); | 1138 recursivelySetTemplateModel(div, model); |
| 606 | 1139 |
| 607 performMicrotaskCheckpoint(); | 1140 return new Future(() { |
| 1141 var t1 = document.getElementById('t1'); |
| 1142 var instance = t1.nextElementSibling; |
| 1143 expect(instance.text, 'Crew member: Leela, Job title: Captain'); |
| 608 | 1144 |
| 609 var t1 = document.getElementById('t1'); | 1145 var t2 = document.getElementById('t2'); |
| 610 var instance = t1.nextElementSibling; | 1146 instance = t2.nextElementSibling; |
| 611 expect(instance.text, 'Crew member: Leela, Job title: Captain'); | 1147 expect(instance.text, 'Crew member: Fry, Job title: Delivery boy'); |
| 612 | 1148 |
| 613 var t2 = document.getElementById('t2'); | 1149 expect(div.children.length, 4); |
| 614 instance = t2.nextElementSibling; | 1150 expect(div.nodes.length, 4); |
| 615 expect(instance.text, 'Crew member: Fry, Job title: Delivery boy'); | |
| 616 | 1151 |
| 617 expect(div.children.length, 4); | 1152 expect(div.nodes[1].tagName, 'P'); |
| 618 expect(div.nodes.length, 4); | 1153 expect(div.nodes[3].tagName, 'P'); |
| 619 | 1154 }); |
| 620 expect(div.nodes[1].tagName, 'P'); | |
| 621 expect(div.nodes[3].tagName, 'P'); | |
| 622 }); | 1155 }); |
| 623 | 1156 |
| 624 observeTest('DefaultStyles', () { | 1157 test('DefaultStyles', () { |
| 625 var t = new Element.tag('template'); | 1158 var t = new Element.tag('template'); |
| 626 TemplateBindExtension.decorate(t); | 1159 TemplateBindExtension.decorate(t); |
| 627 | 1160 |
| 628 document.body.append(t); | 1161 document.body.append(t); |
| 629 expect(t.getComputedStyle().display, 'none'); | 1162 expect(t.getComputedStyle().display, 'none'); |
| 630 | 1163 |
| 631 t.remove(); | 1164 t.remove(); |
| 632 }); | 1165 }); |
| 633 | 1166 |
| 634 | 1167 |
| 635 observeTest('Bind', () { | 1168 test('Bind', () { |
| 636 var div = createTestHtml('<template bind="{{}}">Hi {{ name }}</template>'); | 1169 var div = createTestHtml('<template bind="{{}}">Hi {{ name }}</template>'); |
| 637 var model = toObservable({'name': 'Leela'}); | 1170 var model = toObservable({'name': 'Leela'}); |
| 638 recursivelySetTemplateModel(div, model); | 1171 recursivelySetTemplateModel(div, model); |
| 639 | 1172 |
| 640 performMicrotaskCheckpoint(); | 1173 return new Future(() => expect(div.nodes[1].text, 'Hi Leela')); |
| 641 expect(div.nodes[1].text, 'Hi Leela'); | |
| 642 }); | 1174 }); |
| 643 | 1175 |
| 644 observeTest('BindImperative', () { | 1176 test('BindPlaceHolderHasNewLine', () { |
| 645 var div = createTestHtml( | |
| 646 '<template>' | |
| 647 'Hi {{ name }}' | |
| 648 '</template>'); | |
| 649 var t = div.nodes.first; | |
| 650 | |
| 651 var model = toObservable({'name': 'Leela'}); | |
| 652 nodeBind(t).bind('bind', model, ''); | |
| 653 | |
| 654 performMicrotaskCheckpoint(); | |
| 655 expect(div.nodes[1].text, 'Hi Leela'); | |
| 656 }); | |
| 657 | |
| 658 observeTest('BindPlaceHolderHasNewLine', () { | |
| 659 var div = createTestHtml( | 1177 var div = createTestHtml( |
| 660 '<template bind="{{}}">Hi {{\nname\n}}</template>'); | 1178 '<template bind="{{}}">Hi {{\nname\n}}</template>'); |
| 661 var model = toObservable({'name': 'Leela'}); | 1179 var model = toObservable({'name': 'Leela'}); |
| 662 recursivelySetTemplateModel(div, model); | 1180 recursivelySetTemplateModel(div, model); |
| 663 | 1181 |
| 664 performMicrotaskCheckpoint(); | 1182 return new Future(() => expect(div.nodes[1].text, 'Hi Leela')); |
| 665 expect(div.nodes[1].text, 'Hi Leela'); | |
| 666 }); | 1183 }); |
| 667 | 1184 |
| 668 observeTest('BindWithRef', () { | 1185 test('BindWithRef', () { |
| 669 var id = 't${new math.Random().nextInt(100)}'; | 1186 var id = 't${new math.Random().nextInt(100)}'; |
| 670 var div = createTestHtml( | 1187 var div = createTestHtml( |
| 671 '<template id="$id">' | 1188 '<template id="$id">' |
| 672 'Hi {{ name }}' | 1189 'Hi {{ name }}' |
| 673 '</template>' | 1190 '</template>' |
| 674 '<template ref="$id" bind="{{}}"></template>'); | 1191 '<template ref="$id" bind="{{}}"></template>'); |
| 675 | 1192 |
| 676 var t1 = div.nodes.first; | 1193 var t1 = div.nodes.first; |
| 677 var t2 = div.nodes[1]; | 1194 var t2 = div.nodes[1]; |
| 678 | 1195 |
| 679 expect(templateBind(t2).ref, t1); | 1196 expect(templateBind(t2).ref, t1); |
| 680 | 1197 |
| 681 var model = toObservable({'name': 'Fry'}); | 1198 var model = toObservable({'name': 'Fry'}); |
| 682 recursivelySetTemplateModel(div, model); | 1199 recursivelySetTemplateModel(div, model); |
| 683 | 1200 |
| 684 performMicrotaskCheckpoint(); | 1201 return new Future(() => expect(t2.nextNode.text, 'Hi Fry')); |
| 685 expect(t2.nextNode.text, 'Hi Fry'); | |
| 686 }); | 1202 }); |
| 687 | 1203 |
| 688 observeTest('BindWithDynamicRef', () { | 1204 |
| 1205 test('Update Ref', () { |
| 1206 var div = createTestHtml( |
| 1207 '<template id=A>Hi, {{}}</template>' |
| 1208 '<template id=B>Hola, {{}}</template>' |
| 1209 '<template ref=A repeat></template>'); |
| 1210 |
| 1211 var model = new ObservableList.from(['Fry']); |
| 1212 recursivelySetTemplateModel(div, model); |
| 1213 |
| 1214 return new Future(() { |
| 1215 expect(div.nodes.length, 4); |
| 1216 expect('Hi, Fry', div.nodes[3].text); |
| 1217 |
| 1218 div.nodes[2].attributes['ref'] = 'B'; |
| 1219 model.add('Leela'); |
| 1220 |
| 1221 }).then(endOfMicrotask).then((x) { |
| 1222 expect(div.nodes.length, 5); |
| 1223 |
| 1224 expect('Hi, Fry', div.nodes[3].text); |
| 1225 expect('Hola, Leela', div.nodes[4].text); |
| 1226 }); |
| 1227 }); |
| 1228 |
| 1229 test('BindWithDynamicRef', () { |
| 689 var id = 't${new math.Random().nextInt(100)}'; | 1230 var id = 't${new math.Random().nextInt(100)}'; |
| 690 var div = createTestHtml( | 1231 var div = createTestHtml( |
| 691 '<template id="$id">' | 1232 '<template id="$id">' |
| 692 'Hi {{ name }}' | 1233 'Hi {{ name }}' |
| 693 '</template>' | 1234 '</template>' |
| 694 '<template ref="{{ id }}" bind="{{}}"></template>'); | 1235 '<template ref="{{ id }}" bind="{{}}"></template>'); |
| 695 | 1236 |
| 696 var t1 = div.firstChild; | 1237 var t1 = div.firstChild; |
| 697 var t2 = div.nodes[1]; | 1238 var t2 = div.nodes[1]; |
| 698 var model = toObservable({'name': 'Fry', 'id': id }); | 1239 var model = toObservable({'name': 'Fry', 'id': id }); |
| 699 recursivelySetTemplateModel(div, model); | 1240 recursivelySetTemplateModel(div, model); |
| 700 | 1241 |
| 701 performMicrotaskCheckpoint(); | 1242 return new Future(() => expect(t2.nextNode.text, 'Hi Fry')); |
| 702 expect(t2.nextNode.text, 'Hi Fry'); | |
| 703 }); | |
| 704 | |
| 705 observeTest('BindChanged', () { | |
| 706 var model = toObservable({ | |
| 707 'XX': {'name': 'Leela', 'title': 'Captain'}, | |
| 708 'XY': {'name': 'Fry', 'title': 'Delivery boy'}, | |
| 709 'XZ': {'name': 'Zoidberg', 'title': 'Doctor'} | |
| 710 }); | |
| 711 | |
| 712 var div = createTestHtml( | |
| 713 '<template bind="{{ XX }}">Hi {{ name }}</template>'); | |
| 714 | |
| 715 recursivelySetTemplateModel(div, model); | |
| 716 | |
| 717 var t = div.nodes.first; | |
| 718 performMicrotaskCheckpoint(); | |
| 719 | |
| 720 expect(div.nodes.length, 2); | |
| 721 expect(t.nextNode.text, 'Hi Leela'); | |
| 722 | |
| 723 nodeBind(t).bind('bind', model, 'XZ'); | |
| 724 performMicrotaskCheckpoint(); | |
| 725 | |
| 726 expect(div.nodes.length, 2); | |
| 727 expect(t.nextNode.text, 'Hi Zoidberg'); | |
| 728 }); | 1243 }); |
| 729 | 1244 |
| 730 assertNodesAre(div, [arguments]) { | 1245 assertNodesAre(div, [arguments]) { |
| 731 var expectedLength = arguments.length; | 1246 var expectedLength = arguments.length; |
| 732 expect(div.nodes.length, expectedLength + 1); | 1247 expect(div.nodes.length, expectedLength + 1); |
| 733 | 1248 |
| 734 for (var i = 0; i < arguments.length; i++) { | 1249 for (var i = 0; i < arguments.length; i++) { |
| 735 var targetNode = div.nodes[i + 1]; | 1250 var targetNode = div.nodes[i + 1]; |
| 736 expect(targetNode.text, arguments[i]); | 1251 expect(targetNode.text, arguments[i]); |
| 737 } | 1252 } |
| 738 } | 1253 } |
| 739 | 1254 |
| 740 observeTest('Repeat3', () { | 1255 test('Repeat3', () { |
| 741 var div = createTestHtml( | 1256 var div = createTestHtml( |
| 742 '<template repeat="{{ contacts }}">Hi {{ name }}</template>'); | 1257 '<template repeat="{{ contacts }}">Hi {{ name }}</template>'); |
| 743 var t = div.nodes.first; | 1258 var t = div.nodes.first; |
| 744 | 1259 |
| 745 var m = toObservable({ | 1260 var m = toObservable({ |
| 746 'contacts': [ | 1261 'contacts': [ |
| 747 {'name': 'Raf'}, | 1262 {'name': 'Raf'}, |
| 748 {'name': 'Arv'}, | 1263 {'name': 'Arv'}, |
| 749 {'name': 'Neal'} | 1264 {'name': 'Neal'} |
| 750 ] | 1265 ] |
| 751 }); | 1266 }); |
| 752 | 1267 |
| 753 recursivelySetTemplateModel(div, m); | 1268 recursivelySetTemplateModel(div, m); |
| 754 performMicrotaskCheckpoint(); | 1269 return new Future(() { |
| 755 | 1270 |
| 756 assertNodesAre(div, ['Hi Raf', 'Hi Arv', 'Hi Neal']); | 1271 assertNodesAre(div, ['Hi Raf', 'Hi Arv', 'Hi Neal']); |
| 757 | 1272 |
| 758 m['contacts'].add(toObservable({'name': 'Alex'})); | 1273 m['contacts'].add(toObservable({'name': 'Alex'})); |
| 759 performMicrotaskCheckpoint(); | 1274 }).then(endOfMicrotask).then((_) { |
| 760 assertNodesAre(div, ['Hi Raf', 'Hi Arv', 'Hi Neal', 'Hi Alex']); | 1275 assertNodesAre(div, ['Hi Raf', 'Hi Arv', 'Hi Neal', 'Hi Alex']); |
| 761 | 1276 |
| 762 m['contacts'].replaceRange(0, 2, | 1277 m['contacts'].replaceRange(0, 2, |
| 763 toObservable([{'name': 'Rafael'}, {'name': 'Erik'}])); | 1278 toObservable([{'name': 'Rafael'}, {'name': 'Erik'}])); |
| 764 performMicrotaskCheckpoint(); | 1279 }).then(endOfMicrotask).then((_) { |
| 765 assertNodesAre(div, ['Hi Rafael', 'Hi Erik', 'Hi Neal', 'Hi Alex']); | 1280 assertNodesAre(div, ['Hi Rafael', 'Hi Erik', 'Hi Neal', 'Hi Alex']); |
| 766 | 1281 |
| 767 m['contacts'].removeRange(1, 3); | 1282 m['contacts'].removeRange(1, 3); |
| 768 performMicrotaskCheckpoint(); | 1283 }).then(endOfMicrotask).then((_) { |
| 769 assertNodesAre(div, ['Hi Rafael', 'Hi Alex']); | 1284 assertNodesAre(div, ['Hi Rafael', 'Hi Alex']); |
| 770 | 1285 |
| 771 m['contacts'].insertAll(1, | 1286 m['contacts'].insertAll(1, |
| 772 toObservable([{'name': 'Erik'}, {'name': 'Dimitri'}])); | 1287 toObservable([{'name': 'Erik'}, {'name': 'Dimitri'}])); |
| 773 performMicrotaskCheckpoint(); | 1288 }).then(endOfMicrotask).then((_) { |
| 774 assertNodesAre(div, ['Hi Rafael', 'Hi Erik', 'Hi Dimitri', 'Hi Alex']); | 1289 assertNodesAre(div, ['Hi Rafael', 'Hi Erik', 'Hi Dimitri', 'Hi Alex']); |
| 775 | 1290 |
| 776 m['contacts'].replaceRange(0, 1, | 1291 m['contacts'].replaceRange(0, 1, |
| 777 toObservable([{'name': 'Tab'}, {'name': 'Neal'}])); | 1292 toObservable([{'name': 'Tab'}, {'name': 'Neal'}])); |
| 778 performMicrotaskCheckpoint(); | 1293 }).then(endOfMicrotask).then((_) { |
| 779 assertNodesAre(div, ['Hi Tab', 'Hi Neal', 'Hi Erik', 'Hi Dimitri', | 1294 assertNodesAre(div, ['Hi Tab', 'Hi Neal', 'Hi Erik', 'Hi Dimitri', |
| 780 'Hi Alex']); | 1295 'Hi Alex']); |
| 781 | 1296 |
| 782 m['contacts'] = toObservable([{'name': 'Alex'}]); | 1297 m['contacts'] = toObservable([{'name': 'Alex'}]); |
| 783 performMicrotaskCheckpoint(); | 1298 }).then(endOfMicrotask).then((_) { |
| 784 assertNodesAre(div, ['Hi Alex']); | 1299 assertNodesAre(div, ['Hi Alex']); |
| 785 | 1300 |
| 786 m['contacts'].length = 0; | 1301 m['contacts'].length = 0; |
| 787 performMicrotaskCheckpoint(); | 1302 }).then(endOfMicrotask).then((_) { |
| 788 assertNodesAre(div, []); | 1303 assertNodesAre(div, []); |
| 1304 }); |
| 789 }); | 1305 }); |
| 790 | 1306 |
| 791 observeTest('RepeatModelSet', () { | 1307 test('RepeatModelSet', () { |
| 792 var div = createTestHtml( | 1308 var div = createTestHtml( |
| 793 '<template repeat="{{ contacts }}">' | 1309 '<template repeat="{{ contacts }}">' |
| 794 'Hi {{ name }}' | 1310 'Hi {{ name }}' |
| 795 '</template>'); | 1311 '</template>'); |
| 796 var m = toObservable({ | 1312 var m = toObservable({ |
| 797 'contacts': [ | 1313 'contacts': [ |
| 798 {'name': 'Raf'}, | 1314 {'name': 'Raf'}, |
| 799 {'name': 'Arv'}, | 1315 {'name': 'Arv'}, |
| 800 {'name': 'Neal'} | 1316 {'name': 'Neal'} |
| 801 ] | 1317 ] |
| 802 }); | 1318 }); |
| 803 recursivelySetTemplateModel(div, m); | 1319 recursivelySetTemplateModel(div, m); |
| 804 | 1320 return new Future(() { |
| 805 performMicrotaskCheckpoint(); | 1321 var t = div.nodes.first; |
| 806 var t = div.nodes.first; | 1322 assertNodesAre(div, ['Hi Raf', 'Hi Arv', 'Hi Neal']); |
| 807 | 1323 }); |
| 808 assertNodesAre(div, ['Hi Raf', 'Hi Arv', 'Hi Neal']); | |
| 809 }); | 1324 }); |
| 810 | 1325 |
| 811 observeTest('RepeatEmptyPath', () { | 1326 test('RepeatEmptyPath', () { |
| 812 var div = createTestHtml( | 1327 var div = createTestHtml( |
| 813 '<template repeat="{{}}">Hi {{ name }}</template>'); | 1328 '<template repeat="{{}}">Hi {{ name }}</template>'); |
| 814 var t = div.nodes.first; | 1329 var t = div.nodes.first; |
| 815 | 1330 |
| 816 var m = toObservable([ | 1331 var m = toObservable([ |
| 817 {'name': 'Raf'}, | 1332 {'name': 'Raf'}, |
| 818 {'name': 'Arv'}, | 1333 {'name': 'Arv'}, |
| 819 {'name': 'Neal'} | 1334 {'name': 'Neal'} |
| 820 ]); | 1335 ]); |
| 821 recursivelySetTemplateModel(div, m); | 1336 recursivelySetTemplateModel(div, m); |
| 1337 return new Future(() { |
| 822 | 1338 |
| 823 performMicrotaskCheckpoint(); | 1339 assertNodesAre(div, ['Hi Raf', 'Hi Arv', 'Hi Neal']); |
| 824 | 1340 |
| 825 assertNodesAre(div, ['Hi Raf', 'Hi Arv', 'Hi Neal']); | 1341 m.add(toObservable({'name': 'Alex'})); |
| 1342 }).then(endOfMicrotask).then((_) { |
| 1343 assertNodesAre(div, ['Hi Raf', 'Hi Arv', 'Hi Neal', 'Hi Alex']); |
| 826 | 1344 |
| 827 m.add(toObservable({'name': 'Alex'})); | 1345 m.replaceRange(0, 2, toObservable([{'name': 'Rafael'}, {'name': 'Erik'}]))
; |
| 828 performMicrotaskCheckpoint(); | 1346 }).then(endOfMicrotask).then((_) { |
| 829 assertNodesAre(div, ['Hi Raf', 'Hi Arv', 'Hi Neal', 'Hi Alex']); | 1347 assertNodesAre(div, ['Hi Rafael', 'Hi Erik', 'Hi Neal', 'Hi Alex']); |
| 830 | 1348 |
| 831 m.replaceRange(0, 2, toObservable([{'name': 'Rafael'}, {'name': 'Erik'}])); | 1349 m.removeRange(1, 3); |
| 832 performMicrotaskCheckpoint(); | 1350 }).then(endOfMicrotask).then((_) { |
| 833 assertNodesAre(div, ['Hi Rafael', 'Hi Erik', 'Hi Neal', 'Hi Alex']); | 1351 assertNodesAre(div, ['Hi Rafael', 'Hi Alex']); |
| 834 | 1352 |
| 835 m.removeRange(1, 3); | 1353 m.insertAll(1, toObservable([{'name': 'Erik'}, {'name': 'Dimitri'}])); |
| 836 performMicrotaskCheckpoint(); | 1354 }).then(endOfMicrotask).then((_) { |
| 837 assertNodesAre(div, ['Hi Rafael', 'Hi Alex']); | 1355 assertNodesAre(div, ['Hi Rafael', 'Hi Erik', 'Hi Dimitri', 'Hi Alex']); |
| 838 | 1356 |
| 839 m.insertAll(1, toObservable([{'name': 'Erik'}, {'name': 'Dimitri'}])); | 1357 m.replaceRange(0, 1, toObservable([{'name': 'Tab'}, {'name': 'Neal'}])); |
| 840 performMicrotaskCheckpoint(); | 1358 }).then(endOfMicrotask).then((_) { |
| 841 assertNodesAre(div, ['Hi Rafael', 'Hi Erik', 'Hi Dimitri', 'Hi Alex']); | 1359 assertNodesAre(div, ['Hi Tab', 'Hi Neal', 'Hi Erik', 'Hi Dimitri', |
| 1360 'Hi Alex']); |
| 842 | 1361 |
| 843 m.replaceRange(0, 1, toObservable([{'name': 'Tab'}, {'name': 'Neal'}])); | 1362 m.length = 0; |
| 844 performMicrotaskCheckpoint(); | 1363 m.add(toObservable({'name': 'Alex'})); |
| 845 assertNodesAre(div, ['Hi Tab', 'Hi Neal', 'Hi Erik', 'Hi Dimitri', | 1364 }).then(endOfMicrotask).then((_) { |
| 846 'Hi Alex']); | 1365 assertNodesAre(div, ['Hi Alex']); |
| 847 | 1366 }); |
| 848 m.length = 0; | |
| 849 m.add(toObservable({'name': 'Alex'})); | |
| 850 performMicrotaskCheckpoint(); | |
| 851 assertNodesAre(div, ['Hi Alex']); | |
| 852 }); | 1367 }); |
| 853 | 1368 |
| 854 observeTest('RepeatNullModel', () { | 1369 test('RepeatNullModel', () { |
| 855 var div = createTestHtml( | 1370 var div = createTestHtml( |
| 856 '<template repeat="{{}}">Hi {{ name }}</template>'); | 1371 '<template repeat="{{}}">Hi {{ name }}</template>'); |
| 857 var t = div.nodes.first; | 1372 var t = div.nodes.first; |
| 858 | 1373 |
| 859 var m = null; | 1374 var m = null; |
| 860 recursivelySetTemplateModel(div, m); | 1375 recursivelySetTemplateModel(div, m); |
| 861 | 1376 |
| 862 expect(div.nodes.length, 1); | 1377 expect(div.nodes.length, 1); |
| 863 | 1378 |
| 864 t.attributes['iterate'] = ''; | 1379 t.attributes['iterate'] = ''; |
| 865 m = toObservable({}); | 1380 m = toObservable({}); |
| 866 recursivelySetTemplateModel(div, m); | 1381 recursivelySetTemplateModel(div, m); |
| 867 | 1382 return new Future(() => expect(div.nodes.length, 1)); |
| 868 performMicrotaskCheckpoint(); | |
| 869 expect(div.nodes.length, 1); | |
| 870 }); | 1383 }); |
| 871 | 1384 |
| 872 observeTest('RepeatReuse', () { | 1385 test('RepeatReuse', () { |
| 873 var div = createTestHtml( | 1386 var div = createTestHtml( |
| 874 '<template repeat="{{}}">Hi {{ name }}</template>'); | 1387 '<template repeat="{{}}">Hi {{ name }}</template>'); |
| 875 var t = div.nodes.first; | 1388 var t = div.nodes.first; |
| 876 | 1389 |
| 877 var m = toObservable([ | 1390 var m = toObservable([ |
| 878 {'name': 'Raf'}, | 1391 {'name': 'Raf'}, |
| 879 {'name': 'Arv'}, | 1392 {'name': 'Arv'}, |
| 880 {'name': 'Neal'} | 1393 {'name': 'Neal'} |
| 881 ]); | 1394 ]); |
| 882 recursivelySetTemplateModel(div, m); | 1395 recursivelySetTemplateModel(div, m); |
| 883 performMicrotaskCheckpoint(); | |
| 884 | 1396 |
| 885 assertNodesAre(div, ['Hi Raf', 'Hi Arv', 'Hi Neal']); | 1397 var node1, node2, node3; |
| 886 var node1 = div.nodes[1]; | 1398 return new Future(() { |
| 887 var node2 = div.nodes[2]; | 1399 assertNodesAre(div, ['Hi Raf', 'Hi Arv', 'Hi Neal']); |
| 888 var node3 = div.nodes[3]; | 1400 node1 = div.nodes[1]; |
| 1401 node2 = div.nodes[2]; |
| 1402 node3 = div.nodes[3]; |
| 889 | 1403 |
| 890 m.replaceRange(1, 2, toObservable([{'name': 'Erik'}])); | 1404 m.replaceRange(1, 2, toObservable([{'name': 'Erik'}])); |
| 891 performMicrotaskCheckpoint(); | 1405 }).then(endOfMicrotask).then((_) { |
| 892 assertNodesAre(div, ['Hi Raf', 'Hi Erik', 'Hi Neal']); | 1406 assertNodesAre(div, ['Hi Raf', 'Hi Erik', 'Hi Neal']); |
| 893 expect(div.nodes[1], node1, | 1407 expect(div.nodes[1], node1, |
| 894 reason: 'model[0] did not change so the node should not have changed'); | 1408 reason: 'model[0] did not change so the node should not have changed')
; |
| 895 expect(div.nodes[2], isNot(equals(node2)), | 1409 expect(div.nodes[2], isNot(equals(node2)), |
| 896 reason: 'Should not reuse when replacing'); | 1410 reason: 'Should not reuse when replacing'); |
| 897 expect(div.nodes[3], node3, | 1411 expect(div.nodes[3], node3, |
| 898 reason: 'model[2] did not change so the node should not have changed'); | 1412 reason: 'model[2] did not change so the node should not have changed')
; |
| 899 | 1413 |
| 900 node2 = div.nodes[2]; | 1414 node2 = div.nodes[2]; |
| 901 m.insert(0, toObservable({'name': 'Alex'})); | 1415 m.insert(0, toObservable({'name': 'Alex'})); |
| 902 performMicrotaskCheckpoint(); | 1416 }).then(endOfMicrotask).then((_) { |
| 903 assertNodesAre(div, ['Hi Alex', 'Hi Raf', 'Hi Erik', 'Hi Neal']); | 1417 assertNodesAre(div, ['Hi Alex', 'Hi Raf', 'Hi Erik', 'Hi Neal']); |
| 1418 }); |
| 904 }); | 1419 }); |
| 905 | 1420 |
| 906 observeTest('TwoLevelsDeepBug', () { | 1421 test('TwoLevelsDeepBug', () { |
| 907 var div = createTestHtml( | 1422 var div = createTestHtml( |
| 908 '<template bind="{{}}"><span><span>{{ foo }}</span></span></template>'); | 1423 '<template bind="{{}}"><span><span>{{ foo }}</span></span></template>'); |
| 909 | 1424 |
| 910 var model = toObservable({'foo': 'bar'}); | 1425 var model = toObservable({'foo': 'bar'}); |
| 911 recursivelySetTemplateModel(div, model); | 1426 recursivelySetTemplateModel(div, model); |
| 912 performMicrotaskCheckpoint(); | 1427 return new Future(() { |
| 913 | 1428 expect(div.nodes[1].nodes[0].nodes[0].text, 'bar'); |
| 914 expect(div.nodes[1].nodes[0].nodes[0].text, 'bar'); | 1429 }); |
| 915 }); | 1430 }); |
| 916 | 1431 |
| 917 observeTest('Checked', () { | 1432 test('Checked', () { |
| 918 var div = createTestHtml( | 1433 var div = createTestHtml( |
| 919 '<template>' | 1434 '<template bind>' |
| 920 '<input type="checkbox" checked="{{a}}">' | 1435 '<input type="checkbox" checked="{{a}}">' |
| 921 '</template>'); | 1436 '</template>'); |
| 922 var t = div.nodes.first; | 1437 var t = div.nodes.first; |
| 923 var m = toObservable({ | 1438 templateBind(t).model = toObservable({'a': true }); |
| 924 'a': true | 1439 |
| 1440 return new Future(() { |
| 1441 |
| 1442 var instanceInput = t.nextNode; |
| 1443 expect(instanceInput.checked, true); |
| 1444 |
| 1445 instanceInput.click(); |
| 1446 expect(instanceInput.checked, false); |
| 1447 |
| 1448 instanceInput.click(); |
| 1449 expect(instanceInput.checked, true); |
| 925 }); | 1450 }); |
| 926 nodeBind(t).bind('bind', m, ''); | |
| 927 performMicrotaskCheckpoint(); | |
| 928 | |
| 929 var instanceInput = t.nextNode; | |
| 930 expect(instanceInput.checked, true); | |
| 931 | |
| 932 instanceInput.click(); | |
| 933 expect(instanceInput.checked, false); | |
| 934 | |
| 935 instanceInput.click(); | |
| 936 expect(instanceInput.checked, true); | |
| 937 }); | 1451 }); |
| 938 | 1452 |
| 939 nestedHelper(s, start) { | 1453 nestedHelper(s, start) { |
| 940 var div = createTestHtml(s); | 1454 var div = createTestHtml(s); |
| 941 | 1455 |
| 942 var m = toObservable({ | 1456 var m = toObservable({ |
| 943 'a': { | 1457 'a': { |
| 944 'b': 1, | 1458 'b': 1, |
| 945 'c': {'d': 2} | 1459 'c': {'d': 2} |
| 946 }, | 1460 }, |
| 947 }); | 1461 }); |
| 948 | 1462 |
| 949 recursivelySetTemplateModel(div, m); | 1463 recursivelySetTemplateModel(div, m); |
| 950 performMicrotaskCheckpoint(); | 1464 return new Future(() { |
| 951 | 1465 |
| 952 var i = start; | 1466 var i = start; |
| 953 expect(div.nodes[i++].text, '1'); | 1467 expect(div.nodes[i++].text, '1'); |
| 954 expect(div.nodes[i++].tagName, 'TEMPLATE'); | 1468 expect(div.nodes[i++].tagName, 'TEMPLATE'); |
| 955 expect(div.nodes[i++].text, '2'); | 1469 expect(div.nodes[i++].text, '2'); |
| 956 | 1470 |
| 957 m['a']['b'] = 11; | 1471 m['a']['b'] = 11; |
| 958 performMicrotaskCheckpoint(); | 1472 }).then(endOfMicrotask).then((_) { |
| 959 expect(div.nodes[start].text, '11'); | 1473 expect(div.nodes[start].text, '11'); |
| 960 | 1474 |
| 961 m['a']['c'] = toObservable({'d': 22}); | 1475 m['a']['c'] = toObservable({'d': 22}); |
| 962 performMicrotaskCheckpoint(); | 1476 }).then(endOfMicrotask).then((_) { |
| 963 expect(div.nodes[start + 2].text, '22'); | 1477 expect(div.nodes[start + 2].text, '22'); |
| 1478 }); |
| 964 } | 1479 } |
| 965 | 1480 |
| 966 observeTest('Nested', () { | 1481 test('Nested', () => nestedHelper( |
| 967 nestedHelper( | 1482 '<template bind="{{a}}">' |
| 968 '<template bind="{{a}}">' | 1483 '{{b}}' |
| 969 '{{b}}' | 1484 '<template bind="{{c}}">' |
| 970 '<template bind="{{c}}">' | 1485 '{{d}}' |
| 971 '{{d}}' | 1486 '</template>' |
| 972 '</template>' | 1487 '</template>', 1)); |
| 973 '</template>', 1); | |
| 974 }); | |
| 975 | 1488 |
| 976 observeTest('NestedWithRef', () { | 1489 test('NestedWithRef', () => nestedHelper( |
| 977 nestedHelper( | |
| 978 '<template id="inner">{{d}}</template>' | 1490 '<template id="inner">{{d}}</template>' |
| 979 '<template id="outer" bind="{{a}}">' | 1491 '<template id="outer" bind="{{a}}">' |
| 980 '{{b}}' | 1492 '{{b}}' |
| 981 '<template ref="inner" bind="{{c}}"></template>' | 1493 '<template ref="inner" bind="{{c}}"></template>' |
| 982 '</template>', 2); | 1494 '</template>', 2)); |
| 983 }); | |
| 984 | 1495 |
| 985 nestedIterateInstantiateHelper(s, start) { | 1496 nestedIterateInstantiateHelper(s, start) { |
| 986 var div = createTestHtml(s); | 1497 var div = createTestHtml(s); |
| 987 | 1498 |
| 988 var m = toObservable({ | 1499 var m = toObservable({ |
| 989 'a': [ | 1500 'a': [ |
| 990 { | 1501 { |
| 991 'b': 1, | 1502 'b': 1, |
| 992 'c': {'d': 11} | 1503 'c': {'d': 11} |
| 993 }, | 1504 }, |
| 994 { | 1505 { |
| 995 'b': 2, | 1506 'b': 2, |
| 996 'c': {'d': 22} | 1507 'c': {'d': 22} |
| 997 } | 1508 } |
| 998 ] | 1509 ] |
| 999 }); | 1510 }); |
| 1000 | 1511 |
| 1001 recursivelySetTemplateModel(div, m); | 1512 recursivelySetTemplateModel(div, m); |
| 1002 performMicrotaskCheckpoint(); | 1513 return new Future(() { |
| 1003 | 1514 |
| 1004 var i = start; | 1515 var i = start; |
| 1005 expect(div.nodes[i++].text, '1'); | 1516 expect(div.nodes[i++].text, '1'); |
| 1006 expect(div.nodes[i++].tagName, 'TEMPLATE'); | 1517 expect(div.nodes[i++].tagName, 'TEMPLATE'); |
| 1007 expect(div.nodes[i++].text, '11'); | 1518 expect(div.nodes[i++].text, '11'); |
| 1008 expect(div.nodes[i++].text, '2'); | 1519 expect(div.nodes[i++].text, '2'); |
| 1009 expect(div.nodes[i++].tagName, 'TEMPLATE'); | 1520 expect(div.nodes[i++].tagName, 'TEMPLATE'); |
| 1010 expect(div.nodes[i++].text, '22'); | 1521 expect(div.nodes[i++].text, '22'); |
| 1011 | 1522 |
| 1012 m['a'][1] = toObservable({ | 1523 m['a'][1] = toObservable({ |
| 1013 'b': 3, | 1524 'b': 3, |
| 1014 'c': {'d': 33} | 1525 'c': {'d': 33} |
| 1526 }); |
| 1527 |
| 1528 }).then(endOfMicrotask).then((_) { |
| 1529 expect(div.nodes[start + 3].text, '3'); |
| 1530 expect(div.nodes[start + 5].text, '33'); |
| 1015 }); | 1531 }); |
| 1016 | |
| 1017 performMicrotaskCheckpoint(); | |
| 1018 expect(div.nodes[start + 3].text, '3'); | |
| 1019 expect(div.nodes[start + 5].text, '33'); | |
| 1020 } | 1532 } |
| 1021 | 1533 |
| 1022 observeTest('NestedRepeatBind', () { | 1534 test('NestedRepeatBind', () => nestedIterateInstantiateHelper( |
| 1023 nestedIterateInstantiateHelper( | 1535 '<template repeat="{{a}}">' |
| 1024 '<template repeat="{{a}}">' | 1536 '{{b}}' |
| 1025 '{{b}}' | 1537 '<template bind="{{c}}">' |
| 1026 '<template bind="{{c}}">' | |
| 1027 '{{d}}' | |
| 1028 '</template>' | |
| 1029 '</template>', 1); | |
| 1030 }); | |
| 1031 | |
| 1032 observeTest('NestedRepeatBindWithRef', () { | |
| 1033 nestedIterateInstantiateHelper( | |
| 1034 '<template id="inner">' | |
| 1035 '{{d}}' | 1538 '{{d}}' |
| 1036 '</template>' | 1539 '</template>' |
| 1037 '<template repeat="{{a}}">' | 1540 '</template>', 1)); |
| 1038 '{{b}}' | 1541 |
| 1039 '<template ref="inner" bind="{{c}}"></template>' | 1542 test('NestedRepeatBindWithRef', () => nestedIterateInstantiateHelper( |
| 1040 '</template>', 2); | 1543 '<template id="inner">' |
| 1041 }); | 1544 '{{d}}' |
| 1545 '</template>' |
| 1546 '<template repeat="{{a}}">' |
| 1547 '{{b}}' |
| 1548 '<template ref="inner" bind="{{c}}"></template>' |
| 1549 '</template>', 2)); |
| 1042 | 1550 |
| 1043 nestedIterateIterateHelper(s, start) { | 1551 nestedIterateIterateHelper(s, start) { |
| 1044 var div = createTestHtml(s); | 1552 var div = createTestHtml(s); |
| 1045 | 1553 |
| 1046 var m = toObservable({ | 1554 var m = toObservable({ |
| 1047 'a': [ | 1555 'a': [ |
| 1048 { | 1556 { |
| 1049 'b': 1, | 1557 'b': 1, |
| 1050 'c': [{'d': 11}, {'d': 12}] | 1558 'c': [{'d': 11}, {'d': 12}] |
| 1051 }, | 1559 }, |
| 1052 { | 1560 { |
| 1053 'b': 2, | 1561 'b': 2, |
| 1054 'c': [{'d': 21}, {'d': 22}] | 1562 'c': [{'d': 21}, {'d': 22}] |
| 1055 } | 1563 } |
| 1056 ] | 1564 ] |
| 1057 }); | 1565 }); |
| 1058 | 1566 |
| 1059 recursivelySetTemplateModel(div, m); | 1567 recursivelySetTemplateModel(div, m); |
| 1060 performMicrotaskCheckpoint(); | 1568 return new Future(() { |
| 1061 | 1569 |
| 1062 var i = start; | 1570 var i = start; |
| 1063 expect(div.nodes[i++].text, '1'); | 1571 expect(div.nodes[i++].text, '1'); |
| 1064 expect(div.nodes[i++].tagName, 'TEMPLATE'); | 1572 expect(div.nodes[i++].tagName, 'TEMPLATE'); |
| 1065 expect(div.nodes[i++].text, '11'); | 1573 expect(div.nodes[i++].text, '11'); |
| 1066 expect(div.nodes[i++].text, '12'); | 1574 expect(div.nodes[i++].text, '12'); |
| 1067 expect(div.nodes[i++].text, '2'); | 1575 expect(div.nodes[i++].text, '2'); |
| 1068 expect(div.nodes[i++].tagName, 'TEMPLATE'); | 1576 expect(div.nodes[i++].tagName, 'TEMPLATE'); |
| 1069 expect(div.nodes[i++].text, '21'); | 1577 expect(div.nodes[i++].text, '21'); |
| 1070 expect(div.nodes[i++].text, '22'); | 1578 expect(div.nodes[i++].text, '22'); |
| 1071 | 1579 |
| 1072 m['a'][1] = toObservable({ | 1580 m['a'][1] = toObservable({ |
| 1073 'b': 3, | 1581 'b': 3, |
| 1074 'c': [{'d': 31}, {'d': 32}, {'d': 33}] | 1582 'c': [{'d': 31}, {'d': 32}, {'d': 33}] |
| 1583 }); |
| 1584 |
| 1585 i = start + 4; |
| 1586 }).then(endOfMicrotask).then((_) { |
| 1587 expect(div.nodes[start + 4].text, '3'); |
| 1588 expect(div.nodes[start + 6].text, '31'); |
| 1589 expect(div.nodes[start + 7].text, '32'); |
| 1590 expect(div.nodes[start + 8].text, '33'); |
| 1075 }); | 1591 }); |
| 1076 | |
| 1077 i = start + 4; | |
| 1078 performMicrotaskCheckpoint(); | |
| 1079 expect(div.nodes[start + 4].text, '3'); | |
| 1080 expect(div.nodes[start + 6].text, '31'); | |
| 1081 expect(div.nodes[start + 7].text, '32'); | |
| 1082 expect(div.nodes[start + 8].text, '33'); | |
| 1083 } | 1592 } |
| 1084 | 1593 |
| 1085 observeTest('NestedRepeatBind', () { | 1594 test('NestedRepeatBind', () => nestedIterateIterateHelper( |
| 1086 nestedIterateIterateHelper( | 1595 '<template repeat="{{a}}">' |
| 1087 '<template repeat="{{a}}">' | 1596 '{{b}}' |
| 1088 '{{b}}' | 1597 '<template repeat="{{c}}">' |
| 1089 '<template repeat="{{c}}">' | |
| 1090 '{{d}}' | |
| 1091 '</template>' | |
| 1092 '</template>', 1); | |
| 1093 }); | |
| 1094 | |
| 1095 observeTest('NestedRepeatRepeatWithRef', () { | |
| 1096 nestedIterateIterateHelper( | |
| 1097 '<template id="inner">' | |
| 1098 '{{d}}' | 1598 '{{d}}' |
| 1099 '</template>' | 1599 '</template>' |
| 1100 '<template repeat="{{a}}">' | 1600 '</template>', 1)); |
| 1101 '{{b}}' | |
| 1102 '<template ref="inner" repeat="{{c}}"></template>' | |
| 1103 '</template>', 2); | |
| 1104 }); | |
| 1105 | 1601 |
| 1106 observeTest('NestedRepeatSelfRef', () { | 1602 test('NestedRepeatRepeatWithRef', () => nestedIterateIterateHelper( |
| 1603 '<template id="inner">' |
| 1604 '{{d}}' |
| 1605 '</template>' |
| 1606 '<template repeat="{{a}}">' |
| 1607 '{{b}}' |
| 1608 '<template ref="inner" repeat="{{c}}"></template>' |
| 1609 '</template>', 2)); |
| 1610 |
| 1611 test('NestedRepeatSelfRef', () { |
| 1107 var div = createTestHtml( | 1612 var div = createTestHtml( |
| 1108 '<template id="t" repeat="{{}}">' | 1613 '<template id="t" repeat="{{}}">' |
| 1109 '{{name}}' | 1614 '{{name}}' |
| 1110 '<template ref="t" repeat="{{items}}"></template>' | 1615 '<template ref="t" repeat="{{items}}"></template>' |
| 1111 '</template>'); | 1616 '</template>'); |
| 1112 | 1617 |
| 1113 var m = toObservable([ | 1618 var m = toObservable([ |
| 1114 { | 1619 { |
| 1115 'name': 'Item 1', | 1620 'name': 'Item 1', |
| 1116 'items': [ | 1621 'items': [ |
| (...skipping 11 matching lines...) Expand all Loading... |
| 1128 } | 1633 } |
| 1129 ] | 1634 ] |
| 1130 }, | 1635 }, |
| 1131 { | 1636 { |
| 1132 'name': 'Item 2', | 1637 'name': 'Item 2', |
| 1133 'items': [] | 1638 'items': [] |
| 1134 }, | 1639 }, |
| 1135 ]); | 1640 ]); |
| 1136 | 1641 |
| 1137 recursivelySetTemplateModel(div, m); | 1642 recursivelySetTemplateModel(div, m); |
| 1138 performMicrotaskCheckpoint(); | |
| 1139 | 1643 |
| 1140 var i = 1; | 1644 int i = 1; |
| 1141 expect(div.nodes[i++].text, 'Item 1'); | 1645 return new Future(() { |
| 1142 expect(div.nodes[i++].tagName, 'TEMPLATE'); | 1646 expect(div.nodes[i++].text, 'Item 1'); |
| 1143 expect(div.nodes[i++].text, 'Item 1.1'); | 1647 expect(div.nodes[i++].tagName, 'TEMPLATE'); |
| 1144 expect(div.nodes[i++].tagName, 'TEMPLATE'); | 1648 expect(div.nodes[i++].text, 'Item 1.1'); |
| 1145 expect(div.nodes[i++].text, 'Item 1.1.1'); | 1649 expect(div.nodes[i++].tagName, 'TEMPLATE'); |
| 1146 expect(div.nodes[i++].tagName, 'TEMPLATE'); | 1650 expect(div.nodes[i++].text, 'Item 1.1.1'); |
| 1147 expect(div.nodes[i++].text, 'Item 1.2'); | 1651 expect(div.nodes[i++].tagName, 'TEMPLATE'); |
| 1148 expect(div.nodes[i++].tagName, 'TEMPLATE'); | 1652 expect(div.nodes[i++].text, 'Item 1.2'); |
| 1149 expect(div.nodes[i++].text, 'Item 2'); | 1653 expect(div.nodes[i++].tagName, 'TEMPLATE'); |
| 1654 expect(div.nodes[i++].text, 'Item 2'); |
| 1150 | 1655 |
| 1151 m[0] = toObservable({'name': 'Item 1 changed'}); | 1656 m[0] = toObservable({'name': 'Item 1 changed'}); |
| 1152 | 1657 |
| 1153 i = 1; | 1658 i = 1; |
| 1154 performMicrotaskCheckpoint(); | 1659 }).then(endOfMicrotask).then((_) { |
| 1155 expect(div.nodes[i++].text, 'Item 1 changed'); | 1660 expect(div.nodes[i++].text, 'Item 1 changed'); |
| 1156 expect(div.nodes[i++].tagName, 'TEMPLATE'); | 1661 expect(div.nodes[i++].tagName, 'TEMPLATE'); |
| 1157 expect(div.nodes[i++].text, 'Item 2'); | 1662 expect(div.nodes[i++].text, 'Item 2'); |
| 1663 }); |
| 1158 }); | 1664 }); |
| 1159 | 1665 |
| 1160 // Note: we don't need a zone for this test, and we don't want to alter timing | 1666 // Note: we don't need a zone for this test, and we don't want to alter timing |
| 1161 // since we're testing a rather subtle relationship between select and option. | 1667 // since we're testing a rather subtle relationship between select and option. |
| 1162 test('Attribute Template Option/Optgroup', () { | 1668 test('Attribute Template Option/Optgroup', () { |
| 1163 var div = createTestHtml( | 1669 var div = createTestHtml( |
| 1164 '<template bind>' | 1670 '<template bind>' |
| 1165 '<select selectedIndex="{{ selected }}">' | 1671 '<select selectedIndex="{{ selected }}">' |
| 1166 '<optgroup template repeat="{{ groups }}" label="{{ name }}">' | 1672 '<optgroup template repeat="{{ groups }}" label="{{ name }}">' |
| 1167 '<option template repeat="{{ items }}">{{ val }}</option>' | 1673 '<option template repeat="{{ items }}">{{ val }}</option>' |
| (...skipping 36 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1204 | 1710 |
| 1205 completer.complete(); | 1711 completer.complete(); |
| 1206 }); | 1712 }); |
| 1207 })..observe(div, childList: true, subtree: true); | 1713 })..observe(div, childList: true, subtree: true); |
| 1208 | 1714 |
| 1209 Observable.dirtyCheck(); | 1715 Observable.dirtyCheck(); |
| 1210 | 1716 |
| 1211 return completer.future; | 1717 return completer.future; |
| 1212 }); | 1718 }); |
| 1213 | 1719 |
| 1214 observeTest('NestedIterateTableMixedSemanticNative', () { | 1720 test('NestedIterateTableMixedSemanticNative', () { |
| 1215 if (!parserHasNativeTemplate) return; | 1721 if (!parserHasNativeTemplate) return null; |
| 1216 | 1722 |
| 1217 var div = createTestHtml( | 1723 var div = createTestHtml( |
| 1218 '<table><tbody>' | 1724 '<table><tbody>' |
| 1219 '<template repeat="{{}}">' | 1725 '<template repeat="{{}}">' |
| 1220 '<tr>' | 1726 '<tr>' |
| 1221 '<td template repeat="{{}}" class="{{ val }}">{{ val }}</td>' | 1727 '<td template repeat="{{}}" class="{{ val }}">{{ val }}</td>' |
| 1222 '</tr>' | 1728 '</tr>' |
| 1223 '</template>' | 1729 '</template>' |
| 1224 '</tbody></table>'); | 1730 '</tbody></table>'); |
| 1225 | 1731 |
| 1226 var m = toObservable([ | 1732 var m = toObservable([ |
| 1227 [{ 'val': 0 }, { 'val': 1 }], | 1733 [{ 'val': 0 }, { 'val': 1 }], |
| 1228 [{ 'val': 2 }, { 'val': 3 }] | 1734 [{ 'val': 2 }, { 'val': 3 }] |
| 1229 ]); | 1735 ]); |
| 1230 | 1736 |
| 1231 recursivelySetTemplateModel(div, m); | 1737 recursivelySetTemplateModel(div, m); |
| 1232 performMicrotaskCheckpoint(); | 1738 return new Future(() { |
| 1739 var tbody = div.nodes[0].nodes[0]; |
| 1233 | 1740 |
| 1234 var tbody = div.nodes[0].nodes[0]; | 1741 // 1 for the <tr template>, 2 * (1 tr) |
| 1742 expect(tbody.nodes.length, 3); |
| 1235 | 1743 |
| 1236 // 1 for the <tr template>, 2 * (1 tr) | 1744 // 1 for the <td template>, 2 * (1 td) |
| 1237 expect(tbody.nodes.length, 3); | 1745 expect(tbody.nodes[1].nodes.length, 3); |
| 1238 | 1746 |
| 1239 // 1 for the <td template>, 2 * (1 td) | 1747 expect(tbody.nodes[1].nodes[1].text, '0'); |
| 1240 expect(tbody.nodes[1].nodes.length, 3); | 1748 expect(tbody.nodes[1].nodes[2].text, '1'); |
| 1241 | 1749 |
| 1242 expect(tbody.nodes[1].nodes[1].text, '0'); | 1750 // 1 for the <td template>, 2 * (1 td) |
| 1243 expect(tbody.nodes[1].nodes[2].text, '1'); | 1751 expect(tbody.nodes[2].nodes.length, 3); |
| 1752 expect(tbody.nodes[2].nodes[1].text, '2'); |
| 1753 expect(tbody.nodes[2].nodes[2].text, '3'); |
| 1244 | 1754 |
| 1245 // 1 for the <td template>, 2 * (1 td) | 1755 // Asset the 'class' binding is retained on the semantic template (just |
| 1246 expect(tbody.nodes[2].nodes.length, 3); | 1756 // check the last one). |
| 1247 expect(tbody.nodes[2].nodes[1].text, '2'); | 1757 expect(tbody.nodes[2].nodes[2].attributes["class"], '3'); |
| 1248 expect(tbody.nodes[2].nodes[2].text, '3'); | 1758 }); |
| 1249 | |
| 1250 // Asset the 'class' binding is retained on the semantic template (just | |
| 1251 // check the last one). | |
| 1252 expect(tbody.nodes[2].nodes[2].attributes["class"], '3'); | |
| 1253 }); | 1759 }); |
| 1254 | 1760 |
| 1255 observeTest('NestedIterateTable', () { | 1761 test('NestedIterateTable', () { |
| 1256 var div = createTestHtml( | 1762 var div = createTestHtml( |
| 1257 '<table><tbody>' | 1763 '<table><tbody>' |
| 1258 '<tr template repeat="{{}}">' | 1764 '<tr template repeat="{{}}">' |
| 1259 '<td template repeat="{{}}" class="{{ val }}">{{ val }}</td>' | 1765 '<td template repeat="{{}}" class="{{ val }}">{{ val }}</td>' |
| 1260 '</tr>' | 1766 '</tr>' |
| 1261 '</tbody></table>'); | 1767 '</tbody></table>'); |
| 1262 | 1768 |
| 1263 var m = toObservable([ | 1769 var m = toObservable([ |
| 1264 [{ 'val': 0 }, { 'val': 1 }], | 1770 [{ 'val': 0 }, { 'val': 1 }], |
| 1265 [{ 'val': 2 }, { 'val': 3 }] | 1771 [{ 'val': 2 }, { 'val': 3 }] |
| 1266 ]); | 1772 ]); |
| 1267 | 1773 |
| 1268 recursivelySetTemplateModel(div, m); | 1774 recursivelySetTemplateModel(div, m); |
| 1269 performMicrotaskCheckpoint(); | 1775 return new Future(() { |
| 1270 | 1776 |
| 1271 var i = 1; | 1777 var i = 1; |
| 1272 var tbody = div.nodes[0].nodes[0]; | 1778 var tbody = div.nodes[0].nodes[0]; |
| 1273 | 1779 |
| 1274 // 1 for the <tr template>, 2 * (1 tr) | 1780 // 1 for the <tr template>, 2 * (1 tr) |
| 1275 expect(tbody.nodes.length, 3); | 1781 expect(tbody.nodes.length, 3); |
| 1276 | 1782 |
| 1277 // 1 for the <td template>, 2 * (1 td) | 1783 // 1 for the <td template>, 2 * (1 td) |
| 1278 expect(tbody.nodes[1].nodes.length, 3); | 1784 expect(tbody.nodes[1].nodes.length, 3); |
| 1279 expect(tbody.nodes[1].nodes[1].text, '0'); | 1785 expect(tbody.nodes[1].nodes[1].text, '0'); |
| 1280 expect(tbody.nodes[1].nodes[2].text, '1'); | 1786 expect(tbody.nodes[1].nodes[2].text, '1'); |
| 1281 | 1787 |
| 1282 // 1 for the <td template>, 2 * (1 td) | 1788 // 1 for the <td template>, 2 * (1 td) |
| 1283 expect(tbody.nodes[2].nodes.length, 3); | 1789 expect(tbody.nodes[2].nodes.length, 3); |
| 1284 expect(tbody.nodes[2].nodes[1].text, '2'); | 1790 expect(tbody.nodes[2].nodes[1].text, '2'); |
| 1285 expect(tbody.nodes[2].nodes[2].text, '3'); | 1791 expect(tbody.nodes[2].nodes[2].text, '3'); |
| 1286 | 1792 |
| 1287 // Asset the 'class' binding is retained on the semantic template (just | 1793 // Asset the 'class' binding is retained on the semantic template (just |
| 1288 // check the last one). | 1794 // check the last one). |
| 1289 expect(tbody.nodes[2].nodes[2].attributes['class'], '3'); | 1795 expect(tbody.nodes[2].nodes[2].attributes['class'], '3'); |
| 1796 }); |
| 1290 }); | 1797 }); |
| 1291 | 1798 |
| 1292 observeTest('NestedRepeatDeletionOfMultipleSubTemplates', () { | 1799 test('NestedRepeatDeletionOfMultipleSubTemplates', () { |
| 1293 var div = createTestHtml( | 1800 var div = createTestHtml( |
| 1294 '<ul>' | 1801 '<ul>' |
| 1295 '<template repeat="{{}}" id=t1>' | 1802 '<template repeat="{{}}" id=t1>' |
| 1296 '<li>{{name}}' | 1803 '<li>{{name}}' |
| 1297 '<ul>' | 1804 '<ul>' |
| 1298 '<template ref=t1 repaet="{{items}}"></template>' | 1805 '<template ref=t1 repaet="{{items}}"></template>' |
| 1299 '</ul>' | 1806 '</ul>' |
| 1300 '</li>' | 1807 '</li>' |
| 1301 '</template>' | 1808 '</template>' |
| 1302 '</ul>'); | 1809 '</ul>'); |
| 1303 | 1810 |
| 1304 var m = toObservable([ | 1811 var m = toObservable([ |
| 1305 { | 1812 { |
| 1306 'name': 'Item 1', | 1813 'name': 'Item 1', |
| 1307 'items': [ | 1814 'items': [ |
| 1308 { | 1815 { |
| 1309 'name': 'Item 1.1' | 1816 'name': 'Item 1.1' |
| 1310 } | 1817 } |
| 1311 ] | 1818 ] |
| 1312 } | 1819 } |
| 1313 ]); | 1820 ]); |
| 1314 | 1821 |
| 1315 recursivelySetTemplateModel(div, m); | 1822 recursivelySetTemplateModel(div, m); |
| 1316 | 1823 return new Future(() => m.removeAt(0)); |
| 1317 performMicrotaskCheckpoint(); | |
| 1318 m.removeAt(0); | |
| 1319 performMicrotaskCheckpoint(); | |
| 1320 }); | 1824 }); |
| 1321 | 1825 |
| 1322 observeTest('DeepNested', () { | 1826 test('DeepNested', () { |
| 1323 var div = createTestHtml( | 1827 var div = createTestHtml( |
| 1324 '<template bind="{{a}}">' | 1828 '<template bind="{{a}}">' |
| 1325 '<p>' | 1829 '<p>' |
| 1326 '<template bind="{{b}}">' | 1830 '<template bind="{{b}}">' |
| 1327 '{{ c }}' | 1831 '{{ c }}' |
| 1328 '</template>' | 1832 '</template>' |
| 1329 '</p>' | 1833 '</p>' |
| 1330 '</template>'); | 1834 '</template>'); |
| 1331 | 1835 |
| 1332 var m = toObservable({ | 1836 var m = toObservable({ |
| 1333 'a': { | 1837 'a': { |
| 1334 'b': { | 1838 'b': { |
| 1335 'c': 42 | 1839 'c': 42 |
| 1336 } | 1840 } |
| 1337 } | 1841 } |
| 1338 }); | 1842 }); |
| 1339 recursivelySetTemplateModel(div, m); | 1843 recursivelySetTemplateModel(div, m); |
| 1340 performMicrotaskCheckpoint(); | 1844 return new Future(() { |
| 1341 | 1845 expect(div.nodes[1].tagName, 'P'); |
| 1342 expect(div.nodes[1].tagName, 'P'); | 1846 expect(div.nodes[1].nodes.first.tagName, 'TEMPLATE'); |
| 1343 expect(div.nodes[1].nodes.first.tagName, 'TEMPLATE'); | 1847 expect(div.nodes[1].nodes[1].text, '42'); |
| 1344 expect(div.nodes[1].nodes[1].text, '42'); | 1848 }); |
| 1345 }); | 1849 }); |
| 1346 | 1850 |
| 1347 observeTest('TemplateContentRemoved', () { | 1851 test('TemplateContentRemoved', () { |
| 1348 var div = createTestHtml('<template bind="{{}}">{{ }}</template>'); | 1852 var div = createTestHtml('<template bind="{{}}">{{ }}</template>'); |
| 1349 var model = 42; | 1853 var model = 42; |
| 1350 | 1854 |
| 1351 recursivelySetTemplateModel(div, model); | 1855 recursivelySetTemplateModel(div, model); |
| 1352 performMicrotaskCheckpoint(); | 1856 return new Future(() { |
| 1353 expect(div.nodes[1].text, '42'); | 1857 expect(div.nodes[1].text, '42'); |
| 1354 expect(div.nodes[0].text, ''); | 1858 expect(div.nodes[0].text, ''); |
| 1859 }); |
| 1355 }); | 1860 }); |
| 1356 | 1861 |
| 1357 observeTest('TemplateContentRemovedEmptyArray', () { | 1862 test('TemplateContentRemovedEmptyArray', () { |
| 1358 var div = createTestHtml('<template iterate>Remove me</template>'); | 1863 var div = createTestHtml('<template iterate>Remove me</template>'); |
| 1359 var model = toObservable([]); | 1864 var model = toObservable([]); |
| 1360 | 1865 |
| 1361 recursivelySetTemplateModel(div, model); | 1866 recursivelySetTemplateModel(div, model); |
| 1362 performMicrotaskCheckpoint(); | 1867 return new Future(() { |
| 1363 expect(div.nodes.length, 1); | 1868 expect(div.nodes.length, 1); |
| 1364 expect(div.nodes[0].text, ''); | 1869 expect(div.nodes[0].text, ''); |
| 1870 }); |
| 1365 }); | 1871 }); |
| 1366 | 1872 |
| 1367 observeTest('TemplateContentRemovedNested', () { | 1873 test('TemplateContentRemovedNested', () { |
| 1368 var div = createTestHtml( | 1874 var div = createTestHtml( |
| 1369 '<template bind="{{}}">' | 1875 '<template bind="{{}}">' |
| 1370 '{{ a }}' | 1876 '{{ a }}' |
| 1371 '<template bind="{{}}">' | 1877 '<template bind="{{}}">' |
| 1372 '{{ b }}' | 1878 '{{ b }}' |
| 1373 '</template>' | 1879 '</template>' |
| 1374 '</template>'); | 1880 '</template>'); |
| 1375 | 1881 |
| 1376 var model = toObservable({ | 1882 var model = toObservable({ |
| 1377 'a': 1, | 1883 'a': 1, |
| 1378 'b': 2 | 1884 'b': 2 |
| 1379 }); | 1885 }); |
| 1380 recursivelySetTemplateModel(div, model); | 1886 recursivelySetTemplateModel(div, model); |
| 1381 performMicrotaskCheckpoint(); | 1887 return new Future(() { |
| 1382 | 1888 expect(div.nodes[0].text, ''); |
| 1383 expect(div.nodes[0].text, ''); | 1889 expect(div.nodes[1].text, '1'); |
| 1384 expect(div.nodes[1].text, '1'); | 1890 expect(div.nodes[2].text, ''); |
| 1385 expect(div.nodes[2].text, ''); | 1891 expect(div.nodes[3].text, '2'); |
| 1386 expect(div.nodes[3].text, '2'); | 1892 }); |
| 1387 }); | 1893 }); |
| 1388 | 1894 |
| 1389 observeTest('BindWithUndefinedModel', () { | 1895 test('BindWithUndefinedModel', () { |
| 1390 var div = createTestHtml( | 1896 var div = createTestHtml( |
| 1391 '<template bind="{{}}" if="{{}}">{{ a }}</template>'); | 1897 '<template bind="{{}}" if="{{}}">{{ a }}</template>'); |
| 1392 | 1898 |
| 1393 var model = toObservable({'a': 42}); | 1899 var model = toObservable({'a': 42}); |
| 1394 recursivelySetTemplateModel(div, model); | 1900 recursivelySetTemplateModel(div, model); |
| 1395 performMicrotaskCheckpoint(); | 1901 return new Future(() { |
| 1396 expect(div.nodes[1].text, '42'); | 1902 expect(div.nodes[1].text, '42'); |
| 1397 | 1903 |
| 1398 model = null; | 1904 model = null; |
| 1399 recursivelySetTemplateModel(div, model); | 1905 recursivelySetTemplateModel(div, model); |
| 1400 performMicrotaskCheckpoint(); | 1906 }).then(endOfMicrotask).then((_) { |
| 1401 expect(div.nodes.length, 1); | 1907 expect(div.nodes.length, 1); |
| 1402 | 1908 |
| 1403 model = toObservable({'a': 42}); | 1909 model = toObservable({'a': 42}); |
| 1404 recursivelySetTemplateModel(div, model); | 1910 recursivelySetTemplateModel(div, model); |
| 1405 performMicrotaskCheckpoint(); | 1911 }).then(endOfMicrotask).then((_) { |
| 1406 expect(div.nodes[1].text, '42'); | 1912 expect(div.nodes[1].text, '42'); |
| 1913 }); |
| 1407 }); | 1914 }); |
| 1408 | 1915 |
| 1409 observeTest('BindNested', () { | 1916 test('BindNested', () { |
| 1410 var div = createTestHtml( | 1917 var div = createTestHtml( |
| 1411 '<template bind="{{}}">' | 1918 '<template bind="{{}}">' |
| 1412 'Name: {{ name }}' | 1919 'Name: {{ name }}' |
| 1413 '<template bind="{{wife}}" if="{{wife}}">' | 1920 '<template bind="{{wife}}" if="{{wife}}">' |
| 1414 'Wife: {{ name }}' | 1921 'Wife: {{ name }}' |
| 1415 '</template>' | 1922 '</template>' |
| 1416 '<template bind="{{child}}" if="{{child}}">' | 1923 '<template bind="{{child}}" if="{{child}}">' |
| 1417 'Child: {{ name }}' | 1924 'Child: {{ name }}' |
| 1418 '</template>' | 1925 '</template>' |
| 1419 '</template>'); | 1926 '</template>'); |
| 1420 | 1927 |
| 1421 var m = toObservable({ | 1928 var m = toObservable({ |
| 1422 'name': 'Hermes', | 1929 'name': 'Hermes', |
| 1423 'wife': { | 1930 'wife': { |
| 1424 'name': 'LaBarbara' | 1931 'name': 'LaBarbara' |
| 1425 } | 1932 } |
| 1426 }); | 1933 }); |
| 1427 recursivelySetTemplateModel(div, m); | 1934 recursivelySetTemplateModel(div, m); |
| 1428 performMicrotaskCheckpoint(); | 1935 return new Future(() { |
| 1936 expect(div.nodes.length, 5); |
| 1937 expect(div.nodes[1].text, 'Name: Hermes'); |
| 1938 expect(div.nodes[3].text, 'Wife: LaBarbara'); |
| 1429 | 1939 |
| 1430 expect(div.nodes.length, 5); | 1940 m['child'] = toObservable({'name': 'Dwight'}); |
| 1431 expect(div.nodes[1].text, 'Name: Hermes'); | 1941 }).then(endOfMicrotask).then((_) { |
| 1432 expect(div.nodes[3].text, 'Wife: LaBarbara'); | 1942 expect(div.nodes.length, 6); |
| 1943 expect(div.nodes[5].text, 'Child: Dwight'); |
| 1433 | 1944 |
| 1434 m['child'] = toObservable({'name': 'Dwight'}); | 1945 m.remove('wife'); |
| 1435 performMicrotaskCheckpoint(); | 1946 }).then(endOfMicrotask).then((_) { |
| 1436 expect(div.nodes.length, 6); | 1947 expect(div.nodes.length, 5); |
| 1437 expect(div.nodes[5].text, 'Child: Dwight'); | 1948 expect(div.nodes[4].text, 'Child: Dwight'); |
| 1438 | 1949 }); |
| 1439 m.remove('wife'); | |
| 1440 performMicrotaskCheckpoint(); | |
| 1441 expect(div.nodes.length, 5); | |
| 1442 expect(div.nodes[4].text, 'Child: Dwight'); | |
| 1443 }); | 1950 }); |
| 1444 | 1951 |
| 1445 observeTest('BindRecursive', () { | 1952 test('BindRecursive', () { |
| 1446 var div = createTestHtml( | 1953 var div = createTestHtml( |
| 1447 '<template bind="{{}}" if="{{}}" id="t">' | 1954 '<template bind="{{}}" if="{{}}" id="t">' |
| 1448 'Name: {{ name }}' | 1955 'Name: {{ name }}' |
| 1449 '<template bind="{{friend}}" if="{{friend}}" ref="t"></template>' | 1956 '<template bind="{{friend}}" if="{{friend}}" ref="t"></template>' |
| 1450 '</template>'); | 1957 '</template>'); |
| 1451 | 1958 |
| 1452 var m = toObservable({ | 1959 var m = toObservable({ |
| 1453 'name': 'Fry', | 1960 'name': 'Fry', |
| 1454 'friend': { | 1961 'friend': { |
| 1455 'name': 'Bender' | 1962 'name': 'Bender' |
| 1456 } | 1963 } |
| 1457 }); | 1964 }); |
| 1458 recursivelySetTemplateModel(div, m); | 1965 recursivelySetTemplateModel(div, m); |
| 1459 performMicrotaskCheckpoint(); | 1966 return new Future(() { |
| 1967 expect(div.nodes.length, 5); |
| 1968 expect(div.nodes[1].text, 'Name: Fry'); |
| 1969 expect(div.nodes[3].text, 'Name: Bender'); |
| 1460 | 1970 |
| 1461 expect(div.nodes.length, 5); | 1971 m['friend']['friend'] = toObservable({'name': 'Leela'}); |
| 1462 expect(div.nodes[1].text, 'Name: Fry'); | 1972 }).then(endOfMicrotask).then((_) { |
| 1463 expect(div.nodes[3].text, 'Name: Bender'); | 1973 expect(div.nodes.length, 7); |
| 1974 expect(div.nodes[5].text, 'Name: Leela'); |
| 1464 | 1975 |
| 1465 m['friend']['friend'] = toObservable({'name': 'Leela'}); | 1976 m['friend'] = toObservable({'name': 'Leela'}); |
| 1466 performMicrotaskCheckpoint(); | 1977 }).then(endOfMicrotask).then((_) { |
| 1467 expect(div.nodes.length, 7); | 1978 expect(div.nodes.length, 5); |
| 1468 expect(div.nodes[5].text, 'Name: Leela'); | 1979 expect(div.nodes[3].text, 'Name: Leela'); |
| 1469 | 1980 }); |
| 1470 m['friend'] = toObservable({'name': 'Leela'}); | |
| 1471 performMicrotaskCheckpoint(); | |
| 1472 expect(div.nodes.length, 5); | |
| 1473 expect(div.nodes[3].text, 'Name: Leela'); | |
| 1474 }); | 1981 }); |
| 1475 | 1982 |
| 1476 observeTest('Template - Self is terminator', () { | 1983 test('Template - Self is terminator', () { |
| 1477 var div = createTestHtml( | 1984 var div = createTestHtml( |
| 1478 '<template repeat>{{ foo }}' | 1985 '<template repeat>{{ foo }}' |
| 1479 '<template bind></template>' | 1986 '<template bind></template>' |
| 1480 '</template>'); | 1987 '</template>'); |
| 1481 | 1988 |
| 1482 var m = toObservable([{ 'foo': 'bar' }]); | 1989 var m = toObservable([{ 'foo': 'bar' }]); |
| 1483 recursivelySetTemplateModel(div, m); | 1990 recursivelySetTemplateModel(div, m); |
| 1484 performMicrotaskCheckpoint(); | 1991 return new Future(() { |
| 1485 | 1992 |
| 1486 m.add(toObservable({ 'foo': 'baz' })); | 1993 m.add(toObservable({ 'foo': 'baz' })); |
| 1487 recursivelySetTemplateModel(div, m); | 1994 recursivelySetTemplateModel(div, m); |
| 1488 performMicrotaskCheckpoint(); | 1995 }).then(endOfMicrotask).then((_) { |
| 1489 | 1996 |
| 1490 expect(div.nodes.length, 5); | 1997 expect(div.nodes.length, 5); |
| 1491 expect(div.nodes[1].text, 'bar'); | 1998 expect(div.nodes[1].text, 'bar'); |
| 1492 expect(div.nodes[3].text, 'baz'); | 1999 expect(div.nodes[3].text, 'baz'); |
| 2000 }); |
| 1493 }); | 2001 }); |
| 1494 | 2002 |
| 1495 observeTest('Template - Same Contents, Different Array has no effect', () { | 2003 test('Template - Same Contents, Different Array has no effect', () { |
| 1496 if (!MutationObserver.supported) return; | 2004 if (!MutationObserver.supported) return null; |
| 1497 | 2005 |
| 1498 var div = createTestHtml('<template repeat>{{ foo }}</template>'); | 2006 var div = createTestHtml('<template repeat>{{ foo }}</template>'); |
| 1499 | 2007 |
| 1500 var m = toObservable([{ 'foo': 'bar' }, { 'foo': 'bat'}]); | 2008 var m = toObservable([{ 'foo': 'bar' }, { 'foo': 'bat'}]); |
| 1501 recursivelySetTemplateModel(div, m); | 2009 recursivelySetTemplateModel(div, m); |
| 1502 performMicrotaskCheckpoint(); | 2010 var observer = new MutationObserver((x, y) {}); |
| 2011 return new Future(() { |
| 2012 observer.observe(div, childList: true); |
| 1503 | 2013 |
| 1504 var observer = new MutationObserver((records, _) {}); | 2014 var template = div.firstChild; |
| 1505 observer.observe(div, childList: true); | 2015 templateBind(template).model = new ObservableList.from(m); |
| 1506 | 2016 }).then(endOfMicrotask).then((_) { |
| 1507 var template = div.firstChild; | 2017 var records = observer.takeRecords(); |
| 1508 nodeBind(template).bind('repeat', toObservable(m.toList()), ''); | 2018 expect(records.length, 0); |
| 1509 performMicrotaskCheckpoint(); | 2019 }); |
| 1510 var records = observer.takeRecords(); | |
| 1511 expect(records.length, 0); | |
| 1512 }); | 2020 }); |
| 1513 | 2021 |
| 1514 observeTest('RecursiveRef', () { | 2022 test('RecursiveRef', () { |
| 1515 var div = createTestHtml( | 2023 var div = createTestHtml( |
| 1516 '<template bind>' | 2024 '<template bind>' |
| 1517 '<template id=src>{{ foo }}</template>' | 2025 '<template id=src>{{ foo }}</template>' |
| 1518 '<template bind ref=src></template>' | 2026 '<template bind ref=src></template>' |
| 1519 '</template>'); | 2027 '</template>'); |
| 1520 | 2028 |
| 1521 var m = toObservable({'foo': 'bar'}); | 2029 var m = toObservable({'foo': 'bar'}); |
| 1522 recursivelySetTemplateModel(div, m); | 2030 recursivelySetTemplateModel(div, m); |
| 1523 performMicrotaskCheckpoint(); | 2031 return new Future(() { |
| 1524 | 2032 expect(div.nodes.length, 4); |
| 1525 expect(div.nodes.length, 4); | 2033 expect(div.nodes[3].text, 'bar'); |
| 1526 expect(div.nodes[3].text, 'bar'); | 2034 }); |
| 1527 }); | 2035 }); |
| 1528 | 2036 |
| 1529 observeTest('ChangeFromBindToRepeat', () { | 2037 test('ChangeRefId', () { |
| 1530 var div = createTestHtml( | |
| 1531 '<template bind="{{a}}">' | |
| 1532 '{{ length }}' | |
| 1533 '</template>'); | |
| 1534 var template = div.nodes.first; | |
| 1535 | |
| 1536 // Note: this test data is a little different from the JS version, because | |
| 1537 // we allow binding to the "length" field of the Map in preference to | |
| 1538 // binding keys. | |
| 1539 var m = toObservable({ | |
| 1540 'a': [ | |
| 1541 [], | |
| 1542 { 'b': [1,2,3,4] }, | |
| 1543 // Note: this will use the Map "length" property, not the "length" key. | |
| 1544 {'length': 42, 'c': 123} | |
| 1545 ] | |
| 1546 }); | |
| 1547 recursivelySetTemplateModel(div, m); | |
| 1548 performMicrotaskCheckpoint(); | |
| 1549 | |
| 1550 expect(div.nodes.length, 2); | |
| 1551 expect(div.nodes[1].text, '3'); | |
| 1552 | |
| 1553 nodeBind(template) | |
| 1554 ..unbind('bind') | |
| 1555 ..bind('repeat', m, 'a'); | |
| 1556 performMicrotaskCheckpoint(); | |
| 1557 expect(div.nodes.length, 4); | |
| 1558 expect(div.nodes[1].text, '0'); | |
| 1559 expect(div.nodes[2].text, '1'); | |
| 1560 expect(div.nodes[3].text, '2'); | |
| 1561 | |
| 1562 nodeBind(template).unbind('repeat'); | |
| 1563 nodeBind(template).bind('bind', m, 'a.1.b'); | |
| 1564 | |
| 1565 performMicrotaskCheckpoint(); | |
| 1566 expect(div.nodes.length, 2); | |
| 1567 expect(div.nodes[1].text, '4'); | |
| 1568 }); | |
| 1569 | |
| 1570 observeTest('ChangeRefId', () { | |
| 1571 var div = createTestHtml( | 2038 var div = createTestHtml( |
| 1572 '<template id="a">a:{{ }}</template>' | 2039 '<template id="a">a:{{ }}</template>' |
| 1573 '<template id="b">b:{{ }}</template>' | 2040 '<template id="b">b:{{ }}</template>' |
| 1574 '<template repeat="{{}}">' | 2041 '<template repeat="{{}}">' |
| 1575 '<template ref="a" bind="{{}}"></template>' | 2042 '<template ref="a" bind="{{}}"></template>' |
| 1576 '</template>'); | 2043 '</template>'); |
| 1577 var model = toObservable([]); | 2044 var model = toObservable([]); |
| 1578 recursivelySetTemplateModel(div, model); | 2045 recursivelySetTemplateModel(div, model); |
| 1579 performMicrotaskCheckpoint(); | 2046 return new Future(() { |
| 2047 expect(div.nodes.length, 3); |
| 1580 | 2048 |
| 1581 expect(div.nodes.length, 3); | 2049 document.getElementById('a').id = 'old-a'; |
| 2050 document.getElementById('b').id = 'a'; |
| 1582 | 2051 |
| 1583 document.getElementById('a').id = 'old-a'; | 2052 model..add(1)..add(2); |
| 1584 document.getElementById('b').id = 'a'; | 2053 }).then(endOfMicrotask).then((_) { |
| 1585 | 2054 |
| 1586 model..add(1)..add(2); | 2055 expect(div.nodes.length, 7); |
| 1587 performMicrotaskCheckpoint(); | 2056 expect(div.nodes[4].text, 'b:1'); |
| 1588 | 2057 expect(div.nodes[6].text, 'b:2'); |
| 1589 expect(div.nodes.length, 7); | 2058 }); |
| 1590 expect(div.nodes[4].text, 'b:1'); | |
| 1591 expect(div.nodes[6].text, 'b:2'); | |
| 1592 }); | 2059 }); |
| 1593 | 2060 |
| 1594 observeTest('Content', () { | 2061 test('Content', () { |
| 1595 var div = createTestHtml( | 2062 var div = createTestHtml( |
| 1596 '<template><a></a></template>' | 2063 '<template><a></a></template>' |
| 1597 '<template><b></b></template>'); | 2064 '<template><b></b></template>'); |
| 1598 var templateA = div.nodes.first; | 2065 var templateA = div.nodes.first; |
| 1599 var templateB = div.nodes.last; | 2066 var templateB = div.nodes.last; |
| 1600 var contentA = templateBind(templateA).content; | 2067 var contentA = templateBind(templateA).content; |
| 1601 var contentB = templateBind(templateB).content; | 2068 var contentB = templateBind(templateB).content; |
| 1602 expect(contentA, isNotNull); | 2069 expect(contentA, isNotNull); |
| 1603 | 2070 |
| 1604 expect(templateA.ownerDocument, isNot(equals(contentA.ownerDocument))); | 2071 expect(templateA.ownerDocument, isNot(equals(contentA.ownerDocument))); |
| 1605 expect(templateB.ownerDocument, isNot(equals(contentB.ownerDocument))); | 2072 expect(templateB.ownerDocument, isNot(equals(contentB.ownerDocument))); |
| 1606 | 2073 |
| 1607 expect(templateB.ownerDocument, templateA.ownerDocument); | 2074 expect(templateB.ownerDocument, templateA.ownerDocument); |
| 1608 expect(contentB.ownerDocument, contentA.ownerDocument); | 2075 expect(contentB.ownerDocument, contentA.ownerDocument); |
| 1609 | 2076 |
| 1610 expect(templateA.ownerDocument.window, window); | 2077 expect(templateA.ownerDocument.window, window); |
| 1611 expect(templateB.ownerDocument.window, window); | 2078 expect(templateB.ownerDocument.window, window); |
| 1612 | 2079 |
| 1613 expect(contentA.ownerDocument.window, null); | 2080 expect(contentA.ownerDocument.window, null); |
| 1614 expect(contentB.ownerDocument.window, null); | 2081 expect(contentB.ownerDocument.window, null); |
| 1615 | 2082 |
| 1616 expect(contentA.nodes.last, contentA.nodes.first); | 2083 expect(contentA.nodes.last, contentA.nodes.first); |
| 1617 expect(contentA.nodes.first.tagName, 'A'); | 2084 expect(contentA.nodes.first.tagName, 'A'); |
| 1618 | 2085 |
| 1619 expect(contentB.nodes.last, contentB.nodes.first); | 2086 expect(contentB.nodes.last, contentB.nodes.first); |
| 1620 expect(contentB.nodes.first.tagName, 'B'); | 2087 expect(contentB.nodes.first.tagName, 'B'); |
| 1621 }); | 2088 }); |
| 1622 | 2089 |
| 1623 observeTest('NestedContent', () { | 2090 test('NestedContent', () { |
| 1624 var div = createTestHtml( | 2091 var div = createTestHtml( |
| 1625 '<template>' | 2092 '<template>' |
| 1626 '<template></template>' | 2093 '<template></template>' |
| 1627 '</template>'); | 2094 '</template>'); |
| 1628 var templateA = div.nodes.first; | 2095 var templateA = div.nodes.first; |
| 1629 var templateB = templateBind(templateA).content.nodes.first; | 2096 var templateB = templateBind(templateA).content.nodes.first; |
| 1630 | 2097 |
| 1631 expect(templateB.ownerDocument, templateBind(templateA) | 2098 expect(templateB.ownerDocument, templateBind(templateA) |
| 1632 .content.ownerDocument); | 2099 .content.ownerDocument); |
| 1633 expect(templateBind(templateB).content.ownerDocument, | 2100 expect(templateBind(templateB).content.ownerDocument, |
| 1634 templateBind(templateA).content.ownerDocument); | 2101 templateBind(templateA).content.ownerDocument); |
| 1635 }); | 2102 }); |
| 1636 | 2103 |
| 1637 observeTest('BindShadowDOM', () { | 2104 test('BindShadowDOM', () { |
| 1638 if (ShadowRoot.supported) { | 2105 if (!ShadowRoot.supported) return null; |
| 1639 var root = createShadowTestHtml( | 2106 |
| 1640 '<template bind="{{}}">Hi {{ name }}</template>'); | 2107 var root = createShadowTestHtml( |
| 1641 var model = toObservable({'name': 'Leela'}); | 2108 '<template bind="{{}}">Hi {{ name }}</template>'); |
| 1642 recursivelySetTemplateModel(root, model); | 2109 var model = toObservable({'name': 'Leela'}); |
| 1643 performMicrotaskCheckpoint(); | 2110 recursivelySetTemplateModel(root, model); |
| 1644 expect(root.nodes[1].text, 'Hi Leela'); | 2111 return new Future(() => expect(root.nodes[1].text, 'Hi Leela')); |
| 1645 } | |
| 1646 }); | 2112 }); |
| 1647 | 2113 |
| 1648 // Dart note: this test seems gone from JS. Keeping for posterity sake. | 2114 // Dart note: this test seems gone from JS. Keeping for posterity sake. |
| 1649 observeTest('BindShadowDOM createInstance', () { | 2115 test('BindShadowDOM createInstance', () { |
| 1650 if (ShadowRoot.supported) { | 2116 if (!ShadowRoot.supported) return null; |
| 1651 var model = toObservable({'name': 'Leela'}); | |
| 1652 var template = new Element.html('<template>Hi {{ name }}</template>'); | |
| 1653 var root = createShadowTestHtml(''); | |
| 1654 root.nodes.add(templateBind(template).createInstance(model)); | |
| 1655 | 2117 |
| 1656 performMicrotaskCheckpoint(); | 2118 var model = toObservable({'name': 'Leela'}); |
| 2119 var template = new Element.html('<template>Hi {{ name }}</template>'); |
| 2120 var root = createShadowTestHtml(''); |
| 2121 root.nodes.add(templateBind(template).createInstance(model)); |
| 2122 |
| 2123 return new Future(() { |
| 1657 expect(root.text, 'Hi Leela'); | 2124 expect(root.text, 'Hi Leela'); |
| 1658 | 2125 |
| 1659 model['name'] = 'Fry'; | 2126 model['name'] = 'Fry'; |
| 1660 performMicrotaskCheckpoint(); | 2127 }).then(endOfMicrotask).then((_) { |
| 1661 expect(root.text, 'Hi Fry'); | 2128 expect(root.text, 'Hi Fry'); |
| 1662 } | 2129 }); |
| 1663 }); | 2130 }); |
| 1664 | 2131 |
| 1665 observeTest('BindShadowDOM Template Ref', () { | 2132 test('BindShadowDOM Template Ref', () { |
| 1666 if (ShadowRoot.supported) { | 2133 if (!ShadowRoot.supported) return null; |
| 1667 var root = createShadowTestHtml( | 2134 var root = createShadowTestHtml( |
| 1668 '<template id=foo>Hi</template><template bind ref=foo></template>'); | 2135 '<template id=foo>Hi</template><template bind ref=foo></template>'); |
| 1669 recursivelySetTemplateModel(root, toObservable({})); | 2136 recursivelySetTemplateModel(root, toObservable({})); |
| 1670 performMicrotaskCheckpoint(); | 2137 return new Future(() => expect(root.nodes.length, 3)); |
| 1671 expect(root.nodes.length, 3); | |
| 1672 } | |
| 1673 }); | 2138 }); |
| 1674 | 2139 |
| 1675 // https://github.com/toolkitchen/mdv/issues/8 | 2140 // https://github.com/Polymer/TemplateBinding/issues/8 |
| 1676 observeTest('UnbindingInNestedBind', () { | 2141 test('UnbindingInNestedBind', () { |
| 1677 var div = createTestHtml( | 2142 var div = createTestHtml( |
| 1678 '<template bind="{{outer}}" if="{{outer}}" syntax="testHelper">' | 2143 '<template bind="{{outer}}" if="{{outer}}" syntax="testHelper">' |
| 1679 '<template bind="{{inner}}" if="{{inner}}">' | 2144 '<template bind="{{inner}}" if="{{inner}}">' |
| 1680 '{{ age }}' | 2145 '{{ age }}' |
| 1681 '</template>' | 2146 '</template>' |
| 1682 '</template>'); | 2147 '</template>'); |
| 1683 | 2148 |
| 1684 var syntax = new UnbindingInNestedBindSyntax(); | 2149 var syntax = new UnbindingInNestedBindSyntax(); |
| 1685 var model = toObservable({ | 2150 var model = toObservable({'outer': {'inner': {'age': 42}}}); |
| 1686 'outer': { | |
| 1687 'inner': { | |
| 1688 'age': 42 | |
| 1689 } | |
| 1690 } | |
| 1691 }); | |
| 1692 | 2151 |
| 1693 recursivelySetTemplateModel(div, model, syntax); | 2152 recursivelySetTemplateModel(div, model, syntax); |
| 1694 | 2153 |
| 1695 performMicrotaskCheckpoint(); | 2154 return new Future(() { |
| 1696 expect(syntax.count, 1); | 2155 expect(syntax.count, 1); |
| 1697 | 2156 |
| 1698 var inner = model['outer']['inner']; | 2157 var inner = model['outer']['inner']; |
| 1699 model['outer'] = null; | 2158 model['outer'] = null; |
| 1700 | 2159 |
| 1701 performMicrotaskCheckpoint(); | 2160 }).then(endOfMicrotask).then((_) { |
| 1702 expect(syntax.count, 1); | 2161 expect(syntax.count, 1); |
| 1703 | 2162 |
| 1704 model['outer'] = toObservable({'inner': {'age': 2}}); | 2163 model['outer'] = toObservable({'inner': {'age': 2}}); |
| 1705 syntax.expectedAge = 2; | 2164 syntax.expectedAge = 2; |
| 1706 | 2165 |
| 1707 performMicrotaskCheckpoint(); | 2166 }).then(endOfMicrotask).then((_) { |
| 1708 expect(syntax.count, 2); | 2167 expect(syntax.count, 2); |
| 2168 }); |
| 1709 }); | 2169 }); |
| 1710 | 2170 |
| 1711 // https://github.com/toolkitchen/mdv/issues/8 | 2171 // https://github.com/toolkitchen/mdv/issues/8 |
| 1712 observeTest('DontCreateInstancesForAbandonedIterators', () { | 2172 test('DontCreateInstancesForAbandonedIterators', () { |
| 1713 var div = createTestHtml( | 2173 var div = createTestHtml( |
| 1714 '<template bind="{{}} {{}}">' | 2174 '<template bind="{{}} {{}}">' |
| 1715 '<template bind="{{}}">Foo' | 2175 '<template bind="{{}}">Foo</template>' |
| 1716 '</template>' | |
| 1717 '</template>'); | 2176 '</template>'); |
| 1718 recursivelySetTemplateModel(div, null); | 2177 recursivelySetTemplateModel(div, null); |
| 1719 performMicrotaskCheckpoint(); | 2178 return nextMicrotask; |
| 1720 }); | 2179 }); |
| 1721 | 2180 |
| 1722 observeTest('CreateInstance', () { | 2181 test('CreateInstance', () { |
| 1723 var div = createTestHtml( | 2182 var div = createTestHtml( |
| 1724 '<template bind="{{a}}">' | 2183 '<template bind="{{a}}">' |
| 1725 '<template bind="{{b}}">' | 2184 '<template bind="{{b}}">' |
| 1726 '{{ foo }}:{{ replaceme }}' | 2185 '{{ foo }}:{{ replaceme }}' |
| 1727 '</template>' | 2186 '</template>' |
| 1728 '</template>'); | 2187 '</template>'); |
| 1729 var outer = templateBind(div.nodes.first); | 2188 var outer = templateBind(div.nodes.first); |
| 1730 var model = toObservable({'b': {'foo': 'bar'}}); | 2189 var model = toObservable({'b': {'foo': 'bar'}}); |
| 1731 | 2190 |
| 1732 var host = new DivElement(); | 2191 var host = new DivElement(); |
| 1733 var instance = outer.createInstance(model, new TestBindingSyntax()); | 2192 var instance = outer.createInstance(model, new TestBindingSyntax()); |
| 1734 expect(outer.content.nodes.first, | 2193 expect(outer.content.nodes.first, |
| 1735 templateBind(instance.nodes.first).ref); | 2194 templateBind(instance.nodes.first).ref); |
| 1736 | 2195 |
| 1737 host.append(instance); | 2196 host.append(instance); |
| 1738 performMicrotaskCheckpoint(); | 2197 return new Future(() { |
| 1739 expect(host.firstChild.nextNode.text, 'bar:replaced'); | 2198 expect(host.firstChild.nextNode.text, 'bar:replaced'); |
| 2199 }); |
| 1740 }); | 2200 }); |
| 1741 | 2201 |
| 1742 observeTest('Bootstrap', () { | 2202 test('Repeat - svg', () { |
| 2203 var div = createTestHtml( |
| 2204 '<svg width="400" height="110">' |
| 2205 '<template repeat>' |
| 2206 '<rect width="{{ width }}" height="{{ height }}" />' |
| 2207 '</template>' |
| 2208 '</svg>'); |
| 2209 |
| 2210 var model = toObservable([{ 'width': 10, 'height': 11 }, |
| 2211 { 'width': 20, 'height': 21 }]); |
| 2212 var svg = div.firstChild; |
| 2213 var template = svg.firstChild; |
| 2214 templateBind(template).model = model; |
| 2215 |
| 2216 return new Future(() { |
| 2217 expect(svg.nodes.length, 3); |
| 2218 expect(svg.nodes[1].attributes['width'], '10'); |
| 2219 expect(svg.nodes[1].attributes['height'], '11'); |
| 2220 expect(svg.nodes[2].attributes['width'], '20'); |
| 2221 expect(svg.nodes[2].attributes['height'], '21'); |
| 2222 }); |
| 2223 }); |
| 2224 |
| 2225 test('Bootstrap', () { |
| 1743 var div = new DivElement(); | 2226 var div = new DivElement(); |
| 1744 div.innerHtml = | 2227 div.innerHtml = |
| 1745 '<template>' | 2228 '<template>' |
| 1746 '<div></div>' | 2229 '<div></div>' |
| 1747 '<template>' | 2230 '<template>' |
| 1748 'Hello' | 2231 'Hello' |
| 1749 '</template>' | 2232 '</template>' |
| 1750 '</template>'; | 2233 '</template>'; |
| 1751 | 2234 |
| 1752 TemplateBindExtension.bootstrap(div); | 2235 TemplateBindExtension.bootstrap(div); |
| (...skipping 13 matching lines...) Expand all Loading... |
| 1766 '</template>'; | 2249 '</template>'; |
| 1767 | 2250 |
| 1768 TemplateBindExtension.bootstrap(template); | 2251 TemplateBindExtension.bootstrap(template); |
| 1769 template2 = templateBind(templateBind(template).content.nodes.first); | 2252 template2 = templateBind(templateBind(template).content.nodes.first); |
| 1770 expect(template2.content.nodes.length, 2); | 2253 expect(template2.content.nodes.length, 2); |
| 1771 var template3 = templateBind(template2.content.nodes.first.nextNode); | 2254 var template3 = templateBind(template2.content.nodes.first.nextNode); |
| 1772 expect(template3.content.nodes.length, 1); | 2255 expect(template3.content.nodes.length, 1); |
| 1773 expect(template3.content.nodes.first.text, 'Hello'); | 2256 expect(template3.content.nodes.first.text, 'Hello'); |
| 1774 }); | 2257 }); |
| 1775 | 2258 |
| 1776 observeTest('issue-285', () { | 2259 test('issue-285', () { |
| 1777 var div = createTestHtml( | 2260 var div = createTestHtml( |
| 1778 '<template>' | 2261 '<template>' |
| 1779 '<template bind if="{{show}}">' | 2262 '<template bind if="{{show}}">' |
| 1780 '<template id=del repeat="{{items}}">' | 2263 '<template id=del repeat="{{items}}">' |
| 1781 '{{}}' | 2264 '{{}}' |
| 1782 '</template>' | 2265 '</template>' |
| 1783 '</template>' | 2266 '</template>' |
| 1784 '</template>'); | 2267 '</template>'); |
| 1785 | 2268 |
| 1786 var template = div.firstChild; | 2269 var template = div.firstChild; |
| 1787 | 2270 |
| 1788 var model = toObservable({ | 2271 var model = toObservable({ |
| 1789 'show': true, | 2272 'show': true, |
| 1790 'items': [1] | 2273 'items': [1] |
| 1791 }); | 2274 }); |
| 1792 | 2275 |
| 1793 div.append(templateBind(template).createInstance(model, | 2276 div.append(templateBind(template).createInstance(model, |
| 1794 new Issue285Syntax())); | 2277 new Issue285Syntax())); |
| 1795 | 2278 |
| 1796 performMicrotaskCheckpoint(); | 2279 return new Future(() { |
| 1797 expect(template.nextNode.nextNode.nextNode.text, '2'); | 2280 expect(template.nextNode.nextNode.nextNode.text, '2'); |
| 1798 model['show'] = false; | 2281 model['show'] = false; |
| 1799 performMicrotaskCheckpoint(); | 2282 }).then(endOfMicrotask).then((_) { |
| 1800 model['show'] = true; | 2283 model['show'] = true; |
| 1801 performMicrotaskCheckpoint(); | 2284 }).then(endOfMicrotask).then((_) { |
| 1802 expect(template.nextNode.nextNode.nextNode.text, '2'); | 2285 expect(template.nextNode.nextNode.nextNode.text, '2'); |
| 2286 }); |
| 1803 }); | 2287 }); |
| 1804 | 2288 |
| 1805 observeTest('issue-141', () { | 2289 test('issue-141', () { |
| 1806 var div = createTestHtml( | 2290 var div = createTestHtml( |
| 1807 '<template bind>' + | 2291 '<template bind>' |
| 1808 '<div foo="{{foo1}} {{foo2}}" bar="{{bar}}"></div>' + | 2292 '<div foo="{{foo1}} {{foo2}}" bar="{{bar}}"></div>' |
| 1809 '</template>'); | 2293 '</template>'); |
| 1810 | 2294 |
| 1811 var model = toObservable({ | 2295 var model = toObservable({ |
| 1812 'foo1': 'foo1Value', | 2296 'foo1': 'foo1Value', |
| 1813 'foo2': 'foo2Value', | 2297 'foo2': 'foo2Value', |
| 1814 'bar': 'barValue' | 2298 'bar': 'barValue' |
| 1815 }); | 2299 }); |
| 1816 | 2300 |
| 1817 recursivelySetTemplateModel(div, model); | 2301 recursivelySetTemplateModel(div, model); |
| 1818 performMicrotaskCheckpoint(); | 2302 return new Future(() { |
| 2303 expect(div.lastChild.attributes['bar'], 'barValue'); |
| 2304 }); |
| 2305 }); |
| 1819 | 2306 |
| 1820 expect(div.lastChild.attributes['bar'], 'barValue'); | 2307 test('issue-18', () { |
| 2308 var delegate = new Issue18Syntax(); |
| 2309 |
| 2310 var div = createTestHtml( |
| 2311 '<template bind>' |
| 2312 '<div class="foo: {{ bar }}"></div>' |
| 2313 '</template>'); |
| 2314 |
| 2315 var model = toObservable({'bar': 2}); |
| 2316 |
| 2317 recursivelySetTemplateModel(div, model, delegate); |
| 2318 |
| 2319 return new Future(() { |
| 2320 expect(div.lastChild.attributes['class'], 'foo: 2'); |
| 2321 }); |
| 2322 }); |
| 2323 |
| 2324 test('issue-152', () { |
| 2325 var div = createTestHtml('<template ref=notThere></template>'); |
| 2326 var template = div.firstChild; |
| 2327 |
| 2328 // if a ref cannot be located, a template will continue to use itself |
| 2329 // as the source of template instances. |
| 2330 expect(template, templateBind(template).ref); |
| 1821 }); | 2331 }); |
| 1822 } | 2332 } |
| 1823 | 2333 |
| 1824 compatTests() { | 2334 compatTests() { |
| 1825 observeTest('underbar bindings', () { | 2335 test('underbar bindings', () { |
| 1826 var div = createTestHtml( | 2336 var div = createTestHtml( |
| 1827 '<template bind>' | 2337 '<template bind>' |
| 1828 '<div _style="color: {{ color }};"></div>' | 2338 '<div _style="color: {{ color }};"></div>' |
| 1829 '<img _src="{{ url }}">' | 2339 '<img _src="{{ url }}">' |
| 1830 '<a _href="{{ url2 }}">Link</a>' | 2340 '<a _href="{{ url2 }}">Link</a>' |
| 1831 '<input type="number" _value="{{ number }}">' | 2341 '<input type="number" _value="{{ number }}">' |
| 1832 '</template>'); | 2342 '</template>'); |
| 1833 | 2343 |
| 1834 var model = toObservable({ | 2344 var model = toObservable({ |
| 1835 'color': 'red', | 2345 'color': 'red', |
| 1836 'url': 'pic.jpg', | 2346 'url': 'pic.jpg', |
| 1837 'url2': 'link.html', | 2347 'url2': 'link.html', |
| 1838 'number': 4 | 2348 'number': 4 |
| 1839 }); | 2349 }); |
| 1840 | 2350 |
| 1841 recursivelySetTemplateModel(div, model); | 2351 recursivelySetTemplateModel(div, model); |
| 1842 performMicrotaskCheckpoint(); | 2352 return new Future(() { |
| 2353 var subDiv = div.firstChild.nextNode; |
| 2354 expect(subDiv.attributes['style'], 'color: red;'); |
| 1843 | 2355 |
| 1844 var subDiv = div.firstChild.nextNode; | 2356 var img = subDiv.nextNode; |
| 1845 expect(subDiv.attributes['style'], 'color: red;'); | 2357 expect(img.attributes['src'], 'pic.jpg'); |
| 1846 | 2358 |
| 1847 var img = subDiv.nextNode; | 2359 var a = img.nextNode; |
| 1848 expect(img.attributes['src'], 'pic.jpg'); | 2360 expect(a.attributes['href'], 'link.html'); |
| 1849 | 2361 |
| 1850 var a = img.nextNode; | 2362 var input = a.nextNode; |
| 1851 expect(a.attributes['href'], 'link.html'); | 2363 expect(input.value, '4'); |
| 1852 | 2364 }); |
| 1853 var input = a.nextNode; | |
| 1854 expect(input.value, '4'); | |
| 1855 }); | 2365 }); |
| 1856 } | 2366 } |
| 1857 | 2367 |
| 1858 class Issue285Syntax extends BindingDelegate { | 2368 class Issue285Syntax extends BindingDelegate { |
| 1859 prepareInstanceModel(template) { | 2369 prepareInstanceModel(template) { |
| 1860 if (template.id == 'del') return (val) => val * 2; | 2370 if (template.id == 'del') return (val) => val * 2; |
| 1861 } | 2371 } |
| 1862 } | 2372 } |
| 1863 | 2373 |
| 1864 class TestBindingSyntax extends BindingDelegate { | 2374 class TestBindingSyntax extends BindingDelegate { |
| 1865 prepareBinding(String path, name, node) { | 2375 prepareBinding(String path, name, node) { |
| 1866 if (path.trim() == 'replaceme') { | 2376 if (path.trim() == 'replaceme') { |
| 1867 return (x, y) => new ObservableBox('replaced'); | 2377 return (m, n, oneTime) => new PathObserver('replaced', ''); |
| 1868 } | 2378 } |
| 1869 return null; | 2379 return null; |
| 1870 } | 2380 } |
| 1871 } | 2381 } |
| 1872 | 2382 |
| 1873 class UnbindingInNestedBindSyntax extends BindingDelegate { | 2383 class UnbindingInNestedBindSyntax extends BindingDelegate { |
| 1874 int expectedAge = 42; | 2384 int expectedAge = 42; |
| 1875 int count = 0; | 2385 int count = 0; |
| 1876 | 2386 |
| 1877 prepareBinding(path, name, node) { | 2387 prepareBinding(path, name, node) { |
| 1878 if (name != 'text' || path != 'age') return null; | 2388 if (name != 'text' || path != 'age') return null; |
| 1879 | 2389 |
| 1880 return (model, node) { | 2390 return (model, _, oneTime) { |
| 1881 expect(model['age'], expectedAge); | 2391 expect(model['age'], expectedAge); |
| 1882 count++; | 2392 count++; |
| 1883 return model; | 2393 return new PathObserver(model, path); |
| 1884 }; | 2394 }; |
| 1885 } | 2395 } |
| 1886 } | 2396 } |
| 2397 |
| 2398 class Issue18Syntax extends BindingDelegate { |
| 2399 prepareBinding(path, name, node) { |
| 2400 if (name != 'class') return null; |
| 2401 |
| 2402 return (model, _, oneTime) => new PathObserver(model, path); |
| 2403 } |
| 2404 } |
| OLD | NEW |