OLD | NEW |
(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 } |
OLD | NEW |