Chromium Code Reviews| OLD | NEW |
|---|---|
| (Empty) | |
| 1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file | |
| 2 // for details. All rights reserved. Use of this source code is governed by a | |
| 3 // BSD-style license that can be found in the LICENSE file. | |
| 4 | |
| 5 library dart.js; | |
| 6 | |
| 7 import 'dart:_foreign_helper' show JS; | |
| 8 import 'dart:_js_helper' show convertDartClosureToJS; | |
| 9 | |
| 10 JsObject get context { | |
| 11 return new JsObject._fromJs(JS('=Object', 'window')); | |
| 12 } | |
| 13 | |
| 14 JsObject jsify(dynamic data) => data == null ? null : new JsObject._json(data); | |
| 15 | |
| 16 class Callback implements Serializable<JsFunction> { | |
| 17 final Function _f; // here to allow capture in closure | |
| 18 final bool _withThis; // here to allow capture in closure | |
| 19 dynamic _jsFunction; | |
| 20 | |
| 21 Callback._(this._f, this._withThis) { | |
| 22 _jsFunction = JS('=Object', r''' | |
| 23 (function(){ | |
| 24 var f = #; | |
| 25 return function(){ | |
| 26 return f(this, Array.prototype.slice.apply(arguments)); | |
| 27 }; | |
| 28 }).apply(this)''', convertDartClosureToJS(_call, 2)); | |
| 29 } | |
| 30 | |
| 31 factory Callback(Function f) => new Callback._(f, false); | |
| 32 factory Callback.withThis(Function f) => new Callback._(f, true); | |
| 33 | |
| 34 _call(thisArg, List args) { | |
| 35 final arguments = new List.from(args); | |
| 36 if (_withThis) arguments.insert(0, thisArg); | |
| 37 final dartArgs = arguments.map(_convertToDart).toList(); | |
| 38 return _convertToJS(Function.apply(_f, dartArgs)); | |
| 39 } | |
| 40 | |
| 41 JsFunction toJs() => new JsFunction._fromJs(_jsFunction); | |
| 42 } | |
| 43 | |
| 44 class JsObject implements Serializable<JsObject> { | |
| 45 final dynamic _jsObject; | |
| 46 | |
| 47 JsObject._fromJs(this._jsObject); | |
| 48 | |
| 49 factory JsObject(Serializable<JsFunction> constructor, [List arguments]) { | |
| 50 final constr = _convertToJS(constructor); | |
| 51 if (arguments == null) { | |
| 52 return new JsObject._fromJs(JS('=Object', 'new #()', constr)); | |
| 53 } | |
| 54 final args = arguments.map(_convertToJS).toList(); | |
| 55 switch (args.length) { | |
| 56 case 0: | |
| 57 return new JsObject._fromJs(JS('=Object', 'new #()', constr)); | |
| 58 case 1: | |
| 59 return new JsObject._fromJs(JS('=Object', 'new #(#)', constr, args[0])); | |
| 60 case 2: | |
| 61 return new JsObject._fromJs(JS('=Object', 'new #(#,#)', constr, args[0], | |
| 62 args[1])); | |
| 63 case 3: | |
| 64 return new JsObject._fromJs(JS('=Object', 'new #(#,#,#)', constr, | |
| 65 args[0], args[1], args[2])); | |
| 66 case 4: | |
| 67 return new JsObject._fromJs(JS('=Object', 'new #(#,#,#,#)', constr, | |
| 68 args[0], args[1], args[2], args[3])); | |
| 69 case 5: | |
| 70 return new JsObject._fromJs(JS('=Object', 'new #(#,#,#,#,#)', constr, | |
| 71 args[0], args[1], args[2], args[3], args[4])); | |
| 72 case 6: | |
| 73 return new JsObject._fromJs(JS('=Object', 'new #(#,#,#,#,#,#)', constr, | |
| 74 args[0], args[1], args[2], args[3], args[4], args[5])); | |
| 75 case 7: | |
| 76 return new JsObject._fromJs(JS('=Object', 'new #(#,#,#,#,#,#,#)', | |
| 77 constr, args[0], args[1], args[2], args[3], args[4], args[5], | |
| 78 args[6])); | |
| 79 case 8: | |
| 80 return new JsObject._fromJs(JS('=Object', 'new #(#,#,#,#,#,#,#,#)', | |
| 81 constr, args[0], args[1], args[2], args[3], args[4], args[5], | |
| 82 args[6], args[7])); | |
| 83 case 9: | |
| 84 return new JsObject._fromJs(JS('=Object', 'new #(#,#,#,#,#,#,#,#,#)', | |
| 85 constr, args[0], args[1], args[2], args[3], args[4], args[5], | |
| 86 args[6], args[7], args[8])); | |
| 87 case 10: | |
| 88 return new JsObject._fromJs(JS('=Object', 'new #(#,#,#,#,#,#,#,#,#,#)', | |
| 89 constr, args[0], args[1], args[2], args[3], args[4], args[5], | |
| 90 args[6], args[7], args[8], args[9])); | |
| 91 } | |
| 92 return new JsObject._fromJs(JS('=Object', r'''(function(){ | |
| 93 var Type = function(){}; | |
| 94 Type.prototype = #.prototype; | |
| 95 var instance = new Type(); | |
| 96 ret = #.apply(instance, #); | |
| 97 ret = Object(ret) === ret ? ret : instance; | |
| 98 return ret; | |
| 99 })()''', constr, constr, args)); | |
| 100 } | |
| 101 | |
| 102 factory JsObject._json(data) => new JsObject._fromJs(_convertDataTree(data)); | |
| 103 | |
| 104 static _convertDataTree(data) { | |
| 105 if (data is Map) { | |
| 106 final convertedData = JS('=Object', '{}'); | |
| 107 for (var key in data.keys) { | |
| 108 JS('=Object', '#[#]=#', convertedData, key, | |
| 109 _convertDataTree(data[key])); | |
| 110 } | |
| 111 return convertedData; | |
| 112 } else if (data is Iterable) { | |
| 113 return data.map(_convertDataTree).toList(); | |
| 114 } else { | |
| 115 return _convertToJS(data); | |
| 116 } | |
| 117 } | |
| 118 | |
| 119 JsObject toJs() => this; | |
| 120 | |
| 121 operator[](key) => | |
| 122 _convertToDart(JS('=Object', '#[#]', _convertToJS(this), key)); | |
| 123 operator[]=(key, value) => JS('void', '#[#]=#', _convertToJS(this), key, | |
| 124 _convertToJS(value)); | |
| 125 | |
| 126 operator==(other) => identical(this, other) || | |
| 127 (other is JsObject && JS('bool', '# == #', _convertToJS(this), | |
| 128 _convertToJS(other))); | |
| 129 | |
| 130 bool hasProperty(String property) => JS('bool', '# in #', property, | |
| 131 _convertToJS(this)); | |
| 132 | |
| 133 void deleteProperty(String name) { | |
| 134 JS('void', 'delete #[#]', _convertToJS(this), name); | |
| 135 } | |
| 136 | |
| 137 bool instanceof(Serializable<JsFunction> type) => | |
| 138 JS('bool', '# instanceof #', _convertToJS(this), _convertToJS(type)); | |
| 139 | |
| 140 String toString() { | |
| 141 try { | |
| 142 return JS('String', '#.toString()', _convertToJS(this)); | |
| 143 } catch(e) { | |
| 144 return super.toString(); | |
| 145 } | |
| 146 } | |
| 147 | |
| 148 callMethod(String name, [List args]) => | |
| 149 _convertToDart(JS('=Object', '#[#].apply(#, #)', _convertToJS(this), name, | |
| 150 _convertToJS(this), | |
| 151 args == null ? null : args.map(_convertToJS).toList())); | |
| 152 } | |
| 153 | |
| 154 class JsFunction extends JsObject implements Serializable<JsFunction> { | |
| 155 JsFunction._fromJs(jsObject) : super._fromJs(jsObject); | |
| 156 apply(thisArg, [List args]) => | |
| 157 _convertToDart(JS('=Object', '#.apply(#, #)', _convertToJS(this), | |
| 158 _convertToJS(thisArg), | |
| 159 args == null ? null : args.map(_convertToJS).toList())); | |
| 160 } | |
| 161 | |
| 162 abstract class Serializable<T> { | |
| 163 T toJs(); | |
| 164 } | |
| 165 | |
| 166 class _DartProxy { | |
|
vsm
2013/06/20 18:43:58
This might need to be defined in JS, as a JS type.
alexandre.ardhuin
2013/06/27 17:05:36
Done. I have added a DartProxy javascript object.
| |
| 167 dynamic o; | |
| 168 _DartProxy(this.o); | |
| 169 } | |
| 170 | |
| 171 dynamic _convertToJS(dynamic o) { | |
| 172 if (o == null) { | |
| 173 return null; | |
| 174 } else if (o is String || o is num || o is bool) { | |
| 175 return o; | |
| 176 } else if (o is JsObject) { | |
| 177 return o._jsObject; | |
| 178 } else if (o is Serializable) { | |
| 179 return _convertToJS(o.toJs()); | |
| 180 } else if (o is Function) { | |
| 181 return _convertToJS(new Callback(o)); | |
| 182 } else { | |
| 183 return new _DartProxy(o); | |
|
vsm
2013/06/20 18:43:58
It feels dangerous to pass out a direct pointer to
alexandre.ardhuin
2013/06/27 17:05:36
Done.
| |
| 184 } | |
| 185 } | |
| 186 | |
| 187 dynamic _convertToDart(dynamic o) { | |
|
vsm
2013/06/20 18:43:58
I think we need to always access "o" via JS notati
alexandre.ardhuin
2013/06/27 17:05:36
Done.
| |
| 188 if (o == null) { | |
| 189 return null; | |
| 190 } else if (o is num || o is String || o is bool) { | |
| 191 return o; | |
| 192 } else if (JS('bool', '# instanceof Function', o)) { | |
| 193 return new JsFunction._fromJs(JS('=Object', '#', o)); | |
| 194 } else if (o is _DartProxy) { | |
| 195 return o.o; | |
| 196 } else { | |
| 197 return new JsObject._fromJs(JS('=Object', '#', o)); | |
| 198 } | |
| 199 } | |
| OLD | NEW |