Index: tests/html/js_tests.dart |
diff --git a/tests/html/js_tests.dart b/tests/html/js_tests.dart |
new file mode 100644 |
index 0000000000000000000000000000000000000000..1447bc0a7a86cb94408b3b1a096546cdb0e105d5 |
--- /dev/null |
+++ b/tests/html/js_tests.dart |
@@ -0,0 +1,492 @@ |
+// Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
+// for details. All rights reserved. Use of this source code is governed by a |
+// BSD-style license that can be found in the LICENSE file. |
+ |
+library js_tests; |
+ |
+import 'dart:async'; |
+import 'dart:html'; |
+import 'dart:js'; |
+ |
+import '../../pkg/unittest/lib/unittest.dart'; |
+import '../../pkg/unittest/lib/html_config.dart'; |
+ |
+class Foo implements Serializable<JsObject> { |
+ final JsObject _proxy; |
+ |
+ Foo(num a) : this._proxy = new JsObject(context['Foo'], [a]); |
+ |
+ JsObject toJs() => _proxy; |
+ |
+ num get a => _proxy['a']; |
+ num bar() => _proxy.callMethod('bar'); |
+} |
+ |
+class Color implements Serializable<String> { |
+ static final RED = new Color._("red"); |
+ static final BLUE = new Color._("blue"); |
+ String _value; |
+ Color._(this._value); |
+ String toJs() => this._value; |
+} |
+ |
+main() { |
+ useHtmlConfiguration(); |
+ |
+ test('test scope', () { |
+ var ctx; |
+ scoped(() { |
+ ctx = context; |
+ }); |
+ scoped(() { |
+ expect(() => ctx['x'], throws); |
+ }); |
+ }); |
+ |
+ test('read global field', () { |
+ expect(context['x'], equals(42)); |
+ expect(context['y'], isNull); |
+ }); |
+ |
+ test('read global field with underscore', () { |
+ expect(context['_x'], equals(123)); |
+ expect(context['y'], isNull); |
+ }); |
+ |
+ test('js instantiation : new Foo()', () { |
+ final Foo2 = context['container']['Foo']; |
+ final foo = new JsObject(Foo2, [42]); |
+ expect(foo['a'], 42); |
+ expect(Foo2['b'], 38); |
+ }); |
+ |
+ test('js instantiation : new Array()', () { |
+ final a = new JsObject(context['Array']); |
+ expect(a, isNotNull); |
+ expect(a['length'], equals(0)); |
+ |
+ a.callMethod('push', ["value 1"]); |
+ expect(a['length'], equals(1)); |
+ expect(a[0], equals("value 1")); |
+ |
+ a.callMethod('pop'); |
+ expect(a['length'], equals(0)); |
+ }); |
+ |
+ test('js instantiation : new Date()', () { |
+ final a = new JsObject(context['Date']); |
+ expect(a.callMethod('getTime'), isNotNull); |
+ }); |
+ |
+ test('js instantiation : new Date(12345678)', () { |
+ final a = new JsObject(context['Date'], [12345678]); |
+ expect(a.callMethod('getTime'), equals(12345678)); |
+ }); |
+ |
+ test('js instantiation : new Date("December 17, 1995 03:24:00 GMT+01:00")', |
+ () { |
+ final a = new JsObject(context['Date'], |
+ ["December 17, 1995 03:24:00 GMT+01:00"]); |
+ expect(a.callMethod('getTime'), equals(819167040000)); |
+ }); |
+ |
+ test('js instantiation : new Date(1995,11,17)', () { |
+ // Note: JS Date counts months from 0 while Dart counts from 1. |
+ final a = new JsObject(context['Date'], [1995, 11, 17]); |
+ final b = new DateTime(1995, 12, 17); |
+ expect(a.callMethod('getTime'), equals(b.millisecondsSinceEpoch)); |
+ }); |
+ |
+ test('js instantiation : new Date(1995,11,17,3,24,0)', () { |
+ // Note: JS Date counts months from 0 while Dart counts from 1. |
+ final a = new JsObject(context['Date'], |
+ [1995, 11, 17, 3, 24, 0]); |
+ final b = new DateTime(1995, 12, 17, 3, 24, 0); |
+ expect(a.callMethod('getTime'), equals(b.millisecondsSinceEpoch)); |
+ }); |
+ |
+ test('js instantiation : new Object()', () { |
+ final a = new JsObject(context['Object']); |
+ expect(a, isNotNull); |
+ |
+ a['attr'] = "value"; |
+ expect(a['attr'], equals("value")); |
+ }); |
+ |
+ test(r'js instantiation : new RegExp("^\w+$")', () { |
+ final a = new JsObject(context['RegExp'], [r'^\w+$']); |
+ expect(a, isNotNull); |
+ expect(a.callMethod('test', ['true']), isTrue); |
+ expect(a.callMethod('test', [' false']), isFalse); |
+ }); |
+ |
+ test('js instantiation via map notation : new Array()', () { |
+ final a = new JsObject(context['Array']); |
+ expect(a, isNotNull); |
+ expect(a['length'], equals(0)); |
+ |
+ a['push'].apply(a, ["value 1"]); |
+ expect(a['length'], equals(1)); |
+ expect(a[0], equals("value 1")); |
+ |
+ a['pop'].apply(a); |
+ expect(a['length'], equals(0)); |
+ }); |
+ |
+ test('js instantiation via map notation : new Date()', () { |
+ final a = new JsObject(context['Date']); |
+ expect(a['getTime'].apply(a), isNotNull); |
+ }); |
+ |
+ test('js instantiation : typed array', () { |
+ final codeUnits = "test".codeUnits; |
+ final buf = new JsObject(context['ArrayBuffer'], [codeUnits.length]); |
+ final bufView = new JsObject(context['Uint8Array'], [buf]); |
+ for (var i = 0; i < codeUnits.length; i++) { |
+ bufView[i] = codeUnits[i]; |
+ } |
+ }); |
+ |
+ test('write global field', () { |
+ context['y'] = 42; |
+ expect(context['y'], equals(42)); |
+ }); |
+ |
+ test('get JS JsFunction', () { |
+ var razzle = context['razzle']; |
+ expect(razzle.apply(context), equals(42)); |
+ }); |
+ |
+ test('call JS function', () { |
+ expect(context.callMethod('razzle'), equals(42)); |
+ expect(() => context.callMethod('dazzle'), throwsA(isNoSuchMethodError)); |
+ }); |
+ |
+ test('call JS function via map notation', () { |
+ expect(context['razzle'].apply(context), equals(42)); |
+ expect(() => context['dazzle'].apply(context), throwsA(isNoSuchMethodError)); |
+ }); |
+ |
+ test('call JS function with varargs', () { |
+ expect(context.callMethod('varArgs', [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]), |
+ equals(55)); |
+ }); |
+ |
+ test('allocate JS object', () { |
+ var foo = new JsObject(context['Foo'], [42]); |
+ expect(foo['a'], equals(42)); |
+ expect(foo.callMethod('bar'), equals(42)); |
+ expect(() => foo.callMethod('baz'), throwsA(isNoSuchMethodError)); |
+ }); |
+ |
+ test('call toString()', () { |
+ var foo = new JsObject(context['Foo'], [42]); |
+ expect(foo.toString(), equals("I'm a Foo a=42")); |
+ var container = context['container']; |
+ expect(container.toString(), equals("[object Object]")); |
+ }); |
+ |
+ test('allocate simple JS array', () { |
+ final list = [1, 2, 3, 4, 5, 6, 7, 8]; |
+ var array = jsify(list); |
+ expect(context.callMethod('isArray', [array]), isTrue); |
+ expect(array['length'], equals(list.length)); |
+ for (var i = 0; i < list.length ; i++) { |
+ expect(array[i], equals(list[i])); |
+ } |
+ }); |
+ |
+ test('allocate JS array with iterable', () { |
+ final set = new Set.from([1, 2, 3, 4, 5, 6, 7, 8]); |
+ var array = jsify(set); |
+ expect(context.callMethod('isArray', [array]), isTrue); |
+ expect(array['length'], equals(set.length)); |
+ for (var i = 0; i < array['length'] ; i++) { |
+ expect(set.contains(array[i]), isTrue); |
+ } |
+ }); |
+ |
+ test('allocate simple JS map', () { |
+ var map = {'a': 1, 'b': 2, 'c': 3}; |
+ var jsMap = jsify(map); |
+ expect(!context.callMethod('isArray', [jsMap]), isTrue); |
+ for (final key in map.keys) { |
+ expect(context.callMethod('checkMap', [jsMap, key, map[key]]), isTrue); |
+ } |
+ }); |
+ |
+ test('allocate complex JS object', () { |
+ final object = |
+ { |
+ 'a': [1, [2, 3]], |
+ 'b': { |
+ 'c': 3, |
+ 'd': new JsObject(context['Foo'], [42]) |
+ }, |
+ 'e': null |
+ }; |
+ var jsObject = jsify(object); |
+ expect(jsObject['a'][0], equals(object['a'][0])); |
+ expect(jsObject['a'][1][0], equals(object['a'][1][0])); |
+ expect(jsObject['a'][1][1], equals(object['a'][1][1])); |
+ expect(jsObject['b']['c'], equals(object['b']['c'])); |
+ expect(jsObject['b']['d'], equals(object['b']['d'])); |
+ expect(jsObject['b']['d'].callMethod('bar'), equals(42)); |
+ expect(jsObject['e'], isNull); |
+ }); |
+ |
+ test('invoke Dart callback from JS', () { |
+ expect(() => context.callMethod('invokeCallback'), throws); |
+ |
+ context['callback'] = new Callback.once(() => 42); |
+ expect(context.callMethod('invokeCallback'), equals(42)); |
+ expect(() => context.callMethod('invokeCallback'), throws); |
+ }); |
+ |
+ test('callback as parameter', () { |
+ expect(context.callMethod('getTypeOf', [context['razzle']]), equals("function")); |
+ }); |
+ |
+ test('invoke Dart callback from JS with this', () { |
+ final constructor = new Callback.once(($this, arg1) { |
+ $this['a'] = 42; |
+ $this['b'] = jsify(["a", arg1]); |
+ }, withThis: true); |
+ var o = new JsObject(constructor, ["b"]); |
+ expect(o['a'], equals(42)); |
+ expect(o['b'][0], equals("a")); |
+ expect(o['b'][1], equals("b")); |
+ }); |
+ |
+ test('invoke Dart callback from JS with 11 parameters', () { |
+ context['callbackWith11params'] = new Callback.once((p1, p2, p3, p4, |
+ p5, p6, p7, p8, p9, p10, p11) => '$p1$p2$p3$p4$p5$p6$p7$p8$p9$p10' |
+ '$p11'); |
+ expect(context.callMethod('invokeCallbackWith11params'), equals('1234567891011')); |
+ }); |
+ |
+ test('create a Dart callback outside a scope', () { |
+ // Note, the test framework does not guarantee that each test runs as a |
+ // separate event. This test creates a new asynchronous event and |
+ // ensures that a callback can be created without a scope (i.e., that the |
+ // scope is created on demand). |
+ final subtest = () { |
+ var callback = new Callback.once(() => 42); |
+ context['callback'] = callback; |
+ expect(context.callMethod('invokeCallback'), equals(42)); |
+ }; |
+ |
+ runAsync(expectAsync0(subtest)); |
+ }); |
+ |
+ test('global scope', () { |
+ var x; |
+ var y; |
+ scoped(() { |
+ x = new JsObject(context['Foo'], [42]); |
+ y = new JsObject(context['Foo'], [38]); |
+ expect(x['a'], equals(42)); |
+ expect(y['a'], equals(38)); |
+ retain(y); |
+ }); |
+ scoped(() { |
+ expect(() => x['a'], throws); |
+ expect(y['a'], equals(38)); |
+ release(y); |
+ expect(() => y['a'], throws); |
+ }); |
+ }); |
+ |
+ test('global scope for Serializable', () { |
+ Foo x; |
+ Foo y; |
+ scoped(() { |
+ x = new Foo(42); |
+ y = new Foo(38); |
+ expect(x.a, equals(42)); |
+ expect(y.a, equals(38)); |
+ retain(y); |
+ }); |
+ scoped(() { |
+ expect(() => x.a, throws); |
+ expect(y.a, equals(38)); |
+ release(y); |
+ expect(() => y.a, throws); |
+ }); |
+ }); |
+ |
+ test('pass unattached Dom Element', () { |
+ final div = new DivElement(); |
+ div.classes.add('a'); |
+ expect(context.callMethod('getElementAttribute',[div, 'class']), equals('a')); |
+ }); |
+ |
+ test('pass unattached Dom Element two times on same call', () { |
+ final div = new DivElement(); |
+ div.classes.add('a'); |
+ expect(context.callMethod('addClassAttributes', [jsify([div, div])]), equals('aa')); |
+ }); |
+ |
+ test('pass Dom Element attached to an unattached element', () { |
+ final div = new DivElement(); |
+ div.classes.add('a'); |
+ final container = new DivElement(); |
+ container.children.add(div); |
+ expect(context.callMethod('getElementAttribute', [div, 'class']), equals('a')); |
+ }); |
+ |
+ test('pass 2 Dom Elements attached to an unattached element', () { |
+ final div1 = new DivElement(); |
+ div1.classes.add('a'); |
+ final div2 = new DivElement(); |
+ div2.classes.add('b'); |
+ final container = new DivElement(); |
+ container.children.add(div1); |
+ container.children.add(div2); |
+ final f = context['addClassAttributes']; |
+ expect(f.apply(context, [jsify([div1, div2])]), equals('ab')); |
+ }); |
+ |
+ test('pass multiple Dom Elements unattached to document', () { |
+ // A is alone |
+ // 1 and 3 are brother |
+ // 2 is child of 3 |
+ final divA = new DivElement()..classes.add('A'); |
+ final div1 = new DivElement()..classes.add('1'); |
+ final div2 = new DivElement()..classes.add('2'); |
+ final div3 = new DivElement()..classes.add('3')..children.add(div2); |
+ final container = new DivElement()..children.addAll([div1, div3]); |
+ final f = context['addClassAttributes']; |
+ expect(f.apply(context, [jsify([divA, div1, div2, div3])]), equals('A123')); |
+ expect(f.apply(context, [jsify([divA, div1, div3, div2])]), equals('A132')); |
+ expect(f.apply(context, [jsify([divA, div1, div1, div3, divA, div2, div3])]), |
+ equals('A113A23')); |
+ expect(!document.documentElement.contains(divA), isTrue); |
+ expect(!document.documentElement.contains(div1), isTrue); |
+ expect(!document.documentElement.contains(div2), isTrue); |
+ expect(!document.documentElement.contains(div3), isTrue); |
+ expect(!document.documentElement.contains(container), isTrue); |
+ }); |
+ |
+ test('pass one Dom Elements unattached and another attached', () { |
+ final div1 = new DivElement()..classes.add('1'); |
+ final div2 = new DivElement()..classes.add('2'); |
+ document.documentElement.children.add(div2); |
+ final f = context['addClassAttributes']; |
+ expect(f.apply(context, [jsify([div1, div2])]), equals('12')); |
+ expect(!document.documentElement.contains(div1), isTrue); |
+ expect(document.documentElement.contains(div2), isTrue); |
+ }); |
+ |
+ test('pass documentElement', () { |
+ expect(context.callMethod('returnElement', [document.documentElement]), |
+ equals(document.documentElement)); |
+ }); |
+ |
+ test('retrieve unattached Dom Element', () { |
+ var result = context.callMethod('getNewDivElement'); |
+ expect(result is DivElement, isTrue); |
+ expect(!document.documentElement.contains(result), isTrue); |
+ }); |
+ |
+ test('element of foreign document should not be serialized', () { |
+ final foreignDoc = context['foreignDoc']; |
+ final root = foreignDoc['documentElement']; |
+ expect(root is JsObject, isTrue); |
+ final element = root['firstChild']; |
+ expect(element is JsObject, isTrue); |
+ expect(element.callMethod('getAttribute', ['id']), equals('abc')); |
+ }); |
+ |
+ test('return a JS proxy to JavaScript', () { |
+ var result = context.callMethod('testJsMap', [ |
+ new Callback.once(() => jsify({ 'value': 42 }))]); |
+ expect(result, 42); |
+ }); |
+ |
+ test('dispose a callback', () { |
+ var x = 0; |
+ final callback = new Callback.many(() => x++); |
+ context['callback'] = callback; |
+ expect(context.callMethod('invokeCallback'), equals(0)); |
+ expect(context.callMethod('invokeCallback'), equals(1)); |
+ callback.dispose(); |
+ expect(() => context.callMethod('invokeCallback'), throws); |
+ }); |
+ |
+ test('test proxy equality', () { |
+ var foo1 = new JsObject(context['Foo'], [1]); |
+ var foo2 = new JsObject(context['Foo'], [2]); |
+ context['foo'] = foo1; |
+ context['foo'] = foo2; |
+ expect(foo1, isNot(equals(context['foo']))); |
+ expect(foo2, equals(context['foo'])); |
+ }); |
+ |
+ test('test instanceof', () { |
+ var foo = new JsObject(context['Foo'], [1]); |
+ expect(foo.instanceof(context['Foo']), isTrue); |
+ expect(foo.instanceof(context['Object']), isTrue); |
+ expect(foo.instanceof(context['String']), isFalse); |
+ }); |
+ |
+ test('test deleteProperty', () { |
+ var object = jsify({}); |
+ object['a'] = 1; |
+ expect(context['Object'].callMethod('keys', [object])['length'], 1); |
+ expect(context['Object'].callMethod('keys', [object])[0], "a"); |
+ object.deleteProperty("a"); |
+ expect(context['Object'].callMethod('keys', [object])['length'], 0); |
+ }); |
+ |
+ test('test hasProperty', () { |
+ var object = jsify({}); |
+ object['a'] = 1; |
+ expect(object.hasProperty('a'), isTrue); |
+ expect(object.hasProperty('b'), isFalse); |
+ }); |
+ |
+ test('test index get and set', () { |
+ final myArray = context['myArray']; |
+ expect(myArray['length'], equals(1)); |
+ expect(myArray[0], equals("value1")); |
+ myArray[0] = "value2"; |
+ expect(myArray['length'], equals(1)); |
+ expect(myArray[0], equals("value2")); |
+ |
+ final foo = new JsObject(context['Foo'], [1]); |
+ foo["getAge"] = new Callback.once(() => 10); |
+ expect(foo.callMethod('getAge'), equals(10)); |
+ }); |
+ |
+ test('test experimental apis', () { |
+ var depth = $experimentalEnterScope(); |
+ expect(context['x'], equals(42)); |
+ $experimentalExitScope(depth); |
+ }); |
+ |
+ test('access a property of a function', () { |
+ expect(context.callMethod('Bar'), "ret_value"); |
+ expect(context['Bar']['foo'], "property_value"); |
+ }); |
+ |
+ test('retrieve same dart Object', () { |
+ final date = new DateTime.now(); |
+ context['dartDate'] = date; |
+ expect(context['dartDate'], equals(date)); |
+ }); |
+ |
+ test('usage of Serializable', () { |
+ final red = Color.RED; |
+ context['color'] = red; |
+ expect(context['color'], equals(red._value)); |
+ }); |
+ |
+ test('check for leaks', () { |
+ // Verify that the number of live objects is zero. |
+ final verifyNoLeaks = expectAsync0(() => expect(0, proxyCount())); |
+ // Run this check asychnronously to ensure that any current scope is |
+ // cleared first. |
+ runAsync(verifyNoLeaks); |
+ }); |
+} |