OLD | NEW |
(Empty) | |
| 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 |
| 3 // BSD-style license that can be found in the LICENSE file. |
| 4 |
| 5 library template_binding.test.custom_element_bindings_test; |
| 6 |
| 7 import 'dart:async'; |
| 8 import 'dart:html'; |
| 9 import 'dart:collection' show MapView; |
| 10 import 'package:template_binding/template_binding.dart'; |
| 11 import 'package:observe/observe.dart'; |
| 12 import 'package:unittest/html_config.dart'; |
| 13 import 'package:unittest/unittest.dart'; |
| 14 import 'package:web_components/polyfill.dart'; |
| 15 import 'utils.dart'; |
| 16 import 'package:observe/mirrors_used.dart'; // make test smaller |
| 17 import 'package:smoke/mirrors.dart' as smoke; |
| 18 |
| 19 Future _registered; |
| 20 |
| 21 main() => dirtyCheckZone().run(() { |
| 22 smoke.useMirrors(); |
| 23 useHtmlConfiguration(); |
| 24 |
| 25 _registered = customElementsReady.then((_) { |
| 26 document.registerElement('my-custom-element', MyCustomElement); |
| 27 }); |
| 28 |
| 29 group('Custom Element Bindings', customElementBindingsTest); |
| 30 }); |
| 31 |
| 32 customElementBindingsTest() { |
| 33 setUp(() { |
| 34 document.body.append(testDiv = new DivElement()); |
| 35 return _registered; |
| 36 }); |
| 37 |
| 38 tearDown(() { |
| 39 testDiv.remove(); |
| 40 testDiv = null; |
| 41 }); |
| 42 |
| 43 test('override bind/bindFinished', () { |
| 44 var element = new MyCustomElement(); |
| 45 var model = toObservable({'a': new Point(123, 444), 'b': new Monster(100)}); |
| 46 |
| 47 var pointBinding = nodeBind(element) |
| 48 .bind('my-point', new PathObserver(model, 'a')); |
| 49 |
| 50 var scaryBinding = nodeBind(element) |
| 51 .bind('scary-monster', new PathObserver(model, 'b')); |
| 52 |
| 53 expect(element.attributes, isNot(contains('my-point'))); |
| 54 expect(element.attributes, isNot(contains('scary-monster'))); |
| 55 |
| 56 expect(element.myPoint, model['a']); |
| 57 expect(element.scaryMonster, model['b']); |
| 58 |
| 59 model['a'] = null; |
| 60 return new Future(() { |
| 61 expect(element.myPoint, null); |
| 62 expect(element.bindFinishedCalled, 0); |
| 63 pointBinding.close(); |
| 64 |
| 65 model['a'] = new Point(1, 2); |
| 66 model['b'] = new Monster(200); |
| 67 }).then(endOfMicrotask).then((_) { |
| 68 expect(element.scaryMonster, model['b']); |
| 69 expect(element.myPoint, null, reason: 'a was unbound'); |
| 70 |
| 71 scaryBinding.close(); |
| 72 model['b'] = null; |
| 73 }).then(endOfMicrotask).then((_) { |
| 74 expect(element.scaryMonster.health, 200); |
| 75 expect(element.bindFinishedCalled, 0); |
| 76 }); |
| 77 }); |
| 78 |
| 79 test('template bind uses overridden custom element bind', () { |
| 80 |
| 81 var model = toObservable({'a': new Point(123, 444), 'b': new Monster(100)}); |
| 82 var div = createTestHtml('<template bind>' |
| 83 '<my-custom-element my-point="{{a}}" scary-monster="{{b}}">' |
| 84 '</my-custom-element>' |
| 85 '</template>'); |
| 86 |
| 87 templateBind(div.query('template')).model = model; |
| 88 var element; |
| 89 return new Future(() { |
| 90 element = div.nodes[1]; |
| 91 |
| 92 expect(element is MyCustomElement, true, |
| 93 reason: '$element should be a MyCustomElement'); |
| 94 |
| 95 expect(element.myPoint, model['a']); |
| 96 expect(element.scaryMonster, model['b']); |
| 97 |
| 98 expect(element.attributes, isNot(contains('my-point'))); |
| 99 expect(element.attributes, isNot(contains('scary-monster'))); |
| 100 |
| 101 expect(element.bindFinishedCalled, 1); |
| 102 |
| 103 model['a'] = null; |
| 104 }).then(endOfMicrotask).then((_) { |
| 105 expect(element.myPoint, null); |
| 106 expect(element.bindFinishedCalled, 1); |
| 107 |
| 108 |
| 109 templateBind(div.query('template')).model = null; |
| 110 }).then(endOfMicrotask).then((_) { |
| 111 // Note: the detached element |
| 112 expect(element.parentNode is DocumentFragment, true, |
| 113 reason: 'removed element is added back to its document fragment'); |
| 114 expect(element.parentNode.parentNode, null, |
| 115 reason: 'document fragment is detached'); |
| 116 expect(element.bindFinishedCalled, 1); |
| 117 |
| 118 model['a'] = new Point(1, 2); |
| 119 model['b'] = new Monster(200); |
| 120 }).then(endOfMicrotask).then((_) { |
| 121 expect(element.myPoint, null, reason: 'model was unbound'); |
| 122 expect(element.scaryMonster.health, 100, reason: 'model was unbound'); |
| 123 expect(element.bindFinishedCalled, 1); |
| 124 }); |
| 125 }); |
| 126 |
| 127 } |
| 128 |
| 129 class Monster { |
| 130 int health; |
| 131 Monster(this.health); |
| 132 } |
| 133 |
| 134 /** Demonstrates a custom element overriding bind/bindFinished. */ |
| 135 class MyCustomElement extends HtmlElement implements NodeBindExtension { |
| 136 Point myPoint; |
| 137 Monster scaryMonster; |
| 138 int bindFinishedCalled = 0; |
| 139 |
| 140 factory MyCustomElement() => new Element.tag('my-custom-element'); |
| 141 |
| 142 MyCustomElement.created() : super.created(); |
| 143 |
| 144 Bindable bind(String name, value, {oneTime: false}) { |
| 145 switch (name) { |
| 146 case 'my-point': |
| 147 case 'scary-monster': |
| 148 attributes.remove(name); |
| 149 if (oneTime) { |
| 150 _setProperty(name, value); |
| 151 return null; |
| 152 } |
| 153 _setProperty(name, value.open((x) => _setProperty(name, x))); |
| 154 |
| 155 if (!enableBindingsReflection) return value; |
| 156 if (bindings == null) bindings = {}; |
| 157 var old = bindings[name]; |
| 158 if (old != null) old.close(); |
| 159 return bindings[name] = value; |
| 160 } |
| 161 return nodeBindFallback(this).bind(name, value, oneTime: oneTime); |
| 162 } |
| 163 |
| 164 void bindFinished() { |
| 165 bindFinishedCalled++; |
| 166 } |
| 167 |
| 168 get bindings => nodeBindFallback(this).bindings; |
| 169 set bindings(x) => nodeBindFallback(this).bindings = x; |
| 170 get templateInstance => nodeBindFallback(this).templateInstance; |
| 171 |
| 172 void _setProperty(String property, newValue) { |
| 173 if (property == 'my-point') myPoint = newValue; |
| 174 if (property == 'scary-monster') scaryMonster = newValue; |
| 175 } |
| 176 } |
| 177 |
OLD | NEW |