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.node_bind_test; | 5 library template_binding.test.node_bind_test; |
6 | 6 |
7 import 'dart:async'; | 7 import 'dart:async'; |
8 import 'dart:html'; | 8 import 'dart:html'; |
9 | 9 |
10 import 'package:observe/observe.dart' | 10 import 'package:observe/observe.dart' |
11 show toObservable, PathObserver, PropertyPath; | 11 show toObservable, PathObserver, PropertyPath; |
12 import 'package:template_binding/template_binding.dart' show nodeBind; | 12 import 'package:template_binding/template_binding.dart' |
| 13 show nodeBind, enableBindingsReflection; |
13 | 14 |
14 import 'package:unittest/html_config.dart'; | 15 import 'package:unittest/html_config.dart'; |
15 import 'package:unittest/unittest.dart'; | 16 import 'package:unittest/unittest.dart'; |
16 import 'utils.dart'; | 17 import 'utils.dart'; |
17 | 18 |
18 // Note: this file ported from | 19 // Ported from: https://github.com/Polymer/NodeBind/blob/master/tests/tests.js |
19 // https://github.com/toolkitchen/mdv/blob/master/tests/node_bindings.js | 20 |
| 21 var bindings; |
20 | 22 |
21 main() => dirtyCheckZone().run(() { | 23 main() => dirtyCheckZone().run(() { |
22 useHtmlConfiguration(); | 24 useHtmlConfiguration(); |
23 | 25 |
24 setUp(() { | 26 setUp(() { |
25 document.body.append(testDiv = new DivElement()); | 27 document.body.append(testDiv = new DivElement()); |
| 28 bindings = []; |
26 }); | 29 }); |
27 | 30 |
28 tearDown(() { | 31 tearDown(() { |
29 testDiv.remove(); | 32 testDiv.remove(); |
30 testDiv = null; | 33 testDiv = null; |
| 34 for (var b in bindings) if (b != null) b.close(); |
| 35 bindings = null; |
31 }); | 36 }); |
32 | 37 |
33 group('Text bindings', testBindings); | 38 group('Text bindings', testBindings); |
34 group('Element attribute bindings', elementBindings); | 39 group('Element attribute bindings', elementBindings); |
35 group('Form Element bindings', formBindings); | 40 group('Form Element bindings', formBindings); |
36 }); | 41 }); |
37 | 42 |
38 testBindings() { | 43 testBindings() { |
39 test('Basic', () { | 44 test('Basic', () { |
40 var text = new Text('hi'); | 45 var text = new Text('hi'); |
41 var model = toObservable({'a': 1}); | 46 var model = toObservable({'a': 1}); |
42 nodeBind(text).bind('text', new PathObserver(model, 'a')); | 47 bindings.add(nodeBind(text).bind('text', new PathObserver(model, 'a'))); |
43 expect(text.text, '1'); | 48 expect(text.text, '1'); |
44 | 49 |
45 model['a'] = 2; | 50 model['a'] = 2; |
46 return new Future(() { | 51 return new Future(() { |
47 expect(text.text, '2'); | 52 expect(text.text, '2'); |
48 | |
49 nodeBind(text).unbind('text'); | |
50 model['a'] = 3; | |
51 }).then(endOfMicrotask).then((_) { | |
52 // TODO(rafaelw): Throw on binding to unavailable property? | |
53 expect(text.text, '2'); | |
54 }); | 53 }); |
55 }); | 54 }); |
56 | 55 |
57 test('oneTime', () { | 56 test('oneTime', () { |
58 var text = new Text('hi'); | 57 var text = new Text('hi'); |
59 nodeBind(text).bind('text', 1, oneTime: true); | 58 bindings.add(nodeBind(text).bind('text', 1, oneTime: true)); |
60 expect(text.text, '1'); | 59 expect(text.text, '1'); |
61 }); | 60 }); |
62 | 61 |
63 test('No Path', () { | 62 test('No Path', () { |
64 var text = new Text('hi'); | 63 var text = new Text('hi'); |
65 var model = 1; | 64 var model = 1; |
66 nodeBind(text).bind('text', new PathObserver(model)); | 65 bindings.add(nodeBind(text).bind('text', new PathObserver(model))); |
67 expect(text.text, '1'); | 66 expect(text.text, '1'); |
68 }); | 67 }); |
69 | 68 |
70 test('Path unreachable', () { | 69 test('Path unreachable', () { |
71 var text = testDiv.append(new Text('hi')); | 70 var text = testDiv.append(new Text('hi')); |
72 var model = 1; | 71 var model = 1; |
73 var pathObserver = new PathObserver(model, 'a'); | 72 var pathObserver = new PathObserver(model, 'a'); |
74 expect(() => nodeBind(text).bind('text', pathObserver), throws); | 73 expect(() => nodeBind(text).bind('text', pathObserver), throws); |
75 expect(text.text, 'hi'); | 74 expect(text.text, 'hi'); |
76 }); | 75 }); |
77 | 76 |
78 test('Observer is Model', () { | 77 test('Observer is Model', () { |
79 var text = new Text(''); | 78 // Dart note: we don't have _allObserversCount so we use binding reflection |
80 var model = toObservable({'a': {'b': {'c': 1}}}); | 79 // instead. |
81 var observer = new PathObserver(model, 'a.b.c'); | 80 enableBindingsReflection = true; |
82 nodeBind(text).bind('text', observer); | |
83 expect(text.text, '1'); | |
84 | 81 |
85 var binding = nodeBind(text).bindings['text']; | 82 // This future is here so we can turn off bindings reflection reliably. |
86 expect(binding, observer, reason: 'should reuse observer'); | 83 Text text; |
| 84 return new Future(() { |
| 85 text = new Text(''); |
| 86 var model = toObservable({'a': {'b': {'c': 1}}}); |
| 87 var observer = new PathObserver(model, 'a.b.c'); |
| 88 bindings.add(nodeBind(text).bind('text', observer)); |
| 89 expect(text.text, '1'); |
87 | 90 |
88 model['a']['b']['c'] = 2; | 91 var binding = nodeBind(text).bindings['text']; |
89 return new Future(() { | 92 expect(binding, observer, reason: 'should reuse observer'); |
| 93 |
| 94 model['a']['b']['c'] = 2; |
| 95 }).then(endOfMicrotask).then((_) { |
90 expect(text.text, '2'); | 96 expect(text.text, '2'); |
91 nodeBind(text).unbind('text'); | 97 }).whenComplete(() { |
| 98 enableBindingsReflection = false; |
92 }); | 99 }); |
93 }); | 100 }); |
94 } | 101 } |
95 | 102 |
96 elementBindings() { | 103 elementBindings() { |
| 104 |
97 test('Basic', () { | 105 test('Basic', () { |
98 var el = new DivElement(); | 106 var el = new DivElement(); |
99 var model = toObservable({'a': '1'}); | 107 var model = toObservable({'a': '1'}); |
100 nodeBind(el).bind('foo', new PathObserver(model, 'a')); | 108 bindings.add(nodeBind(el).bind('foo', new PathObserver(model, 'a'))); |
101 | 109 |
102 return new Future(() { | 110 return new Future(() { |
103 expect(el.attributes['foo'], '1'); | 111 expect(el.attributes['foo'], '1'); |
104 model['a'] = '2'; | 112 model['a'] = '2'; |
105 }).then(endOfMicrotask).then((_) { | 113 }).then(endOfMicrotask).then((_) { |
106 expect(el.attributes['foo'], '2'); | 114 expect(el.attributes['foo'], '2'); |
107 model['a'] = 232.2; | 115 model['a'] = 232.2; |
108 }).then(endOfMicrotask).then((_) { | 116 }).then(endOfMicrotask).then((_) { |
109 expect(el.attributes['foo'], '232.2'); | 117 expect(el.attributes['foo'], '232.2'); |
110 model['a'] = 232; | 118 model['a'] = 232; |
111 }).then(endOfMicrotask).then((_) { | 119 }).then(endOfMicrotask).then((_) { |
112 expect(el.attributes['foo'], '232'); | 120 expect(el.attributes['foo'], '232'); |
113 model['a'] = null; | 121 model['a'] = null; |
114 }).then(endOfMicrotask).then((_) { | 122 }).then(endOfMicrotask).then((_) { |
115 expect(el.attributes['foo'], ''); | 123 expect(el.attributes['foo'], ''); |
116 }); | 124 }); |
117 }); | 125 }); |
118 | 126 |
| 127 // Dart specific test |
| 128 test('enableBindingsReflection defaults to off', () { |
| 129 expect(enableBindingsReflection, false); |
| 130 |
| 131 var el = new DivElement(); |
| 132 var model = toObservable({'a': '1'}); |
| 133 bindings.add(nodeBind(el).bind('foo', new PathObserver(model, 'a'))); |
| 134 |
| 135 expect(nodeBind(el).bindings, null); |
| 136 }); |
| 137 |
| 138 test('enableBindingsReflection', () { |
| 139 enableBindingsReflection = true; |
| 140 try { |
| 141 var el = testDiv.append(new DivElement()); |
| 142 var model = toObservable({'a': '1'}); |
| 143 bindings.add(nodeBind(el).bind('foo', new PathObserver(model, 'a'))); |
| 144 bindings.add(nodeBind(el).bind('bar', new PathObserver(model, 'a'))); |
| 145 bindings.add(nodeBind(el).bind('baz', new PathObserver(model, 'a'))); |
| 146 |
| 147 expect(nodeBind(el).bindings.keys, |
| 148 unorderedEquals(['foo', 'bar', 'baz'])); |
| 149 |
| 150 } finally { |
| 151 enableBindingsReflection = false; |
| 152 } |
| 153 }); |
| 154 |
119 test('oneTime', () { | 155 test('oneTime', () { |
120 var el = testDiv.append(new DivElement()); | 156 var el = testDiv.append(new DivElement()); |
121 var model = toObservable({'a': '1'}); | 157 var model = toObservable({'a': '1'}); |
122 nodeBind(el).bind('foo', 1, oneTime: true); | 158 bindings.add(nodeBind(el).bind('foo', 1, oneTime: true)); |
123 expect('1', el.attributes['foo']); | 159 expect('1', el.attributes['foo']); |
124 }); | 160 }); |
125 | 161 |
126 test('No Path', () { | 162 test('No Path', () { |
127 var el = testDiv.append(new DivElement()); | 163 var el = testDiv.append(new DivElement()); |
128 var model = 1; | 164 var model = 1; |
129 nodeBind(el).bind('foo', new PathObserver(model)); | 165 bindings.add(nodeBind(el).bind('foo', new PathObserver(model))); |
130 return new Future(() { | 166 return new Future(() { |
131 expect(el.attributes['foo'], '1'); | 167 expect(el.attributes['foo'], '1'); |
132 }); | 168 }); |
133 }); | 169 }); |
134 | 170 |
135 test('Path unreachable', () { | 171 test('Path unreachable', () { |
136 var el = testDiv.append(new DivElement()); | 172 var el = testDiv.append(new DivElement()); |
137 var model = toObservable({}); | 173 var model = toObservable({}); |
138 nodeBind(el).bind('foo', new PathObserver(model, 'bar')); | 174 bindings.add(nodeBind(el).bind('foo', new PathObserver(model, 'bar'))); |
139 return new Future(() { | 175 return new Future(() { |
140 expect(el.attributes['foo'], ''); | 176 expect(el.attributes['foo'], ''); |
141 }); | 177 }); |
142 }); | 178 }); |
143 | 179 |
144 test('Dashes', () { | 180 test('Dashes', () { |
145 var el = testDiv.append(new DivElement()); | 181 var el = testDiv.append(new DivElement()); |
146 var model = toObservable({'a': '1'}); | 182 var model = toObservable({'a': '1'}); |
147 nodeBind(el).bind('foo-bar', new PathObserver(model, 'a')); | 183 bindings.add(nodeBind(el).bind('foo-bar', new PathObserver(model, 'a'))); |
148 return new Future(() { | 184 return new Future(() { |
149 expect(el.attributes['foo-bar'], '1'); | 185 expect(el.attributes['foo-bar'], '1'); |
150 model['a'] = '2'; | 186 model['a'] = '2'; |
151 | 187 |
152 }).then(endOfMicrotask).then((_) { | 188 }).then(endOfMicrotask).then((_) { |
153 expect(el.attributes['foo-bar'], '2'); | 189 expect(el.attributes['foo-bar'], '2'); |
154 }); | 190 }); |
155 }); | 191 }); |
156 | 192 |
157 test('Element.id, Element.hidden?', () { | 193 test('Element.id, Element.hidden?', () { |
158 var element = new DivElement(); | 194 var element = new DivElement(); |
159 var model = toObservable({'a': 1, 'b': 2}); | 195 var model = toObservable({'a': 1, 'b': 2}); |
160 nodeBind(element).bind('hidden?', new PathObserver(model, 'a')); | 196 bindings.add( |
161 nodeBind(element).bind('id', new PathObserver(model, 'b')); | 197 nodeBind(element).bind('hidden?', new PathObserver(model, 'a'))); |
| 198 bindings.add(nodeBind(element).bind('id', new PathObserver(model, 'b'))); |
162 | 199 |
163 expect(element.attributes, contains('hidden')); | 200 expect(element.attributes, contains('hidden')); |
164 expect(element.attributes['hidden'], ''); | 201 expect(element.attributes['hidden'], ''); |
165 expect(element.id, '2'); | 202 expect(element.id, '2'); |
166 | 203 |
167 model['a'] = null; | 204 model['a'] = null; |
168 return new Future(() { | 205 return new Future(() { |
169 expect(element.attributes, isNot(contains('hidden')), | 206 expect(element.attributes, isNot(contains('hidden')), |
170 reason: 'null is false-y'); | 207 reason: 'null is false-y'); |
171 | 208 |
172 model['a'] = false; | 209 model['a'] = false; |
173 }).then(endOfMicrotask).then((_) { | 210 }).then(endOfMicrotask).then((_) { |
174 expect(element.attributes, isNot(contains('hidden'))); | 211 expect(element.attributes, isNot(contains('hidden'))); |
175 | 212 |
176 model['a'] = 'foo'; | 213 model['a'] = 'foo'; |
177 model['b'] = 'x'; | 214 model['b'] = 'x'; |
178 }).then(endOfMicrotask).then((_) { | 215 }).then(endOfMicrotask).then((_) { |
179 expect(element.attributes, contains('hidden')); | 216 expect(element.attributes, contains('hidden')); |
180 expect(element.attributes['hidden'], ''); | 217 expect(element.attributes['hidden'], ''); |
181 expect(element.id, 'x'); | 218 expect(element.id, 'x'); |
182 }); | 219 }); |
183 }); | 220 }); |
184 | 221 |
185 test('Element.id - path unreachable', () { | 222 test('Element.id - path unreachable', () { |
186 var element = testDiv.append(new DivElement()); | 223 var element = testDiv.append(new DivElement()); |
187 var model = toObservable({}); | 224 var model = toObservable({}); |
188 nodeBind(element).bind('id', new PathObserver(model, 'a')); | 225 bindings.add(nodeBind(element).bind('id', new PathObserver(model, 'a'))); |
189 return new Future(() => expect(element.id, '')); | 226 return new Future(() => expect(element.id, '')); |
190 }); | 227 }); |
191 } | 228 } |
192 | 229 |
193 formBindings() { | 230 formBindings() { |
194 inputTextAreaValueTest(String tagName) { | 231 inputTextAreaValueTest(String tagName) { |
195 var el = new Element.tag(tagName); | 232 var el = new Element.tag(tagName); |
196 testDiv.nodes.add(el); | 233 testDiv.nodes.add(el); |
197 var model = toObservable({'x': 42}); | 234 var model = toObservable({'x': 42}); |
198 nodeBind(el).bind('value', new PathObserver(model, 'x')); | 235 bindings.add(nodeBind(el).bind('value', new PathObserver(model, 'x'))); |
199 expect(el.value, '42'); | 236 expect(el.value, '42'); |
200 | 237 |
201 model['x'] = 'Hi'; | 238 model['x'] = 'Hi'; |
202 expect(el.value, '42', reason: 'changes delivered async'); | 239 expect(el.value, '42', reason: 'changes delivered async'); |
203 return new Future(() { | 240 return new Future(() { |
204 expect(el.value, 'Hi'); | 241 expect(el.value, 'Hi'); |
205 | 242 |
206 el.value = 'changed'; | 243 el.value = 'changed'; |
207 dispatchEvent('input', el); | 244 dispatchEvent('input', el); |
208 expect(model['x'], 'changed'); | 245 expect(model['x'], 'changed'); |
209 | |
210 nodeBind(el).unbind('value'); | |
211 | |
212 el.value = 'changed again'; | |
213 dispatchEvent('input', el); | |
214 expect(model['x'], 'changed'); | |
215 | |
216 nodeBind(el).bind('value', new PathObserver(model, 'x')); | |
217 model['x'] = null; | |
218 }).then(endOfMicrotask).then((_) { | |
219 expect(el.value, ''); | |
220 }); | 246 }); |
221 } | 247 } |
222 | 248 |
223 inputTextAreaValueOnetime(String tagName) { | 249 inputTextAreaValueOnetime(String tagName) { |
224 var el = testDiv.append(new Element.tag(tagName)); | 250 var el = testDiv.append(new Element.tag(tagName)); |
225 nodeBind(el).bind('value', 42, oneTime: true); | 251 bindings.add(nodeBind(el).bind('value', 42, oneTime: true)); |
226 expect(el.value, '42'); | 252 expect(el.value, '42'); |
227 } | 253 } |
228 | 254 |
229 inputTextAreaNoPath(String tagName) { | 255 inputTextAreaNoPath(String tagName) { |
230 var el = testDiv.append(new Element.tag(tagName)); | 256 var el = testDiv.append(new Element.tag(tagName)); |
231 var model = 42; | 257 var model = 42; |
232 nodeBind(el).bind('value', new PathObserver(model)); | 258 bindings.add(nodeBind(el).bind('value', new PathObserver(model))); |
233 expect(el.value, '42'); | 259 expect(el.value, '42'); |
234 } | 260 } |
235 | 261 |
236 inputTextAreaPathUnreachable(String tagName) { | 262 inputTextAreaPathUnreachable(String tagName) { |
237 var el = testDiv.append(new Element.tag(tagName)); | 263 var el = testDiv.append(new Element.tag(tagName)); |
238 var model = toObservable({}); | 264 var model = toObservable({}); |
239 nodeBind(el).bind('value', new PathObserver(model, 'a')); | 265 bindings.add(nodeBind(el).bind('value', new PathObserver(model, 'a'))); |
240 expect(el.value, ''); | 266 expect(el.value, ''); |
241 } | 267 } |
242 | 268 |
243 test('Input.value', | 269 test('Input.value', |
244 () => inputTextAreaValueTest('input')); | 270 () => inputTextAreaValueTest('input')); |
245 | 271 |
246 test('Input.value - oneTime', | 272 test('Input.value - oneTime', |
247 () => inputTextAreaValueOnetime('input')); | 273 () => inputTextAreaValueOnetime('input')); |
248 | 274 |
249 test('Input.value - no path', | 275 test('Input.value - no path', |
(...skipping 11 matching lines...) Expand all Loading... |
261 test('TextArea.value - no path', | 287 test('TextArea.value - no path', |
262 () => inputTextAreaNoPath('textarea')); | 288 () => inputTextAreaNoPath('textarea')); |
263 | 289 |
264 test('TextArea.value - path unreachable', | 290 test('TextArea.value - path unreachable', |
265 () => inputTextAreaPathUnreachable('textarea')); | 291 () => inputTextAreaPathUnreachable('textarea')); |
266 | 292 |
267 test('Radio Input', () { | 293 test('Radio Input', () { |
268 var input = new InputElement(); | 294 var input = new InputElement(); |
269 input.type = 'radio'; | 295 input.type = 'radio'; |
270 var model = toObservable({'x': true}); | 296 var model = toObservable({'x': true}); |
271 nodeBind(input).bind('checked', new PathObserver(model, 'x')); | 297 bindings.add(nodeBind(input).bind('checked', new PathObserver(model, 'x'))); |
272 expect(input.checked, true); | 298 expect(input.checked, true); |
273 | 299 |
274 model['x'] = false; | 300 model['x'] = false; |
275 expect(input.checked, true); | 301 expect(input.checked, true); |
276 return new Future(() { | 302 return new Future(() { |
277 expect(input.checked, false,reason: 'model change should update checked'); | 303 expect(input.checked, false,reason: 'model change should update checked'); |
278 | 304 |
279 input.checked = true; | 305 input.checked = true; |
280 dispatchEvent('change', input); | 306 dispatchEvent('change', input); |
281 expect(model['x'], true, reason: 'input.checked should set model'); | 307 expect(model['x'], true, reason: 'input.checked should set model'); |
282 | 308 |
283 nodeBind(input).unbind('checked'); | 309 bindings[0].close(); |
284 | 310 |
285 input.checked = false; | 311 input.checked = false; |
286 dispatchEvent('change', input); | 312 dispatchEvent('change', input); |
287 expect(model['x'], true, | 313 expect(model['x'], true, |
288 reason: 'disconnected binding should not fire'); | 314 reason: 'disconnected binding should not fire'); |
289 }); | 315 }); |
290 }); | 316 }); |
291 | 317 |
292 test('Input.value - user value rejected', () { | 318 test('Input.value - user value rejected', () { |
293 var model = toObservable({'val': 'ping'}); | 319 var model = toObservable({'val': 'ping'}); |
294 | 320 |
| 321 var rejector = new PathObserver(model, 'val'); |
| 322 rejector.open(() { |
| 323 model['val'] = 'ping'; |
| 324 }); |
| 325 |
295 var el = new InputElement(); | 326 var el = new InputElement(); |
296 nodeBind(el).bind('value', new PathObserver(model, 'val')); | 327 bindings.add(nodeBind(el).bind('value', new PathObserver(model, 'val'))); |
| 328 |
297 return new Future(() { | 329 return new Future(() { |
298 expect(el.value, 'ping'); | 330 expect(el.value, 'ping'); |
299 | 331 |
300 el.value = 'pong'; | 332 el.value = 'pong'; |
301 dispatchEvent('input', el); | 333 dispatchEvent('input', el); |
302 expect(model['val'], 'pong'); | |
303 | 334 |
304 // Try a deep path. | |
305 model = toObservable({'a': {'b': {'c': 'ping'}}}); | |
306 | |
307 nodeBind(el).bind('value', new PathObserver(model, 'a.b.c')); | |
308 }).then(endOfMicrotask).then((_) { | 335 }).then(endOfMicrotask).then((_) { |
| 336 // rejector will have set the bound value back to 'ping'. |
309 expect(el.value, 'ping'); | 337 expect(el.value, 'ping'); |
310 | 338 |
311 el.value = 'pong'; | 339 rejector.close(); |
312 dispatchEvent('input', el); | |
313 expect(new PropertyPath('a.b.c').getValueFrom(model), 'pong'); | |
314 | |
315 // Start with the model property being absent. | |
316 model['a']['b'].remove('c'); | |
317 }).then(endOfMicrotask).then((_) { | |
318 expect(el.value, ''); | |
319 | |
320 el.value = 'pong'; | |
321 dispatchEvent('input', el); | |
322 expect(new PropertyPath('a.b.c').getValueFrom(model), 'pong'); | |
323 }).then(endOfMicrotask).then((_) { | |
324 | |
325 // Model property unreachable (and unsettable). | |
326 model['a'].remove('b'); | |
327 }).then(endOfMicrotask).then((_) { | |
328 expect(el.value, ''); | |
329 | |
330 el.value = 'pong'; | |
331 dispatchEvent('input', el); | |
332 expect(new PropertyPath('a.b.c').getValueFrom(model), null); | |
333 }); | 340 }); |
334 }); | 341 }); |
335 | 342 |
336 test('Checkbox Input.checked', () { | 343 test('Checkbox Input.checked', () { |
337 var el = testDiv.append(new InputElement()); | 344 var el = testDiv.append(new InputElement()); |
338 el.type = 'checkbox'; | 345 el.type = 'checkbox'; |
339 | 346 |
340 var model = toObservable({'x': true}); | 347 var model = toObservable({'x': true}); |
341 nodeBind(el).bind('checked', new PathObserver(model, 'x')); | 348 bindings.add(nodeBind(el).bind('checked', new PathObserver(model, 'x'))); |
342 expect(el.checked, true); | 349 expect(el.checked, true); |
343 | 350 |
344 model['x'] = false; | 351 model['x'] = false; |
345 expect(el.checked, true, reason: 'changes delivered async'); | 352 expect(el.checked, true, reason: 'changes delivered async'); |
346 return new Future(() { | 353 return new Future(() { |
347 expect(el.checked, false); | 354 expect(el.checked, false); |
348 | 355 |
349 el.click(); | 356 el.click(); |
350 expect(model['x'], true); | 357 expect(model['x'], true); |
351 }).then(endOfMicrotask).then((_) { | 358 }).then(endOfMicrotask).then((_) { |
352 | 359 |
353 el.click(); | 360 el.click(); |
354 expect(model['x'], false); | 361 expect(model['x'], false); |
355 }); | 362 }); |
356 }); | 363 }); |
357 | 364 |
358 test('Checkbox Input.checked - oneTime', () { | 365 test('Checkbox Input.checked - oneTime', () { |
359 var input = testDiv.append(new InputElement()); | 366 var input = testDiv.append(new InputElement()); |
360 input.type = 'checkbox'; | 367 input.type = 'checkbox'; |
361 nodeBind(input).bind('checked', true, oneTime: true); | 368 bindings.add(nodeBind(input).bind('checked', true, oneTime: true)); |
362 expect(input.checked, true, reason: 'checked was set'); | 369 expect(input.checked, true, reason: 'checked was set'); |
363 }); | 370 }); |
364 | 371 |
365 test('Checkbox Input.checked - path unreachable', () { | 372 test('Checkbox Input.checked - path unreachable', () { |
366 var input = testDiv.append(new InputElement()); | 373 var input = testDiv.append(new InputElement()); |
367 input.type = 'checkbox'; | 374 input.type = 'checkbox'; |
368 var model = toObservable({}); | 375 var model = toObservable({}); |
369 nodeBind(input).bind('checked', new PathObserver(model, 'x')); | 376 bindings.add(nodeBind(input).bind('checked', new PathObserver(model, 'x'))); |
370 expect(input.checked, false); | 377 expect(input.checked, false); |
371 }); | 378 }); |
372 | 379 |
373 test('Checkbox Input.checked 2', () { | 380 test('Checkbox Input.checked 2', () { |
374 var model = toObservable({'val': true}); | 381 var model = toObservable({'val': true}); |
375 | 382 |
376 var el = testDiv.append(new InputElement()); | 383 var el = testDiv.append(new InputElement()); |
377 el.type = 'checkbox'; | 384 el.type = 'checkbox'; |
378 nodeBind(el).bind('checked', new PathObserver(model, 'val')); | 385 bindings.add(nodeBind(el).bind('checked', new PathObserver(model, 'val'))); |
379 return new Future(() { | 386 return new Future(() { |
380 expect(el.checked, true); | 387 expect(el.checked, true); |
381 | 388 |
382 model['val'] = false; | 389 model['val'] = false; |
383 }).then(endOfMicrotask).then((_) { | 390 }).then(endOfMicrotask).then((_) { |
384 expect(el.checked, false); | 391 expect(el.checked, false); |
385 | 392 |
386 el.click(); | 393 el.click(); |
387 expect(model['val'], true); | 394 expect(model['val'], true); |
388 | 395 |
(...skipping 10 matching lines...) Expand all Loading... |
399 el.dispatchEvent(new MouseEvent('click', view: window)); | 406 el.dispatchEvent(new MouseEvent('click', view: window)); |
400 }); | 407 }); |
401 }); | 408 }); |
402 | 409 |
403 test('Checkbox Input.checked - binding updated on click', () { | 410 test('Checkbox Input.checked - binding updated on click', () { |
404 var model = toObservable({'val': true}); | 411 var model = toObservable({'val': true}); |
405 | 412 |
406 var el = new InputElement(); | 413 var el = new InputElement(); |
407 testDiv.append(el); | 414 testDiv.append(el); |
408 el.type = 'checkbox'; | 415 el.type = 'checkbox'; |
409 nodeBind(el).bind('checked', new PathObserver(model, 'val')); | 416 bindings.add(nodeBind(el).bind('checked', new PathObserver(model, 'val'))); |
410 return new Future(() { | 417 return new Future(() { |
411 expect(el.checked, true); | 418 expect(el.checked, true); |
412 | 419 |
413 int fired = 0; | 420 int fired = 0; |
414 el.onClick.listen((_) { | 421 el.onClick.listen((_) { |
415 fired++; | 422 fired++; |
416 expect(model['val'], false); | 423 expect(model['val'], false); |
417 }); | 424 }); |
418 | 425 |
419 el.dispatchEvent(new MouseEvent('click', view: window)); | 426 el.dispatchEvent(new MouseEvent('click', view: window)); |
420 | 427 |
421 expect(fired, 1, reason: 'events dispatched synchronously'); | 428 expect(fired, 1, reason: 'events dispatched synchronously'); |
422 }); | 429 }); |
423 }); | 430 }); |
424 | 431 |
425 test('Checkbox Input.checked - binding updated on change', () { | 432 test('Checkbox Input.checked - binding updated on change', () { |
426 var model = toObservable({'val': true}); | 433 var model = toObservable({'val': true}); |
427 | 434 |
428 var el = new InputElement(); | 435 var el = new InputElement(); |
429 testDiv.append(el); | 436 testDiv.append(el); |
430 el.type = 'checkbox'; | 437 el.type = 'checkbox'; |
431 nodeBind(el).bind('checked', new PathObserver(model, 'val')); | 438 bindings.add(nodeBind(el).bind('checked', new PathObserver(model, 'val'))); |
432 return new Future(() { | 439 return new Future(() { |
433 expect(el.checked, true); | 440 expect(el.checked, true); |
434 | 441 |
435 int fired = 0; | 442 int fired = 0; |
436 el.onChange.listen((_) { | 443 el.onChange.listen((_) { |
437 fired++; | 444 fired++; |
438 expect(model['val'], false); | 445 expect(model['val'], false); |
439 }); | 446 }); |
440 | 447 |
441 el.dispatchEvent(new MouseEvent('click', view: window)); | 448 el.dispatchEvent(new MouseEvent('click', view: window)); |
442 | 449 |
443 expect(fired, 1, reason: 'events dispatched synchronously'); | 450 expect(fired, 1, reason: 'events dispatched synchronously'); |
444 }); | 451 }); |
445 }); | 452 }); |
446 | 453 |
447 test('Radio Input.checked', () { | 454 test('Radio Input.checked', () { |
448 var input = testDiv.append(new InputElement()); | 455 var input = testDiv.append(new InputElement()); |
449 input.type = 'radio'; | 456 input.type = 'radio'; |
450 var model = toObservable({'x': true}); | 457 var model = toObservable({'x': true}); |
451 nodeBind(input).bind('checked', new PathObserver(model, 'x')); | 458 bindings.add(nodeBind(input).bind('checked', new PathObserver(model, 'x'))); |
452 expect(input.checked, true); | 459 expect(input.checked, true); |
453 | 460 |
454 model['x'] = false; | 461 model['x'] = false; |
455 expect(input.checked, true); | 462 expect(input.checked, true); |
456 return new Future(() { | 463 return new Future(() { |
457 expect(input.checked, false); | 464 expect(input.checked, false); |
458 | 465 |
459 input.checked = true; | 466 input.checked = true; |
460 dispatchEvent('change', input); | 467 dispatchEvent('change', input); |
461 expect(model['x'], true); | 468 expect(model['x'], true); |
462 | |
463 nodeBind(input).unbind('checked'); | |
464 | |
465 input.checked = false; | |
466 dispatchEvent('change', input); | |
467 expect(model['x'], true); | |
468 }); | 469 }); |
469 }); | 470 }); |
470 | 471 |
471 test('Radio Input.checked - oneTime', () { | 472 test('Radio Input.checked - oneTime', () { |
472 var input = testDiv.append(new InputElement()); | 473 var input = testDiv.append(new InputElement()); |
473 input.type = 'radio'; | 474 input.type = 'radio'; |
474 nodeBind(input).bind('checked', true, oneTime: true); | 475 bindings.add(nodeBind(input).bind('checked', true, oneTime: true)); |
475 expect(input.checked, true, reason: 'checked was set'); | 476 expect(input.checked, true, reason: 'checked was set'); |
476 }); | 477 }); |
477 | 478 |
478 radioInputChecked2(host) { | 479 radioInputChecked2(host) { |
479 var model = toObservable({'val1': true, 'val2': false, 'val3': false, | 480 var model = toObservable({'val1': true, 'val2': false, 'val3': false, |
480 'val4': true}); | 481 'val4': true}); |
481 var RADIO_GROUP_NAME = 'test'; | 482 var RADIO_GROUP_NAME = 'test'; |
482 | 483 |
483 var container = host.append(new DivElement()); | 484 var container = host.append(new DivElement()); |
484 | 485 |
485 var el1 = container.append(new InputElement()); | 486 var el1 = container.append(new InputElement()); |
486 el1.type = 'radio'; | 487 el1.type = 'radio'; |
487 el1.name = RADIO_GROUP_NAME; | 488 el1.name = RADIO_GROUP_NAME; |
488 nodeBind(el1).bind('checked', new PathObserver(model, 'val1')); | 489 bindings.add( |
| 490 nodeBind(el1).bind('checked', new PathObserver(model, 'val1'))); |
489 | 491 |
490 var el2 = container.append(new InputElement()); | 492 var el2 = container.append(new InputElement()); |
491 el2.type = 'radio'; | 493 el2.type = 'radio'; |
492 el2.name = RADIO_GROUP_NAME; | 494 el2.name = RADIO_GROUP_NAME; |
493 nodeBind(el2).bind('checked', new PathObserver(model, 'val2')); | 495 bindings.add( |
| 496 nodeBind(el2).bind('checked', new PathObserver(model, 'val2'))); |
494 | 497 |
495 var el3 = container.append(new InputElement()); | 498 var el3 = container.append(new InputElement()); |
496 el3.type = 'radio'; | 499 el3.type = 'radio'; |
497 el3.name = RADIO_GROUP_NAME; | 500 el3.name = RADIO_GROUP_NAME; |
498 nodeBind(el3).bind('checked', new PathObserver(model, 'val3')); | 501 bindings.add( |
| 502 nodeBind(el3).bind('checked', new PathObserver(model, 'val3'))); |
499 | 503 |
500 var el4 = container.append(new InputElement()); | 504 var el4 = container.append(new InputElement()); |
501 el4.type = 'radio'; | 505 el4.type = 'radio'; |
502 el4.name = 'othergroup'; | 506 el4.name = 'othergroup'; |
503 nodeBind(el4).bind('checked', new PathObserver(model, 'val4')); | 507 bindings.add( |
| 508 nodeBind(el4).bind('checked', new PathObserver(model, 'val4'))); |
504 | 509 |
505 return new Future(() { | 510 return new Future(() { |
506 expect(el1.checked, true); | 511 expect(el1.checked, true); |
507 expect(el2.checked, false); | 512 expect(el2.checked, false); |
508 expect(el3.checked, false); | 513 expect(el3.checked, false); |
509 expect(el4.checked, true); | 514 expect(el4.checked, true); |
510 | 515 |
511 model['val1'] = false; | 516 model['val1'] = false; |
512 model['val2'] = true; | 517 model['val2'] = true; |
513 }).then(endOfMicrotask).then((_) { | 518 }).then(endOfMicrotask).then((_) { |
(...skipping 17 matching lines...) Expand all Loading... |
531 expect(model['val4'], true); | 536 expect(model['val4'], true); |
532 }); | 537 }); |
533 } | 538 } |
534 | 539 |
535 test('Radio Input.checked 2', () => radioInputChecked2(testDiv)); | 540 test('Radio Input.checked 2', () => radioInputChecked2(testDiv)); |
536 | 541 |
537 test('Radio Input.checked 2 - ShadowRoot', () { | 542 test('Radio Input.checked 2 - ShadowRoot', () { |
538 if (!ShadowRoot.supported) return null; | 543 if (!ShadowRoot.supported) return null; |
539 | 544 |
540 var shadowRoot = new DivElement().createShadowRoot(); | 545 var shadowRoot = new DivElement().createShadowRoot(); |
541 return radioInputChecked2(shadowRoot) | 546 return radioInputChecked2(shadowRoot); |
542 .whenComplete(() => unbindAll(shadowRoot)); | |
543 }); | 547 }); |
544 | 548 |
545 radioInputCheckedMultipleForms(host) { | 549 radioInputCheckedMultipleForms(host) { |
546 var model = toObservable({'val1': true, 'val2': false, 'val3': false, | 550 var model = toObservable({'val1': true, 'val2': false, 'val3': false, |
547 'val4': true}); | 551 'val4': true}); |
548 var RADIO_GROUP_NAME = 'test'; | 552 var RADIO_GROUP_NAME = 'test'; |
549 | 553 |
550 var container = testDiv.append(new DivElement()); | 554 var container = testDiv.append(new DivElement()); |
551 var form1 = new FormElement(); | 555 var form1 = new FormElement(); |
552 container.append(form1); | 556 container.append(form1); |
553 var form2 = new FormElement(); | 557 var form2 = new FormElement(); |
554 container.append(form2); | 558 container.append(form2); |
555 | 559 |
556 var el1 = new InputElement(); | 560 var el1 = new InputElement(); |
557 form1.append(el1); | 561 form1.append(el1); |
558 el1.type = 'radio'; | 562 el1.type = 'radio'; |
559 el1.name = RADIO_GROUP_NAME; | 563 el1.name = RADIO_GROUP_NAME; |
560 nodeBind(el1).bind('checked', new PathObserver(model, 'val1')); | 564 bindings.add( |
| 565 nodeBind(el1).bind('checked', new PathObserver(model, 'val1'))); |
561 | 566 |
562 var el2 = new InputElement(); | 567 var el2 = new InputElement(); |
563 form1.append(el2); | 568 form1.append(el2); |
564 el2.type = 'radio'; | 569 el2.type = 'radio'; |
565 el2.name = RADIO_GROUP_NAME; | 570 el2.name = RADIO_GROUP_NAME; |
566 nodeBind(el2).bind('checked', new PathObserver(model, 'val2')); | 571 bindings.add( |
| 572 nodeBind(el2).bind('checked', new PathObserver(model, 'val2'))); |
567 | 573 |
568 var el3 = new InputElement(); | 574 var el3 = new InputElement(); |
569 form2.append(el3); | 575 form2.append(el3); |
570 el3.type = 'radio'; | 576 el3.type = 'radio'; |
571 el3.name = RADIO_GROUP_NAME; | 577 el3.name = RADIO_GROUP_NAME; |
572 nodeBind(el3).bind('checked', new PathObserver(model, 'val3')); | 578 bindings.add( |
| 579 nodeBind(el3).bind('checked', new PathObserver(model, 'val3'))); |
573 | 580 |
574 var el4 = new InputElement(); | 581 var el4 = new InputElement(); |
575 form2.append(el4); | 582 form2.append(el4); |
576 el4.type = 'radio'; | 583 el4.type = 'radio'; |
577 el4.name = RADIO_GROUP_NAME; | 584 el4.name = RADIO_GROUP_NAME; |
578 nodeBind(el4).bind('checked', new PathObserver(model, 'val4')); | 585 bindings.add( |
| 586 nodeBind(el4).bind('checked', new PathObserver(model, 'val4'))); |
579 | 587 |
580 return new Future(() { | 588 return new Future(() { |
581 expect(el1.checked, true); | 589 expect(el1.checked, true); |
582 expect(el2.checked, false); | 590 expect(el2.checked, false); |
583 expect(el3.checked, false); | 591 expect(el3.checked, false); |
584 expect(el4.checked, true); | 592 expect(el4.checked, true); |
585 | 593 |
586 el2.checked = true; | 594 el2.checked = true; |
587 dispatchEvent('change', el2); | 595 dispatchEvent('change', el2); |
588 expect(model['val1'], false); | 596 expect(model['val1'], false); |
(...skipping 15 matching lines...) Expand all Loading... |
604 } | 612 } |
605 | 613 |
606 test('Radio Input.checked - multiple forms', () { | 614 test('Radio Input.checked - multiple forms', () { |
607 return radioInputCheckedMultipleForms(testDiv); | 615 return radioInputCheckedMultipleForms(testDiv); |
608 }); | 616 }); |
609 | 617 |
610 test('Radio Input.checked - multiple forms - ShadowRoot', () { | 618 test('Radio Input.checked - multiple forms - ShadowRoot', () { |
611 if (!ShadowRoot.supported) return null; | 619 if (!ShadowRoot.supported) return null; |
612 | 620 |
613 var shadowRoot = new DivElement().createShadowRoot(); | 621 var shadowRoot = new DivElement().createShadowRoot(); |
614 return radioInputCheckedMultipleForms(shadowRoot) | 622 return radioInputCheckedMultipleForms(shadowRoot); |
615 .whenComplete(() => unbindAll(shadowRoot)); | |
616 }); | 623 }); |
617 | 624 |
618 test('Select.selectedIndex', () { | 625 test('Select.selectedIndex', () { |
619 var select = new SelectElement(); | 626 var select = new SelectElement(); |
620 testDiv.append(select); | 627 testDiv.append(select); |
621 var option0 = select.append(new OptionElement()); | 628 var option0 = select.append(new OptionElement()); |
622 var option1 = select.append(new OptionElement()); | 629 var option1 = select.append(new OptionElement()); |
623 var option2 = select.append(new OptionElement()); | 630 var option2 = select.append(new OptionElement()); |
624 | 631 |
625 var model = toObservable({'val': 2}); | 632 var model = toObservable({'val': 2}); |
626 | 633 |
627 nodeBind(select).bind('selectedIndex', new PathObserver(model, 'val')); | 634 bindings.add( |
| 635 nodeBind(select).bind('selectedIndex', new PathObserver(model, 'val'))); |
628 return new Future(() { | 636 return new Future(() { |
629 expect(select.selectedIndex, 2); | 637 expect(select.selectedIndex, 2); |
630 | 638 |
631 select.selectedIndex = 1; | 639 select.selectedIndex = 1; |
632 dispatchEvent('change', select); | 640 dispatchEvent('change', select); |
633 expect(model['val'], 1); | 641 expect(model['val'], 1); |
634 }); | 642 }); |
635 }); | 643 }); |
636 | 644 |
637 test('Select.selectedIndex - oneTime', () { | 645 test('Select.selectedIndex - oneTime', () { |
638 var select = new SelectElement(); | 646 var select = new SelectElement(); |
639 testDiv.append(select); | 647 testDiv.append(select); |
640 var option0 = select.append(new OptionElement()); | 648 var option0 = select.append(new OptionElement()); |
641 var option1 = select.append(new OptionElement()); | 649 var option1 = select.append(new OptionElement()); |
642 var option2 = select.append(new OptionElement()); | 650 var option2 = select.append(new OptionElement()); |
643 | 651 |
644 nodeBind(select).bind('selectedIndex', 2, oneTime: true); | 652 bindings.add(nodeBind(select).bind('selectedIndex', 2, oneTime: true)); |
645 return new Future(() => expect(select.selectedIndex, 2)); | 653 return new Future(() => expect(select.selectedIndex, 2)); |
646 }); | 654 }); |
647 | 655 |
648 test('Select.selectedIndex - invalid path', () { | 656 test('Select.selectedIndex - invalid path', () { |
649 var select = new SelectElement(); | 657 var select = new SelectElement(); |
650 testDiv.append(select); | 658 testDiv.append(select); |
651 var option0 = select.append(new OptionElement()); | 659 var option0 = select.append(new OptionElement()); |
652 var option1 = select.append(new OptionElement()); | 660 var option1 = select.append(new OptionElement()); |
653 option1.selected = true; | 661 option1.selected = true; |
654 var option2 = select.append(new OptionElement()); | 662 var option2 = select.append(new OptionElement()); |
655 | 663 |
656 var model = toObservable({'val': 'foo'}); | 664 var model = toObservable({'val': 'foo'}); |
657 | 665 |
658 nodeBind(select).bind('selectedIndex', new PathObserver(model, 'val')); | 666 bindings.add( |
| 667 nodeBind(select).bind('selectedIndex', new PathObserver(model, 'val'))); |
659 return new Future(() => expect(select.selectedIndex, 0)); | 668 return new Future(() => expect(select.selectedIndex, 0)); |
660 }); | 669 }); |
661 | 670 |
662 test('Select.selectedIndex - path unreachable', () { | 671 test('Select.selectedIndex - path unreachable', () { |
663 var select = new SelectElement(); | 672 var select = new SelectElement(); |
664 testDiv.append(select); | 673 testDiv.append(select); |
665 var option0 = select.append(new OptionElement()); | 674 var option0 = select.append(new OptionElement()); |
666 var option1 = select.append(new OptionElement()); | 675 var option1 = select.append(new OptionElement()); |
667 option1.selected = true; | 676 option1.selected = true; |
668 var option2 = select.append(new OptionElement()); | 677 var option2 = select.append(new OptionElement()); |
669 | 678 |
670 var model = toObservable({}); | 679 var model = toObservable({}); |
671 | 680 |
672 nodeBind(select).bind('selectedIndex', new PathObserver(model, 'val')); | 681 bindings.add( |
| 682 nodeBind(select).bind('selectedIndex', new PathObserver(model, 'val'))); |
673 return new Future(() => expect(select.selectedIndex, 0)); | 683 return new Future(() => expect(select.selectedIndex, 0)); |
674 }); | 684 }); |
675 | 685 |
676 test('Option.value', () { | 686 test('Option.value', () { |
677 var option = testDiv.append(new OptionElement()); | 687 var option = testDiv.append(new OptionElement()); |
678 var model = toObservable({'x': 42}); | 688 var model = toObservable({'x': 42}); |
679 nodeBind(option).bind('value', new PathObserver(model, 'x')); | 689 bindings.add(nodeBind(option).bind('value', new PathObserver(model, 'x'))); |
680 expect(option.value, '42'); | 690 expect(option.value, '42'); |
681 | 691 |
682 model['x'] = 'Hi'; | 692 model['x'] = 'Hi'; |
683 expect(option.value, '42'); | 693 expect(option.value, '42'); |
684 return new Future(() => expect(option.value, 'Hi')); | 694 return new Future(() => expect(option.value, 'Hi')); |
685 }); | 695 }); |
686 | 696 |
687 test('Option.value - oneTime', () { | 697 test('Option.value - oneTime', () { |
688 var option = testDiv.append(new OptionElement()); | 698 var option = testDiv.append(new OptionElement()); |
689 nodeBind(option).bind('value', 42, oneTime: true); | 699 bindings.add(nodeBind(option).bind('value', 42, oneTime: true)); |
690 expect(option.value, '42'); | 700 expect(option.value, '42'); |
691 }); | 701 }); |
692 | 702 |
693 test('Select.value', () { | 703 test('Select.value', () { |
694 var select = testDiv.append(new SelectElement()); | 704 var select = testDiv.append(new SelectElement()); |
695 testDiv.append(select); | 705 testDiv.append(select); |
696 var option0 = select.append(new OptionElement()); | 706 var option0 = select.append(new OptionElement()); |
697 var option1 = select.append(new OptionElement()); | 707 var option1 = select.append(new OptionElement()); |
698 var option2 = select.append(new OptionElement()); | 708 var option2 = select.append(new OptionElement()); |
699 | 709 |
700 var model = toObservable({ | 710 var model = toObservable({ |
701 'opt0': 'a', | 711 'opt0': 'a', |
702 'opt1': 'b', | 712 'opt1': 'b', |
703 'opt2': 'c', | 713 'opt2': 'c', |
704 'selected': 'b' | 714 'selected': 'b' |
705 }); | 715 }); |
706 | 716 |
707 nodeBind(option0).bind('value', new PathObserver(model, 'opt0')); | 717 bindings.add( |
708 nodeBind(option1).bind('value', new PathObserver(model, 'opt1')); | 718 nodeBind(option0).bind('value', new PathObserver(model, 'opt0'))); |
709 nodeBind(option2).bind('value', new PathObserver(model, 'opt2')); | 719 bindings.add( |
710 | 720 nodeBind(option1).bind('value', new PathObserver(model, 'opt1'))); |
711 nodeBind(select).bind('value', new PathObserver(model, 'selected')); | 721 bindings.add( |
| 722 nodeBind(option2).bind('value', new PathObserver(model, 'opt2'))); |
| 723 bindings.add( |
| 724 nodeBind(select).bind('value', new PathObserver(model, 'selected'))); |
712 return new Future(() { | 725 return new Future(() { |
713 expect(select.value, 'b'); | 726 expect(select.value, 'b'); |
714 | 727 |
715 select.value = 'c'; | 728 select.value = 'c'; |
716 dispatchEvent('change', select); | 729 dispatchEvent('change', select); |
717 expect(model['selected'], 'c'); | 730 expect(model['selected'], 'c'); |
718 | 731 |
719 model['opt2'] = 'X'; | 732 model['opt2'] = 'X'; |
720 }).then(endOfMicrotask).then((_) { | 733 }).then(endOfMicrotask).then((_) { |
721 expect(select.value, 'X'); | 734 expect(select.value, 'X'); |
722 expect(model['selected'], 'X'); | 735 expect(model['selected'], 'X'); |
723 | 736 |
724 model['selected'] = 'a'; | 737 model['selected'] = 'a'; |
725 }).then(endOfMicrotask).then((_) { | 738 }).then(endOfMicrotask).then((_) { |
726 expect(select.value, 'a'); | 739 expect(select.value, 'a'); |
727 }); | 740 }); |
728 }); | 741 }); |
729 } | 742 } |
OLD | NEW |