dart-sdk/client/dart.js
2012-08-26 21:55:31 +00:00

412 lines
13 KiB
JavaScript

// 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.
// Bootstrap support for Dart scripts on the page as this script.
if (navigator.webkitStartDart) {
if (!navigator.webkitStartDart()) {
document.body.innerHTML = 'This build has expired. Please download a new Dartium at http://www.dartlang.org/dartium/index.html';
}
} else {
// TODO:
// - Support in-browser compilation.
// - Handle inline Dart scripts.
window.addEventListener("DOMContentLoaded", function (e) {
// Fall back to compiled JS. Run through all the scripts and
// replace them if they have a type that indicate that they source
// in Dart code.
//
// <script type="application/dart" src="..."></script>
//
var scripts = document.getElementsByTagName("script");
var length = scripts.length;
for (var i = 0; i < length; ++i) {
if (scripts[i].type == "application/dart") {
// Remap foo.dart to foo.dart.js.
if (scripts[i].src && scripts[i].src != '') {
var script = document.createElement('script');
script.src = scripts[i].src + '.js';
var parent = scripts[i].parentNode;
parent.replaceChild(script, scripts[i]);
}
}
}
}, false);
}
// ---------------------------------------------------------------------------
// Experimental support for JS interoperability
// ---------------------------------------------------------------------------
function SendPortSync() {
}
function ReceivePortSync() {
this.id = ReceivePortSync.id++;
ReceivePortSync.map[this.id] = this;
}
(function() {
// Serialize:
// - primitives / null: unchanged
// - lists: [ 'list', id, list of recursively serialized elements ]
// - maps: [ 'map', id, map of keys and recursively serialized values ]
// - functions: [ 'funcref', function-proxy-id, function-proxy-send-port ]
// - objects: [ 'objref', object-proxy-id, object-proxy-send-port ]
function serialize(message) {
var visited = [];
function checkedSerialization(obj, serializer) {
// Implementation detail: for now use linear search.
// Another option is expando, but it may prohibit
// VM optimizations (like putting object into slow mode
// on property deletion.)
var id = visited.indexOf(obj);
if (id != -1) return [ 'ref', id ];
var id = visited.length;
visited.push(obj);
return serializer(id);
}
function doSerialize(message) {
if (message == null) {
return null; // Convert undefined to null.
} else if (typeof(message) == 'string' ||
typeof(message) == 'number' ||
typeof(message) == 'boolean') {
return message;
} else if (message instanceof Array) {
return checkedSerialization(message, function(id) {
var values = new Array(message.length);
for (var i = 0; i < message.length; i++) {
values[i] = doSerialize(message[i]);
}
return [ 'list', id, values ];
});
} else if (message instanceof LocalSendPortSync) {
return [ 'sendport', 'nativejs', message.receivePort.id ];
} else if (message instanceof DartSendPortSync) {
return [ 'sendport', 'dart', message.isolateId, message.portId ];
} else if (message instanceof Function) {
return [ 'funcref', functionRefTable.makeRef(message),
doSerialize(functionRefTable.sendPort) ];
} else if (message instanceof HTMLElement) {
var id = elementId(message);
// Verify that the element is connected to the document.
// Otherwise, we will not be able to find it on the other side.
getElement(id);
return [ 'element', id ];
} else if (message instanceof DartProxy) {
return [ 'objref', message._id, doSerialize(message._port) ];
} else if (message.__proto__ != {}.__proto__) {
// TODO(vsm): Is the above portable and what we want?
// Proxy non-map Objects.
return [ 'objref', jsRefTable.makeRef(message),
doSerialize(jsRefTable.sendPort) ];
} else {
return checkedSerialization(message, function(id) {
var keys = Object.getOwnPropertyNames(message);
var values = new Array(keys.length);
for (var i = 0; i < keys.length; i++) {
values[i] = doSerialize(message[keys[i]]);
}
return [ 'map', id, keys, values ];
});
}
}
return doSerialize(message);
}
function deserialize(message) {
return deserializeHelper(message);
}
function deserializeHelper(message) {
if (message == null ||
typeof(message) == 'string' ||
typeof(message) == 'number' ||
typeof(message) == 'boolean') {
return message;
}
switch (message[0]) {
case 'map': return deserializeMap(message);
case 'sendport': return deserializeSendPort(message);
case 'list': return deserializeList(message);
case 'funcref': return deserializeFunction(message);
case 'objref': return deserializeProxy(message);
case 'element': return deserializeElement(message);
default: throw 'unimplemented';
}
}
function deserializeMap(message) {
var result = { };
var id = message[1];
var keys = message[2];
var values = message[3];
for (var i = 0, length = keys.length; i < length; i++) {
var key = deserializeHelper(keys[i]);
var value = deserializeHelper(values[i]);
result[key] = value;
}
return result;
}
function deserializeSendPort(message) {
var tag = message[1];
switch (tag) {
case 'nativejs':
var id = message[2];
return new LocalSendPortSync(ReceivePortSync.map[id]);
case 'dart':
var isolateId = message[2];
var portId = message[3];
return new DartSendPortSync(isolateId, portId);
default:
throw 'Illegal SendPortSync type: $tag';
}
}
function deserializeList(message) {
var values = message[2];
var length = values.length;
var result = new Array(length);
for (var i = 0; i < length; i++) {
result[i] = deserializeHelper(values[i]);
}
return result;
}
function deserializeFunction(message) {
var ref = message[1];
var sendPort = deserializeSendPort(message[2]);
// Number of arguments is not used as of now
// we cannot find it out for Dart function in pure Dart.
return _makeFunctionFromRef(ref, sendPort);
}
function deserializeProxy(message) {
var id = message[1];
var port = deserializeSendPort(message[2]);
if (port instanceof LocalSendPortSync) {
return jsRefTable.map[id];
} else if (port instanceof DartSendPortSync) {
return new DartProxy(port, id);
}
throw 'Illegal proxy object: ' + message;
}
function deserializeElement(message) {
var id = message[1];
return getElement(id);
}
window.registerPort = function(name, port) {
var stringified = JSON.stringify(serialize(port));
window.localStorage['dart-port:' + name] = stringified;
};
window.lookupPort = function(name) {
var stringified = window.localStorage['dart-port:' + name];
return deserialize(JSON.parse(stringified));
};
ReceivePortSync.id = 0;
ReceivePortSync.map = {};
ReceivePortSync.dispatchCall = function(id, message) {
// TODO(vsm): Handle and propagate exceptions.
var deserialized = deserialize(message);
var result = ReceivePortSync.map[id].callback(deserialized);
return serialize(result);
};
ReceivePortSync.prototype.receive = function(callback) {
this.callback = callback;
};
ReceivePortSync.prototype.toSendPort = function() {
return new LocalSendPortSync(this);
};
ReceivePortSync.prototype.close = function() {
delete ReceivePortSync.map[this.id];
};
if (navigator.webkitStartDart) {
window.addEventListener('js-sync-message', function(event) {
var data = JSON.parse(event.data);
var deserialized = deserialize(data.message);
var result = ReceivePortSync.map[data.id].callback(deserialized);
// TODO(vsm): Handle and propagate exceptions.
dispatchEvent('js-result', serialize(result));
}, false);
}
function LocalSendPortSync(receivePort) {
this.receivePort = receivePort;
}
LocalSendPortSync.prototype = new SendPortSync();
LocalSendPortSync.prototype.callSync = function(message) {
// TODO(vsm): Do a direct deepcopy.
message = deserialize(serialize(message));
return this.receivePort.callback(message);
}
function DartSendPortSync(isolateId, portId) {
this.isolateId = isolateId;
this.portId = portId;
}
DartSendPortSync.prototype = new SendPortSync();
function dispatchEvent(receiver, message) {
var string = JSON.stringify(message);
var event = document.createEvent('TextEvent');
event.initTextEvent(receiver, false, false, window, string);
window.dispatchEvent(event);
}
DartSendPortSync.prototype.callSync = function(message) {
var serialized = serialize(message);
var target = 'dart-port-' + this.isolateId + '-' + this.portId;
// TODO(vsm): Make this re-entrant.
// TODO(vsm): Set this up set once, on the first call.
var source = target + '-result';
var result = null;
var listener = function (e) {
result = JSON.parse(e.data);
};
window.addEventListener(source, listener, false);
dispatchEvent(target, [source, serialized]);
window.removeEventListener(source, listener, false);
return deserialize(result);
}
// Proxy support
function RefTable(name) {
// TODO(vsm): Fix leaks, particularly in dart2js case.
this.name = name;
this.map = {};
this.id = 0;
this.initialized = false;
this.port = new ReceivePortSync();
this.sendPort = this.port.toSendPort();
}
RefTable.prototype.nextId = function () { return this.id++; }
RefTable.prototype.makeRef = function (obj) {
this.initializeOnce();
// TODO(vsm): Cache refs for each obj.
var ref = this.name + '-' + this.nextId();
this.map[ref] = obj;
return ref;
}
RefTable.prototype.initializeOnce = function () {
if (!this.initialized) {
this.initialize();
}
this.initialized = true;
}
// Overridable initialization on first use hook.
RefTable.prototype.initialize = function () {}
RefTable.prototype.get = function (ref) {
return this.map[ref];
}
function FunctionRefTable() {}
FunctionRefTable.prototype = new RefTable('func-ref');
FunctionRefTable.prototype.initialize = function () {
map = this.map;
this.port.receive(function (message) {
var id = message[0];
var args = message[1];
var f = map[id];
// TODO(vsm): Should we capture this automatically?
return f.apply(null, args);
});
}
var functionRefTable = new FunctionRefTable();
function JSRefTable() {}
JSRefTable.prototype = new RefTable('js-ref');
JSRefTable.prototype.initialize = function () {
map = this.map;
this.port.receive(function (message) {
// TODO(vsm): Support a mechanism to register a handler here.
var receiver = map[message[0]];
var method = message[1];
var args = message[2];
if (method.indexOf("get:") == 0) {
// Getter.
var field = method.substring(4);
if (field in receiver && args.length == 0) {
return [ 'return', receiver[field] ];
}
} else if (method.indexOf("set:") == 0) {
// Setter.
var field = method.substring(4);
if (field in receiver && args.length == 1) {
return [ 'return', receiver[field] = args[0] ];
}
} else {
var f = receiver[method];
if (f) {
try {
var result = f.apply(receiver, args);
return [ 'return', result ];
} catch (e) {
return [ 'exception', e ];
}
}
}
return [ 'none' ];
});
}
var jsRefTable = new JSRefTable();
function DartProxy(port, id) {
this._port = port;
this._id = id;
}
// Leaking implementation.
// TODO(vsm): provide proper, backend-specific implementation.
function _makeFunctionFromRef(ref, sendPort) {
return function() {
return sendPort.callSync([ref, Array.prototype.slice.call(arguments)]);
}
}
var localNextElementId = 0;
var _DART_ID = 'data-dart_id';
function elementId(e) {
if (e.hasAttribute(_DART_ID)) return e.getAttribute(_DART_ID);
var id = (localNextElementId++).toString();
e.setAttribute(_DART_ID, id);
return id;
}
function getElement(id) {
var list = document.querySelectorAll('[' + _DART_ID + '="' + id + '"]');
if (list.length > 1) throw 'Non unique ID: ' + id;
if (list.length == 0) {
throw 'Element must be attached to the document: ' + id;
}
return list[0];
}
})();