Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(123)

Side by Side Diff: tests/html/js_tests.dart

Issue 15782009: RFC: introduce dart:js (Closed) Base URL: http://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: Created 7 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
(Empty)
1 // Copyright (c) 2012, 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 js_tests;
6
7 import 'dart:async';
8 import 'dart:html';
9 import 'dart:js';
10
11 import '../../pkg/unittest/lib/unittest.dart';
12 import '../../pkg/unittest/lib/html_config.dart';
13
14 class Foo implements Serializable<JsObject> {
15 final JsObject _proxy;
16
17 Foo(num a) : this._proxy = new JsObject(context['Foo'], [a]);
18
19 JsObject toJs() => _proxy;
20
21 num get a => _proxy['a'];
22 num bar() => _proxy.callMethod('bar');
23 }
24
25 class Color implements Serializable<String> {
26 static final RED = new Color._("red");
27 static final BLUE = new Color._("blue");
28 String _value;
29 Color._(this._value);
30 String toJs() => this._value;
31 }
32
33 main() {
34 useHtmlConfiguration();
35
36 test('test scope', () {
37 var ctx;
38 scoped(() {
39 ctx = context;
40 });
41 scoped(() {
42 expect(() => ctx['x'], throws);
43 });
44 });
45
46 test('read global field', () {
47 expect(context['x'], equals(42));
48 expect(context['y'], isNull);
49 });
50
51 test('read global field with underscore', () {
52 expect(context['_x'], equals(123));
53 expect(context['y'], isNull);
54 });
55
56 test('js instantiation : new Foo()', () {
57 final Foo2 = context['container']['Foo'];
58 final foo = new JsObject(Foo2, [42]);
59 expect(foo['a'], 42);
60 expect(Foo2['b'], 38);
61 });
62
63 test('js instantiation : new Array()', () {
64 final a = new JsObject(context['Array']);
65 expect(a, isNotNull);
66 expect(a['length'], equals(0));
67
68 a.callMethod('push', ["value 1"]);
69 expect(a['length'], equals(1));
70 expect(a[0], equals("value 1"));
71
72 a.callMethod('pop');
73 expect(a['length'], equals(0));
74 });
75
76 test('js instantiation : new Date()', () {
77 final a = new JsObject(context['Date']);
78 expect(a.callMethod('getTime'), isNotNull);
79 });
80
81 test('js instantiation : new Date(12345678)', () {
82 final a = new JsObject(context['Date'], [12345678]);
83 expect(a.callMethod('getTime'), equals(12345678));
84 });
85
86 test('js instantiation : new Date("December 17, 1995 03:24:00 GMT+01:00")',
87 () {
88 final a = new JsObject(context['Date'],
89 ["December 17, 1995 03:24:00 GMT+01:00"]);
90 expect(a.callMethod('getTime'), equals(819167040000));
91 });
92
93 test('js instantiation : new Date(1995,11,17)', () {
94 // Note: JS Date counts months from 0 while Dart counts from 1.
95 final a = new JsObject(context['Date'], [1995, 11, 17]);
96 final b = new DateTime(1995, 12, 17);
97 expect(a.callMethod('getTime'), equals(b.millisecondsSinceEpoch));
98 });
99
100 test('js instantiation : new Date(1995,11,17,3,24,0)', () {
101 // Note: JS Date counts months from 0 while Dart counts from 1.
102 final a = new JsObject(context['Date'],
103 [1995, 11, 17, 3, 24, 0]);
104 final b = new DateTime(1995, 12, 17, 3, 24, 0);
105 expect(a.callMethod('getTime'), equals(b.millisecondsSinceEpoch));
106 });
107
108 test('js instantiation : new Object()', () {
109 final a = new JsObject(context['Object']);
110 expect(a, isNotNull);
111
112 a['attr'] = "value";
113 expect(a['attr'], equals("value"));
114 });
115
116 test(r'js instantiation : new RegExp("^\w+$")', () {
117 final a = new JsObject(context['RegExp'], [r'^\w+$']);
118 expect(a, isNotNull);
119 expect(a.callMethod('test', ['true']), isTrue);
120 expect(a.callMethod('test', [' false']), isFalse);
121 });
122
123 test('js instantiation via map notation : new Array()', () {
124 final a = new JsObject(context['Array']);
125 expect(a, isNotNull);
126 expect(a['length'], equals(0));
127
128 a['push'].apply(a, ["value 1"]);
129 expect(a['length'], equals(1));
130 expect(a[0], equals("value 1"));
131
132 a['pop'].apply(a);
133 expect(a['length'], equals(0));
134 });
135
136 test('js instantiation via map notation : new Date()', () {
137 final a = new JsObject(context['Date']);
138 expect(a['getTime'].apply(a), isNotNull);
139 });
140
141 test('js instantiation : typed array', () {
142 final codeUnits = "test".codeUnits;
143 final buf = new JsObject(context['ArrayBuffer'], [codeUnits.length]);
144 final bufView = new JsObject(context['Uint8Array'], [buf]);
145 for (var i = 0; i < codeUnits.length; i++) {
146 bufView[i] = codeUnits[i];
147 }
148 });
149
150 test('write global field', () {
151 context['y'] = 42;
152 expect(context['y'], equals(42));
153 });
154
155 test('get JS JsFunction', () {
156 var razzle = context['razzle'];
157 expect(razzle.apply(context), equals(42));
158 });
159
160 test('call JS function', () {
161 expect(context.callMethod('razzle'), equals(42));
162 expect(() => context.callMethod('dazzle'), throwsA(isNoSuchMethodError));
163 });
164
165 test('call JS function via map notation', () {
166 expect(context['razzle'].apply(context), equals(42));
167 expect(() => context['dazzle'].apply(context), throwsA(isNoSuchMethodError)) ;
168 });
169
170 test('call JS function with varargs', () {
171 expect(context.callMethod('varArgs', [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]),
172 equals(55));
173 });
174
175 test('allocate JS object', () {
176 var foo = new JsObject(context['Foo'], [42]);
177 expect(foo['a'], equals(42));
178 expect(foo.callMethod('bar'), equals(42));
179 expect(() => foo.callMethod('baz'), throwsA(isNoSuchMethodError));
180 });
181
182 test('call toString()', () {
183 var foo = new JsObject(context['Foo'], [42]);
184 expect(foo.toString(), equals("I'm a Foo a=42"));
185 var container = context['container'];
186 expect(container.toString(), equals("[object Object]"));
187 });
188
189 test('allocate simple JS array', () {
190 final list = [1, 2, 3, 4, 5, 6, 7, 8];
191 var array = jsify(list);
192 expect(context.callMethod('isArray', [array]), isTrue);
193 expect(array['length'], equals(list.length));
194 for (var i = 0; i < list.length ; i++) {
195 expect(array[i], equals(list[i]));
196 }
197 });
198
199 test('allocate JS array with iterable', () {
200 final set = new Set.from([1, 2, 3, 4, 5, 6, 7, 8]);
201 var array = jsify(set);
202 expect(context.callMethod('isArray', [array]), isTrue);
203 expect(array['length'], equals(set.length));
204 for (var i = 0; i < array['length'] ; i++) {
205 expect(set.contains(array[i]), isTrue);
206 }
207 });
208
209 test('allocate simple JS map', () {
210 var map = {'a': 1, 'b': 2, 'c': 3};
211 var jsMap = jsify(map);
212 expect(!context.callMethod('isArray', [jsMap]), isTrue);
213 for (final key in map.keys) {
214 expect(context.callMethod('checkMap', [jsMap, key, map[key]]), isTrue);
215 }
216 });
217
218 test('allocate complex JS object', () {
219 final object =
220 {
221 'a': [1, [2, 3]],
222 'b': {
223 'c': 3,
224 'd': new JsObject(context['Foo'], [42])
225 },
226 'e': null
227 };
228 var jsObject = jsify(object);
229 expect(jsObject['a'][0], equals(object['a'][0]));
230 expect(jsObject['a'][1][0], equals(object['a'][1][0]));
231 expect(jsObject['a'][1][1], equals(object['a'][1][1]));
232 expect(jsObject['b']['c'], equals(object['b']['c']));
233 expect(jsObject['b']['d'], equals(object['b']['d']));
234 expect(jsObject['b']['d'].callMethod('bar'), equals(42));
235 expect(jsObject['e'], isNull);
236 });
237
238 test('invoke Dart callback from JS', () {
239 expect(() => context.callMethod('invokeCallback'), throws);
240
241 context['callback'] = new Callback.once(() => 42);
242 expect(context.callMethod('invokeCallback'), equals(42));
243 expect(() => context.callMethod('invokeCallback'), throws);
244 });
245
246 test('callback as parameter', () {
247 expect(context.callMethod('getTypeOf', [context['razzle']]), equals("functio n"));
248 });
249
250 test('invoke Dart callback from JS with this', () {
251 final constructor = new Callback.once(($this, arg1) {
252 $this['a'] = 42;
253 $this['b'] = jsify(["a", arg1]);
254 }, withThis: true);
255 var o = new JsObject(constructor, ["b"]);
256 expect(o['a'], equals(42));
257 expect(o['b'][0], equals("a"));
258 expect(o['b'][1], equals("b"));
259 });
260
261 test('invoke Dart callback from JS with 11 parameters', () {
262 context['callbackWith11params'] = new Callback.once((p1, p2, p3, p4,
263 p5, p6, p7, p8, p9, p10, p11) => '$p1$p2$p3$p4$p5$p6$p7$p8$p9$p10'
264 '$p11');
265 expect(context.callMethod('invokeCallbackWith11params'), equals('12345678910 11'));
266 });
267
268 test('create a Dart callback outside a scope', () {
269 // Note, the test framework does not guarantee that each test runs as a
270 // separate event. This test creates a new asynchronous event and
271 // ensures that a callback can be created without a scope (i.e., that the
272 // scope is created on demand).
273 final subtest = () {
274 var callback = new Callback.once(() => 42);
275 context['callback'] = callback;
276 expect(context.callMethod('invokeCallback'), equals(42));
277 };
278
279 runAsync(expectAsync0(subtest));
280 });
281
282 test('global scope', () {
283 var x;
284 var y;
285 scoped(() {
286 x = new JsObject(context['Foo'], [42]);
287 y = new JsObject(context['Foo'], [38]);
288 expect(x['a'], equals(42));
289 expect(y['a'], equals(38));
290 retain(y);
291 });
292 scoped(() {
293 expect(() => x['a'], throws);
294 expect(y['a'], equals(38));
295 release(y);
296 expect(() => y['a'], throws);
297 });
298 });
299
300 test('global scope for Serializable', () {
301 Foo x;
302 Foo y;
303 scoped(() {
304 x = new Foo(42);
305 y = new Foo(38);
306 expect(x.a, equals(42));
307 expect(y.a, equals(38));
308 retain(y);
309 });
310 scoped(() {
311 expect(() => x.a, throws);
312 expect(y.a, equals(38));
313 release(y);
314 expect(() => y.a, throws);
315 });
316 });
317
318 test('pass unattached Dom Element', () {
319 final div = new DivElement();
320 div.classes.add('a');
321 expect(context.callMethod('getElementAttribute',[div, 'class']), equals('a') );
322 });
323
324 test('pass unattached Dom Element two times on same call', () {
325 final div = new DivElement();
326 div.classes.add('a');
327 expect(context.callMethod('addClassAttributes', [jsify([div, div])]), equals ('aa'));
328 });
329
330 test('pass Dom Element attached to an unattached element', () {
331 final div = new DivElement();
332 div.classes.add('a');
333 final container = new DivElement();
334 container.children.add(div);
335 expect(context.callMethod('getElementAttribute', [div, 'class']), equals('a' ));
336 });
337
338 test('pass 2 Dom Elements attached to an unattached element', () {
339 final div1 = new DivElement();
340 div1.classes.add('a');
341 final div2 = new DivElement();
342 div2.classes.add('b');
343 final container = new DivElement();
344 container.children.add(div1);
345 container.children.add(div2);
346 final f = context['addClassAttributes'];
347 expect(f.apply(context, [jsify([div1, div2])]), equals('ab'));
348 });
349
350 test('pass multiple Dom Elements unattached to document', () {
351 // A is alone
352 // 1 and 3 are brother
353 // 2 is child of 3
354 final divA = new DivElement()..classes.add('A');
355 final div1 = new DivElement()..classes.add('1');
356 final div2 = new DivElement()..classes.add('2');
357 final div3 = new DivElement()..classes.add('3')..children.add(div2);
358 final container = new DivElement()..children.addAll([div1, div3]);
359 final f = context['addClassAttributes'];
360 expect(f.apply(context, [jsify([divA, div1, div2, div3])]), equals('A123'));
361 expect(f.apply(context, [jsify([divA, div1, div3, div2])]), equals('A132'));
362 expect(f.apply(context, [jsify([divA, div1, div1, div3, divA, div2, div3])]) ,
363 equals('A113A23'));
364 expect(!document.documentElement.contains(divA), isTrue);
365 expect(!document.documentElement.contains(div1), isTrue);
366 expect(!document.documentElement.contains(div2), isTrue);
367 expect(!document.documentElement.contains(div3), isTrue);
368 expect(!document.documentElement.contains(container), isTrue);
369 });
370
371 test('pass one Dom Elements unattached and another attached', () {
372 final div1 = new DivElement()..classes.add('1');
373 final div2 = new DivElement()..classes.add('2');
374 document.documentElement.children.add(div2);
375 final f = context['addClassAttributes'];
376 expect(f.apply(context, [jsify([div1, div2])]), equals('12'));
377 expect(!document.documentElement.contains(div1), isTrue);
378 expect(document.documentElement.contains(div2), isTrue);
379 });
380
381 test('pass documentElement', () {
382 expect(context.callMethod('returnElement', [document.documentElement]),
383 equals(document.documentElement));
384 });
385
386 test('retrieve unattached Dom Element', () {
387 var result = context.callMethod('getNewDivElement');
388 expect(result is DivElement, isTrue);
389 expect(!document.documentElement.contains(result), isTrue);
390 });
391
392 test('element of foreign document should not be serialized', () {
393 final foreignDoc = context['foreignDoc'];
394 final root = foreignDoc['documentElement'];
395 expect(root is JsObject, isTrue);
396 final element = root['firstChild'];
397 expect(element is JsObject, isTrue);
398 expect(element.callMethod('getAttribute', ['id']), equals('abc'));
399 });
400
401 test('return a JS proxy to JavaScript', () {
402 var result = context.callMethod('testJsMap', [
403 new Callback.once(() => jsify({ 'value': 42 }))]);
404 expect(result, 42);
405 });
406
407 test('dispose a callback', () {
408 var x = 0;
409 final callback = new Callback.many(() => x++);
410 context['callback'] = callback;
411 expect(context.callMethod('invokeCallback'), equals(0));
412 expect(context.callMethod('invokeCallback'), equals(1));
413 callback.dispose();
414 expect(() => context.callMethod('invokeCallback'), throws);
415 });
416
417 test('test proxy equality', () {
418 var foo1 = new JsObject(context['Foo'], [1]);
419 var foo2 = new JsObject(context['Foo'], [2]);
420 context['foo'] = foo1;
421 context['foo'] = foo2;
422 expect(foo1, isNot(equals(context['foo'])));
423 expect(foo2, equals(context['foo']));
424 });
425
426 test('test instanceof', () {
427 var foo = new JsObject(context['Foo'], [1]);
428 expect(foo.instanceof(context['Foo']), isTrue);
429 expect(foo.instanceof(context['Object']), isTrue);
430 expect(foo.instanceof(context['String']), isFalse);
431 });
432
433 test('test deleteProperty', () {
434 var object = jsify({});
435 object['a'] = 1;
436 expect(context['Object'].callMethod('keys', [object])['length'], 1);
437 expect(context['Object'].callMethod('keys', [object])[0], "a");
438 object.deleteProperty("a");
439 expect(context['Object'].callMethod('keys', [object])['length'], 0);
440 });
441
442 test('test hasProperty', () {
443 var object = jsify({});
444 object['a'] = 1;
445 expect(object.hasProperty('a'), isTrue);
446 expect(object.hasProperty('b'), isFalse);
447 });
448
449 test('test index get and set', () {
450 final myArray = context['myArray'];
451 expect(myArray['length'], equals(1));
452 expect(myArray[0], equals("value1"));
453 myArray[0] = "value2";
454 expect(myArray['length'], equals(1));
455 expect(myArray[0], equals("value2"));
456
457 final foo = new JsObject(context['Foo'], [1]);
458 foo["getAge"] = new Callback.once(() => 10);
459 expect(foo.callMethod('getAge'), equals(10));
460 });
461
462 test('test experimental apis', () {
463 var depth = $experimentalEnterScope();
464 expect(context['x'], equals(42));
465 $experimentalExitScope(depth);
466 });
467
468 test('access a property of a function', () {
469 expect(context.callMethod('Bar'), "ret_value");
470 expect(context['Bar']['foo'], "property_value");
471 });
472
473 test('retrieve same dart Object', () {
474 final date = new DateTime.now();
475 context['dartDate'] = date;
476 expect(context['dartDate'], equals(date));
477 });
478
479 test('usage of Serializable', () {
480 final red = Color.RED;
481 context['color'] = red;
482 expect(context['color'], equals(red._value));
483 });
484
485 test('check for leaks', () {
486 // Verify that the number of live objects is zero.
487 final verifyNoLeaks = expectAsync0(() => expect(0, proxyCount()));
488 // Run this check asychnronously to ensure that any current scope is
489 // cleared first.
490 runAsync(verifyNoLeaks);
491 });
492 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698