3097 lines
281 KiB
JavaScript
3097 lines
281 KiB
JavaScript
(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
|
|
var ConnectorClass, adaptConnector;
|
|
|
|
ConnectorClass = require("./ConnectorClass");
|
|
|
|
adaptConnector = function(connector, engine, HB, execution_listener) {
|
|
var applyHB, encode_state_vector, f, getHB, getStateVector, name, parse_state_vector, send_;
|
|
for (name in ConnectorClass) {
|
|
f = ConnectorClass[name];
|
|
connector[name] = f;
|
|
}
|
|
connector.setIsBoundToY();
|
|
send_ = function(o) {
|
|
if ((o.uid.creator === HB.getUserId()) && (typeof o.uid.op_number !== "string") && (HB.getUserId() !== "_temp")) {
|
|
return connector.broadcast(o);
|
|
}
|
|
};
|
|
if (connector.invokeSync != null) {
|
|
HB.setInvokeSyncHandler(connector.invokeSync);
|
|
}
|
|
execution_listener.push(send_);
|
|
encode_state_vector = function(v) {
|
|
var value, _results;
|
|
_results = [];
|
|
for (name in v) {
|
|
value = v[name];
|
|
_results.push({
|
|
user: name,
|
|
state: value
|
|
});
|
|
}
|
|
return _results;
|
|
};
|
|
parse_state_vector = function(v) {
|
|
var s, state_vector, _i, _len;
|
|
state_vector = {};
|
|
for (_i = 0, _len = v.length; _i < _len; _i++) {
|
|
s = v[_i];
|
|
state_vector[s.user] = s.state;
|
|
}
|
|
return state_vector;
|
|
};
|
|
getStateVector = function() {
|
|
return encode_state_vector(HB.getOperationCounter());
|
|
};
|
|
getHB = function(v) {
|
|
var hb, json, state_vector;
|
|
state_vector = parse_state_vector(v);
|
|
hb = HB._encode(state_vector);
|
|
json = {
|
|
hb: hb,
|
|
state_vector: encode_state_vector(HB.getOperationCounter())
|
|
};
|
|
return json;
|
|
};
|
|
applyHB = function(hb, fromHB) {
|
|
return engine.applyOp(hb, fromHB);
|
|
};
|
|
connector.getStateVector = getStateVector;
|
|
connector.getHB = getHB;
|
|
connector.applyHB = applyHB;
|
|
if (connector.receive_handlers == null) {
|
|
connector.receive_handlers = [];
|
|
}
|
|
return connector.receive_handlers.push(function(sender, op) {
|
|
if (op.uid.creator !== HB.getUserId()) {
|
|
return engine.applyOp(op);
|
|
}
|
|
});
|
|
};
|
|
|
|
module.exports = adaptConnector;
|
|
|
|
|
|
},{"./ConnectorClass":2}],2:[function(require,module,exports){
|
|
module.exports = {
|
|
init: function(options) {
|
|
var req;
|
|
req = (function(_this) {
|
|
return function(name, choices) {
|
|
if (options[name] != null) {
|
|
if ((choices == null) || choices.some(function(c) {
|
|
return c === options[name];
|
|
})) {
|
|
return _this[name] = options[name];
|
|
} else {
|
|
throw new Error("You can set the '" + name + "' option to one of the following choices: " + JSON.encode(choices));
|
|
}
|
|
} else {
|
|
throw new Error("You must specify " + name + ", when initializing the Connector!");
|
|
}
|
|
};
|
|
})(this);
|
|
req("syncMethod", ["syncAll", "master-slave"]);
|
|
req("role", ["master", "slave"]);
|
|
req("user_id");
|
|
if (typeof this.on_user_id_set === "function") {
|
|
this.on_user_id_set(this.user_id);
|
|
}
|
|
if (options.perform_send_again != null) {
|
|
this.perform_send_again = options.perform_send_again;
|
|
} else {
|
|
this.perform_send_again = true;
|
|
}
|
|
if (this.role === "master") {
|
|
this.syncMethod = "syncAll";
|
|
}
|
|
this.is_synced = false;
|
|
this.connections = {};
|
|
if (this.receive_handlers == null) {
|
|
this.receive_handlers = [];
|
|
}
|
|
this.connections = {};
|
|
this.current_sync_target = null;
|
|
this.sent_hb_to_all_users = false;
|
|
return this.is_initialized = true;
|
|
},
|
|
onUserEvent: function(f) {
|
|
if (this.connections_listeners == null) {
|
|
this.connections_listeners = [];
|
|
}
|
|
return this.connections_listeners.push(f);
|
|
},
|
|
isRoleMaster: function() {
|
|
return this.role === "master";
|
|
},
|
|
isRoleSlave: function() {
|
|
return this.role === "slave";
|
|
},
|
|
findNewSyncTarget: function() {
|
|
var c, user, _ref;
|
|
this.current_sync_target = null;
|
|
if (this.syncMethod === "syncAll") {
|
|
_ref = this.connections;
|
|
for (user in _ref) {
|
|
c = _ref[user];
|
|
if (!c.is_synced) {
|
|
this.performSync(user);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (this.current_sync_target == null) {
|
|
this.setStateSynced();
|
|
}
|
|
return null;
|
|
},
|
|
userLeft: function(user) {
|
|
var f, _i, _len, _ref, _results;
|
|
delete this.connections[user];
|
|
this.findNewSyncTarget();
|
|
if (this.connections_listeners != null) {
|
|
_ref = this.connections_listeners;
|
|
_results = [];
|
|
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
|
f = _ref[_i];
|
|
_results.push(f({
|
|
action: "userLeft",
|
|
user: user
|
|
}));
|
|
}
|
|
return _results;
|
|
}
|
|
},
|
|
userJoined: function(user, role) {
|
|
var f, _base, _i, _len, _ref, _results;
|
|
if (role == null) {
|
|
throw new Error("Internal: You must specify the role of the joined user! E.g. userJoined('uid:3939','slave')");
|
|
}
|
|
if ((_base = this.connections)[user] == null) {
|
|
_base[user] = {};
|
|
}
|
|
this.connections[user].is_synced = false;
|
|
if ((!this.is_synced) || this.syncMethod === "syncAll") {
|
|
if (this.syncMethod === "syncAll") {
|
|
this.performSync(user);
|
|
} else if (role === "master") {
|
|
this.performSyncWithMaster(user);
|
|
}
|
|
}
|
|
if (this.connections_listeners != null) {
|
|
_ref = this.connections_listeners;
|
|
_results = [];
|
|
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
|
f = _ref[_i];
|
|
_results.push(f({
|
|
action: "userJoined",
|
|
user: user,
|
|
role: role
|
|
}));
|
|
}
|
|
return _results;
|
|
}
|
|
},
|
|
whenSynced: function(args) {
|
|
if (args.constructor === Function) {
|
|
args = [args];
|
|
}
|
|
if (this.is_synced) {
|
|
return args[0].apply(this, args.slice(1));
|
|
} else {
|
|
if (this.compute_when_synced == null) {
|
|
this.compute_when_synced = [];
|
|
}
|
|
return this.compute_when_synced.push(args);
|
|
}
|
|
},
|
|
onReceive: function(f) {
|
|
return this.receive_handlers.push(f);
|
|
},
|
|
|
|
/*
|
|
* Broadcast a message to all connected peers.
|
|
* @param message {Object} The message to broadcast.
|
|
*
|
|
broadcast: (message)->
|
|
throw new Error "You must implement broadcast!"
|
|
|
|
*
|
|
* Send a message to a peer, or set of peers
|
|
*
|
|
send: (peer_s, message)->
|
|
throw new Error "You must implement send!"
|
|
*/
|
|
performSync: function(user) {
|
|
var hb, o, _hb, _i, _len;
|
|
if (this.current_sync_target == null) {
|
|
this.current_sync_target = user;
|
|
this.send(user, {
|
|
sync_step: "getHB",
|
|
send_again: "true",
|
|
data: this.getStateVector()
|
|
});
|
|
if (!this.sent_hb_to_all_users) {
|
|
this.sent_hb_to_all_users = true;
|
|
hb = this.getHB([]).hb;
|
|
_hb = [];
|
|
for (_i = 0, _len = hb.length; _i < _len; _i++) {
|
|
o = hb[_i];
|
|
_hb.push(o);
|
|
if (_hb.length > 10) {
|
|
this.broadcast({
|
|
sync_step: "applyHB_",
|
|
data: _hb
|
|
});
|
|
_hb = [];
|
|
}
|
|
}
|
|
return this.broadcast({
|
|
sync_step: "applyHB",
|
|
data: _hb
|
|
});
|
|
}
|
|
}
|
|
},
|
|
performSyncWithMaster: function(user) {
|
|
var hb, o, _hb, _i, _len;
|
|
this.current_sync_target = user;
|
|
this.send(user, {
|
|
sync_step: "getHB",
|
|
send_again: "true",
|
|
data: this.getStateVector()
|
|
});
|
|
hb = this.getHB([]).hb;
|
|
_hb = [];
|
|
for (_i = 0, _len = hb.length; _i < _len; _i++) {
|
|
o = hb[_i];
|
|
_hb.push(o);
|
|
if (_hb.length > 10) {
|
|
this.broadcast({
|
|
sync_step: "applyHB_",
|
|
data: _hb
|
|
});
|
|
_hb = [];
|
|
}
|
|
}
|
|
return this.broadcast({
|
|
sync_step: "applyHB",
|
|
data: _hb
|
|
});
|
|
},
|
|
setStateSynced: function() {
|
|
var args, el, f, _i, _len, _ref;
|
|
if (!this.is_synced) {
|
|
this.is_synced = true;
|
|
if (this.compute_when_synced != null) {
|
|
_ref = this.compute_when_synced;
|
|
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
|
el = _ref[_i];
|
|
f = el[0];
|
|
args = el.slice(1);
|
|
f.apply(args);
|
|
}
|
|
delete this.compute_when_synced;
|
|
}
|
|
return null;
|
|
}
|
|
},
|
|
whenReceivedStateVector: function(f) {
|
|
if (this.when_received_state_vector_listeners == null) {
|
|
this.when_received_state_vector_listeners = [];
|
|
}
|
|
return this.when_received_state_vector_listeners.push(f);
|
|
},
|
|
receiveMessage: function(sender, res) {
|
|
var data, f, hb, o, sendApplyHB, send_again, _hb, _i, _j, _k, _len, _len1, _len2, _ref, _ref1, _results;
|
|
if (res.sync_step == null) {
|
|
_ref = this.receive_handlers;
|
|
_results = [];
|
|
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
|
f = _ref[_i];
|
|
_results.push(f(sender, res));
|
|
}
|
|
return _results;
|
|
} else {
|
|
if (sender === this.user_id) {
|
|
return;
|
|
}
|
|
if (res.sync_step === "getHB") {
|
|
if (this.when_received_state_vector_listeners != null) {
|
|
_ref1 = this.when_received_state_vector_listeners;
|
|
for (_j = 0, _len1 = _ref1.length; _j < _len1; _j++) {
|
|
f = _ref1[_j];
|
|
f.call(this, res.data);
|
|
}
|
|
}
|
|
delete this.when_received_state_vector_listeners;
|
|
data = this.getHB(res.data);
|
|
hb = data.hb;
|
|
_hb = [];
|
|
if (this.is_synced) {
|
|
sendApplyHB = (function(_this) {
|
|
return function(m) {
|
|
return _this.send(sender, m);
|
|
};
|
|
})(this);
|
|
} else {
|
|
sendApplyHB = (function(_this) {
|
|
return function(m) {
|
|
return _this.broadcast(m);
|
|
};
|
|
})(this);
|
|
}
|
|
for (_k = 0, _len2 = hb.length; _k < _len2; _k++) {
|
|
o = hb[_k];
|
|
_hb.push(o);
|
|
if (_hb.length > 10) {
|
|
sendApplyHB({
|
|
sync_step: "applyHB_",
|
|
data: _hb
|
|
});
|
|
_hb = [];
|
|
}
|
|
}
|
|
sendApplyHB({
|
|
sync_step: "applyHB",
|
|
data: _hb
|
|
});
|
|
if ((res.send_again != null) && this.perform_send_again) {
|
|
send_again = (function(_this) {
|
|
return function(sv) {
|
|
return function() {
|
|
var _l, _len3;
|
|
hb = _this.getHB(sv).hb;
|
|
for (_l = 0, _len3 = hb.length; _l < _len3; _l++) {
|
|
o = hb[_l];
|
|
_hb.push(o);
|
|
if (_hb.length > 10) {
|
|
_this.send(sender, {
|
|
sync_step: "applyHB_",
|
|
data: _hb
|
|
});
|
|
_hb = [];
|
|
}
|
|
}
|
|
return _this.send(sender, {
|
|
sync_step: "applyHB",
|
|
data: _hb,
|
|
sent_again: "true"
|
|
});
|
|
};
|
|
};
|
|
})(this)(data.state_vector);
|
|
return setTimeout(send_again, 3000);
|
|
}
|
|
} else if (res.sync_step === "applyHB") {
|
|
this.applyHB(res.data, sender === this.current_sync_target);
|
|
if ((this.syncMethod === "syncAll" || (res.sent_again != null)) && (!this.is_synced) && ((this.current_sync_target === sender) || (this.current_sync_target == null))) {
|
|
this.connections[sender].is_synced = true;
|
|
return this.findNewSyncTarget();
|
|
}
|
|
} else if (res.sync_step === "applyHB_") {
|
|
return this.applyHB(res.data, sender === this.current_sync_target);
|
|
}
|
|
}
|
|
},
|
|
parseMessageFromXml: function(m) {
|
|
var parse_array, parse_object;
|
|
parse_array = function(node) {
|
|
var n, _i, _len, _ref, _results;
|
|
_ref = node.children;
|
|
_results = [];
|
|
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
|
n = _ref[_i];
|
|
if (n.getAttribute("isArray") === "true") {
|
|
_results.push(parse_array(n));
|
|
} else {
|
|
_results.push(parse_object(n));
|
|
}
|
|
}
|
|
return _results;
|
|
};
|
|
parse_object = function(node) {
|
|
var int, json, n, name, value, _i, _len, _ref, _ref1;
|
|
json = {};
|
|
_ref = node.attrs;
|
|
for (name in _ref) {
|
|
value = _ref[name];
|
|
int = parseInt(value);
|
|
if (isNaN(int) || ("" + int) !== value) {
|
|
json[name] = value;
|
|
} else {
|
|
json[name] = int;
|
|
}
|
|
}
|
|
_ref1 = node.children;
|
|
for (_i = 0, _len = _ref1.length; _i < _len; _i++) {
|
|
n = _ref1[_i];
|
|
name = n.name;
|
|
if (n.getAttribute("isArray") === "true") {
|
|
json[name] = parse_array(n);
|
|
} else {
|
|
json[name] = parse_object(n);
|
|
}
|
|
}
|
|
return json;
|
|
};
|
|
return parse_object(m);
|
|
},
|
|
encodeMessageToXml: function(m, json) {
|
|
var encode_array, encode_object;
|
|
encode_object = function(m, json) {
|
|
var name, value;
|
|
for (name in json) {
|
|
value = json[name];
|
|
if (value == null) {
|
|
|
|
} else if (value.constructor === Object) {
|
|
encode_object(m.c(name), value);
|
|
} else if (value.constructor === Array) {
|
|
encode_array(m.c(name), value);
|
|
} else {
|
|
m.setAttribute(name, value);
|
|
}
|
|
}
|
|
return m;
|
|
};
|
|
encode_array = function(m, array) {
|
|
var e, _i, _len;
|
|
m.setAttribute("isArray", "true");
|
|
for (_i = 0, _len = array.length; _i < _len; _i++) {
|
|
e = array[_i];
|
|
if (e.constructor === Object) {
|
|
encode_object(m.c("array-element"), e);
|
|
} else {
|
|
encode_array(m.c("array-element"), e);
|
|
}
|
|
}
|
|
return m;
|
|
};
|
|
if (json.constructor === Object) {
|
|
return encode_object(m.c("y", {
|
|
xmlns: "http://y.ninja/connector-stanza"
|
|
}), json);
|
|
} else if (json.constructor === Array) {
|
|
return encode_array(m.c("y", {
|
|
xmlns: "http://y.ninja/connector-stanza"
|
|
}), json);
|
|
} else {
|
|
throw new Error("I can't encode this json!");
|
|
}
|
|
},
|
|
setIsBoundToY: function() {
|
|
if (typeof this.on_bound_to_y === "function") {
|
|
this.on_bound_to_y();
|
|
}
|
|
delete this.when_bound_to_y;
|
|
return this.is_bound_to_y = true;
|
|
}
|
|
};
|
|
|
|
|
|
},{}],3:[function(require,module,exports){
|
|
var Engine;
|
|
|
|
if (typeof window !== "undefined" && window !== null) {
|
|
window.unprocessed_counter = 0;
|
|
}
|
|
|
|
if (typeof window !== "undefined" && window !== null) {
|
|
window.unprocessed_exec_counter = 0;
|
|
}
|
|
|
|
if (typeof window !== "undefined" && window !== null) {
|
|
window.unprocessed_types = [];
|
|
}
|
|
|
|
Engine = (function() {
|
|
function Engine(HB, types) {
|
|
this.HB = HB;
|
|
this.types = types;
|
|
this.unprocessed_ops = [];
|
|
}
|
|
|
|
Engine.prototype.parseOperation = function(json) {
|
|
var type;
|
|
type = this.types[json.type];
|
|
if ((type != null ? type.parse : void 0) != null) {
|
|
return type.parse(json);
|
|
} else {
|
|
throw new Error("You forgot to specify a parser for type " + json.type + ". The message is " + (JSON.stringify(json)) + ".");
|
|
}
|
|
};
|
|
|
|
|
|
/*
|
|
applyOpsBundle: (ops_json)->
|
|
ops = []
|
|
for o in ops_json
|
|
ops.push @parseOperation o
|
|
for o in ops
|
|
if not o.execute()
|
|
@unprocessed_ops.push o
|
|
@tryUnprocessed()
|
|
*/
|
|
|
|
Engine.prototype.applyOpsCheckDouble = function(ops_json) {
|
|
var o, _i, _len, _results;
|
|
_results = [];
|
|
for (_i = 0, _len = ops_json.length; _i < _len; _i++) {
|
|
o = ops_json[_i];
|
|
if (this.HB.getOperation(o.uid) == null) {
|
|
_results.push(this.applyOp(o));
|
|
} else {
|
|
_results.push(void 0);
|
|
}
|
|
}
|
|
return _results;
|
|
};
|
|
|
|
Engine.prototype.applyOps = function(ops_json) {
|
|
return this.applyOp(ops_json);
|
|
};
|
|
|
|
Engine.prototype.applyOp = function(op_json_array, fromHB) {
|
|
var o, op_json, _i, _len;
|
|
if (fromHB == null) {
|
|
fromHB = false;
|
|
}
|
|
if (op_json_array.constructor !== Array) {
|
|
op_json_array = [op_json_array];
|
|
}
|
|
for (_i = 0, _len = op_json_array.length; _i < _len; _i++) {
|
|
op_json = op_json_array[_i];
|
|
if (fromHB) {
|
|
op_json.fromHB = "true";
|
|
}
|
|
o = this.parseOperation(op_json);
|
|
o.parsed_from_json = op_json;
|
|
if (op_json.fromHB != null) {
|
|
o.fromHB = op_json.fromHB;
|
|
}
|
|
if (this.HB.getOperation(o) != null) {
|
|
|
|
} else if (((!this.HB.isExpectedOperation(o)) && (o.fromHB == null)) || (!o.execute())) {
|
|
this.unprocessed_ops.push(o);
|
|
if (typeof window !== "undefined" && window !== null) {
|
|
window.unprocessed_types.push(o.type);
|
|
}
|
|
}
|
|
}
|
|
return this.tryUnprocessed();
|
|
};
|
|
|
|
Engine.prototype.tryUnprocessed = function() {
|
|
var old_length, op, unprocessed, _i, _len, _ref;
|
|
while (true) {
|
|
old_length = this.unprocessed_ops.length;
|
|
unprocessed = [];
|
|
_ref = this.unprocessed_ops;
|
|
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
|
op = _ref[_i];
|
|
if (this.HB.getOperation(op) != null) {
|
|
|
|
} else if ((!this.HB.isExpectedOperation(op) && (op.fromHB == null)) || (!op.execute())) {
|
|
unprocessed.push(op);
|
|
}
|
|
}
|
|
this.unprocessed_ops = unprocessed;
|
|
if (this.unprocessed_ops.length === old_length) {
|
|
break;
|
|
}
|
|
}
|
|
if (this.unprocessed_ops.length !== 0) {
|
|
return this.HB.invokeSync();
|
|
}
|
|
};
|
|
|
|
return Engine;
|
|
|
|
})();
|
|
|
|
module.exports = Engine;
|
|
|
|
|
|
},{}],4:[function(require,module,exports){
|
|
var HistoryBuffer,
|
|
__bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; };
|
|
|
|
HistoryBuffer = (function() {
|
|
function HistoryBuffer(user_id) {
|
|
this.user_id = user_id;
|
|
this.emptyGarbage = __bind(this.emptyGarbage, this);
|
|
this.operation_counter = {};
|
|
this.buffer = {};
|
|
this.change_listeners = [];
|
|
this.garbage = [];
|
|
this.trash = [];
|
|
this.performGarbageCollection = true;
|
|
this.garbageCollectTimeout = 30000;
|
|
this.reserved_identifier_counter = 0;
|
|
setTimeout(this.emptyGarbage, this.garbageCollectTimeout);
|
|
}
|
|
|
|
HistoryBuffer.prototype.setUserId = function(user_id, state_vector) {
|
|
var buff, counter_diff, o, o_name, _base, _name, _ref;
|
|
this.user_id = user_id;
|
|
if ((_base = this.buffer)[_name = this.user_id] == null) {
|
|
_base[_name] = [];
|
|
}
|
|
buff = this.buffer[this.user_id];
|
|
counter_diff = state_vector[this.user_id] || 0;
|
|
if (this.buffer._temp != null) {
|
|
_ref = this.buffer._temp;
|
|
for (o_name in _ref) {
|
|
o = _ref[o_name];
|
|
o.uid.creator = this.user_id;
|
|
o.uid.op_number += counter_diff;
|
|
buff[o.uid.op_number] = o;
|
|
}
|
|
}
|
|
this.operation_counter[this.user_id] = (this.operation_counter._temp || 0) + counter_diff;
|
|
delete this.operation_counter._temp;
|
|
return delete this.buffer._temp;
|
|
};
|
|
|
|
HistoryBuffer.prototype.emptyGarbage = function() {
|
|
var o, _i, _len, _ref;
|
|
_ref = this.garbage;
|
|
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
|
o = _ref[_i];
|
|
if (typeof o.cleanup === "function") {
|
|
o.cleanup();
|
|
}
|
|
}
|
|
this.garbage = this.trash;
|
|
this.trash = [];
|
|
if (this.garbageCollectTimeout !== -1) {
|
|
this.garbageCollectTimeoutId = setTimeout(this.emptyGarbage, this.garbageCollectTimeout);
|
|
}
|
|
return void 0;
|
|
};
|
|
|
|
HistoryBuffer.prototype.getUserId = function() {
|
|
return this.user_id;
|
|
};
|
|
|
|
HistoryBuffer.prototype.addToGarbageCollector = function() {
|
|
var o, _i, _len, _results;
|
|
if (this.performGarbageCollection) {
|
|
_results = [];
|
|
for (_i = 0, _len = arguments.length; _i < _len; _i++) {
|
|
o = arguments[_i];
|
|
if (o != null) {
|
|
_results.push(this.garbage.push(o));
|
|
} else {
|
|
_results.push(void 0);
|
|
}
|
|
}
|
|
return _results;
|
|
}
|
|
};
|
|
|
|
HistoryBuffer.prototype.stopGarbageCollection = function() {
|
|
this.performGarbageCollection = false;
|
|
this.setManualGarbageCollect();
|
|
this.garbage = [];
|
|
return this.trash = [];
|
|
};
|
|
|
|
HistoryBuffer.prototype.setManualGarbageCollect = function() {
|
|
this.garbageCollectTimeout = -1;
|
|
clearTimeout(this.garbageCollectTimeoutId);
|
|
return this.garbageCollectTimeoutId = void 0;
|
|
};
|
|
|
|
HistoryBuffer.prototype.setGarbageCollectTimeout = function(garbageCollectTimeout) {
|
|
this.garbageCollectTimeout = garbageCollectTimeout;
|
|
};
|
|
|
|
HistoryBuffer.prototype.getReservedUniqueIdentifier = function() {
|
|
return {
|
|
creator: '_',
|
|
op_number: "_" + (this.reserved_identifier_counter++)
|
|
};
|
|
};
|
|
|
|
HistoryBuffer.prototype.getOperationCounter = function(user_id) {
|
|
var ctn, res, user, _ref;
|
|
if (user_id == null) {
|
|
res = {};
|
|
_ref = this.operation_counter;
|
|
for (user in _ref) {
|
|
ctn = _ref[user];
|
|
res[user] = ctn;
|
|
}
|
|
return res;
|
|
} else {
|
|
return this.operation_counter[user_id];
|
|
}
|
|
};
|
|
|
|
HistoryBuffer.prototype.isExpectedOperation = function(o) {
|
|
var _base, _name;
|
|
if ((_base = this.operation_counter)[_name = o.uid.creator] == null) {
|
|
_base[_name] = 0;
|
|
}
|
|
o.uid.op_number <= this.operation_counter[o.uid.creator];
|
|
return true;
|
|
};
|
|
|
|
HistoryBuffer.prototype._encode = function(state_vector) {
|
|
var json, o, o_json, o_next, o_number, o_prev, u_name, unknown, user, _ref;
|
|
if (state_vector == null) {
|
|
state_vector = {};
|
|
}
|
|
json = [];
|
|
unknown = function(user, o_number) {
|
|
if ((user == null) || (o_number == null)) {
|
|
throw new Error("dah!");
|
|
}
|
|
return (state_vector[user] == null) || state_vector[user] <= o_number;
|
|
};
|
|
_ref = this.buffer;
|
|
for (u_name in _ref) {
|
|
user = _ref[u_name];
|
|
if (u_name === "_") {
|
|
continue;
|
|
}
|
|
for (o_number in user) {
|
|
o = user[o_number];
|
|
if ((o.uid.noOperation == null) && unknown(u_name, o_number)) {
|
|
o_json = o._encode();
|
|
if (o.next_cl != null) {
|
|
o_next = o.next_cl;
|
|
while ((o_next.next_cl != null) && unknown(o_next.uid.creator, o_next.uid.op_number)) {
|
|
o_next = o_next.next_cl;
|
|
}
|
|
o_json.next = o_next.getUid();
|
|
} else if (o.prev_cl != null) {
|
|
o_prev = o.prev_cl;
|
|
while ((o_prev.prev_cl != null) && unknown(o_prev.uid.creator, o_prev.uid.op_number)) {
|
|
o_prev = o_prev.prev_cl;
|
|
}
|
|
o_json.prev = o_prev.getUid();
|
|
}
|
|
json.push(o_json);
|
|
}
|
|
}
|
|
}
|
|
return json;
|
|
};
|
|
|
|
HistoryBuffer.prototype.getNextOperationIdentifier = function(user_id) {
|
|
var uid;
|
|
if (user_id == null) {
|
|
user_id = this.user_id;
|
|
}
|
|
if (this.operation_counter[user_id] == null) {
|
|
this.operation_counter[user_id] = 0;
|
|
}
|
|
uid = {
|
|
'creator': user_id,
|
|
'op_number': this.operation_counter[user_id]
|
|
};
|
|
this.operation_counter[user_id]++;
|
|
return uid;
|
|
};
|
|
|
|
HistoryBuffer.prototype.getOperation = function(uid) {
|
|
var o, _ref;
|
|
if (uid.uid != null) {
|
|
uid = uid.uid;
|
|
}
|
|
o = (_ref = this.buffer[uid.creator]) != null ? _ref[uid.op_number] : void 0;
|
|
if ((uid.sub != null) && (o != null)) {
|
|
return o.retrieveSub(uid.sub);
|
|
} else {
|
|
return o;
|
|
}
|
|
};
|
|
|
|
HistoryBuffer.prototype.addOperation = function(o) {
|
|
if (this.buffer[o.uid.creator] == null) {
|
|
this.buffer[o.uid.creator] = {};
|
|
}
|
|
if (this.buffer[o.uid.creator][o.uid.op_number] != null) {
|
|
throw new Error("You must not overwrite operations!");
|
|
}
|
|
if ((o.uid.op_number.constructor !== String) && (!this.isExpectedOperation(o)) && (o.fromHB == null)) {
|
|
throw new Error("this operation was not expected!");
|
|
}
|
|
this.addToCounter(o);
|
|
this.buffer[o.uid.creator][o.uid.op_number] = o;
|
|
return o;
|
|
};
|
|
|
|
HistoryBuffer.prototype.removeOperation = function(o) {
|
|
var _ref;
|
|
return (_ref = this.buffer[o.uid.creator]) != null ? delete _ref[o.uid.op_number] : void 0;
|
|
};
|
|
|
|
HistoryBuffer.prototype.setInvokeSyncHandler = function(f) {
|
|
return this.invokeSync = f;
|
|
};
|
|
|
|
HistoryBuffer.prototype.invokeSync = function() {};
|
|
|
|
HistoryBuffer.prototype.renewStateVector = function(state_vector) {
|
|
var state, user, _results;
|
|
_results = [];
|
|
for (user in state_vector) {
|
|
state = state_vector[user];
|
|
if (((this.operation_counter[user] == null) || (this.operation_counter[user] < state_vector[user])) && (state_vector[user] != null)) {
|
|
_results.push(this.operation_counter[user] = state_vector[user]);
|
|
} else {
|
|
_results.push(void 0);
|
|
}
|
|
}
|
|
return _results;
|
|
};
|
|
|
|
HistoryBuffer.prototype.addToCounter = function(o) {
|
|
var _base, _name;
|
|
if ((_base = this.operation_counter)[_name = o.uid.creator] == null) {
|
|
_base[_name] = 0;
|
|
}
|
|
if (o.uid.op_number === this.operation_counter[o.uid.creator]) {
|
|
this.operation_counter[o.uid.creator]++;
|
|
}
|
|
while (this.buffer[o.uid.creator][this.operation_counter[o.uid.creator]] != null) {
|
|
this.operation_counter[o.uid.creator]++;
|
|
}
|
|
return void 0;
|
|
};
|
|
|
|
return HistoryBuffer;
|
|
|
|
})();
|
|
|
|
module.exports = HistoryBuffer;
|
|
|
|
|
|
},{}],5:[function(require,module,exports){
|
|
var YObject;
|
|
|
|
YObject = (function() {
|
|
function YObject(_object) {
|
|
var name, val, _ref;
|
|
this._object = _object != null ? _object : {};
|
|
if (this._object.constructor === Object) {
|
|
_ref = this._object;
|
|
for (name in _ref) {
|
|
val = _ref[name];
|
|
if (val.constructor === Object) {
|
|
this._object[name] = new YObject(val);
|
|
}
|
|
}
|
|
} else {
|
|
throw new Error("Y.Object accepts Json Objects only");
|
|
}
|
|
}
|
|
|
|
YObject.prototype._name = "Object";
|
|
|
|
YObject.prototype._getModel = function(types, ops) {
|
|
var n, o, _ref;
|
|
if (this._model == null) {
|
|
this._model = new ops.MapManager(this).execute();
|
|
_ref = this._object;
|
|
for (n in _ref) {
|
|
o = _ref[n];
|
|
this._model.val(n, o);
|
|
}
|
|
}
|
|
delete this._object;
|
|
return this._model;
|
|
};
|
|
|
|
YObject.prototype._setModel = function(_model) {
|
|
this._model = _model;
|
|
return delete this._object;
|
|
};
|
|
|
|
YObject.prototype.observe = function(f) {
|
|
this._model.observe(f);
|
|
return this;
|
|
};
|
|
|
|
YObject.prototype.unobserve = function(f) {
|
|
this._model.unobserve(f);
|
|
return this;
|
|
};
|
|
|
|
YObject.prototype.val = function(name, content) {
|
|
var n, res, v, _ref;
|
|
if (this._model != null) {
|
|
return this._model.val.apply(this._model, arguments);
|
|
} else {
|
|
if (content != null) {
|
|
return this._object[name] = content;
|
|
} else if (name != null) {
|
|
return this._object[name];
|
|
} else {
|
|
res = {};
|
|
_ref = this._object;
|
|
for (n in _ref) {
|
|
v = _ref[n];
|
|
res[n] = v;
|
|
}
|
|
return res;
|
|
}
|
|
}
|
|
};
|
|
|
|
YObject.prototype["delete"] = function(name) {
|
|
this._model["delete"](name);
|
|
return this;
|
|
};
|
|
|
|
return YObject;
|
|
|
|
})();
|
|
|
|
if (typeof window !== "undefined" && window !== null) {
|
|
if (window.Y != null) {
|
|
window.Y.Object = YObject;
|
|
} else {
|
|
throw new Error("You must first import Y!");
|
|
}
|
|
}
|
|
|
|
if (typeof module !== "undefined" && module !== null) {
|
|
module.exports = YObject;
|
|
}
|
|
|
|
|
|
},{}],6:[function(require,module,exports){
|
|
var __slice = [].slice,
|
|
__hasProp = {}.hasOwnProperty,
|
|
__extends = function(child, parent) { for (var key in parent) { if (__hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; };
|
|
|
|
module.exports = function() {
|
|
var execution_listener, ops;
|
|
ops = {};
|
|
execution_listener = [];
|
|
ops.Operation = (function() {
|
|
function Operation(custom_type, uid, content, content_operations) {
|
|
var name, op;
|
|
if (custom_type != null) {
|
|
this.custom_type = custom_type;
|
|
}
|
|
this.is_deleted = false;
|
|
this.garbage_collected = false;
|
|
this.event_listeners = [];
|
|
if (uid != null) {
|
|
this.uid = uid;
|
|
}
|
|
if (content === void 0) {
|
|
|
|
} else if ((content != null) && (content.creator != null)) {
|
|
this.saveOperation('content', content);
|
|
} else {
|
|
this.content = content;
|
|
}
|
|
if (content_operations != null) {
|
|
this.content_operations = {};
|
|
for (name in content_operations) {
|
|
op = content_operations[name];
|
|
this.saveOperation(name, op, 'content_operations');
|
|
}
|
|
}
|
|
}
|
|
|
|
Operation.prototype.type = "Operation";
|
|
|
|
Operation.prototype.getContent = function(name) {
|
|
var content, n, v, _ref, _ref1;
|
|
if (this.content != null) {
|
|
if (this.content.getCustomType != null) {
|
|
return this.content.getCustomType();
|
|
} else if (this.content.constructor === Object) {
|
|
if (name != null) {
|
|
if (this.content[name] != null) {
|
|
return this.content[name];
|
|
} else {
|
|
return this.content_operations[name].getCustomType();
|
|
}
|
|
} else {
|
|
content = {};
|
|
_ref = this.content;
|
|
for (n in _ref) {
|
|
v = _ref[n];
|
|
content[n] = v;
|
|
}
|
|
if (this.content_operations != null) {
|
|
_ref1 = this.content_operations;
|
|
for (n in _ref1) {
|
|
v = _ref1[n];
|
|
v = v.getCustomType();
|
|
content[n] = v;
|
|
}
|
|
}
|
|
return content;
|
|
}
|
|
} else {
|
|
return this.content;
|
|
}
|
|
} else {
|
|
return this.content;
|
|
}
|
|
};
|
|
|
|
Operation.prototype.retrieveSub = function() {
|
|
throw new Error("sub properties are not enable on this operation type!");
|
|
};
|
|
|
|
Operation.prototype.observe = function(f) {
|
|
return this.event_listeners.push(f);
|
|
};
|
|
|
|
Operation.prototype.unobserve = function(f) {
|
|
return this.event_listeners = this.event_listeners.filter(function(g) {
|
|
return f !== g;
|
|
});
|
|
};
|
|
|
|
Operation.prototype.deleteAllObservers = function() {
|
|
return this.event_listeners = [];
|
|
};
|
|
|
|
Operation.prototype["delete"] = function() {
|
|
(new ops.Delete(void 0, this)).execute();
|
|
return null;
|
|
};
|
|
|
|
Operation.prototype.callEvent = function() {
|
|
var callon;
|
|
if (this.custom_type != null) {
|
|
callon = this.getCustomType();
|
|
} else {
|
|
callon = this;
|
|
}
|
|
return this.forwardEvent.apply(this, [callon].concat(__slice.call(arguments)));
|
|
};
|
|
|
|
Operation.prototype.forwardEvent = function() {
|
|
var args, f, op, _i, _len, _ref, _results;
|
|
op = arguments[0], args = 2 <= arguments.length ? __slice.call(arguments, 1) : [];
|
|
_ref = this.event_listeners;
|
|
_results = [];
|
|
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
|
f = _ref[_i];
|
|
_results.push(f.call.apply(f, [op].concat(__slice.call(args))));
|
|
}
|
|
return _results;
|
|
};
|
|
|
|
Operation.prototype.isDeleted = function() {
|
|
return this.is_deleted;
|
|
};
|
|
|
|
Operation.prototype.applyDelete = function(garbagecollect) {
|
|
if (garbagecollect == null) {
|
|
garbagecollect = true;
|
|
}
|
|
if (!this.garbage_collected) {
|
|
this.is_deleted = true;
|
|
if (garbagecollect) {
|
|
this.garbage_collected = true;
|
|
return this.HB.addToGarbageCollector(this);
|
|
}
|
|
}
|
|
};
|
|
|
|
Operation.prototype.cleanup = function() {
|
|
this.HB.removeOperation(this);
|
|
return this.deleteAllObservers();
|
|
};
|
|
|
|
Operation.prototype.setParent = function(parent) {
|
|
this.parent = parent;
|
|
};
|
|
|
|
Operation.prototype.getParent = function() {
|
|
return this.parent;
|
|
};
|
|
|
|
Operation.prototype.getUid = function() {
|
|
var map_uid;
|
|
if (this.uid.noOperation == null) {
|
|
return this.uid;
|
|
} else {
|
|
if (this.uid.alt != null) {
|
|
map_uid = this.uid.alt.cloneUid();
|
|
map_uid.sub = this.uid.sub;
|
|
return map_uid;
|
|
} else {
|
|
return void 0;
|
|
}
|
|
}
|
|
};
|
|
|
|
Operation.prototype.cloneUid = function() {
|
|
var n, uid, v, _ref;
|
|
uid = {};
|
|
_ref = this.getUid();
|
|
for (n in _ref) {
|
|
v = _ref[n];
|
|
uid[n] = v;
|
|
}
|
|
return uid;
|
|
};
|
|
|
|
Operation.prototype.execute = function() {
|
|
var l, _i, _len;
|
|
if (this.validateSavedOperations()) {
|
|
this.is_executed = true;
|
|
if (this.uid == null) {
|
|
this.uid = this.HB.getNextOperationIdentifier();
|
|
}
|
|
if (this.uid.noOperation == null) {
|
|
this.HB.addOperation(this);
|
|
for (_i = 0, _len = execution_listener.length; _i < _len; _i++) {
|
|
l = execution_listener[_i];
|
|
l(this._encode());
|
|
}
|
|
}
|
|
return this;
|
|
} else {
|
|
return false;
|
|
}
|
|
};
|
|
|
|
Operation.prototype.saveOperation = function(name, op, base) {
|
|
var dest, last_path, path, paths, _base, _i, _len;
|
|
if (base == null) {
|
|
base = "this";
|
|
}
|
|
if ((op != null) && (op._getModel != null)) {
|
|
op = op._getModel(this.custom_types, this.operations);
|
|
}
|
|
if (op == null) {
|
|
|
|
} else if ((op.execute != null) || !((op.op_number != null) && (op.creator != null))) {
|
|
if (base === "this") {
|
|
return this[name] = op;
|
|
} else {
|
|
dest = this[base];
|
|
paths = name.split("/");
|
|
last_path = paths.pop();
|
|
for (_i = 0, _len = paths.length; _i < _len; _i++) {
|
|
path = paths[_i];
|
|
dest = dest[path];
|
|
}
|
|
return dest[last_path] = op;
|
|
}
|
|
} else {
|
|
if (this.unchecked == null) {
|
|
this.unchecked = {};
|
|
}
|
|
if ((_base = this.unchecked)[base] == null) {
|
|
_base[base] = {};
|
|
}
|
|
return this.unchecked[base][name] = op;
|
|
}
|
|
};
|
|
|
|
Operation.prototype.validateSavedOperations = function() {
|
|
var base, base_name, dest, last_path, name, op, op_uid, path, paths, success, uninstantiated, _i, _len, _ref;
|
|
uninstantiated = {};
|
|
success = true;
|
|
_ref = this.unchecked;
|
|
for (base_name in _ref) {
|
|
base = _ref[base_name];
|
|
for (name in base) {
|
|
op_uid = base[name];
|
|
op = this.HB.getOperation(op_uid);
|
|
if (op) {
|
|
if (base_name === "this") {
|
|
this[name] = op;
|
|
} else {
|
|
dest = this[base_name];
|
|
paths = name.split("/");
|
|
last_path = paths.pop();
|
|
for (_i = 0, _len = paths.length; _i < _len; _i++) {
|
|
path = paths[_i];
|
|
dest = dest[path];
|
|
}
|
|
dest[last_path] = op;
|
|
}
|
|
} else {
|
|
if (uninstantiated[base_name] == null) {
|
|
uninstantiated[base_name] = {};
|
|
}
|
|
uninstantiated[base_name][name] = op_uid;
|
|
success = false;
|
|
}
|
|
}
|
|
}
|
|
if (!success) {
|
|
this.unchecked = uninstantiated;
|
|
return false;
|
|
} else {
|
|
delete this.unchecked;
|
|
return this;
|
|
}
|
|
};
|
|
|
|
Operation.prototype.getCustomType = function() {
|
|
var Type, t, _i, _len, _ref;
|
|
if (this.custom_type == null) {
|
|
return this;
|
|
} else {
|
|
if (this.custom_type.constructor === String) {
|
|
Type = this.custom_types;
|
|
_ref = this.custom_type.split(".");
|
|
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
|
t = _ref[_i];
|
|
Type = Type[t];
|
|
}
|
|
this.custom_type = new Type();
|
|
this.custom_type._setModel(this);
|
|
}
|
|
return this.custom_type;
|
|
}
|
|
};
|
|
|
|
Operation.prototype._encode = function(json) {
|
|
var n, o, operations, _ref, _ref1;
|
|
if (json == null) {
|
|
json = {};
|
|
}
|
|
json.type = this.type;
|
|
json.uid = this.getUid();
|
|
if (this.custom_type != null) {
|
|
if (this.custom_type.constructor === String) {
|
|
json.custom_type = this.custom_type;
|
|
} else {
|
|
json.custom_type = this.custom_type._name;
|
|
}
|
|
}
|
|
if (((_ref = this.content) != null ? _ref.getUid : void 0) != null) {
|
|
json.content = this.content.getUid();
|
|
} else {
|
|
json.content = this.content;
|
|
}
|
|
if (this.content_operations != null) {
|
|
operations = {};
|
|
_ref1 = this.content_operations;
|
|
for (n in _ref1) {
|
|
o = _ref1[n];
|
|
if (o._getModel != null) {
|
|
o = o._getModel(this.custom_types, this.operations);
|
|
}
|
|
operations[n] = o.getUid();
|
|
}
|
|
json.content_operations = operations;
|
|
}
|
|
return json;
|
|
};
|
|
|
|
return Operation;
|
|
|
|
})();
|
|
ops.Delete = (function(_super) {
|
|
__extends(Delete, _super);
|
|
|
|
function Delete(custom_type, uid, deletes) {
|
|
this.saveOperation('deletes', deletes);
|
|
Delete.__super__.constructor.call(this, custom_type, uid);
|
|
}
|
|
|
|
Delete.prototype.type = "Delete";
|
|
|
|
Delete.prototype._encode = function() {
|
|
return {
|
|
'type': "Delete",
|
|
'uid': this.getUid(),
|
|
'deletes': this.deletes.getUid()
|
|
};
|
|
};
|
|
|
|
Delete.prototype.execute = function() {
|
|
var res;
|
|
if (this.validateSavedOperations()) {
|
|
res = Delete.__super__.execute.apply(this, arguments);
|
|
if (res) {
|
|
this.deletes.applyDelete(this);
|
|
}
|
|
return res;
|
|
} else {
|
|
return false;
|
|
}
|
|
};
|
|
|
|
return Delete;
|
|
|
|
})(ops.Operation);
|
|
ops.Delete.parse = function(o) {
|
|
var deletes_uid, uid;
|
|
uid = o['uid'], deletes_uid = o['deletes'];
|
|
return new this(null, uid, deletes_uid);
|
|
};
|
|
ops.Insert = (function(_super) {
|
|
__extends(Insert, _super);
|
|
|
|
function Insert(custom_type, content, content_operations, parent, uid, prev_cl, next_cl, origin) {
|
|
this.saveOperation('parent', parent);
|
|
this.saveOperation('prev_cl', prev_cl);
|
|
this.saveOperation('next_cl', next_cl);
|
|
if (origin != null) {
|
|
this.saveOperation('origin', origin);
|
|
} else {
|
|
this.saveOperation('origin', prev_cl);
|
|
}
|
|
Insert.__super__.constructor.call(this, custom_type, uid, content, content_operations);
|
|
}
|
|
|
|
Insert.prototype.type = "Insert";
|
|
|
|
Insert.prototype.val = function() {
|
|
return this.getContent();
|
|
};
|
|
|
|
Insert.prototype.getNext = function(i) {
|
|
var n;
|
|
if (i == null) {
|
|
i = 1;
|
|
}
|
|
n = this;
|
|
while (i > 0 && (n.next_cl != null)) {
|
|
n = n.next_cl;
|
|
if (!n.is_deleted) {
|
|
i--;
|
|
}
|
|
}
|
|
if (n.is_deleted) {
|
|
null;
|
|
}
|
|
return n;
|
|
};
|
|
|
|
Insert.prototype.getPrev = function(i) {
|
|
var n;
|
|
if (i == null) {
|
|
i = 1;
|
|
}
|
|
n = this;
|
|
while (i > 0 && (n.prev_cl != null)) {
|
|
n = n.prev_cl;
|
|
if (!n.is_deleted) {
|
|
i--;
|
|
}
|
|
}
|
|
if (n.is_deleted) {
|
|
return null;
|
|
} else {
|
|
return n;
|
|
}
|
|
};
|
|
|
|
Insert.prototype.applyDelete = function(o) {
|
|
var callLater, garbagecollect;
|
|
if (this.deleted_by == null) {
|
|
this.deleted_by = [];
|
|
}
|
|
callLater = false;
|
|
if ((this.parent != null) && !this.is_deleted && (o != null)) {
|
|
callLater = true;
|
|
}
|
|
if (o != null) {
|
|
this.deleted_by.push(o);
|
|
}
|
|
garbagecollect = false;
|
|
if (this.next_cl.isDeleted()) {
|
|
garbagecollect = true;
|
|
}
|
|
Insert.__super__.applyDelete.call(this, garbagecollect);
|
|
if (callLater) {
|
|
this.parent.callOperationSpecificDeleteEvents(this, o);
|
|
}
|
|
if ((this.prev_cl != null) && this.prev_cl.isDeleted() && this.prev_cl.garbage_collected !== true) {
|
|
return this.prev_cl.applyDelete();
|
|
}
|
|
};
|
|
|
|
Insert.prototype.cleanup = function() {
|
|
var d, o, _i, _len, _ref;
|
|
if (this.next_cl.isDeleted()) {
|
|
_ref = this.deleted_by;
|
|
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
|
d = _ref[_i];
|
|
d.cleanup();
|
|
}
|
|
o = this.next_cl;
|
|
while (o.type !== "Delimiter") {
|
|
if (o.origin === this) {
|
|
o.origin = this.prev_cl;
|
|
}
|
|
o = o.next_cl;
|
|
}
|
|
this.prev_cl.next_cl = this.next_cl;
|
|
this.next_cl.prev_cl = this.prev_cl;
|
|
if (this.content instanceof ops.Operation && !(this.content instanceof ops.Insert)) {
|
|
this.content.referenced_by--;
|
|
if (this.content.referenced_by <= 0 && !this.content.is_deleted) {
|
|
this.content.applyDelete();
|
|
}
|
|
}
|
|
delete this.content;
|
|
return Insert.__super__.cleanup.apply(this, arguments);
|
|
}
|
|
};
|
|
|
|
Insert.prototype.getDistanceToOrigin = function() {
|
|
var d, o;
|
|
d = 0;
|
|
o = this.prev_cl;
|
|
while (true) {
|
|
if (this.origin === o) {
|
|
break;
|
|
}
|
|
d++;
|
|
o = o.prev_cl;
|
|
}
|
|
return d;
|
|
};
|
|
|
|
Insert.prototype.execute = function() {
|
|
var distance_to_origin, i, o, oDistance, _base;
|
|
if (!this.validateSavedOperations()) {
|
|
return false;
|
|
} else {
|
|
if (this.content instanceof ops.Operation) {
|
|
this.content.insert_parent = this;
|
|
if ((_base = this.content).referenced_by == null) {
|
|
_base.referenced_by = 0;
|
|
}
|
|
this.content.referenced_by++;
|
|
}
|
|
if (this.parent != null) {
|
|
if (this.prev_cl == null) {
|
|
this.prev_cl = this.parent.beginning;
|
|
}
|
|
if (this.origin == null) {
|
|
this.origin = this.prev_cl;
|
|
} else if (this.origin === "Delimiter") {
|
|
this.origin = this.parent.beginning;
|
|
}
|
|
if (this.next_cl == null) {
|
|
this.next_cl = this.parent.end;
|
|
}
|
|
}
|
|
if (this.prev_cl != null) {
|
|
distance_to_origin = this.getDistanceToOrigin();
|
|
o = this.prev_cl.next_cl;
|
|
i = distance_to_origin;
|
|
while (true) {
|
|
if (o !== this.next_cl) {
|
|
oDistance = o.getDistanceToOrigin();
|
|
if (oDistance === i) {
|
|
if (o.uid.creator < this.uid.creator) {
|
|
this.prev_cl = o;
|
|
distance_to_origin = i + 1;
|
|
} else {
|
|
|
|
}
|
|
} else if (oDistance < i) {
|
|
if (i - distance_to_origin <= oDistance) {
|
|
this.prev_cl = o;
|
|
distance_to_origin = i + 1;
|
|
} else {
|
|
|
|
}
|
|
} else {
|
|
break;
|
|
}
|
|
i++;
|
|
o = o.next_cl;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
this.next_cl = this.prev_cl.next_cl;
|
|
this.prev_cl.next_cl = this;
|
|
this.next_cl.prev_cl = this;
|
|
}
|
|
this.setParent(this.prev_cl.getParent());
|
|
Insert.__super__.execute.apply(this, arguments);
|
|
this.parent.callOperationSpecificInsertEvents(this);
|
|
return this;
|
|
}
|
|
};
|
|
|
|
Insert.prototype.getPosition = function() {
|
|
var position, prev;
|
|
position = 0;
|
|
prev = this.prev_cl;
|
|
while (true) {
|
|
if (prev instanceof ops.Delimiter) {
|
|
break;
|
|
}
|
|
if (!prev.isDeleted()) {
|
|
position++;
|
|
}
|
|
prev = prev.prev_cl;
|
|
}
|
|
return position;
|
|
};
|
|
|
|
Insert.prototype._encode = function(json) {
|
|
if (json == null) {
|
|
json = {};
|
|
}
|
|
json.prev = this.prev_cl.getUid();
|
|
json.next = this.next_cl.getUid();
|
|
if (this.origin.type === "Delimiter") {
|
|
json.origin = "Delimiter";
|
|
} else if (this.origin !== this.prev_cl) {
|
|
json.origin = this.origin.getUid();
|
|
}
|
|
json.parent = this.parent.getUid();
|
|
return Insert.__super__._encode.call(this, json);
|
|
};
|
|
|
|
return Insert;
|
|
|
|
})(ops.Operation);
|
|
ops.Insert.parse = function(json) {
|
|
var content, content_operations, next, origin, parent, prev, uid;
|
|
content = json['content'], content_operations = json['content_operations'], uid = json['uid'], prev = json['prev'], next = json['next'], origin = json['origin'], parent = json['parent'];
|
|
return new this(null, content, content_operations, parent, uid, prev, next, origin);
|
|
};
|
|
ops.Delimiter = (function(_super) {
|
|
__extends(Delimiter, _super);
|
|
|
|
function Delimiter(prev_cl, next_cl, origin) {
|
|
this.saveOperation('prev_cl', prev_cl);
|
|
this.saveOperation('next_cl', next_cl);
|
|
this.saveOperation('origin', prev_cl);
|
|
Delimiter.__super__.constructor.call(this, null, {
|
|
noOperation: true
|
|
});
|
|
}
|
|
|
|
Delimiter.prototype.type = "Delimiter";
|
|
|
|
Delimiter.prototype.applyDelete = function() {
|
|
var o;
|
|
Delimiter.__super__.applyDelete.call(this);
|
|
o = this.prev_cl;
|
|
while (o != null) {
|
|
o.applyDelete();
|
|
o = o.prev_cl;
|
|
}
|
|
return void 0;
|
|
};
|
|
|
|
Delimiter.prototype.cleanup = function() {
|
|
return Delimiter.__super__.cleanup.call(this);
|
|
};
|
|
|
|
Delimiter.prototype.execute = function() {
|
|
var _ref, _ref1;
|
|
if (((_ref = this.unchecked) != null ? _ref['next_cl'] : void 0) != null) {
|
|
return Delimiter.__super__.execute.apply(this, arguments);
|
|
} else if ((_ref1 = this.unchecked) != null ? _ref1['prev_cl'] : void 0) {
|
|
if (this.validateSavedOperations()) {
|
|
if (this.prev_cl.next_cl != null) {
|
|
throw new Error("Probably duplicated operations");
|
|
}
|
|
this.prev_cl.next_cl = this;
|
|
return Delimiter.__super__.execute.apply(this, arguments);
|
|
} else {
|
|
return false;
|
|
}
|
|
} else if ((this.prev_cl != null) && (this.prev_cl.next_cl == null)) {
|
|
delete this.prev_cl.unchecked.next_cl;
|
|
this.prev_cl.next_cl = this;
|
|
return Delimiter.__super__.execute.apply(this, arguments);
|
|
} else if ((this.prev_cl != null) || (this.next_cl != null) || true) {
|
|
return Delimiter.__super__.execute.apply(this, arguments);
|
|
}
|
|
};
|
|
|
|
Delimiter.prototype._encode = function() {
|
|
var _ref, _ref1;
|
|
return {
|
|
'type': this.type,
|
|
'uid': this.getUid(),
|
|
'prev': (_ref = this.prev_cl) != null ? _ref.getUid() : void 0,
|
|
'next': (_ref1 = this.next_cl) != null ? _ref1.getUid() : void 0
|
|
};
|
|
};
|
|
|
|
return Delimiter;
|
|
|
|
})(ops.Operation);
|
|
ops.Delimiter.parse = function(json) {
|
|
var next, prev, uid;
|
|
uid = json['uid'], prev = json['prev'], next = json['next'];
|
|
return new this(uid, prev, next);
|
|
};
|
|
return {
|
|
'operations': ops,
|
|
'execution_listener': execution_listener
|
|
};
|
|
};
|
|
|
|
|
|
},{}],7:[function(require,module,exports){
|
|
var RBTReeByIndex, basic_ops_uninitialized,
|
|
__hasProp = {}.hasOwnProperty,
|
|
__extends = function(child, parent) { for (var key in parent) { if (__hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; };
|
|
|
|
basic_ops_uninitialized = require("./Basic");
|
|
|
|
RBTReeByIndex = require('bintrees/lib/rbtree_by_index');
|
|
|
|
module.exports = function() {
|
|
var basic_ops, ops;
|
|
basic_ops = basic_ops_uninitialized();
|
|
ops = basic_ops.operations;
|
|
ops.MapManager = (function(_super) {
|
|
__extends(MapManager, _super);
|
|
|
|
function MapManager(custom_type, uid, content, content_operations) {
|
|
this._map = {};
|
|
MapManager.__super__.constructor.call(this, custom_type, uid, content, content_operations);
|
|
}
|
|
|
|
MapManager.prototype.type = "MapManager";
|
|
|
|
MapManager.prototype.applyDelete = function() {
|
|
var name, p, _ref;
|
|
_ref = this._map;
|
|
for (name in _ref) {
|
|
p = _ref[name];
|
|
p.applyDelete();
|
|
}
|
|
return MapManager.__super__.applyDelete.call(this);
|
|
};
|
|
|
|
MapManager.prototype.cleanup = function() {
|
|
return MapManager.__super__.cleanup.call(this);
|
|
};
|
|
|
|
MapManager.prototype.map = function(f) {
|
|
var n, v, _ref;
|
|
_ref = this._map;
|
|
for (n in _ref) {
|
|
v = _ref[n];
|
|
f(n, v);
|
|
}
|
|
return void 0;
|
|
};
|
|
|
|
MapManager.prototype.val = function(name, content) {
|
|
var o, prop, rep, res, result, _ref;
|
|
if (arguments.length > 1) {
|
|
if ((content != null) && (content._getModel != null)) {
|
|
rep = content._getModel(this.custom_types, this.operations);
|
|
} else {
|
|
rep = content;
|
|
}
|
|
this.retrieveSub(name).replace(rep);
|
|
return this.getCustomType();
|
|
} else if (name != null) {
|
|
prop = this._map[name];
|
|
if ((prop != null) && !prop.isContentDeleted()) {
|
|
res = prop.val();
|
|
if (res instanceof ops.Operation) {
|
|
return res.getCustomType();
|
|
} else {
|
|
return res;
|
|
}
|
|
} else {
|
|
return void 0;
|
|
}
|
|
} else {
|
|
result = {};
|
|
_ref = this._map;
|
|
for (name in _ref) {
|
|
o = _ref[name];
|
|
if (!o.isContentDeleted()) {
|
|
result[name] = o.val();
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
};
|
|
|
|
MapManager.prototype["delete"] = function(name) {
|
|
var _ref;
|
|
if ((_ref = this._map[name]) != null) {
|
|
_ref.deleteContent();
|
|
}
|
|
return this;
|
|
};
|
|
|
|
MapManager.prototype.retrieveSub = function(property_name) {
|
|
var event_properties, event_this, rm, rm_uid;
|
|
if (this._map[property_name] == null) {
|
|
event_properties = {
|
|
name: property_name
|
|
};
|
|
event_this = this;
|
|
rm_uid = {
|
|
noOperation: true,
|
|
sub: property_name,
|
|
alt: this
|
|
};
|
|
rm = new ops.ReplaceManager(null, event_properties, event_this, rm_uid);
|
|
this._map[property_name] = rm;
|
|
rm.setParent(this, property_name);
|
|
rm.execute();
|
|
}
|
|
return this._map[property_name];
|
|
};
|
|
|
|
return MapManager;
|
|
|
|
})(ops.Operation);
|
|
ops.MapManager.parse = function(json) {
|
|
var content, content_operations, custom_type, uid;
|
|
uid = json['uid'], custom_type = json['custom_type'], content = json['content'], content_operations = json['content_operations'];
|
|
return new this(custom_type, uid, content, content_operations);
|
|
};
|
|
ops.ListManager = (function(_super) {
|
|
__extends(ListManager, _super);
|
|
|
|
function ListManager(custom_type, uid, content, content_operations) {
|
|
this.beginning = new ops.Delimiter(void 0, void 0);
|
|
this.end = new ops.Delimiter(this.beginning, void 0);
|
|
this.beginning.next_cl = this.end;
|
|
this.beginning.execute();
|
|
this.end.execute();
|
|
this.shortTree = new RBTreeByIndex();
|
|
this.completeTree = new RBTreeByIndex();
|
|
ListManager.__super__.constructor.call(this, custom_type, uid, content, content_operations);
|
|
}
|
|
|
|
ListManager.prototype.type = "ListManager";
|
|
|
|
ListManager.prototype.applyDelete = function() {
|
|
var o;
|
|
o = this.beginning;
|
|
while (o != null) {
|
|
o.applyDelete();
|
|
o = o.next_cl;
|
|
}
|
|
return ListManager.__super__.applyDelete.call(this);
|
|
};
|
|
|
|
ListManager.prototype.cleanup = function() {
|
|
return ListManager.__super__.cleanup.call(this);
|
|
};
|
|
|
|
ListManager.prototype.toJson = function(transform_to_value) {
|
|
var i, o, val, _i, _len, _results;
|
|
if (transform_to_value == null) {
|
|
transform_to_value = false;
|
|
}
|
|
val = this.val();
|
|
_results = [];
|
|
for (o = _i = 0, _len = val.length; _i < _len; o = ++_i) {
|
|
i = val[o];
|
|
if (o instanceof ops.Object) {
|
|
_results.push(o.toJson(transform_to_value));
|
|
} else if (o instanceof ops.ListManager) {
|
|
_results.push(o.toJson(transform_to_value));
|
|
} else if (transform_to_value && o instanceof ops.Operation) {
|
|
_results.push(o.val());
|
|
} else {
|
|
_results.push(o);
|
|
}
|
|
}
|
|
return _results;
|
|
};
|
|
|
|
ListManager.prototype.execute = function() {
|
|
if (this.validateSavedOperations()) {
|
|
this.beginning.setParent(this);
|
|
this.end.setParent(this);
|
|
return ListManager.__super__.execute.apply(this, arguments);
|
|
} else {
|
|
return false;
|
|
}
|
|
};
|
|
|
|
ListManager.prototype.getLastOperation = function() {
|
|
return this.end.prev_cl;
|
|
};
|
|
|
|
ListManager.prototype.getFirstOperation = function() {
|
|
return this.beginning.next_cl;
|
|
};
|
|
|
|
ListManager.prototype.getNextNonDeleted = function(start) {
|
|
var operation;
|
|
if (start.isDeleted() || (start.node == null)) {
|
|
operation = start.next_cl;
|
|
while (!(operation instanceof ops.Delimiter)) {
|
|
if (operation.is_deleted) {
|
|
operation = operation.next_cl;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
operation = start.node.next.node;
|
|
if (!operation) {
|
|
return false;
|
|
}
|
|
}
|
|
return operation;
|
|
};
|
|
|
|
ListManager.prototype.getPrevNonDeleted = function(start) {
|
|
var operation;
|
|
if (start.isDeleted() || (start.node == null)) {
|
|
operation = start.prev_cl;
|
|
while (!(operation instanceof ops.Delimiter)) {
|
|
if (operation.is_deleted) {
|
|
operation = operation.prev_cl;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
operation = start.node.prev.node;
|
|
if (!operation) {
|
|
return false;
|
|
}
|
|
}
|
|
return operation;
|
|
};
|
|
|
|
ListManager.prototype.toArray = function() {
|
|
return this.shortTree.map(function(operation) {
|
|
return operation.val();
|
|
});
|
|
};
|
|
|
|
ListManager.prototype.map = function(fun) {
|
|
return this.shortTree.map(fun);
|
|
};
|
|
|
|
ListManager.prototype.fold = function(init, fun) {
|
|
return this.shortTree.map(function(operation) {
|
|
return init = fun(init, operation);
|
|
});
|
|
};
|
|
|
|
ListManager.prototype.val = function(pos) {
|
|
if (pos != null) {
|
|
return this.shortTree.find(pos).val();
|
|
} else {
|
|
return this.toArray();
|
|
}
|
|
};
|
|
|
|
ListManager.prototype.ref = function(pos) {
|
|
if (pos != null) {
|
|
return this.shortTree.find(pos);
|
|
} else {
|
|
return this.shortTree.map(function(operation) {
|
|
return operation;
|
|
});
|
|
}
|
|
};
|
|
|
|
ListManager.prototype.getOperationByPosition = function(position) {
|
|
if (position === 0) {
|
|
return this.beginning;
|
|
} else if (position === this.shortTree.size + 1) {
|
|
return this.end;
|
|
} else {
|
|
return this.shortTree.find(position - 1);
|
|
}
|
|
};
|
|
|
|
ListManager.prototype.push = function(content) {
|
|
return this.insertAfter(this.end.prev_cl, [content]);
|
|
};
|
|
|
|
ListManager.prototype.insertAfterHelper = function(root, content) {
|
|
var right;
|
|
if (!root.right) {
|
|
root.bt.right = content;
|
|
return content.bt.parent = root;
|
|
} else {
|
|
return right = root.next_cl;
|
|
}
|
|
};
|
|
|
|
ListManager.prototype.insertAfter = function(left, contents) {
|
|
var c, leftNode, right, rightNode, tmp, _i, _len;
|
|
if (left === this.beginning) {
|
|
leftNode = null;
|
|
rightNode = this.shortTree.findNode(0);
|
|
right = rightNode ? rightNode.data : this.end;
|
|
} else {
|
|
rightNode = left.node.next;
|
|
leftNode = left.node;
|
|
right = rightNode ? rightNode.data : this.end;
|
|
}
|
|
left = right.prev_cl;
|
|
if (contents instanceof ops.Operation) {
|
|
tmp = new ops.Insert(null, content, null, void 0, void 0, left, right);
|
|
tmp.execute();
|
|
} else {
|
|
for (_i = 0, _len = contents.length; _i < _len; _i++) {
|
|
c = contents[_i];
|
|
if ((c != null) && (c._name != null) && (c._getModel != null)) {
|
|
c = c._getModel(this.custom_types, this.operations);
|
|
}
|
|
tmp = new ops.Insert(null, c, null, void 0, void 0, left, right);
|
|
tmp.execute();
|
|
leftNode = tmp.node;
|
|
left = tmp;
|
|
}
|
|
}
|
|
return this;
|
|
};
|
|
|
|
ListManager.prototype.insert = function(position, contents) {
|
|
var ith;
|
|
ith = this.getOperationByPosition(position);
|
|
return this.insertAfter(ith, contents);
|
|
};
|
|
|
|
ListManager.prototype.deleteRef = function(operation, length, dir) {
|
|
var deleteOperation, i, nextOperation, _i;
|
|
if (length == null) {
|
|
length = 1;
|
|
}
|
|
if (dir == null) {
|
|
dir = 'right';
|
|
}
|
|
nextOperation = (function(_this) {
|
|
return function(operation) {
|
|
if (dir === 'right') {
|
|
return _this.getNextNonDeleted(operation);
|
|
} else {
|
|
return _this.getPrevNonDeleted(operation);
|
|
}
|
|
};
|
|
})(this);
|
|
for (i = _i = 0; 0 <= length ? _i < length : _i > length; i = 0 <= length ? ++_i : --_i) {
|
|
if (operation instanceof ops.Delimiter) {
|
|
break;
|
|
}
|
|
deleteOperation = (new ops.Delete(null, void 0, operation)).execute();
|
|
operation = nextOperation(operation);
|
|
}
|
|
return this;
|
|
};
|
|
|
|
ListManager.prototype["delete"] = function(position, length) {
|
|
var operation;
|
|
if (length == null) {
|
|
length = 1;
|
|
}
|
|
operation = this.getOperationByPosition(position + length);
|
|
return this.deleteRef(operation, length, 'left');
|
|
};
|
|
|
|
ListManager.prototype.callOperationSpecificInsertEvents = function(operation) {
|
|
var getContentType, next, nextNode, prev, prevNode;
|
|
prev = (this.getPrevNonDeleted(operation)) || this.beginning;
|
|
prevNode = prev ? prev.node : null;
|
|
next = (this.getNextNonDeleted(operation)) || this.end;
|
|
nextNode = next ? next.node : null;
|
|
operation.node = operation.node || (this.shortTree.insert_between(prevNode, nextNode, operation));
|
|
operation.completeNode = operation.completeNode || (this.completeTree.insert_between(operation.prev_cl.completeNode, operation.next_cl.completeNode, operation));
|
|
getContentType = function(content) {
|
|
if (content instanceof ops.Operation) {
|
|
return content.getCustomType();
|
|
} else {
|
|
return content;
|
|
}
|
|
};
|
|
return this.callEvent([
|
|
{
|
|
type: "insert",
|
|
reference: operation,
|
|
position: operation.node.position(),
|
|
object: this.getCustomType(),
|
|
changedBy: operation.uid.creator,
|
|
value: getContentType(operation.val())
|
|
}
|
|
]);
|
|
};
|
|
|
|
ListManager.prototype.callOperationSpecificDeleteEvents = function(operation, del_op) {
|
|
var position;
|
|
if (operation.node) {
|
|
position = operation.node.position();
|
|
this.shortTree.remove_node(operation.node);
|
|
operation.node = null;
|
|
}
|
|
return this.callEvent([
|
|
{
|
|
type: "delete",
|
|
reference: operation,
|
|
position: position,
|
|
object: this.getCustomType(),
|
|
length: 1,
|
|
changedBy: del_op.uid.creator,
|
|
oldValue: operation.val()
|
|
}
|
|
]);
|
|
};
|
|
|
|
return ListManager;
|
|
|
|
})(ops.Operation);
|
|
ops.ListManager.parse = function(json) {
|
|
var content, content_operations, custom_type, uid;
|
|
uid = json['uid'], custom_type = json['custom_type'], content = json['content'], content_operations = json['content_operations'];
|
|
return new this(custom_type, uid, content, content_operations);
|
|
};
|
|
ops.Composition = (function(_super) {
|
|
__extends(Composition, _super);
|
|
|
|
function Composition(custom_type, _composition_value, composition_value_operations, uid, tmp_composition_ref) {
|
|
var n, o;
|
|
this._composition_value = _composition_value;
|
|
Composition.__super__.constructor.call(this, custom_type, uid);
|
|
if (tmp_composition_ref != null) {
|
|
this.tmp_composition_ref = tmp_composition_ref;
|
|
} else {
|
|
this.composition_ref = this.end.prev_cl;
|
|
}
|
|
if (composition_value_operations != null) {
|
|
this.composition_value_operations = {};
|
|
for (n in composition_value_operations) {
|
|
o = composition_value_operations[n];
|
|
this.saveOperation(n, o, '_composition_value');
|
|
}
|
|
}
|
|
}
|
|
|
|
Composition.prototype.type = "Composition";
|
|
|
|
Composition.prototype.execute = function() {
|
|
var composition_ref;
|
|
if (this.validateSavedOperations()) {
|
|
this.getCustomType()._setCompositionValue(this._composition_value);
|
|
delete this._composition_value;
|
|
if (this.tmp_composition_ref) {
|
|
composition_ref = this.HB.getOperation(this.tmp_composition_ref);
|
|
if (composition_ref != null) {
|
|
delete this.tmp_composition_ref;
|
|
this.composition_ref = composition_ref;
|
|
}
|
|
}
|
|
return Composition.__super__.execute.apply(this, arguments);
|
|
} else {
|
|
return false;
|
|
}
|
|
};
|
|
|
|
Composition.prototype.callOperationSpecificInsertEvents = function(operation) {
|
|
var o;
|
|
if (this.tmp_composition_ref != null) {
|
|
if (operation.uid.creator === this.tmp_composition_ref.creator && operation.uid.op_number === this.tmp_composition_ref.op_number) {
|
|
this.composition_ref = operation;
|
|
delete this.tmp_composition_ref;
|
|
operation = operation.next_cl;
|
|
if (operation === this.end) {
|
|
return;
|
|
}
|
|
} else {
|
|
return;
|
|
}
|
|
}
|
|
o = this.end.prev_cl;
|
|
while (o !== operation) {
|
|
this.getCustomType()._unapply(o.undo_delta);
|
|
o = o.prev_cl;
|
|
}
|
|
while (o !== this.end) {
|
|
o.undo_delta = this.getCustomType()._apply(o.val());
|
|
o = o.next_cl;
|
|
}
|
|
this.composition_ref = this.end.prev_cl;
|
|
return this.callEvent([
|
|
{
|
|
type: "update",
|
|
changedBy: operation.uid.creator,
|
|
newValue: this.val()
|
|
}
|
|
]);
|
|
};
|
|
|
|
Composition.prototype.callOperationSpecificDeleteEvents = function(operation, del_op) {};
|
|
|
|
Composition.prototype.applyDelta = function(delta, operations) {
|
|
(new ops.Insert(null, delta, operations, this, null, this.end.prev_cl, this.end)).execute();
|
|
return void 0;
|
|
};
|
|
|
|
Composition.prototype._encode = function(json) {
|
|
var custom, n, o, _ref;
|
|
if (json == null) {
|
|
json = {};
|
|
}
|
|
custom = this.getCustomType()._getCompositionValue();
|
|
json.composition_value = custom.composition_value;
|
|
if (custom.composition_value_operations != null) {
|
|
json.composition_value_operations = {};
|
|
_ref = custom.composition_value_operations;
|
|
for (n in _ref) {
|
|
o = _ref[n];
|
|
json.composition_value_operations[n] = o.getUid();
|
|
}
|
|
}
|
|
if (this.composition_ref != null) {
|
|
json.composition_ref = this.composition_ref.getUid();
|
|
} else {
|
|
json.composition_ref = this.tmp_composition_ref;
|
|
}
|
|
return Composition.__super__._encode.call(this, json);
|
|
};
|
|
|
|
return Composition;
|
|
|
|
})(ops.ListManager);
|
|
ops.Composition.parse = function(json) {
|
|
var composition_ref, composition_value, composition_value_operations, custom_type, uid;
|
|
uid = json['uid'], custom_type = json['custom_type'], composition_value = json['composition_value'], composition_value_operations = json['composition_value_operations'], composition_ref = json['composition_ref'];
|
|
return new this(custom_type, composition_value, composition_value_operations, uid, composition_ref);
|
|
};
|
|
ops.ReplaceManager = (function(_super) {
|
|
__extends(ReplaceManager, _super);
|
|
|
|
function ReplaceManager(custom_type, event_properties, event_this, uid) {
|
|
this.event_properties = event_properties;
|
|
this.event_this = event_this;
|
|
if (this.event_properties['object'] == null) {
|
|
this.event_properties['object'] = this.event_this.getCustomType();
|
|
}
|
|
ReplaceManager.__super__.constructor.call(this, custom_type, uid);
|
|
}
|
|
|
|
ReplaceManager.prototype.type = "ReplaceManager";
|
|
|
|
ReplaceManager.prototype.callEventDecorator = function(events) {
|
|
var event, name, prop, _i, _len, _ref;
|
|
if (!this.isDeleted()) {
|
|
for (_i = 0, _len = events.length; _i < _len; _i++) {
|
|
event = events[_i];
|
|
_ref = this.event_properties;
|
|
for (name in _ref) {
|
|
prop = _ref[name];
|
|
event[name] = prop;
|
|
}
|
|
}
|
|
this.event_this.callEvent(events);
|
|
}
|
|
return void 0;
|
|
};
|
|
|
|
ReplaceManager.prototype.callOperationSpecificInsertEvents = function(operation) {
|
|
var old_value;
|
|
if (operation.next_cl.type === "Delimiter" && operation.prev_cl.type !== "Delimiter") {
|
|
if (!operation.is_deleted) {
|
|
old_value = operation.prev_cl.val();
|
|
this.callEventDecorator([
|
|
{
|
|
type: "update",
|
|
changedBy: operation.uid.creator,
|
|
oldValue: old_value
|
|
}
|
|
]);
|
|
}
|
|
operation.prev_cl.applyDelete();
|
|
} else if (operation.next_cl.type !== "Delimiter") {
|
|
operation.applyDelete();
|
|
} else {
|
|
this.callEventDecorator([
|
|
{
|
|
type: "add",
|
|
changedBy: operation.uid.creator
|
|
}
|
|
]);
|
|
}
|
|
return void 0;
|
|
};
|
|
|
|
ReplaceManager.prototype.callOperationSpecificDeleteEvents = function(operation, del_op) {
|
|
if (operation.next_cl.type === "Delimiter") {
|
|
return this.callEventDecorator([
|
|
{
|
|
type: "delete",
|
|
changedBy: del_op.uid.creator,
|
|
oldValue: operation.val()
|
|
}
|
|
]);
|
|
}
|
|
};
|
|
|
|
ReplaceManager.prototype.replace = function(content, replaceable_uid) {
|
|
var o, relp;
|
|
o = this.getLastOperation();
|
|
relp = (new ops.Insert(null, content, null, this, replaceable_uid, o, o.next_cl)).execute();
|
|
return void 0;
|
|
};
|
|
|
|
ReplaceManager.prototype.isContentDeleted = function() {
|
|
return this.getLastOperation().isDeleted();
|
|
};
|
|
|
|
ReplaceManager.prototype.deleteContent = function() {
|
|
var last_op;
|
|
last_op = this.getLastOperation();
|
|
if ((!last_op.isDeleted()) && last_op.type !== "Delimiter") {
|
|
(new ops.Delete(null, void 0, this.getLastOperation().uid)).execute();
|
|
}
|
|
return void 0;
|
|
};
|
|
|
|
ReplaceManager.prototype.val = function() {
|
|
var o;
|
|
o = this.getLastOperation();
|
|
return typeof o.val === "function" ? o.val() : void 0;
|
|
};
|
|
|
|
return ReplaceManager;
|
|
|
|
})(ops.ListManager);
|
|
return basic_ops;
|
|
};
|
|
|
|
|
|
},{"./Basic":6,"bintrees/lib/rbtree_by_index":9}],8:[function(require,module,exports){
|
|
var Engine, HistoryBuffer, adaptConnector, createY, structured_ops_uninitialized;
|
|
|
|
structured_ops_uninitialized = require("./Operations/Structured");
|
|
|
|
HistoryBuffer = require("./HistoryBuffer");
|
|
|
|
Engine = require("./Engine");
|
|
|
|
adaptConnector = require("./ConnectorAdapter");
|
|
|
|
createY = function(connector) {
|
|
var HB, ct, engine, model, ops, ops_manager, user_id;
|
|
if (connector.user_id != null) {
|
|
user_id = connector.user_id;
|
|
} else {
|
|
user_id = "_temp";
|
|
connector.when_received_state_vector_listeners = [
|
|
function(state_vector) {
|
|
return HB.setUserId(this.user_id, state_vector);
|
|
}
|
|
];
|
|
}
|
|
HB = new HistoryBuffer(user_id);
|
|
ops_manager = structured_ops_uninitialized(HB, this.constructor);
|
|
ops = ops_manager.operations;
|
|
engine = new Engine(HB, ops);
|
|
adaptConnector(connector, engine, HB, ops_manager.execution_listener);
|
|
ops.Operation.prototype.HB = HB;
|
|
ops.Operation.prototype.operations = ops;
|
|
ops.Operation.prototype.engine = engine;
|
|
ops.Operation.prototype.connector = connector;
|
|
ops.Operation.prototype.custom_types = this.constructor;
|
|
ct = new createY.Object();
|
|
model = new ops.MapManager(ct, HB.getReservedUniqueIdentifier()).execute();
|
|
ct._setModel(model);
|
|
return ct;
|
|
};
|
|
|
|
module.exports = createY;
|
|
|
|
if (typeof window !== "undefined" && window !== null) {
|
|
window.Y = createY;
|
|
}
|
|
|
|
createY.Object = require("./ObjectType");
|
|
|
|
|
|
},{"./ConnectorAdapter":1,"./Engine":3,"./HistoryBuffer":4,"./ObjectType":5,"./Operations/Structured":7}],9:[function(require,module,exports){
|
|
var TreeBase = require('./treebase');
|
|
/** algorithm from Cormen, Leiserson - Introduction to algorithm **/
|
|
|
|
function RBTree(comparator) {
|
|
this._root = null;
|
|
this._nil = new Node('nil');
|
|
this._nil.red = false;
|
|
this._nil.weight = 0;
|
|
|
|
Object.defineProperty(this, 'size', {
|
|
get: function() {
|
|
return this._root ? this._root.weight : 0;
|
|
}
|
|
});
|
|
}
|
|
|
|
RBTree.prototype = new TreeBase();
|
|
|
|
RBTree.prototype.insert = function(position, data) {
|
|
var nodeToInsert = new Node(data);
|
|
if (this.insert_node(position, nodeToInsert)) {
|
|
return nodeToInsert;
|
|
} else {
|
|
return false;
|
|
}
|
|
};
|
|
|
|
RBTree.prototype.insert_node = function(position, nodeToInsert) {
|
|
var node = this._root;
|
|
var insertAfter;
|
|
|
|
if (!node) {
|
|
this._root = nodeToInsert;
|
|
this._root.red = false;
|
|
return true;
|
|
}
|
|
|
|
while (true) {
|
|
if (node.weight === position) { // Insert after max of node subtree
|
|
insertAfter = node.max_tree(function(node) {
|
|
node.weight += 1;
|
|
});
|
|
insertAfter.set_child('right', nodeToInsert);
|
|
nodeToInsert.parent = insertAfter;
|
|
break;
|
|
} else {
|
|
left = node.get_child('left');
|
|
right = node.get_child('right');
|
|
if (!left && position === 0) {
|
|
node.weight += 1;
|
|
node.set_child('left', nodeToInsert);
|
|
nodeToInsert.parent = node;
|
|
break;
|
|
|
|
} else if (left && left.weight >= position) {
|
|
node.weight += 1;
|
|
node = left;
|
|
|
|
} else if (right) {
|
|
position -= (left? left.weight: 0) + 1;
|
|
node.weight += 1;
|
|
node = right;
|
|
|
|
} else {
|
|
node.weight += 1;
|
|
node.set_child('right', nodeToInsert);
|
|
nodeToInsert.parent = node;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
this.insert_correction(nodeToInsert);
|
|
return true;
|
|
};
|
|
|
|
RBTree.prototype.insert_between = function(onLeft, onRight, data) {
|
|
var newNode = new Node(data);
|
|
// onLeft and onRight are neighbors, so they can't have an element in between.
|
|
// For example, if onLeft exists, its right neighbor is either nil or right. If it's onRight,
|
|
// then onRight has no child. If neither left and right exist, the tree is empty.
|
|
if (onLeft && !onLeft.right) {
|
|
onLeft.right = newNode;
|
|
newNode.parent = onLeft;
|
|
|
|
this.insert_correction(newNode);
|
|
} else if (onRight && !onRight.left) {
|
|
onRight.left = newNode;
|
|
newNode.parent = onRight;
|
|
|
|
this.insert_correction(newNode);
|
|
} else {
|
|
this.insert_node(0, newNode);
|
|
}
|
|
newNode.traverse_up(function(node, parent) {
|
|
parent.weight = (parent.left ? parent.left.weight : 0) +
|
|
(parent.right ? parent.right.weight : 0) + 1;
|
|
});
|
|
|
|
return newNode;
|
|
};
|
|
|
|
RBTree.prototype.rotate = function(side, node) {
|
|
// all the comment are with the assumption that side === 'left'
|
|
var neighbor;
|
|
|
|
// get right neighbor
|
|
neighbor = node.get_child(side, true);
|
|
// neighbor's left tree becomes node's right tree
|
|
node.set_child(side, neighbor.get_child(side), true);
|
|
|
|
// if this right tree was non-empty
|
|
if (neighbor.get_child(side)) {
|
|
neighbor.get_child(side).parent = node; // attach neighbor's left child to node
|
|
}
|
|
neighbor.parent = node.parent; // link neighbor's parent to node's parent
|
|
|
|
if (!node.parent) { // no parent === is_root
|
|
this._root = neighbor;
|
|
} else if (node === node.parent.get_child(side)){ // node is left child
|
|
node.parent.set_child(side, neighbor); // node's parent left child is now neighbor
|
|
} else {
|
|
node.parent.set_child(side, neighbor, true); // node's parent right child is now neighbor
|
|
}
|
|
|
|
neighbor.set_child(side, node); // attach node on left of child
|
|
node.parent = neighbor; // set node's parent to neighbor
|
|
|
|
// update node's weight first, then
|
|
node.weight = (node.left? node.left.weight: 0) + (node.right? node.right.weight: 0) + 1;
|
|
neighbor.weight = (neighbor.left? neighbor.left.weight: 0) + (neighbor.right? neighbor.right.weight: 0) + 1;
|
|
};
|
|
|
|
RBTree.prototype.insert_correction = function(node) {
|
|
var self = this;
|
|
function helper(side) {
|
|
var uncle;
|
|
uncle = node.parent.parent.get_child(side, true);
|
|
|
|
if (uncle && uncle.red) { // if uncle is undefined, it's a leaf so it's black
|
|
node.parent.red = false;
|
|
uncle.red = false;
|
|
node.parent.parent.red = true;
|
|
|
|
node = node.parent.parent;
|
|
} else {
|
|
if (node === node.parent.get_child(side, true)) {
|
|
node = node.parent;
|
|
self.rotate(side, node);
|
|
}
|
|
|
|
node.parent.red = false;
|
|
node.parent.parent.red = true;
|
|
|
|
var oppositeSide = side === 'left'? 'right': 'left';
|
|
self.rotate(oppositeSide, node.parent.parent);
|
|
}
|
|
}
|
|
|
|
|
|
while (node.parent && node.parent.red) { // is there's no node.parent, then it means the parent
|
|
// is nil which is black
|
|
|
|
// if node's parent is on left of his parent
|
|
if (node.parent === node.parent.parent.get_child('left')) { // Check that there is a grandparent
|
|
helper('left');
|
|
} else if (node.parent === node.parent.parent.get_child('right')){
|
|
helper('right');
|
|
}
|
|
}
|
|
|
|
this._root.red = false;
|
|
|
|
};
|
|
|
|
RBTree.prototype.find = function() {
|
|
var node = this.findNode.apply(this, arguments);
|
|
if (node) {
|
|
return node.data;
|
|
} else {
|
|
return null;
|
|
}
|
|
};
|
|
|
|
/** Find the node at position @position and return it. If no node is found, returns null.
|
|
* It is also possible to pass a function as second argument. It will be called
|
|
* with each node traversed except for the one found.
|
|
**/
|
|
RBTree.prototype.findNode = function(position, fun) {
|
|
// if the weight is 'n', the biggest index is n-1, so we check that position >= size
|
|
if (position >= this.size || position < 0) {
|
|
return null;
|
|
}
|
|
var node = this._root;
|
|
|
|
if (!node) {
|
|
return null;
|
|
}
|
|
|
|
// Find node to delete
|
|
while (position > 0 || (position === 0 && node.left)) {
|
|
left = node.left;
|
|
right = node.right;
|
|
|
|
if (left && left.weight > position) {
|
|
// when there's a left neighbor with a weight > position to remove,
|
|
// go into this subtree and decrease the subtree weight
|
|
if (fun) {
|
|
fun(node);
|
|
}
|
|
node = left;
|
|
|
|
} else if ((!left && position === 0) || (left && left.weight === position)) {
|
|
break;
|
|
} else if (right) {
|
|
// when there's a right subtee, go into it and decrease the position
|
|
// by the weight of the left subtree - 1 + 1 (for the previous node)
|
|
if (fun) {
|
|
fun(node);
|
|
}
|
|
node = right;
|
|
position -= (left? left.weight: 0) + 1;
|
|
|
|
} else {
|
|
// this should not happen, except if the position is greater than the size of the tree
|
|
throw new Error('this should not happen');
|
|
}
|
|
}
|
|
return node;
|
|
};
|
|
|
|
RBTree.prototype.remove = function(position) {
|
|
// if the weight is 'n', the biggest index is n-1, so we check that position >= size
|
|
var nodeToRemove;
|
|
|
|
nodeToRemove = this.findNode(position, function(node) { node.weight --; });
|
|
|
|
return this.remove_helper(nodeToRemove);
|
|
};
|
|
|
|
RBTree.prototype.remove_helper = function(nodeToRemove) {
|
|
var left, right, nextNode, childNode, parent;
|
|
|
|
// if there's only one child, replace the nodeToRemove to delete by it's child and update
|
|
// the refs.
|
|
// if there's two children, find it's successor and replace the nodeToRemove to delete by his successor
|
|
// and update the refs
|
|
|
|
if (!nodeToRemove.left || !nodeToRemove.right) {
|
|
nextNode = nodeToRemove;
|
|
} else {
|
|
nextNode = nodeToRemove.getNext(function(node) {
|
|
node.weight -= 1;
|
|
});
|
|
}
|
|
|
|
if (nextNode.left) {
|
|
childNode = nextNode.left;
|
|
parent = nextNode.parent;
|
|
} else {
|
|
childNode = nextNode.right;
|
|
parent = nextNode.parent;
|
|
}
|
|
|
|
if (childNode) {
|
|
childNode.parent = nextNode.parent;
|
|
}
|
|
|
|
|
|
if (!nextNode.parent) {
|
|
this._root = childNode;
|
|
} else {
|
|
if (nextNode === nextNode.parent.left) {
|
|
nextNode.parent.left = childNode;
|
|
} else {
|
|
nextNode.parent.right = childNode;
|
|
}
|
|
}
|
|
|
|
// replace nodeToRemove's data by nextNode's (same as removing nodeToRemove, then inserting nextNode at his place
|
|
// but easier and more efficient)
|
|
if (nextNode !== nodeToRemove) {
|
|
nodeToRemove.data = nextNode.data;
|
|
nodeToRemove.weight = (nodeToRemove.left? nodeToRemove.left.weight: 0) +
|
|
(nodeToRemove.right? nodeToRemove.right.weight: 0) + 1;
|
|
|
|
}
|
|
|
|
if (!nextNode.red) {
|
|
this.remove_correction(childNode, parent);
|
|
}
|
|
|
|
return nextNode;
|
|
};
|
|
|
|
RBTree.prototype.remove_node = function(node) {
|
|
node.traverse_up(function(node, parent) {
|
|
parent.weight --;
|
|
});
|
|
|
|
return this.remove_helper(node);
|
|
};
|
|
|
|
RBTree.prototype.remove_correction = function(node, parent) {
|
|
var self = this;
|
|
var oppositeSide;
|
|
var helper = function(side) {
|
|
var neighbor = parent.get_child(side, true);
|
|
|
|
if (neighbor && neighbor.red) {
|
|
neighbor.red = false;
|
|
parent.red = true;
|
|
|
|
self.rotate(side, parent);
|
|
neighbor = parent.get_child(side, true);
|
|
}
|
|
|
|
if ((!neighbor.left || !neighbor.left.red) && (!neighbor.right || !neighbor.right.red )) {
|
|
neighbor.red = true;
|
|
node = parent;
|
|
parent = node.parent;
|
|
} else {
|
|
if (!neighbor.get_child(side, true) || !neighbor.get_child(side, true).red) {
|
|
if (neighbor.get_child(side)) {
|
|
neighbor.get_child(side).red = false;
|
|
}
|
|
neighbor.red = true;
|
|
|
|
oppositeSide = side === 'left'? 'right': 'left';
|
|
self.rotate(oppositeSide, neighbor);
|
|
neighbor = parent.get_child(side, true);
|
|
}
|
|
|
|
neighbor.red = parent? parent.red: false;
|
|
parent.red = false;
|
|
if (neighbor.get_child(side, true)) {
|
|
neighbor.get_child(side, true).red = false;
|
|
}
|
|
|
|
self.rotate(side, parent);
|
|
node = self._root;
|
|
parent = node.parent;
|
|
}
|
|
};
|
|
while (node !== this._root && (!node || !node.red)) {
|
|
if (node === parent.get_child('left')) {
|
|
tmp = helper('left');
|
|
} else if (node === parent.get_child('right')) {
|
|
tmp = helper('right');
|
|
}
|
|
}
|
|
|
|
if (node) {
|
|
node.red = false;
|
|
}
|
|
};
|
|
|
|
|
|
function Node(data, name) {
|
|
this._dataBindName = name || 'node';
|
|
this._data = data;
|
|
|
|
this.left = null;
|
|
this.right = null;
|
|
this.parent = null;
|
|
this.red = true;
|
|
this.weight = 1;
|
|
|
|
Object.defineProperty(this, 'next', {
|
|
get: function() {
|
|
return this.getNext();
|
|
}
|
|
});
|
|
Object.defineProperty(this, 'prev', {
|
|
get: function() {
|
|
return this.getPrev();
|
|
}
|
|
});
|
|
|
|
Object.defineProperty(this, 'data', {
|
|
get: function() {
|
|
return this._data;
|
|
},
|
|
set: function(data) {
|
|
this._data = data;
|
|
// bind the data to this
|
|
data[this._dataBindName] = this;
|
|
}
|
|
});
|
|
|
|
}
|
|
|
|
Node.prototype.get_child = function(side, opposite) {
|
|
if (opposite) {
|
|
side = (side === 'left')? 'right': 'left';
|
|
}
|
|
return side === 'left'? this.left: this.right;
|
|
};
|
|
|
|
Node.prototype.set_child = function(side, node, opposite) {
|
|
if (opposite) {
|
|
side = (side === 'left')? 'right': 'left';
|
|
}
|
|
|
|
if (side === 'left') {
|
|
this.left = node;
|
|
} else {
|
|
this.right = node;
|
|
}
|
|
};
|
|
|
|
Node.prototype.max_tree = function(fun) {
|
|
var node = this;
|
|
|
|
if (fun) {
|
|
fun(node);
|
|
}
|
|
|
|
while (node.right) {
|
|
node = node.right;
|
|
if (fun) {
|
|
fun(node);
|
|
}
|
|
}
|
|
|
|
return node;
|
|
};
|
|
|
|
Node.prototype.min_tree = function(fun) {
|
|
var node = this;
|
|
|
|
if (fun) {
|
|
fun(node);
|
|
}
|
|
|
|
while (node.left) {
|
|
node = node.left;
|
|
if (fun) {
|
|
fun(node);
|
|
}
|
|
}
|
|
|
|
return node;
|
|
};
|
|
|
|
Node.prototype.getNext = function(fun) {
|
|
var node, parent;
|
|
if (this.get_child('right')) {
|
|
return this.get_child('right').min_tree(fun);
|
|
}
|
|
if (fun) {
|
|
fun(this);
|
|
}
|
|
node = this;
|
|
parent = this.parent;
|
|
while (parent && node === parent.get_child('right')) {
|
|
node = parent;
|
|
parent = node.parent;
|
|
if (fun) {
|
|
fun(parent);
|
|
}
|
|
}
|
|
return parent;
|
|
};
|
|
Node.prototype.getPrev = function(fun) {
|
|
var node, parent;
|
|
if (this.get_child('left')) {
|
|
return this.get_child('left').max_tree(fun);
|
|
}
|
|
if (fun) {
|
|
fun(this);
|
|
}
|
|
node = this;
|
|
parent = this.parent;
|
|
while (parent && node === parent.get_child('left')) {
|
|
node = parent;
|
|
parent = node.parent;
|
|
if (fun) {
|
|
fun(parent);
|
|
}
|
|
}
|
|
return parent;
|
|
};
|
|
|
|
|
|
/** Traverse the tree upwards until it reaches the top. For each node traversed,
|
|
* call the function passed as argument with arguments node, parent.
|
|
* The function will be called h-1 times, h being the height of the branch.
|
|
**/
|
|
Node.prototype.traverse_up = function(fun) {
|
|
var parent = this.parent;
|
|
var node = this;
|
|
|
|
while(parent) {
|
|
fun(node, parent);
|
|
node = parent;
|
|
parent = parent.parent;
|
|
}
|
|
};
|
|
|
|
Node.prototype.depth = function() {
|
|
var depth = 0;
|
|
this.traverse_up(function() {
|
|
depth ++;
|
|
});
|
|
return depth;
|
|
};
|
|
|
|
Node.prototype.position = function() {
|
|
var position = this.left? this.left.weight: 0;
|
|
var countFun = function(node, parent) {
|
|
if (parent.right === node) {
|
|
// for the left subtree
|
|
if (parent.left) {
|
|
position += parent.left.weight;
|
|
}
|
|
position += 1; // for the parent
|
|
}
|
|
};
|
|
|
|
this.traverse_up(countFun);
|
|
return position;
|
|
};
|
|
|
|
module.exports = RBTree;
|
|
|
|
},{"./treebase":10}],10:[function(require,module,exports){
|
|
|
|
function TreeBase() {}
|
|
|
|
// removes all nodes from the tree
|
|
TreeBase.prototype.clear = function() {
|
|
this._root = null;
|
|
this.size = 0;
|
|
};
|
|
|
|
// returns node data if found, null otherwise
|
|
TreeBase.prototype.find = function(data) {
|
|
var res = this._root;
|
|
|
|
while(res !== null) {
|
|
var c = this._comparator(data, res.data);
|
|
if(c === 0) {
|
|
return res.data;
|
|
}
|
|
else {
|
|
res = res.get_child(c > 0);
|
|
}
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
// returns iterator to node if found, null otherwise
|
|
TreeBase.prototype.findIter = function(data) {
|
|
var res = this._root;
|
|
var iter = this.iterator();
|
|
|
|
while(res !== null) {
|
|
var c = this._comparator(data, res.data);
|
|
if(c === 0) {
|
|
iter._cursor = res;
|
|
return iter;
|
|
}
|
|
else {
|
|
iter._ancestors.push(res);
|
|
res = res.get_child(c > 0);
|
|
}
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
// Returns an interator to the tree node at or immediately after the item
|
|
TreeBase.prototype.lowerBound = function(item) {
|
|
var cur = this._root;
|
|
var iter = this.iterator();
|
|
var cmp = this._comparator;
|
|
|
|
while(cur !== null) {
|
|
var c = cmp(item, cur.data);
|
|
if(c === 0) {
|
|
iter._cursor = cur;
|
|
return iter;
|
|
}
|
|
iter._ancestors.push(cur);
|
|
cur = cur.get_child(c > 0);
|
|
}
|
|
|
|
for(var i=iter._ancestors.length - 1; i >= 0; --i) {
|
|
cur = iter._ancestors[i];
|
|
if(cmp(item, cur.data) < 0) {
|
|
iter._cursor = cur;
|
|
iter._ancestors.length = i;
|
|
return iter;
|
|
}
|
|
}
|
|
|
|
iter._ancestors.length = 0;
|
|
return iter;
|
|
};
|
|
|
|
// Returns an interator to the tree node immediately after the item
|
|
TreeBase.prototype.upperBound = function(item) {
|
|
var iter = this.lowerBound(item);
|
|
var cmp = this._comparator;
|
|
|
|
while(cmp(iter.data(), item) === 0) {
|
|
iter.next();
|
|
}
|
|
|
|
return iter;
|
|
};
|
|
|
|
// returns null if tree is empty
|
|
TreeBase.prototype.min = function() {
|
|
var res = this._root;
|
|
if(res === null) {
|
|
return null;
|
|
}
|
|
|
|
while(res.left !== null) {
|
|
res = res.left;
|
|
}
|
|
|
|
return res.data;
|
|
};
|
|
|
|
// returns null if tree is empty
|
|
TreeBase.prototype.max = function() {
|
|
var res = this._root;
|
|
if(res === null) {
|
|
return null;
|
|
}
|
|
|
|
while(res.right !== null) {
|
|
res = res.right;
|
|
}
|
|
|
|
return res.data;
|
|
};
|
|
|
|
// returns a null iterator
|
|
// call next() or prev() to point to an element
|
|
TreeBase.prototype.iterator = function() {
|
|
return new Iterator(this);
|
|
};
|
|
|
|
|
|
TreeBase.prototype.eachNode = function(cb) {
|
|
var it=this.iterator(), node;
|
|
var index = 0;
|
|
while((node = it.next()) !== null) {
|
|
cb(node, index);
|
|
index++;
|
|
}
|
|
};
|
|
|
|
// calls cb on each node's data, in order
|
|
TreeBase.prototype.each = function(cb) {
|
|
this.eachNode(function(node, index) {
|
|
cb(node.data, index);
|
|
});
|
|
};
|
|
|
|
|
|
TreeBase.prototype.mapNode = function(cb) {
|
|
var it=this.iterator(), node;
|
|
var results = [];
|
|
var index = 0;
|
|
while((node = it.next()) !== null) {
|
|
results.push(cb(node, index));
|
|
index ++;
|
|
}
|
|
return results;
|
|
};
|
|
|
|
// calls cb on each node-s data, store the result and return it
|
|
TreeBase.prototype.map = function(cb) {
|
|
return this.mapNode(function(node, index) {
|
|
return cb(node.data, index);
|
|
});
|
|
};
|
|
|
|
function Iterator(tree) {
|
|
this._tree = tree;
|
|
this._ancestors = [];
|
|
this._cursor = null;
|
|
}
|
|
|
|
Iterator.prototype.data = function() {
|
|
return this._cursor !== null ? this._cursor.data : null;
|
|
};
|
|
|
|
// if null-iterator, returns first node
|
|
// otherwise, returns next node
|
|
Iterator.prototype.next = function() {
|
|
if(this._cursor === null) {
|
|
var root = this._tree._root;
|
|
if(root !== null) {
|
|
this._minNode(root);
|
|
}
|
|
}
|
|
else {
|
|
if(this._cursor.right === null) {
|
|
// no greater node in subtree, go up to parent
|
|
// if coming from a right child, continue up the stack
|
|
var save;
|
|
do {
|
|
save = this._cursor;
|
|
if(this._ancestors.length) {
|
|
this._cursor = this._ancestors.pop();
|
|
}
|
|
else {
|
|
this._cursor = null;
|
|
break;
|
|
}
|
|
} while(this._cursor.right === save);
|
|
}
|
|
else {
|
|
// get the next node from the subtree
|
|
this._ancestors.push(this._cursor);
|
|
this._minNode(this._cursor.right);
|
|
}
|
|
}
|
|
return this._cursor !== null ? this._cursor : null;
|
|
};
|
|
|
|
// if null-iterator, returns last node
|
|
// otherwise, returns previous node
|
|
Iterator.prototype.prev = function() {
|
|
if(this._cursor === null) {
|
|
var root = this._tree._root;
|
|
if(root !== null) {
|
|
this._maxNode(root);
|
|
}
|
|
}
|
|
else {
|
|
if(this._cursor.left === null) {
|
|
var save;
|
|
do {
|
|
save = this._cursor;
|
|
if(this._ancestors.length) {
|
|
this._cursor = this._ancestors.pop();
|
|
}
|
|
else {
|
|
this._cursor = null;
|
|
break;
|
|
}
|
|
} while(this._cursor.left === save);
|
|
}
|
|
else {
|
|
this._ancestors.push(this._cursor);
|
|
this._maxNode(this._cursor.left);
|
|
}
|
|
}
|
|
return this._cursor !== null ? this._cursor : null;
|
|
};
|
|
|
|
Iterator.prototype._minNode = function(start) {
|
|
while(start.left !== null) {
|
|
this._ancestors.push(start);
|
|
start = start.left;
|
|
}
|
|
this._cursor = start;
|
|
};
|
|
|
|
Iterator.prototype._maxNode = function(start) {
|
|
while(start.right !== null) {
|
|
this._ancestors.push(start);
|
|
start = start.right;
|
|
}
|
|
this._cursor = start;
|
|
};
|
|
|
|
module.exports = TreeBase;
|
|
|
|
},{}]},{},[8])
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/home/ccc/2/yjs/node_modules/gulp-browserify/node_modules/browserify/node_modules/browser-pack/_prelude.js","/home/ccc/2/yjs/lib/ConnectorAdapter.coffee","/home/ccc/2/yjs/lib/ConnectorClass.coffee","/home/ccc/2/yjs/lib/Engine.coffee","/home/ccc/2/yjs/lib/HistoryBuffer.coffee","/home/ccc/2/yjs/lib/ObjectType.coffee","/home/ccc/2/yjs/lib/Operations/Basic.coffee","/home/ccc/2/yjs/lib/Operations/Structured.coffee","/home/ccc/2/yjs/lib/y.coffee","/home/ccc/2/yjs/node_modules/bintrees/lib/rbtree_by_index.js","/home/ccc/2/yjs/node_modules/bintrees/lib/treebase.js"],"names":[],"mappings":"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,CAAC,SAAS,CAAC,SAAS,CAAC,SAAxB,GAAoC,SAjBpC,CAAA;AAAA,EAkBA,GAAG,CAAC,SAAS,CAAC,SAAS,CAAC,YAAxB,GAAuC,IAAI,CAAC,WAlB5C,CAAA;AAAA,EAoBA,EAAA,GAAS,IAAA,OAAO,CAAC,MAAR,CAAA,CApBT,CAAA;AAAA,EAqBA,KAAA,GAAY,IAAA,GAAG,CAAC,UAAJ,CAAe,EAAf,EAAmB,EAAE,CAAC,2BAAH,CAAA,CAAnB,CAAoD,CAAC,OAArD,CAAA,CArBZ,CAAA;AAAA,EAsBA,EAAE,CAAC,SAAH,CAAa,KAAb,CAtBA,CAAA;SAuBA,GAxBQ;AAAA,CANV,CAAA;;AAAA,MAgCM,CAAC,OAAP,GAAiB,OAhCjB,CAAA;;AAiCA,IAAG,gDAAH;AACE,EAAA,MAAM,CAAC,CAAP,GAAW,OAAX,CADF;CAjCA;;AAAA,OAoCO,CAAC,MAAR,GAAiB,OAAA,CAAQ,cAAR,CApCjB,CAAA;;;;ACDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7gBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error(\"Cannot find module '\"+o+\"'\")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","\nConnectorClass = require \"./ConnectorClass\"\n#\n# @param {Engine} engine The transformation engine\n# @param {HistoryBuffer} HB\n# @param {Array<Function>} execution_listener You must ensure that whenever an operation is executed, every function in this Array is called.\n#\nadaptConnector = (connector, engine, HB, execution_listener)->\n\n  for name, f of ConnectorClass\n    connector[name] = f\n\n  connector.setIsBoundToY()\n\n  send_ = (o)->\n    if (o.uid.creator is HB.getUserId()) and\n        (typeof o.uid.op_number isnt \"string\") and # TODO: i don't think that we need this anymore..\n        (HB.getUserId() isnt \"_temp\")\n      connector.broadcast o\n\n  if connector.invokeSync?\n    HB.setInvokeSyncHandler connector.invokeSync\n\n  execution_listener.push send_\n  # For the XMPPConnector: lets send it as an array\n  # therefore, we have to restructure it later\n  encode_state_vector = (v)->\n    for name,value of v\n      user: name\n      state: value\n  parse_state_vector = (v)->\n    state_vector = {}\n    for s in v\n      state_vector[s.user] = s.state\n    state_vector\n\n  getStateVector = ()->\n    encode_state_vector HB.getOperationCounter()\n\n  getHB = (v)->\n    state_vector = parse_state_vector v\n    hb = HB._encode state_vector\n    json =\n      hb: hb\n      state_vector: encode_state_vector HB.getOperationCounter()\n    json\n\n  applyHB = (hb, fromHB)->\n    engine.applyOp hb, fromHB\n\n  connector.getStateVector = getStateVector\n  connector.getHB = getHB\n  connector.applyHB = applyHB\n\n  connector.receive_handlers ?= []\n  connector.receive_handlers.push (sender, op)->\n    if op.uid.creator isnt HB.getUserId()\n      engine.applyOp op\n\n\nmodule.exports = adaptConnector\n","\nmodule.exports =\n  #\n  # @params new Connector(options)\n  #   @param options.syncMethod {String}  is either \"syncAll\" or \"master-slave\".\n  #   @param options.role {String} The role of this client\n  #            (slave or master (only used when syncMethod is master-slave))\n  #   @param options.perform_send_again {Boolean} Whetehr to whether to resend the HB after some time period. This reduces sync errors, but has some overhead (optional)\n  #\n  init: (options)->\n    req = (name, choices)=>\n      if options[name]?\n        if (not choices?) or choices.some((c)->c is options[name])\n          @[name] = options[name]\n        else\n          throw new Error \"You can set the '\"+name+\"' option to one of the following choices: \"+JSON.encode(choices)\n      else\n        throw new Error \"You must specify \"+name+\", when initializing the Connector!\"\n\n    req \"syncMethod\", [\"syncAll\", \"master-slave\"]\n    req \"role\", [\"master\", \"slave\"]\n    req \"user_id\"\n    @on_user_id_set?(@user_id)\n\n    # whether to resend the HB after some time period. This reduces sync errors.\n    # But this is not necessary in the test-connector\n    if options.perform_send_again?\n      @perform_send_again = options.perform_send_again\n    else\n      @perform_send_again = true\n\n    # A Master should sync with everyone! TODO: really? - for now its safer this way!\n    if @role is \"master\"\n      @syncMethod = \"syncAll\"\n\n    # is set to true when this is synced with all other connections\n    @is_synced = false\n    # Peerjs Connections: key: conn-id, value: object\n    @connections = {}\n    # List of functions that shall process incoming data\n    @receive_handlers ?= []\n\n    # whether this instance is bound to any y instance\n    @connections = {}\n    @current_sync_target = null\n    @sent_hb_to_all_users = false\n    @is_initialized = true\n\n  onUserEvent: (f)->\n    @connections_listeners ?= []\n    @connections_listeners.push f\n\n  isRoleMaster: ->\n    @role is \"master\"\n\n  isRoleSlave: ->\n    @role is \"slave\"\n\n  findNewSyncTarget: ()->\n    @current_sync_target = null\n    if @syncMethod is \"syncAll\"\n      for user, c of @connections\n        if not c.is_synced\n          @performSync user\n          break\n    if not @current_sync_target?\n      @setStateSynced()\n    null\n\n  userLeft: (user)->\n    delete @connections[user]\n    @findNewSyncTarget()\n    if @connections_listeners?\n      for f in @connections_listeners\n        f {\n          action: \"userLeft\"\n          user: user\n        }\n\n\n  userJoined: (user, role)->\n    if not role?\n      throw new Error \"Internal: You must specify the role of the joined user! E.g. userJoined('uid:3939','slave')\"\n    # a user joined the room\n    @connections[user] ?= {}\n    @connections[user].is_synced = false\n\n    if (not @is_synced) or @syncMethod is \"syncAll\"\n      if @syncMethod is \"syncAll\"\n        @performSync user\n      else if role is \"master\"\n        # TODO: What if there are two masters? Prevent sending everything two times!\n        @performSyncWithMaster user\n\n    if @connections_listeners?\n      for f in @connections_listeners\n        f {\n          action: \"userJoined\"\n          user: user\n          role: role\n        }\n\n  #\n  # Execute a function _when_ we are connected. If not connected, wait until connected.\n  # @param f {Function} Will be executed on the Connector context.\n  #\n  whenSynced: (args)->\n    if args.constructor is Function\n      args = [args]\n    if @is_synced\n      args[0].apply this, args[1..]\n    else\n      @compute_when_synced ?= []\n      @compute_when_synced.push args\n\n  #\n  # Execute an function when a message is received.\n  # @param f {Function} Will be executed on the PeerJs-Connector context. f will be called with (sender_id, broadcast {true|false}, message).\n  #\n  onReceive: (f)->\n    @receive_handlers.push f\n\n  ###\n  # Broadcast a message to all connected peers.\n  # @param message {Object} The message to broadcast.\n  #\n  broadcast: (message)->\n    throw new Error \"You must implement broadcast!\"\n\n  #\n  # Send a message to a peer, or set of peers\n  #\n  send: (peer_s, message)->\n    throw new Error \"You must implement send!\"\n  ###\n\n  #\n  # perform a sync with a specific user.\n  #\n  performSync: (user)->\n    if not @current_sync_target?\n      @current_sync_target = user\n      @send user,\n        sync_step: \"getHB\"\n        send_again: \"true\"\n        data: @getStateVector()\n      if not @sent_hb_to_all_users\n        @sent_hb_to_all_users = true\n\n        hb = @getHB([]).hb\n        _hb = []\n        for o in hb\n          _hb.push o\n          if _hb.length > 10\n            @broadcast\n              sync_step: \"applyHB_\"\n              data: _hb\n            _hb = []\n        @broadcast\n          sync_step: \"applyHB\"\n          data: _hb\n\n\n\n  #\n  # When a master node joined the room, perform this sync with him. It will ask the master for the HB,\n  # and will broadcast his own HB\n  #\n  performSyncWithMaster: (user)->\n    @current_sync_target = user\n    @send user,\n      sync_step: \"getHB\"\n      send_again: \"true\"\n      data: @getStateVector()\n    hb = @getHB([]).hb\n    _hb = []\n    for o in hb\n      _hb.push o\n      if _hb.length > 10\n        @broadcast\n          sync_step: \"applyHB_\"\n          data: _hb\n        _hb = []\n    @broadcast\n      sync_step: \"applyHB\"\n      data: _hb\n\n  #\n  # You are sure that all clients are synced, call this function.\n  #\n  setStateSynced: ()->\n    if not @is_synced\n      @is_synced = true\n      if @compute_when_synced?\n        for el in @compute_when_synced\n          f = el[0]\n          args = el[1..]\n          f.apply(args)\n        delete @compute_when_synced\n      null\n\n  # executed when the a state_vector is received. listener will be called only once!\n  whenReceivedStateVector: (f)->\n    @when_received_state_vector_listeners ?= []\n    @when_received_state_vector_listeners.push f\n\n\n  #\n  # You received a raw message, and you know that it is intended for to Yjs. Then call this function.\n  #\n  receiveMessage: (sender, res)->\n    if not res.sync_step?\n      for f in @receive_handlers\n        f sender, res\n    else\n      if sender is @user_id\n        return\n      if res.sync_step is \"getHB\"\n        # call listeners\n        if @when_received_state_vector_listeners?\n          for f in @when_received_state_vector_listeners\n            f.call this, res.data\n        delete @when_received_state_vector_listeners\n\n        data = @getHB(res.data)\n        hb = data.hb\n        _hb = []\n        # always broadcast, when not synced.\n        # This reduces errors, when the clients goes offline prematurely.\n        # When this client only syncs to one other clients, but looses connectors,\n        # before syncing to the other clients, the online clients have different states.\n        # Since we do not want to perform regular syncs, this is a good alternative\n        if @is_synced\n          sendApplyHB = (m)=>\n            @send sender, m\n        else\n          sendApplyHB = (m)=>\n            @broadcast m\n\n        for o in hb\n          _hb.push o\n          if _hb.length > 10\n            sendApplyHB\n              sync_step: \"applyHB_\"\n              data: _hb\n            _hb = []\n\n        sendApplyHB\n          sync_step : \"applyHB\"\n          data: _hb\n\n        if res.send_again? and @perform_send_again\n          send_again = do (sv = data.state_vector)=>\n            ()=>\n              hb = @getHB(sv).hb\n              for o in hb\n                _hb.push o\n                if _hb.length > 10\n                  @send sender,\n                    sync_step: \"applyHB_\"\n                    data: _hb\n                  _hb = []\n              @send sender,\n                sync_step: \"applyHB\",\n                data: _hb\n                sent_again: \"true\"\n          setTimeout send_again, 3000\n      else if res.sync_step is \"applyHB\"\n        @applyHB(res.data, sender is @current_sync_target)\n\n        if (@syncMethod is \"syncAll\" or res.sent_again?) and (not @is_synced) and ((@current_sync_target is sender) or (not @current_sync_target?))\n          @connections[sender].is_synced = true\n          @findNewSyncTarget()\n\n      else if res.sync_step is \"applyHB_\"\n        @applyHB(res.data, sender is @current_sync_target)\n\n\n  # Currently, the HB encodes operations as JSON. For the moment I want to keep it\n  # that way. Maybe we support encoding in the HB as XML in the future, but for now I don't want\n  # too much overhead. Y is very likely to get changed a lot in the future\n  #\n  # Because we don't want to encode JSON as string (with character escaping, wich makes it pretty much unreadable)\n  # we encode the JSON as XML.\n  #\n  # When the HB support encoding as XML, the format should look pretty much like this.\n\n  # does not support primitive values as array elements\n  # expects an ltx (less than xml) object\n  parseMessageFromXml: (m)->\n    parse_array = (node)->\n      for n in node.children\n        if n.getAttribute(\"isArray\") is \"true\"\n          parse_array n\n        else\n          parse_object n\n\n    parse_object = (node)->\n      json = {}\n      for name, value  of node.attrs\n        int = parseInt(value)\n        if isNaN(int) or (\"\"+int) isnt value\n          json[name] = value\n        else\n          json[name] = int\n      for n in node.children\n        name = n.name\n        if n.getAttribute(\"isArray\") is \"true\"\n          json[name] = parse_array n\n        else\n          json[name] = parse_object n\n      json\n    parse_object m\n\n  # encode message in xml\n  # we use string because Strophe only accepts an \"xml-string\"..\n  # So {a:4,b:{c:5}} will look like\n  # <y a=\"4\">\n  #   <b c=\"5\"></b>\n  # </y>\n  # m - ltx element\n  # json - guess it ;)\n  #\n  encodeMessageToXml: (m, json)->\n    # attributes is optional\n    encode_object = (m, json)->\n      for name,value of json\n        if not value?\n          # nop\n        else if value.constructor is Object\n          encode_object m.c(name), value\n        else if value.constructor is Array\n          encode_array m.c(name), value\n        else\n          m.setAttribute(name,value)\n      m\n    encode_array = (m, array)->\n      m.setAttribute(\"isArray\",\"true\")\n      for e in array\n        if e.constructor is Object\n          encode_object m.c(\"array-element\"), e\n        else\n          encode_array m.c(\"array-element\"), e\n      m\n    if json.constructor is Object\n      encode_object m.c(\"y\",{xmlns:\"http://y.ninja/connector-stanza\"}), json\n    else if json.constructor is Array\n      encode_array m.c(\"y\",{xmlns:\"http://y.ninja/connector-stanza\"}), json\n    else\n      throw new Error \"I can't encode this json!\"\n\n  setIsBoundToY: ()->\n    @on_bound_to_y?()\n    delete @when_bound_to_y\n    @is_bound_to_y = true\n","\nwindow?.unprocessed_counter = 0 # del this\nwindow?.unprocessed_exec_counter = 0 # TODO\nwindow?.unprocessed_types = []\n\n#\n# @nodoc\n# The Engine handles how and in which order to execute operations and add operations to the HistoryBuffer.\n#\nclass Engine\n\n  #\n  # @param {HistoryBuffer} HB\n  # @param {Object} types list of available types\n  #\n  constructor: (@HB, @types)->\n    @unprocessed_ops = []\n\n  #\n  # Parses an operatio from the json format. It uses the specified parser in your OperationType module.\n  #\n  parseOperation: (json)->\n    type = @types[json.type]\n    if type?.parse?\n      type.parse json\n    else\n      throw new Error \"You forgot to specify a parser for type #{json.type}. The message is #{JSON.stringify json}.\"\n\n\n  #\n  # Apply a set of operations. E.g. the operations you received from another users HB._encode().\n  # @note You must not use this method when you already have ops in your HB!\n  ###\n  applyOpsBundle: (ops_json)->\n    ops = []\n    for o in ops_json\n      ops.push @parseOperation o\n    for o in ops\n      if not o.execute()\n        @unprocessed_ops.push o\n    @tryUnprocessed()\n  ###\n\n  #\n  # Same as applyOps but operations that are already in the HB are not applied.\n  # @see Engine.applyOps\n  #\n  applyOpsCheckDouble: (ops_json)->\n    for o in ops_json\n      if not @HB.getOperation(o.uid)?\n        @applyOp o\n\n  #\n  # Apply a set of operations. (Helper for using applyOp on Arrays)\n  # @see Engine.applyOp\n  applyOps: (ops_json)->\n    @applyOp ops_json\n\n  #\n  # Apply an operation that you received from another peer.\n  # TODO: make this more efficient!!\n  # - operations may only executed in order by creator, order them in object of arrays (key by creator)\n  # - you can probably make something like dependencies (creator1 waits for creator2)\n  applyOp: (op_json_array, fromHB = false)->\n    if op_json_array.constructor isnt Array\n      op_json_array = [op_json_array]\n    for op_json in op_json_array\n      if fromHB\n        op_json.fromHB = \"true\" # execute immediately, if\n      # $parse_and_execute will return false if $o_json was parsed and executed, otherwise the parsed operadion\n      o = @parseOperation op_json\n      o.parsed_from_json = op_json\n      if op_json.fromHB?\n        o.fromHB = op_json.fromHB\n      # @HB.addOperation o\n      if @HB.getOperation(o)?\n        # nop\n      else if ((not @HB.isExpectedOperation(o)) and (not o.fromHB?)) or (not o.execute())\n        @unprocessed_ops.push o\n        window?.unprocessed_types.push o.type # TODO: delete this\n    @tryUnprocessed()\n\n  #\n  # Call this method when you applied a new operation.\n  # It checks if operations that were previously not executable are now executable.\n  #\n  tryUnprocessed: ()->\n    while true\n      old_length = @unprocessed_ops.length\n      unprocessed = []\n      for op in @unprocessed_ops\n        if @HB.getOperation(op)?\n          # nop\n        else if (not @HB.isExpectedOperation(op) and (not op.fromHB?)) or (not op.execute())\n          unprocessed.push op\n      @unprocessed_ops = unprocessed\n      if @unprocessed_ops.length is old_length\n        break\n    if @unprocessed_ops.length isnt 0\n      @HB.invokeSync()\n\n\nmodule.exports = Engine\n\n\n\n\n\n\n\n\n\n\n\n\n","\n#\n# @nodoc\n# An object that holds all applied operations.\n#\n# @note The HistoryBuffer is commonly abbreviated to HB.\n#\nclass HistoryBuffer\n\n  #\n  # Creates an empty HB.\n  # @param {Object} user_id Creator of the HB.\n  #\n  constructor: (@user_id)->\n    @operation_counter = {}\n    @buffer = {}\n    @change_listeners = []\n    @garbage = [] # Will be cleaned on next call of garbageCollector\n    @trash = [] # Is deleted. Wait until it is not used anymore.\n    @performGarbageCollection = true\n    @garbageCollectTimeout = 30000\n    @reserved_identifier_counter = 0\n    setTimeout @emptyGarbage, @garbageCollectTimeout\n\n  # At the beginning (when the user id was not assigned yet),\n  # the operations are added to buffer._temp. When you finally get your user id,\n  # the operations are copies from buffer._temp to buffer[id]. Furthermore, when buffer[id] does already contain operations\n  # (because of a previous session), the uid.op_numbers of the operations have to be reassigned.\n  # This is what this function does. It adds them to buffer[id],\n  # and assigns them the correct uid.op_number and uid.creator\n  setUserId: (@user_id, state_vector)->\n    @buffer[@user_id] ?= []\n    buff = @buffer[@user_id]\n\n    # we assumed that we started with counter = 0.\n    # when we receive tha state_vector, and actually have\n    # counter = 10. Then we have to add 10 to every op_counter\n    counter_diff = state_vector[@user_id] or 0\n\n    if @buffer._temp?\n      for o_name,o of @buffer._temp\n        o.uid.creator = @user_id\n        o.uid.op_number += counter_diff\n        buff[o.uid.op_number] = o\n\n    @operation_counter[@user_id] = (@operation_counter._temp or 0) + counter_diff\n\n    delete @operation_counter._temp\n    delete @buffer._temp\n\n\n  emptyGarbage: ()=>\n    for o in @garbage\n      #if @getOperationCounter(o.uid.creator) > o.uid.op_number\n      o.cleanup?()\n\n    @garbage = @trash\n    @trash = []\n    if @garbageCollectTimeout isnt -1\n      @garbageCollectTimeoutId = setTimeout @emptyGarbage, @garbageCollectTimeout\n    undefined\n\n  #\n  # Get the user id with wich the History Buffer was initialized.\n  #\n  getUserId: ()->\n    @user_id\n\n  addToGarbageCollector: ()->\n    if @performGarbageCollection\n      for o in arguments\n        if o?\n          @garbage.push o\n\n  stopGarbageCollection: ()->\n    @performGarbageCollection = false\n    @setManualGarbageCollect()\n    @garbage = []\n    @trash = []\n\n  setManualGarbageCollect: ()->\n    @garbageCollectTimeout = -1\n    clearTimeout @garbageCollectTimeoutId\n    @garbageCollectTimeoutId = undefined\n\n  setGarbageCollectTimeout: (@garbageCollectTimeout)->\n\n  #\n  # I propose to use it in your Framework, to create something like a root element.\n  # An operation with this identifier is not propagated to other clients.\n  # This is why everybode must create the same operation with this uid.\n  #\n  getReservedUniqueIdentifier: ()->\n    {\n      creator : '_'\n      op_number : \"_#{@reserved_identifier_counter++}\"\n    }\n\n  #\n  # Get the operation counter that describes the current state of the document.\n  #\n  getOperationCounter: (user_id)->\n    if not user_id?\n      res = {}\n      for user,ctn of @operation_counter\n        res[user] = ctn\n      res\n    else\n      @operation_counter[user_id]\n\n  isExpectedOperation: (o)->\n    @operation_counter[o.uid.creator] ?= 0\n    o.uid.op_number <= @operation_counter[o.uid.creator]\n    true #TODO: !! this could break stuff. But I dunno why\n\n  #\n  # Encode this operation in such a way that it can be parsed by remote peers.\n  # TODO: Make this more efficient!\n  _encode: (state_vector={})->\n    json = []\n    unknown = (user, o_number)->\n      if (not user?) or (not o_number?)\n        throw new Error \"dah!\"\n      not state_vector[user]? or state_vector[user] <= o_number\n\n    for u_name,user of @buffer\n      # TODO next, if @state_vector[user] <= state_vector[user]\n      if u_name is \"_\"\n        continue\n      for o_number,o of user\n        if (not o.uid.noOperation?) and unknown(u_name, o_number)\n          # its necessary to send it, and not known in state_vector\n          o_json = o._encode()\n          if o.next_cl? # applies for all ops but the most right delimiter!\n            # search for the next _known_ operation. (When state_vector is {} then this is the Delimiter)\n            o_next = o.next_cl\n            while o_next.next_cl? and unknown(o_next.uid.creator, o_next.uid.op_number)\n              o_next = o_next.next_cl\n            o_json.next = o_next.getUid()\n          else if o.prev_cl? # most right delimiter only!\n            # same as the above with prev.\n            o_prev = o.prev_cl\n            while o_prev.prev_cl? and unknown(o_prev.uid.creator, o_prev.uid.op_number)\n              o_prev = o_prev.prev_cl\n            o_json.prev = o_prev.getUid()\n          json.push o_json\n\n    json\n\n  #\n  # Get the number of operations that were created by a user.\n  # Accordingly you will get the next operation number that is expected from that user.\n  # This will increment the operation counter.\n  #\n  getNextOperationIdentifier: (user_id)->\n    if not user_id?\n      user_id = @user_id\n    if not @operation_counter[user_id]?\n      @operation_counter[user_id] = 0\n    uid =\n      'creator' : user_id\n      'op_number' : @operation_counter[user_id]\n    @operation_counter[user_id]++\n    uid\n\n  #\n  # Retrieve an operation from a unique id.\n  #\n  # when uid has a \"sub\" property, the value of it will be applied\n  # on the operations retrieveSub method (which must! be defined)\n  #\n  getOperation: (uid)->\n    if uid.uid?\n      uid = uid.uid\n    o = @buffer[uid.creator]?[uid.op_number]\n    if uid.sub? and o?\n      o.retrieveSub uid.sub\n    else\n      o\n\n  #\n  # Add an operation to the HB. Note that this will not link it against\n  # other operations (it wont executed)\n  #\n  addOperation: (o)->\n    if not @buffer[o.uid.creator]?\n      @buffer[o.uid.creator] = {}\n    if @buffer[o.uid.creator][o.uid.op_number]?\n      throw new Error \"You must not overwrite operations!\"\n    if (o.uid.op_number.constructor isnt String) and (not @isExpectedOperation(o)) and (not o.fromHB?) # you already do this in the engine, so delete it here!\n      throw new Error \"this operation was not expected!\"\n    @addToCounter(o)\n    @buffer[o.uid.creator][o.uid.op_number] = o\n    o\n\n  removeOperation: (o)->\n    delete @buffer[o.uid.creator]?[o.uid.op_number]\n\n  # When the HB determines inconsistencies, then the invokeSync\n  # handler wil be called, which should somehow invoke the sync with another collaborator.\n  # The parameter of the sync handler is the user_id with wich an inconsistency was determined\n  setInvokeSyncHandler: (f)->\n    @invokeSync = f\n\n  # empty per default # TODO: do i need this?\n  invokeSync: ()->\n\n  # after you received the HB of another user (in the sync process),\n  # you renew your own state_vector to the state_vector of the other user\n  renewStateVector: (state_vector)->\n    for user,state of state_vector\n      if ((not @operation_counter[user]?) or (@operation_counter[user] < state_vector[user])) and state_vector[user]?\n        @operation_counter[user] = state_vector[user]\n\n  #\n  # Increment the operation_counter that defines the current state of the Engine.\n  #\n  addToCounter: (o)->\n    @operation_counter[o.uid.creator] ?= 0\n    # TODO: check if operations are send in order\n    if o.uid.op_number is @operation_counter[o.uid.creator]\n      @operation_counter[o.uid.creator]++\n    while @buffer[o.uid.creator][@operation_counter[o.uid.creator]]?\n      @operation_counter[o.uid.creator]++\n    undefined\n\nmodule.exports = HistoryBuffer\n","\nclass YObject\n\n  constructor: (@_object = {})->\n    if @_object.constructor is Object\n      for name, val of @_object\n        if val.constructor is Object\n          @_object[name] = new YObject(val)\n    else\n      throw new Error \"Y.Object accepts Json Objects only\"\n\n  _name: \"Object\"\n\n  _getModel: (types, ops)->\n    if not @_model?\n      @_model = new ops.MapManager(@).execute()\n      for n,o of @_object\n        @_model.val n, o\n    delete @_object\n    @_model\n\n  _setModel: (@_model)->\n    delete @_object\n\n  observe: (f)->\n    @_model.observe f\n    @\n\n  unobserve: (f)->\n    @_model.unobserve f\n    @\n\n  #\n  # @overload val()\n  #   Get this as a Json object.\n  #   @return [Json]\n  #\n  # @overload val(name)\n  #   Get value of a property.\n  #   @param {String} name Name of the object property.\n  #   @return [*] Depends on the value of the property.\n  #\n  # @overload val(name, content)\n  #   Set a new property.\n  #   @param {String} name Name of the object property.\n  #   @param {Object|String} content Content of the object property.\n  #   @return [Object Type] This object. (supports chaining)\n  #\n  val: (name, content)->\n    if @_model?\n      @_model.val.apply @_model, arguments\n    else\n      if content?\n        @_object[name] = content\n      else if name?\n        @_object[name]\n      else\n        res = {}\n        for n,v of @_object\n          res[n] = v\n        res\n\n  delete: (name)->\n    @_model.delete(name)\n    @\n\nif window?\n  if window.Y?\n    window.Y.Object = YObject\n  else\n    throw new Error \"You must first import Y!\"\n\nif module?\n  module.exports = YObject\n","module.exports = ()->\n  # @see Engine.parse\n  ops = {}\n  execution_listener = []\n\n  #\n  # @private\n  # @abstract\n  # @nodoc\n  # A generic interface to ops.\n  #\n  # An operation has the following methods:\n  # * _encode: encodes an operation (needed only if instance of this operation is sent).\n  # * execute: execute the effects of this operations. Good examples are Insert-type and AddName-type\n  # * val: in the case that the operation holds a value\n  #\n  # Furthermore an encodable operation has a parser. We extend the parser object in order to parse encoded operations.\n  #\n  class ops.Operation\n\n    #\n    # @param {Object} uid A unique identifier.\n    # If uid is undefined, a new uid will be created before at the end of the execution sequence\n    #\n    constructor: (custom_type, uid, content, content_operations)->\n      if custom_type?\n        @custom_type = custom_type\n      @is_deleted = false\n      @garbage_collected = false\n      @event_listeners = [] # TODO: rename to observers or sth like that\n      if uid?\n        @uid = uid\n\n      # see encode to see, why we are doing it this way\n      if content is undefined\n        # nop\n      else if content? and content.creator?\n        @saveOperation 'content', content\n      else\n        @content = content\n      if content_operations?\n        @content_operations = {}\n        for name, op of content_operations\n          @saveOperation name, op, 'content_operations'\n\n    type: \"Operation\"\n\n    getContent: (name)->\n      if @content?\n        if @content.getCustomType?\n          @content.getCustomType()\n        else if @content.constructor is Object\n          if name?\n            if @content[name]?\n              @content[name]\n            else\n              @content_operations[name].getCustomType()\n          else\n            content = {}\n            for n,v of @content\n              content[n] = v\n            if @content_operations?\n              for n,v of @content_operations\n                v = v.getCustomType()\n                content[n] = v\n            content\n        else\n          @content\n      else\n        @content\n\n    retrieveSub: ()->\n      throw new Error \"sub properties are not enable on this operation type!\"\n\n    #\n    # Add an event listener. It depends on the operation which events are supported.\n    # @param {Function} f f is executed in case the event fires.\n    #\n    observe: (f)->\n      @event_listeners.push f\n\n    #\n    # Deletes function from the observer list\n    # @see Operation.observe\n    #\n    # @overload unobserve(event, f)\n    #   @param f     {Function} The function that you want to delete\n    unobserve: (f)->\n      @event_listeners = @event_listeners.filter (g)->\n        f isnt g\n\n    #\n    # Deletes all subscribed event listeners.\n    # This should be called, e.g. after this has been replaced.\n    # (Then only one replace event should fire. )\n    # This is also called in the cleanup method.\n    deleteAllObservers: ()->\n      @event_listeners = []\n\n    delete: ()->\n      (new ops.Delete undefined, @).execute()\n      null\n\n    #\n    # Fire an event.\n    # TODO: Do something with timeouts. You don't want this to fire for every operation (e.g. insert).\n    # TODO: do you need callEvent+forwardEvent? Only one suffices probably\n    callEvent: ()->\n      if @custom_type?\n        callon = @getCustomType()\n      else\n        callon = @\n      @forwardEvent callon, arguments...\n\n    #\n    # Fire an event and specify in which context the listener is called (set 'this').\n    # TODO: do you need this ?\n    forwardEvent: (op, args...)->\n      for f in @event_listeners\n        f.call op, args...\n\n    isDeleted: ()->\n      @is_deleted\n\n    applyDelete: (garbagecollect = true)->\n      if not @garbage_collected\n        #console.log \"applyDelete: #{@type}\"\n        @is_deleted = true\n        if garbagecollect\n          @garbage_collected = true\n          @HB.addToGarbageCollector @\n\n    cleanup: ()->\n      #console.log \"cleanup: #{@type}\"\n      @HB.removeOperation @\n      @deleteAllObservers()\n\n    #\n    # Set the parent of this operation.\n    #\n    setParent: (@parent)->\n\n    #\n    # Get the parent of this operation.\n    #\n    getParent: ()->\n      @parent\n\n    #\n    # Computes a unique identifier (uid) that identifies this operation.\n    #\n    getUid: ()->\n      if not @uid.noOperation?\n        @uid\n      else\n        if @uid.alt? # could be (safely) undefined\n          map_uid = @uid.alt.cloneUid()\n          map_uid.sub = @uid.sub\n          map_uid\n        else\n          undefined\n\n    cloneUid: ()->\n      uid = {}\n      for n,v of @getUid()\n        uid[n] = v\n      uid\n\n    #\n    # @private\n    # If not already done, set the uid\n    # Add this to the HB\n    # Notify the all the listeners.\n    #\n    execute: ()->\n      if @validateSavedOperations()\n        @is_executed = true\n        if not @uid?\n          # When this operation was created without a uid, then set it here.\n          # There is only one other place, where this can be done - before an Insertion\n          # is executed (because we need the creator_id)\n          @uid = @HB.getNextOperationIdentifier()\n        if not @uid.noOperation?\n          @HB.addOperation @\n          for l in execution_listener\n            l @_encode()\n        @\n      else\n        false\n\n    #\n    # @private\n    # Operations may depend on other operations (linked lists, etc.).\n    # The saveOperation and validateSavedOperations methods provide\n    # an easy way to refer to these operations via an uid or object reference.\n    #\n    # For example: We can create a new Delete operation that deletes the operation $o like this\n    #     - var d = new Delete(uid, $o);   or\n    #     - var d = new Delete(uid, $o.getUid());\n    # Either way we want to access $o via d.deletes. In the second case validateSavedOperations must be called first.\n    #\n    # @overload saveOperation(name, op_uid)\n    #   @param {String} name The name of the operation. After validating (with validateSavedOperations) the instantiated operation will be accessible via this[name].\n    #   @param {Object} op_uid A uid that refers to an operation\n    # @overload saveOperation(name, op)\n    #   @param {String} name The name of the operation. After calling this function op is accessible via this[name].\n    #   @param {Operation} op An Operation object\n    #\n    saveOperation: (name, op, base = \"this\")->\n      if op? and op._getModel?\n        op = op._getModel(@custom_types, @operations)\n      #\n      # Every instance of $Operation must have an $execute function.\n      # We use duck-typing to check if op is instantiated since there\n      # could exist multiple classes of $Operation\n      #\n      if not op?\n        # nop\n      else if op.execute? or not (op.op_number? and op.creator?)\n        # is instantiated, or op is string. Currently \"Delimiter\" is saved as string\n        # (in combination with @parent you can retrieve the delimiter..)\n        if base is \"this\"\n          @[name] = op\n        else\n          dest = @[base]\n          paths = name.split(\"/\")\n          last_path = paths.pop()\n          for path in paths\n            dest = dest[path]\n          dest[last_path] = op\n      else\n        # not initialized. Do it when calling $validateSavedOperations()\n        @unchecked ?= {}\n        @unchecked[base] ?= {}\n        @unchecked[base][name] = op\n\n    #\n    # @private\n    # After calling this function all not instantiated operations will be accessible.\n    # @see Operation.saveOperation\n    #\n    # @return [Boolean] Whether it was possible to instantiate all operations.\n    #\n    validateSavedOperations: ()->\n      uninstantiated = {}\n      success = true\n      for base_name, base of @unchecked\n        for name, op_uid of base\n          op = @HB.getOperation op_uid\n          if op\n            if base_name is \"this\"\n              @[name] = op\n            else\n              dest = @[base_name]\n              paths = name.split(\"/\")\n              last_path = paths.pop()\n              for path in paths\n                dest = dest[path]\n              dest[last_path] = op\n          else\n            uninstantiated[base_name] ?= {}\n            uninstantiated[base_name][name] = op_uid\n            success = false\n      if not success\n        @unchecked = uninstantiated\n        return false\n      else\n        delete @unchecked\n        return @\n\n    getCustomType: ()->\n      if not @custom_type?\n        # throw new Error \"This operation was not initialized with a custom type\"\n        @\n      else\n        if @custom_type.constructor is String\n          # has not been initialized yet (only the name is specified)\n          Type = @custom_types\n          for t in @custom_type.split(\".\")\n            Type = Type[t]\n          @custom_type = new Type()\n          @custom_type._setModel @\n        @custom_type\n\n    #\n    # @private\n    # Encode this operation in such a way that it can be parsed by remote peers.\n    #\n    _encode: (json = {})->\n      json.type = @type\n      json.uid = @getUid()\n      if @custom_type?\n        if @custom_type.constructor is String\n          json.custom_type = @custom_type\n        else\n          json.custom_type = @custom_type._name\n\n      if @content?.getUid?\n        json.content = @content.getUid()\n      else\n        json.content = @content\n      if @content_operations?\n        operations = {}\n        for n,o of @content_operations\n          if o._getModel?\n            o = o._getModel(@custom_types, @operations)\n          operations[n] = o.getUid()\n        json.content_operations = operations\n      json\n\n  #\n  # @nodoc\n  # A simple Delete-type operation that deletes an operation.\n  #\n  class ops.Delete extends ops.Operation\n\n    #\n    # @param {Object} uid A unique identifier. If uid is undefined, a new uid will be created.\n    # @param {Object} deletes UID or reference of the operation that this to be deleted.\n    #\n    constructor: (custom_type, uid, deletes)->\n      @saveOperation 'deletes', deletes\n      super custom_type, uid\n\n    type: \"Delete\"\n\n    #\n    # @private\n    # Convert all relevant information of this operation to the json-format.\n    # This result can be sent to other clients.\n    #\n    _encode: ()->\n      {\n        'type': \"Delete\"\n        'uid': @getUid()\n        'deletes': @deletes.getUid()\n      }\n\n    #\n    # @private\n    # Apply the deletion.\n    #\n    execute: ()->\n      if @validateSavedOperations()\n        res = super\n        if res\n          @deletes.applyDelete @\n        res\n      else\n        false\n\n  #\n  # Define how to parse Delete operations.\n  #\n  ops.Delete.parse = (o)->\n    {\n      'uid' : uid\n      'deletes': deletes_uid\n    } = o\n    new this(null, uid, deletes_uid)\n\n  #\n  # @nodoc\n  # A simple insert-type operation.\n  #\n  # An insert operation is always positioned between two other insert operations.\n  # Internally this is realized as associative lists, whereby each insert operation has a predecessor and a successor.\n  # For the sake of efficiency we maintain two lists:\n  #   - The short-list (abbrev. sl) maintains only the operations that are not deleted (unimplemented, good idea?)\n  #   - The complete-list (abbrev. cl) maintains all operations\n  #\n  class ops.Insert extends ops.Operation\n\n    #\n    # @param {Object} uid A unique identifier. If uid is undefined, a new uid will be created.\n    # @param {Operation} prev_cl The predecessor of this operation in the complete-list (cl)\n    # @param {Operation} next_cl The successor of this operation in the complete-list (cl)\n    #\n    constructor: (custom_type, content, content_operations, parent, uid, prev_cl, next_cl, origin)->\n      @saveOperation 'parent', parent\n      @saveOperation 'prev_cl', prev_cl\n      @saveOperation 'next_cl', next_cl\n      if origin?\n        @saveOperation 'origin', origin\n      else\n        @saveOperation 'origin', prev_cl\n      super custom_type, uid, content, content_operations\n\n    type: \"Insert\"\n\n    val: ()->\n      @getContent()\n\n    getNext: (i=1)->\n      n = @\n      while i > 0 and n.next_cl?\n        n = n.next_cl\n        if not n.is_deleted\n          i--\n      if n.is_deleted\n        null\n      n\n\n    getPrev: (i=1)->\n      n = @\n      while i > 0 and n.prev_cl?\n        n = n.prev_cl\n        if not n.is_deleted\n          i--\n      if n.is_deleted\n        null\n      else\n        n\n\n    #\n    # set content to null and other stuff\n    # @private\n    #\n    applyDelete: (o)->\n      @deleted_by ?= []\n      callLater = false\n      if @parent? and not @is_deleted and o? # o? : if not o?, then the delimiter deleted this Insertion. Furthermore, it would be wrong to call it. TODO: make this more expressive and save\n        # call iff wasn't deleted earlyer\n        callLater = true\n      if o?\n        @deleted_by.push o\n      garbagecollect = false\n      if @next_cl.isDeleted()\n        garbagecollect = true\n      super garbagecollect\n      if callLater\n        @parent.callOperationSpecificDeleteEvents(this, o)\n      if @prev_cl? and @prev_cl.isDeleted() and @prev_cl.garbage_collected isnt true\n        # garbage collect prev_cl\n        @prev_cl.applyDelete()\n\n    cleanup: ()->\n      if @next_cl.isDeleted()\n        # delete all ops that delete this insertion\n        for d in @deleted_by\n          d.cleanup()\n\n        # throw new Error \"right is not deleted. inconsistency!, wrararar\"\n        # change origin references to the right\n        o = @next_cl\n        while o.type isnt \"Delimiter\"\n          if o.origin is @\n            o.origin = @prev_cl\n          o = o.next_cl\n        # reconnect left/right\n        @prev_cl.next_cl = @next_cl\n        @next_cl.prev_cl = @prev_cl\n\n        # delete content\n        # - we must not do this in applyDelete, because this would lead to inconsistencies\n        # (e.g. the following operation order must be invertible :\n        #   Insert refers to content, then the content is deleted)\n        # Therefore, we have to do this in the cleanup\n        # * NODE: We never delete Insertions!\n        if @content instanceof ops.Operation and not (@content instanceof ops.Insert)\n          @content.referenced_by--\n          if @content.referenced_by <= 0 and not @content.is_deleted\n            @content.applyDelete()\n        delete @content\n        super\n      # else\n      #   Someone inserted something in the meantime.\n      #   Remember: this can only be garbage collected when next_cl is deleted\n\n    #\n    # @private\n    # The amount of positions that $this operation was moved to the right.\n    #\n    getDistanceToOrigin: ()->\n      d = 0\n      o = @prev_cl\n      while true\n        if @origin is o\n          break\n        d++\n        o = o.prev_cl\n      d\n\n    #\n    # @private\n    # Include this operation in the associative lists.\n    execute: ()->\n      if not @validateSavedOperations()\n        return false\n      else\n        if @content instanceof ops.Operation\n          @content.insert_parent = @ # TODO: this is probably not necessary and only nice for debugging\n          @content.referenced_by ?= 0\n          @content.referenced_by++\n        if @parent?\n          if not @prev_cl?\n            @prev_cl = @parent.beginning\n          if not @origin?\n            @origin = @prev_cl\n          else if @origin is \"Delimiter\"\n            @origin = @parent.beginning\n          if not @next_cl?\n            @next_cl = @parent.end\n        if @prev_cl?\n          distance_to_origin = @getDistanceToOrigin() # most cases: 0\n          o = @prev_cl.next_cl\n          i = distance_to_origin # loop counter\n\n          # $this has to find a unique position between origin and the next known character\n          # case 1: $origin equals $o.origin: the $creator parameter decides if left or right\n          #         let $OL= [o1,o2,o3,o4], whereby $this is to be inserted between o1 and o4\n          #         o2,o3 and o4 origin is 1 (the position of o2)\n          #         there is the case that $this.creator < o2.creator, but o3.creator < $this.creator\n          #         then o2 knows o3. Since on another client $OL could be [o1,o3,o4] the problem is complex\n          #         therefore $this would be always to the right of o3\n          # case 2: $origin < $o.origin\n          #         if current $this insert_position > $o origin: $this ins\n          #         else $insert_position will not change\n          #         (maybe we encounter case 1 later, then this will be to the right of $o)\n          # case 3: $origin > $o.origin\n          #         $this insert_position is to the left of $o (forever!)\n          while true\n            if o isnt @next_cl\n              oDistance = o.getDistanceToOrigin()\n              # $o happened concurrently\n              if oDistance is i\n                # case 1\n                if o.uid.creator < @uid.creator\n                  @prev_cl = o\n                  distance_to_origin = i + 1\n                else\n                  # nop\n              else if oDistance < i\n                # case 2\n                if i - distance_to_origin <= oDistance\n                  @prev_cl = o\n                  distance_to_origin = i + 1\n                else\n                  #nop\n              else\n                # case 3\n                break\n              i++\n              o = o.next_cl\n            else\n              # $this knows that $o exists,\n              break\n          # now reconnect everything\n          @next_cl = @prev_cl.next_cl\n          @prev_cl.next_cl = @\n          @next_cl.prev_cl = @\n\n        @setParent @prev_cl.getParent() # do Insertions always have a parent?\n        super # notify the execution_listeners\n        @parent.callOperationSpecificInsertEvents(this)\n        @\n\n    #\n    # Compute the position of this operation.\n    #\n    getPosition: ()->\n      position = 0\n      prev = @prev_cl\n      while true\n        if prev instanceof ops.Delimiter\n          break\n        if not prev.isDeleted()\n          position++\n        prev = prev.prev_cl\n      position\n\n    #\n    # Convert all relevant information of this operation to the json-format.\n    # This result can be send to other clients.\n    #\n    _encode: (json = {})->\n      json.prev = @prev_cl.getUid()\n      json.next = @next_cl.getUid()\n\n      if @origin.type is \"Delimiter\"\n        json.origin = \"Delimiter\"\n      else if @origin isnt @prev_cl\n        json.origin = @origin.getUid()\n\n      # if not (json.prev? and json.next?)\n      json.parent = @parent.getUid()\n\n      super json\n\n  ops.Insert.parse = (json)->\n    {\n      'content' : content\n      'content_operations' : content_operations\n      'uid' : uid\n      'prev': prev\n      'next': next\n      'origin' : origin\n      'parent' : parent\n    } = json\n    new this null, content, content_operations, parent, uid, prev, next, origin\n\n  #\n  # @nodoc\n  # A delimiter is placed at the end and at the beginning of the associative lists.\n  # This is necessary in order to have a beginning and an end even if the content\n  # of the Engine is empty.\n  #\n  class ops.Delimiter extends ops.Operation\n    #\n    # @param {Object} uid A unique identifier. If uid is undefined, a new uid will be created.\n    # @param {Operation} prev_cl The predecessor of this operation in the complete-list (cl)\n    # @param {Operation} next_cl The successor of this operation in the complete-list (cl)\n    #\n    constructor: (prev_cl, next_cl, origin)->\n      @saveOperation 'prev_cl', prev_cl\n      @saveOperation 'next_cl', next_cl\n      @saveOperation 'origin', prev_cl\n      super null, {noOperation: true}\n\n    type: \"Delimiter\"\n\n    applyDelete: ()->\n      super()\n      o = @prev_cl\n      while o?\n        o.applyDelete()\n        o = o.prev_cl\n      undefined\n\n    cleanup: ()->\n      super()\n\n    #\n    # @private\n    #\n    execute: ()->\n      if @unchecked?['next_cl']?\n        super\n      else if @unchecked?['prev_cl']\n        if @validateSavedOperations()\n          if @prev_cl.next_cl?\n            throw new Error \"Probably duplicated operations\"\n          @prev_cl.next_cl = @\n          super\n        else\n          false\n      else if @prev_cl? and not @prev_cl.next_cl?\n        delete @prev_cl.unchecked.next_cl\n        @prev_cl.next_cl = @\n        super\n      else if @prev_cl? or @next_cl? or true # TODO: are you sure? This can happen right?\n        super\n      #else\n      #  throw new Error \"Delimiter is unsufficient defined!\"\n\n    #\n    # @private\n    #\n    _encode: ()->\n      {\n        'type' : @type\n        'uid' : @getUid()\n        'prev' : @prev_cl?.getUid()\n        'next' : @next_cl?.getUid()\n      }\n\n  ops.Delimiter.parse = (json)->\n    {\n    'uid' : uid\n    'prev' : prev\n    'next' : next\n    } = json\n    new this(uid, prev, next)\n\n  # This is what this module exports after initializing it with the HistoryBuffer\n  {\n    'operations' : ops\n    'execution_listener' : execution_listener\n  }\n","basic_ops_uninitialized = require \"./Basic\"\nRBTReeByIndex = require 'bintrees/lib/rbtree_by_index'\n\nmodule.exports = ()->\n  basic_ops = basic_ops_uninitialized()\n  ops = basic_ops.operations\n\n  #\n  # @nodoc\n  # Manages map like objects. E.g. Json-Type and XML attributes.\n  #\n  class ops.MapManager extends ops.Operation\n\n    #\n    # @param {Object} uid A unique identifier. If uid is undefined, a new uid will be created.\n    #\n    constructor: (custom_type, uid, content, content_operations)->\n      @_map = {}\n      super custom_type, uid, content, content_operations\n\n    type: \"MapManager\"\n\n    applyDelete: ()->\n      for name,p of @_map\n        p.applyDelete()\n      super()\n\n    cleanup: ()->\n      super()\n\n    map: (f)->\n      for n,v of @_map\n        f(n,v)\n      undefined\n\n    #\n    # @see JsonOperations.val\n    #\n    val: (name, content)->\n      if arguments.length > 1\n        if content? and content._getModel?\n          rep = content._getModel(@custom_types, @operations)\n        else\n          rep = content\n        @retrieveSub(name).replace rep\n        @getCustomType()\n      else if name?\n        prop = @_map[name]\n        if prop? and not prop.isContentDeleted()\n          res = prop.val()\n          if res instanceof ops.Operation\n            res.getCustomType()\n          else\n            res\n        else\n          undefined\n      else\n        result = {}\n        for name,o of @_map\n          if not o.isContentDeleted()\n            result[name] = o.val()\n        result\n\n    delete: (name)->\n      @_map[name]?.deleteContent()\n      @\n\n    retrieveSub: (property_name)->\n      if not @_map[property_name]?\n        event_properties =\n          name: property_name\n        event_this = @\n        rm_uid =\n          noOperation: true\n          sub: property_name\n          alt: @\n        rm = new ops.ReplaceManager null, event_properties, event_this, rm_uid # this operation shall not be saved in the HB\n        @_map[property_name] = rm\n        rm.setParent @, property_name\n        rm.execute()\n      @_map[property_name]\n\n  ops.MapManager.parse = (json)->\n    {\n      'uid' : uid\n      'custom_type' : custom_type\n      'content' : content\n      'content_operations' : content_operations\n    } = json\n    new this(custom_type, uid, content, content_operations)\n\n\n\n  #\n  # @nodoc\n  # Manages a list of Insert-type operations.\n  #\n  class ops.ListManager extends ops.Operation\n\n    #\n    # A ListManager maintains a non-empty list that has a beginning and an end (both Delimiters!)\n    # @param {Object} uid A unique identifier. If uid is undefined, a new uid will be created.\n    # @param {Delimiter} beginning Reference or Object.\n    # @param {Delimiter} end Reference or Object.\n    constructor: (custom_type, uid, content, content_operations)->\n      @beginning = new ops.Delimiter undefined, undefined\n      @end =       new ops.Delimiter @beginning, undefined\n      @beginning.next_cl = @end\n      @beginning.execute()\n      @end.execute()\n\n\n      # The short tree is storing the non deleted operations\n      @shortTree = new RBTreeByIndex()\n      # The complete tree is storing all the operations\n      @completeTree = new RBTreeByIndex()\n\n      super custom_type, uid, content, content_operations\n\n    type: \"ListManager\"\n\n\n    applyDelete: ()->\n      o = @beginning\n      while o?\n        o.applyDelete()\n        o = o.next_cl\n      super()\n\n    cleanup: ()->\n      super()\n\n\n    toJson: (transform_to_value = false)->\n      val = @val()\n      for i, o in val\n        if o instanceof ops.Object\n          o.toJson(transform_to_value)\n        else if o instanceof ops.ListManager\n          o.toJson(transform_to_value)\n        else if transform_to_value and o instanceof ops.Operation\n          o.val()\n        else\n          o\n\n    #\n    # @private\n    # @see Operation.execute\n    #\n    execute: ()->\n      if @validateSavedOperations()\n        @beginning.setParent @\n        @end.setParent @\n        super\n      else\n        false\n\n    # Get the element previous to the delemiter at the end\n    getLastOperation: ()->\n      @end.prev_cl\n\n    # similar to the above\n    getFirstOperation: ()->\n      @beginning.next_cl\n\n    # get the next non-deleted operation\n    getNextNonDeleted: (start)->\n      if start.isDeleted() or not start.node?\n        operation = start.next_cl\n        while not ((operation instanceof ops.Delimiter))\n          if operation.is_deleted\n            operation = operation.next_cl\n          else\n            break\n      else\n        operation = start.node.next.node\n        if not operation\n          return false\n\n      operation\n\n    getPrevNonDeleted: (start) ->\n      if start.isDeleted() or not start.node?\n        operation = start.prev_cl\n        while not ((operation instanceof ops.Delimiter))\n          if operation.is_deleted\n            operation = operation.prev_cl\n          else\n            break\n      else\n        operation = start.node.prev.node\n        if not operation\n          return false\n\n      operation\n\n\n    # Transforms the list to an array\n    # Doesn't return left-right delimiter.\n    toArray: ()->\n      @shortTree.map (operation) ->\n        operation.val()\n\n    map: (fun)->\n      @shortTree.map fun\n\n    fold: (init, fun)->\n      @shortTree.map (operation) ->\n        init = fun(init, operation)\n\n    val: (pos)->\n      if pos?\n        @shortTree.find(pos).val()\n      else\n        @toArray()\n\n    ref: (pos)->\n      if pos?\n        @shortTree.find(pos)\n      else\n        @shortTree.map (operation) ->\n          operation\n\n    #\n    # Retrieves the x-th not deleted element.\n    # e.g. \"abc\" : the 1th character is \"a\"\n    # the 0th character is the left Delimiter\n    #\n    getOperationByPosition: (position)->\n      if position == 0\n        @beginning\n      else if position == @shortTree.size + 1\n        @end\n      else\n        @shortTree.find (position-1)\n\n    push: (content)->\n      @insertAfter @end.prev_cl, [content]\n\n    insertAfterHelper: (root, content)->\n      if !root.right\n        root.bt.right = content\n        content.bt.parent = root\n      else\n        right = root.next_cl\n\n\n    insertAfter: (left, contents)->\n      if left is @beginning\n        leftNode = null\n        rightNode = @shortTree.findNode 0\n        right = if rightNode then rightNode.data else @end\n      else\n        # left.node should always exist (insert after a non-deleted element)\n        rightNode = left.node.next\n        leftNode = left.node\n        right = if rightNode then rightNode.data else @end\n\n      left = right.prev_cl\n\n      # TODO: always expect an array as content. Then you can combine this with the other option (else)\n      if contents instanceof ops.Operation\n        tmp = new ops.Insert null, content, null, undefined, undefined, left, right\n        tmp.execute()\n      else\n        for c in contents\n          if c? and c._name? and c._getModel?\n            c = c._getModel(@custom_types, @operations)\n          tmp = new ops.Insert null, c, null, undefined, undefined, left, right\n          tmp.execute()\n          leftNode = tmp.node\n\n          left = tmp\n      @\n\n    #\n    # Inserts an array of content into this list.\n    # @Note: This expects an array as content!\n    #\n    # @return {ListManager Type} This String object.\n    #\n    insert: (position, contents)->\n      ith = @getOperationByPosition position\n      # the (i-1)th character. e.g. \"abc\" the 1th character is \"a\"\n      # the 0th character is the left Delimiter\n      @insertAfter ith, contents\n\n    #\n    # Deletes a part of the word.\n    #\n    # @return {ListManager Type} This String object\n    #\n    deleteRef: (operation, length = 1, dir = 'right') ->\n      nextOperation = (operation) =>\n        if dir == 'right' then @getNextNonDeleted operation else @getPrevNonDeleted operation\n\n      for i in [0...length]\n        if operation instanceof ops.Delimiter\n          break\n        deleteOperation = (new ops.Delete null, undefined, operation).execute()\n\n        operation = nextOperation operation\n      @\n\n    delete: (position, length = 1)->\n      operation = @getOperationByPosition(position+length) # position 0 in this case is the deletion of the first character\n\n      @deleteRef operation, length, 'left'\n\n\n    callOperationSpecificInsertEvents: (operation)->\n      prev = (@getPrevNonDeleted operation) or @beginning\n      prevNode = if prev then prev.node else null\n\n      next = (@getNextNonDeleted operation) or @end\n      nextNode = if next then next.node else null\n      operation.node = operation.node or (@shortTree.insert_between prevNode, nextNode, operation)\n      operation.completeNode = operation.completeNode or (@completeTree.insert_between operation.prev_cl.completeNode, operation.next_cl.completeNode, operation)\n\n      getContentType = (content)->\n        if content instanceof ops.Operation\n          content.getCustomType()\n        else\n          content\n\n      @callEvent [\n        type: \"insert\"\n        reference: operation\n        position: operation.node.position()\n        object: @getCustomType()\n        changedBy: operation.uid.creator\n        value: getContentType operation.val()\n      ]\n\n    callOperationSpecificDeleteEvents: (operation, del_op)->\n      if operation.node\n        position = operation.node.position()\n        @shortTree.remove_node operation.node\n        operation.node = null\n\n      @callEvent [\n        type: \"delete\"\n        reference: operation\n        position: position\n        object: @getCustomType() # TODO: You can combine getPosition + getParent in a more efficient manner! (only left Delimiter will hold @parent)\n        length: 1\n        changedBy: del_op.uid.creator\n        oldValue: operation.val()\n      ]\n\n  ops.ListManager.parse = (json)->\n    {\n      'uid' : uid\n      'custom_type': custom_type\n      'content' : content\n      'content_operations' : content_operations\n    } = json\n    new this(custom_type, uid, content, content_operations)\n\n  class ops.Composition extends ops.ListManager\n\n    constructor: (custom_type, @_composition_value, composition_value_operations, uid, tmp_composition_ref)->\n      # we can't use @seveOperation 'composition_ref', tmp_composition_ref here,\n      # because then there is a \"loop\" (insertion refers to parent, refers to insertion..)\n      # This is why we have to check in @callOperationSpecificInsertEvents until we find it\n      super custom_type, uid\n      if tmp_composition_ref?\n        @tmp_composition_ref = tmp_composition_ref\n      else\n        @composition_ref = @end.prev_cl\n      if composition_value_operations?\n        @composition_value_operations = {}\n        for n,o of composition_value_operations\n          @saveOperation n, o, '_composition_value'\n\n    type: \"Composition\"\n\n    #\n    # @private\n    # @see Operation.execute\n    #\n    execute: ()->\n      if @validateSavedOperations()\n        @getCustomType()._setCompositionValue @_composition_value\n        delete @_composition_value\n        # check if tmp_composition_ref already exists\n        if @tmp_composition_ref\n          composition_ref = @HB.getOperation @tmp_composition_ref\n          if composition_ref?\n            delete @tmp_composition_ref\n            @composition_ref = composition_ref\n        super\n      else\n        false\n\n    #\n    # This is called, when the Insert-operation was successfully executed.\n    #\n    callOperationSpecificInsertEvents: (operation)->\n      if @tmp_composition_ref?\n        if operation.uid.creator is @tmp_composition_ref.creator and operation.uid.op_number is @tmp_composition_ref.op_number\n          @composition_ref = operation\n          delete @tmp_composition_ref\n          operation = operation.next_cl\n          if operation is @end\n            return\n        else\n          return\n\n      o = @end.prev_cl\n      while o isnt operation\n        @getCustomType()._unapply o.undo_delta\n        o = o.prev_cl\n      while o isnt @end\n        o.undo_delta = @getCustomType()._apply o.val()\n        o = o.next_cl\n      @composition_ref = @end.prev_cl\n\n      @callEvent [\n        type: \"update\"\n        changedBy: operation.uid.creator\n        newValue: @val()\n      ]\n\n    callOperationSpecificDeleteEvents: (operation, del_op)->\n      return\n\n    #\n    # Create a new Delta\n    # - inserts new Content at the end of the list\n    # - updates the composition_value\n    # - updates the composition_ref\n    #\n    # @param delta The delta that is applied to the composition_value\n    #\n    applyDelta: (delta, operations)->\n      (new ops.Insert null, delta, operations, @, null, @end.prev_cl, @end).execute()\n      undefined\n\n    #\n    # Encode this operation in such a way that it can be parsed by remote peers.\n    #\n    _encode: (json = {})->\n      custom = @getCustomType()._getCompositionValue()\n      json.composition_value = custom.composition_value\n      if custom.composition_value_operations?\n        json.composition_value_operations = {}\n        for n,o of custom.composition_value_operations\n          json.composition_value_operations[n] = o.getUid()\n      if @composition_ref?\n        json.composition_ref = @composition_ref.getUid()\n      else\n        json.composition_ref = @tmp_composition_ref\n      super json\n\n  ops.Composition.parse = (json)->\n    {\n      'uid' : uid\n      'custom_type': custom_type\n      'composition_value' : composition_value\n      'composition_value_operations' : composition_value_operations\n      'composition_ref' : composition_ref\n    } = json\n    new this(custom_type, composition_value, composition_value_operations, uid, composition_ref)\n\n\n  #\n  # @nodoc\n  # Adds support for replace. The ReplaceManager manages Replaceable operations.\n  # Each Replaceable holds a value that is now replaceable.\n  #\n  # The TextType-type has implemented support for replace\n  # @see TextType\n  #\n  class ops.ReplaceManager extends ops.ListManager\n    #\n    # @param {Object} event_properties Decorates the event that is thrown by the RM\n    # @param {Object} event_this The object on which the event shall be executed\n    # @param {Operation} initial_content Initialize this with a Replaceable that holds the initial_content.\n    # @param {Object} uid A unique identifier. If uid is undefined, a new uid will be created.\n    # @param {Delimiter} beginning Reference or Object.\n    # @param {Delimiter} end Reference or Object.\n    constructor: (custom_type, @event_properties, @event_this, uid)->\n      if not @event_properties['object']?\n        @event_properties['object'] = @event_this.getCustomType()\n      super custom_type, uid\n\n    type: \"ReplaceManager\"\n\n    #\n    # This doesn't throw the same events as the ListManager. Therefore, the\n    # Replaceables also not throw the same events.\n    # So, ReplaceManager and ListManager both implement\n    # these functions that are called when an Insertion is executed (at the end).\n    #\n    #\n    callEventDecorator: (events)->\n      if not @isDeleted()\n        for event in events\n          for name,prop of @event_properties\n            event[name] = prop\n        @event_this.callEvent events\n      undefined\n\n    #\n    # This is called, when the Insert-type was successfully executed.\n    # TODO: consider doing this in a more consistent manner. This could also be\n    # done with execute. But currently, there are no specital Insert-ops for ListManager.\n    #\n    callOperationSpecificInsertEvents: (operation)->\n      if operation.next_cl.type is \"Delimiter\" and operation.prev_cl.type isnt \"Delimiter\"\n        # this replaces another Replaceable\n        if not operation.is_deleted # When this is received from the HB, this could already be deleted!\n          old_value = operation.prev_cl.val()\n          @callEventDecorator [\n            type: \"update\"\n            changedBy: operation.uid.creator\n            oldValue: old_value\n          ]\n        operation.prev_cl.applyDelete()\n      else if operation.next_cl.type isnt \"Delimiter\"\n        # This won't be recognized by the user, because another\n        # concurrent operation is set as the current value of the RM\n        operation.applyDelete()\n      else # prev _and_ next are Delimiters. This is the first created Replaceable in the RM\n        @callEventDecorator [\n          type: \"add\"\n          changedBy: operation.uid.creator\n        ]\n      undefined\n\n    callOperationSpecificDeleteEvents: (operation, del_op)->\n      if operation.next_cl.type is \"Delimiter\"\n        @callEventDecorator [\n          type: \"delete\"\n          changedBy: del_op.uid.creator\n          oldValue: operation.val()\n        ]\n\n\n    #\n    # Replace the existing word with a new word.\n    #\n    # @param content {Operation} The new value of this ReplaceManager.\n    # @param replaceable_uid {UID} Optional: Unique id of the Replaceable that is created\n    #\n    replace: (content, replaceable_uid)->\n      o = @getLastOperation()\n      relp = (new ops.Insert null, content, null, @, replaceable_uid, o, o.next_cl).execute()\n      # TODO: delete repl (for debugging)\n      undefined\n\n    isContentDeleted: ()->\n      @getLastOperation().isDeleted()\n\n    deleteContent: ()->\n      last_op = @getLastOperation()\n      if (not last_op.isDeleted()) and last_op.type isnt \"Delimiter\"\n        (new ops.Delete null, undefined, @getLastOperation().uid).execute()\n      undefined\n\n    #\n    # Get the value of this\n    # @return {String}\n    #\n    val: ()->\n      o = @getLastOperation()\n      #if o instanceof ops.Delimiter\n        # throw new Error \"Replace Manager doesn't contain anything.\"\n      o.val?() # ? - for the case that (currently) the RM does not contain anything (then o is a Delimiter)\n\n  basic_ops\n","\nstructured_ops_uninitialized = require \"./Operations/Structured\"\n\nHistoryBuffer = require \"./HistoryBuffer\"\nEngine = require \"./Engine\"\nadaptConnector = require \"./ConnectorAdapter\"\n\ncreateY = (connector)->\n  if connector.user_id?\n    user_id = connector.user_id # TODO: change to getUniqueId()\n  else\n    user_id = \"_temp\"\n    connector.when_received_state_vector_listeners = [(state_vector)->\n        HB.setUserId this.user_id, state_vector\n      ]\n  HB = new HistoryBuffer user_id\n  ops_manager = structured_ops_uninitialized HB, this.constructor\n  ops = ops_manager.operations\n\n  engine = new Engine HB, ops\n  adaptConnector connector, engine, HB, ops_manager.execution_listener\n\n  ops.Operation.prototype.HB = HB\n  ops.Operation.prototype.operations = ops\n  ops.Operation.prototype.engine = engine\n  ops.Operation.prototype.connector = connector\n  ops.Operation.prototype.custom_types = this.constructor\n\n  ct = new createY.Object()\n  model = new ops.MapManager(ct, HB.getReservedUniqueIdentifier()).execute()\n  ct._setModel model\n  ct\n\nmodule.exports = createY\nif window?\n  window.Y = createY\n\ncreateY.Object = require \"./ObjectType\"\n","var TreeBase = require('./treebase');\n/** algorithm from Cormen, Leiserson - Introduction to algorithm **/\n\nfunction RBTree(comparator) {\n    this._root = null;\n    this._nil = new Node('nil');\n    this._nil.red = false;\n    this._nil.weight = 0;\n\n    Object.defineProperty(this, 'size', {\n        get: function() {\n            return this._root ? this._root.weight : 0;\n        }\n    });\n}\n\nRBTree.prototype = new TreeBase();\n\nRBTree.prototype.insert = function(position, data) {\n  var nodeToInsert = new Node(data);\n  if (this.insert_node(position, nodeToInsert)) {\n    return nodeToInsert;\n  } else {\n    return false;\n  }\n};\n\nRBTree.prototype.insert_node = function(position, nodeToInsert) {\n  var node = this._root;\n  var insertAfter;\n\n  if (!node) {\n    this._root = nodeToInsert;\n    this._root.red = false;\n    return true;\n  }\n\n  while (true) {\n    if (node.weight === position) { // Insert after max of node subtree\n      insertAfter = node.max_tree(function(node) {\n        node.weight += 1;\n      });\n      insertAfter.set_child('right', nodeToInsert);\n      nodeToInsert.parent = insertAfter;\n      break;\n    } else {\n      left = node.get_child('left');\n      right = node.get_child('right');\n      if (!left && position === 0) {\n        node.weight += 1;\n        node.set_child('left', nodeToInsert);\n        nodeToInsert.parent = node;\n        break;\n\n      } else if (left && left.weight >= position) {\n        node.weight += 1;\n        node = left;\n\n      } else if (right) {\n        position -= (left? left.weight: 0) + 1;\n        node.weight += 1;\n        node = right;\n\n      } else {\n        node.weight += 1;\n        node.set_child('right', nodeToInsert);\n        nodeToInsert.parent = node;\n        break;\n      }\n    }\n  }\n\n  this.insert_correction(nodeToInsert);\n  return true;\n};\n\nRBTree.prototype.insert_between = function(onLeft, onRight, data) {\n  var newNode = new Node(data);\n  // onLeft and onRight are neighbors, so they can't have an element in between.\n  // For example, if onLeft exists, its right neighbor is either nil or right. If it's onRight,\n  // then onRight has no child. If neither left and right exist, the tree is empty.\n  if (onLeft && !onLeft.right) {\n    onLeft.right = newNode;\n    newNode.parent = onLeft;\n\n    this.insert_correction(newNode);\n  } else if (onRight && !onRight.left) {\n    onRight.left = newNode;\n    newNode.parent = onRight;\n\n    this.insert_correction(newNode);\n  } else {\n    this.insert_node(0, newNode);\n  }\n  newNode.traverse_up(function(node, parent) {\n    parent.weight = (parent.left ? parent.left.weight : 0) +\n        (parent.right ? parent.right.weight : 0) + 1;\n  });\n\n  return newNode;\n};\n\nRBTree.prototype.rotate = function(side, node) {\n  // all the comment are with the assumption that side === 'left'\n  var neighbor;\n\n  // get right neighbor\n  neighbor = node.get_child(side, true);\n  // neighbor's left tree becomes node's right tree\n  node.set_child(side, neighbor.get_child(side), true);\n\n  // if this right tree was non-empty\n  if (neighbor.get_child(side)) {\n    neighbor.get_child(side).parent = node; // attach neighbor's left child to node\n  }\n  neighbor.parent = node.parent; // link neighbor's parent to node's parent\n\n  if (!node.parent) { // no parent === is_root\n    this._root = neighbor;\n  } else if (node === node.parent.get_child(side)){ // node is left child\n    node.parent.set_child(side, neighbor); // node's parent left child is now neighbor\n  } else {\n    node.parent.set_child(side, neighbor, true); // node's parent right child is now neighbor\n  }\n\n  neighbor.set_child(side, node); // attach node on left of child\n  node.parent = neighbor; // set node's parent to neighbor\n\n  // update node's weight first, then\n  node.weight = (node.left? node.left.weight: 0) + (node.right? node.right.weight: 0) + 1;\n  neighbor.weight = (neighbor.left? neighbor.left.weight: 0) + (neighbor.right? neighbor.right.weight: 0) + 1;\n};\n\nRBTree.prototype.insert_correction = function(node) {\n  var self = this;\n  function helper(side) {\n    var uncle;\n    uncle = node.parent.parent.get_child(side, true);\n\n    if (uncle && uncle.red) { // if uncle is undefined, it's a leaf so it's black\n      node.parent.red        = false;\n      uncle.red              = false;\n      node.parent.parent.red = true;\n\n      node = node.parent.parent;\n    } else {\n      if (node === node.parent.get_child(side, true)) {\n        node = node.parent;\n        self.rotate(side, node);\n      }\n\n      node.parent.red        = false;\n      node.parent.parent.red = true;\n\n      var oppositeSide = side === 'left'? 'right': 'left';\n      self.rotate(oppositeSide, node.parent.parent);\n    }\n  }\n\n\n  while (node.parent && node.parent.red) { // is there's no node.parent, then it means the parent\n    // is nil which is black\n\n    // if node's parent is on left of his parent\n    if (node.parent === node.parent.parent.get_child('left')) { // Check that there is a grandparent\n      helper('left');\n    } else if (node.parent === node.parent.parent.get_child('right')){\n      helper('right');\n    }\n  }\n\n  this._root.red = false;\n\n};\n\nRBTree.prototype.find = function() {\n  var node = this.findNode.apply(this, arguments);\n  if (node) {\n    return node.data;\n  } else {\n    return null;\n  }\n};\n\n/** Find the node at position @position and return it. If no node is found, returns null.\n * It is also possible to pass a function as second argument. It will be called\n * with each node traversed except for the one found.\n**/\nRBTree.prototype.findNode = function(position, fun) {\n  // if the weight is 'n', the biggest index is n-1, so we check that position >= size\n  if (position >= this.size || position < 0) {\n    return null;\n  }\n  var node = this._root;\n\n  if (!node) {\n    return null;\n  }\n\n  // Find node to delete\n  while (position > 0 || (position === 0 && node.left)) {\n    left = node.left;\n    right = node.right;\n\n    if (left && left.weight > position) {\n      // when there's a left neighbor with a weight > position to remove,\n      // go into this subtree and decrease the subtree weight\n      if (fun) {\n        fun(node);\n      }\n      node = left;\n\n    } else if ((!left && position === 0) || (left && left.weight === position)) {\n      break;\n    } else if (right) {\n      // when there's a right subtee, go into it and decrease the position\n      // by the weight of the left subtree - 1 + 1 (for the previous node)\n      if (fun) {\n        fun(node);\n      }\n      node = right;\n      position -= (left? left.weight: 0) + 1;\n\n    } else {\n      // this should not happen, except if the position is greater than the size of the tree\n      throw new Error('this should not happen');\n    }\n  }\n  return node;\n};\n\nRBTree.prototype.remove = function(position) {\n  // if the weight is 'n', the biggest index is n-1, so we check that position >= size\n  var nodeToRemove;\n\n  nodeToRemove = this.findNode(position, function(node) { node.weight --; });\n\n  return this.remove_helper(nodeToRemove);\n};\n\nRBTree.prototype.remove_helper = function(nodeToRemove) {\n  var left, right, nextNode, childNode, parent;\n\n  // if there's only one child, replace the nodeToRemove to delete by it's child and update\n  // the refs.\n  // if there's two children, find it's successor and replace the nodeToRemove to delete by his successor\n  // and update the refs\n\n  if (!nodeToRemove.left || !nodeToRemove.right) {\n    nextNode = nodeToRemove;\n  } else {\n    nextNode = nodeToRemove.getNext(function(node) {\n      node.weight -= 1;\n    });\n  }\n\n  if (nextNode.left) {\n    childNode = nextNode.left;\n    parent = nextNode.parent;\n  } else {\n    childNode = nextNode.right;\n    parent = nextNode.parent;\n  }\n\n  if (childNode) {\n    childNode.parent = nextNode.parent;\n  }\n\n\n  if (!nextNode.parent) {\n    this._root = childNode;\n  } else {\n    if (nextNode === nextNode.parent.left) {\n      nextNode.parent.left = childNode;\n    } else {\n      nextNode.parent.right = childNode;\n    }\n  }\n\n  // replace nodeToRemove's data by nextNode's (same as removing nodeToRemove, then inserting nextNode at his place\n  // but easier and more efficient)\n  if (nextNode !== nodeToRemove) {\n    nodeToRemove.data = nextNode.data;\n    nodeToRemove.weight = (nodeToRemove.left? nodeToRemove.left.weight: 0) +\n      (nodeToRemove.right? nodeToRemove.right.weight: 0) + 1;\n\n  }\n\n  if (!nextNode.red) {\n    this.remove_correction(childNode, parent);\n  }\n\n  return nextNode;\n};\n\nRBTree.prototype.remove_node = function(node) {\n  node.traverse_up(function(node, parent) {\n    parent.weight --;\n  });\n\n  return this.remove_helper(node);\n};\n\nRBTree.prototype.remove_correction = function(node, parent) {\n  var self = this;\n  var oppositeSide;\n  var helper = function(side) {\n    var neighbor = parent.get_child(side, true);\n\n    if (neighbor && neighbor.red) {\n      neighbor.red = false;\n      parent.red   = true;\n\n      self.rotate(side, parent);\n      neighbor = parent.get_child(side, true);\n    }\n\n    if ((!neighbor.left || !neighbor.left.red) && (!neighbor.right || !neighbor.right.red )) {\n      neighbor.red = true;\n      node = parent;\n      parent = node.parent;\n    } else {\n      if (!neighbor.get_child(side, true) || !neighbor.get_child(side, true).red) {\n        if (neighbor.get_child(side)) {\n          neighbor.get_child(side).red = false;\n        }\n        neighbor.red = true;\n\n        oppositeSide = side === 'left'? 'right': 'left';\n        self.rotate(oppositeSide, neighbor);\n        neighbor = parent.get_child(side, true);\n      }\n\n      neighbor.red = parent? parent.red: false;\n      parent.red   = false;\n      if (neighbor.get_child(side, true)) {\n        neighbor.get_child(side, true).red = false;\n      }\n\n      self.rotate(side, parent);\n      node = self._root;\n      parent = node.parent;\n    }\n  };\n  while (node !== this._root && (!node || !node.red)) {\n    if (node === parent.get_child('left')) {\n      tmp = helper('left');\n    } else if (node === parent.get_child('right')) {\n      tmp = helper('right');\n    }\n  }\n\n  if (node) {\n    node.red = false;\n  }\n};\n\n\nfunction Node(data, name) {\n    this._dataBindName = name || 'node';\n    this._data = data;\n\n    this.left = null;\n    this.right = null;\n    this.parent = null;\n    this.red = true;\n    this.weight = 1;\n\n    Object.defineProperty(this, 'next', {\n        get: function() {\n            return this.getNext();\n        }\n    });\n    Object.defineProperty(this, 'prev', {\n        get: function() {\n            return this.getPrev();\n        }\n    });\n\n    Object.defineProperty(this, 'data', {\n        get: function() {\n            return this._data;\n        },\n        set: function(data) {\n            this._data = data;\n            // bind the data to this\n            data[this._dataBindName] = this;\n        }\n    });\n\n}\n\nNode.prototype.get_child = function(side, opposite) {\n  if (opposite) {\n    side = (side === 'left')? 'right': 'left';\n  }\n  return side === 'left'? this.left: this.right;\n};\n\nNode.prototype.set_child = function(side, node, opposite) {\n  if (opposite) {\n    side = (side === 'left')? 'right': 'left';\n  }\n\n  if (side === 'left') {\n    this.left = node;\n  } else {\n    this.right = node;\n  }\n};\n\nNode.prototype.max_tree = function(fun) {\n  var node = this;\n\n  if (fun) {\n    fun(node);\n  }\n\n  while (node.right) {\n    node = node.right;\n    if (fun) {\n      fun(node);\n    }\n  }\n\n  return node;\n};\n\nNode.prototype.min_tree = function(fun) {\n  var node = this;\n\n  if (fun) {\n    fun(node);\n  }\n\n  while (node.left) {\n    node = node.left;\n    if (fun) {\n      fun(node);\n    }\n  }\n\n  return node;\n};\n\nNode.prototype.getNext = function(fun) {\n  var node, parent;\n  if (this.get_child('right')) {\n    return this.get_child('right').min_tree(fun);\n  }\n  if (fun) {\n    fun(this);\n  }\n  node = this;\n  parent = this.parent;\n  while (parent && node === parent.get_child('right')) {\n    node = parent;\n    parent = node.parent;\n    if (fun) {\n      fun(parent);\n    }\n  }\n  return parent;\n};\nNode.prototype.getPrev = function(fun) {\n  var node, parent;\n  if (this.get_child('left')) {\n    return this.get_child('left').max_tree(fun);\n  }\n  if (fun) {\n    fun(this);\n  }\n  node = this;\n  parent = this.parent;\n  while (parent && node === parent.get_child('left')) {\n    node = parent;\n    parent = node.parent;\n    if (fun) {\n      fun(parent);\n    }\n  }\n  return parent;\n};\n\n\n/** Traverse the tree upwards until it reaches the top. For each node traversed,\n  * call the function passed as argument with arguments node, parent.\n  * The function will be called h-1 times, h being the height of the branch.\n  **/\nNode.prototype.traverse_up = function(fun) {\n  var parent = this.parent;\n  var node = this;\n\n  while(parent) {\n    fun(node, parent);\n    node = parent;\n    parent = parent.parent;\n  }\n};\n\nNode.prototype.depth = function() {\n  var depth = 0;\n  this.traverse_up(function() {\n    depth ++;\n  });\n  return depth;\n};\n\nNode.prototype.position = function() {\n  var position = this.left? this.left.weight: 0;\n  var countFun = function(node, parent) {\n    if (parent.right === node) {\n      // for the left subtree\n      if (parent.left) {\n        position += parent.left.weight;\n      }\n      position += 1; // for the parent\n    }\n  };\n\n  this.traverse_up(countFun);\n  return position;\n};\n\nmodule.exports = RBTree;\n","\nfunction TreeBase() {}\n\n// removes all nodes from the tree\nTreeBase.prototype.clear = function() {\n    this._root = null;\n    this.size = 0;\n};\n\n// returns node data if found, null otherwise\nTreeBase.prototype.find = function(data) {\n    var res = this._root;\n\n    while(res !== null) {\n        var c = this._comparator(data, res.data);\n        if(c === 0) {\n            return res.data;\n        }\n        else {\n            res = res.get_child(c > 0);\n        }\n    }\n\n    return null;\n};\n\n// returns iterator to node if found, null otherwise\nTreeBase.prototype.findIter = function(data) {\n    var res = this._root;\n    var iter = this.iterator();\n\n    while(res !== null) {\n        var c = this._comparator(data, res.data);\n        if(c === 0) {\n            iter._cursor = res;\n            return iter;\n        }\n        else {\n            iter._ancestors.push(res);\n            res = res.get_child(c > 0);\n        }\n    }\n\n    return null;\n};\n\n// Returns an interator to the tree node at or immediately after the item\nTreeBase.prototype.lowerBound = function(item) {\n    var cur = this._root;\n    var iter = this.iterator();\n    var cmp = this._comparator;\n\n    while(cur !== null) {\n        var c = cmp(item, cur.data);\n        if(c === 0) {\n            iter._cursor = cur;\n            return iter;\n        }\n        iter._ancestors.push(cur);\n        cur = cur.get_child(c > 0);\n    }\n\n    for(var i=iter._ancestors.length - 1; i >= 0; --i) {\n        cur = iter._ancestors[i];\n        if(cmp(item, cur.data) < 0) {\n            iter._cursor = cur;\n            iter._ancestors.length = i;\n            return iter;\n        }\n    }\n\n    iter._ancestors.length = 0;\n    return iter;\n};\n\n// Returns an interator to the tree node immediately after the item\nTreeBase.prototype.upperBound = function(item) {\n    var iter = this.lowerBound(item);\n    var cmp = this._comparator;\n\n    while(cmp(iter.data(), item) === 0) {\n        iter.next();\n    }\n\n    return iter;\n};\n\n// returns null if tree is empty\nTreeBase.prototype.min = function() {\n    var res = this._root;\n    if(res === null) {\n        return null;\n    }\n\n    while(res.left !== null) {\n        res = res.left;\n    }\n\n    return res.data;\n};\n\n// returns null if tree is empty\nTreeBase.prototype.max = function() {\n    var res = this._root;\n    if(res === null) {\n        return null;\n    }\n\n    while(res.right !== null) {\n        res = res.right;\n    }\n\n    return res.data;\n};\n\n// returns a null iterator\n// call next() or prev() to point to an element\nTreeBase.prototype.iterator = function() {\n    return new Iterator(this);\n};\n\n\nTreeBase.prototype.eachNode = function(cb) {\n    var it=this.iterator(), node;\n    var index = 0;\n    while((node = it.next()) !== null) {\n        cb(node, index);\n        index++;\n    }\n};\n\n// calls cb on each node's data, in order\nTreeBase.prototype.each = function(cb) {\n    this.eachNode(function(node, index) {\n        cb(node.data, index);\n    });\n};\n\n\nTreeBase.prototype.mapNode = function(cb) {\n    var it=this.iterator(), node;\n    var results = [];\n    var index = 0;\n    while((node = it.next()) !== null) {\n        results.push(cb(node, index));\n        index ++;\n    }\n    return results;\n};\n\n// calls cb on each node-s data, store the result and return it\nTreeBase.prototype.map = function(cb) {\n    return this.mapNode(function(node, index) {\n        return cb(node.data, index);\n    });\n};\n\nfunction Iterator(tree) {\n    this._tree = tree;\n    this._ancestors = [];\n    this._cursor = null;\n}\n\nIterator.prototype.data = function() {\n    return this._cursor !== null ? this._cursor.data : null;\n};\n\n// if null-iterator, returns first node\n// otherwise, returns next node\nIterator.prototype.next = function() {\n    if(this._cursor === null) {\n        var root = this._tree._root;\n        if(root !== null) {\n            this._minNode(root);\n        }\n    }\n    else {\n        if(this._cursor.right === null) {\n            // no greater node in subtree, go up to parent\n            // if coming from a right child, continue up the stack\n            var save;\n            do {\n                save = this._cursor;\n                if(this._ancestors.length) {\n                    this._cursor = this._ancestors.pop();\n                }\n                else {\n                    this._cursor = null;\n                    break;\n                }\n            } while(this._cursor.right === save);\n        }\n        else {\n            // get the next node from the subtree\n            this._ancestors.push(this._cursor);\n            this._minNode(this._cursor.right);\n        }\n    }\n    return this._cursor !== null ? this._cursor : null;\n};\n\n// if null-iterator, returns last node\n// otherwise, returns previous node\nIterator.prototype.prev = function() {\n    if(this._cursor === null) {\n        var root = this._tree._root;\n        if(root !== null) {\n            this._maxNode(root);\n        }\n    }\n    else {\n        if(this._cursor.left === null) {\n            var save;\n            do {\n                save = this._cursor;\n                if(this._ancestors.length) {\n                    this._cursor = this._ancestors.pop();\n                }\n                else {\n                    this._cursor = null;\n                    break;\n                }\n            } while(this._cursor.left === save);\n        }\n        else {\n            this._ancestors.push(this._cursor);\n            this._maxNode(this._cursor.left);\n        }\n    }\n    return this._cursor !== null ? this._cursor : null;\n};\n\nIterator.prototype._minNode = function(start) {\n    while(start.left !== null) {\n        this._ancestors.push(start);\n        start = start.left;\n    }\n    this._cursor = start;\n};\n\nIterator.prototype._maxNode = function(start) {\n    while(start.right !== null) {\n        this._ancestors.push(start);\n        start = start.right;\n    }\n    this._cursor = start;\n};\n\nmodule.exports = TreeBase;\n"]}
|