3180 lines
288 KiB
JavaScript
3180 lines
288 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) {
|
|
oDistance = o.getDistanceToOrigin();
|
|
if (o !== this.next_cl) {
|
|
if (o.getDistanceToOrigin() === 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();
|
|
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.getNext = function(start) {
|
|
var o;
|
|
o = start.next_cl;
|
|
while (!(o instanceof ops.Delimiter)) {
|
|
if (o.is_deleted) {
|
|
o = o.next_cl;
|
|
} else if (o instanceof ops.Delimiter) {
|
|
return false;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
return o;
|
|
};
|
|
|
|
ListManager.prototype.toArray = function() {
|
|
var o, result;
|
|
o = this.beginning.next_cl;
|
|
result = [];
|
|
while (o !== this.end) {
|
|
if (!o.is_deleted) {
|
|
result.push(o.val());
|
|
}
|
|
o = o.next_cl;
|
|
}
|
|
return result;
|
|
};
|
|
|
|
ListManager.prototype.map = function(f) {
|
|
var o, result;
|
|
o = this.beginning.next_cl;
|
|
result = [];
|
|
while (o !== this.end) {
|
|
if (!o.is_deleted) {
|
|
result.push(f(o));
|
|
}
|
|
o = o.next_cl;
|
|
}
|
|
return result;
|
|
};
|
|
|
|
ListManager.prototype.fold = function(init, f) {
|
|
var o;
|
|
o = this.beginning.next_cl;
|
|
while (o !== this.end) {
|
|
if (!o.is_deleted) {
|
|
init = f(init, o);
|
|
}
|
|
o = o.next_cl;
|
|
}
|
|
return init;
|
|
};
|
|
|
|
ListManager.prototype.val = function(pos) {
|
|
var o;
|
|
if (pos != null) {
|
|
o = this.getOperationByPosition(pos + 1);
|
|
if (!(o instanceof ops.Delimiter)) {
|
|
return o.val();
|
|
} else {
|
|
throw new Error("this position does not exist");
|
|
}
|
|
} else {
|
|
return this.toArray();
|
|
}
|
|
};
|
|
|
|
ListManager.prototype.ref = function(pos) {
|
|
var o;
|
|
if (pos != null) {
|
|
o = this.getOperationByPosition(pos + 1);
|
|
if (!(o instanceof ops.Delimiter)) {
|
|
return o;
|
|
} else {
|
|
return null;
|
|
}
|
|
} else {
|
|
throw new Error("you must specify a position parameter");
|
|
}
|
|
};
|
|
|
|
ListManager.prototype.getOperationByPosition = function(position) {
|
|
var o;
|
|
o = this.beginning;
|
|
while (true) {
|
|
if (o instanceof ops.Delimiter && (o.prev_cl != null)) {
|
|
o = o.prev_cl;
|
|
while (o.isDeleted() && (o.prev_cl != null)) {
|
|
o = o.prev_cl;
|
|
}
|
|
break;
|
|
}
|
|
if (position <= 0 && !o.isDeleted()) {
|
|
break;
|
|
}
|
|
o = o.next_cl;
|
|
if (!o.isDeleted()) {
|
|
position -= 1;
|
|
}
|
|
}
|
|
return o;
|
|
};
|
|
|
|
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, right, tmp, _i, _len;
|
|
right = left.next_cl;
|
|
while (right.isDeleted()) {
|
|
right = right.next_cl;
|
|
}
|
|
left = right.prev_cl;
|
|
if (contents instanceof ops.Operation) {
|
|
(new ops.Insert(null, content, null, void 0, void 0, left, right)).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)).execute();
|
|
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(o, length) {
|
|
var d, delete_ops, i, _i;
|
|
if (length == null) {
|
|
length = 1;
|
|
}
|
|
delete_ops = [];
|
|
for (i = _i = 0; 0 <= length ? _i < length : _i > length; i = 0 <= length ? ++_i : --_i) {
|
|
if (o instanceof ops.Delimiter) {
|
|
break;
|
|
}
|
|
d = (new ops.Delete(null, void 0, o)).execute();
|
|
o = o.next_cl;
|
|
while ((!(o instanceof ops.Delimiter)) && o.isDeleted()) {
|
|
o = o.next_cl;
|
|
}
|
|
delete_ops.push(d._encode());
|
|
}
|
|
return this;
|
|
};
|
|
|
|
ListManager.prototype["delete"] = function(position, length) {
|
|
var o;
|
|
if (length == null) {
|
|
length = 1;
|
|
}
|
|
o = this.getOperationByPosition(position + 1);
|
|
return this.deleteRef(o, length);
|
|
};
|
|
|
|
ListManager.prototype.callOperationSpecificInsertEvents = function(op) {
|
|
var getContentType;
|
|
getContentType = function(content) {
|
|
if (content instanceof ops.Operation) {
|
|
return content.getCustomType();
|
|
} else {
|
|
return content;
|
|
}
|
|
};
|
|
return this.callEvent([
|
|
{
|
|
type: "insert",
|
|
reference: op,
|
|
position: op.getPosition(),
|
|
object: this.getCustomType(),
|
|
changedBy: op.uid.creator,
|
|
value: getContentType(op.val())
|
|
}
|
|
]);
|
|
};
|
|
|
|
ListManager.prototype.callOperationSpecificDeleteEvents = function(op, del_op) {
|
|
return this.callEvent([
|
|
{
|
|
type: "delete",
|
|
reference: op,
|
|
position: op.getPosition(),
|
|
object: this.getCustomType(),
|
|
length: 1,
|
|
changedBy: del_op.uid.creator,
|
|
oldValue: op.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(op) {
|
|
var o;
|
|
if (this.tmp_composition_ref != null) {
|
|
if (op.uid.creator === this.tmp_composition_ref.creator && op.uid.op_number === this.tmp_composition_ref.op_number) {
|
|
this.composition_ref = op;
|
|
delete this.tmp_composition_ref;
|
|
op = op.next_cl;
|
|
if (op === this.end) {
|
|
return;
|
|
}
|
|
} else {
|
|
return;
|
|
}
|
|
}
|
|
o = this.end.prev_cl;
|
|
while (o !== op) {
|
|
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: op.uid.creator,
|
|
newValue: this.val()
|
|
}
|
|
]);
|
|
};
|
|
|
|
Composition.prototype.callOperationSpecificDeleteEvents = function(op, 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(op) {
|
|
var old_value;
|
|
if (op.next_cl.type === "Delimiter" && op.prev_cl.type !== "Delimiter") {
|
|
if (!op.is_deleted) {
|
|
old_value = op.prev_cl.val();
|
|
this.callEventDecorator([
|
|
{
|
|
type: "update",
|
|
changedBy: op.uid.creator,
|
|
oldValue: old_value
|
|
}
|
|
]);
|
|
}
|
|
op.prev_cl.applyDelete();
|
|
} else if (op.next_cl.type !== "Delimiter") {
|
|
op.applyDelete();
|
|
} else {
|
|
this.callEventDecorator([
|
|
{
|
|
type: "add",
|
|
changedBy: op.uid.creator
|
|
}
|
|
]);
|
|
}
|
|
return void 0;
|
|
};
|
|
|
|
ReplaceManager.prototype.callOperationSpecificDeleteEvents = function(op, del_op) {
|
|
if (op.next_cl.type === "Delimiter") {
|
|
return this.callEventDecorator([
|
|
{
|
|
type: "delete",
|
|
changedBy: del_op.uid.creator,
|
|
oldValue: op.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);
|
|
ops.FastListManager = (function(_super) {
|
|
__extends(FastListManager, _super);
|
|
|
|
function FastListManager() {
|
|
this.shortTree = new RBTreeByIndex();
|
|
FastListManager.__super__.constructor.apply(this, arguments);
|
|
}
|
|
|
|
FastListManager.prototype.type = 'FastListManager';
|
|
|
|
FastListManager.prototype.getNext = function(start) {
|
|
return start.node.next().data;
|
|
};
|
|
|
|
FastListManager.prototype.getPrev = function(start) {
|
|
return start.node.prev().data;
|
|
};
|
|
|
|
FastListManager.prototype.map = function(fun) {
|
|
return this.shortTree.map(function(operation) {
|
|
return fun(operation);
|
|
});
|
|
};
|
|
|
|
FastListManager.prototype.fold = function(init, f) {
|
|
this.shortTree.each(function(operation) {
|
|
return init = f(init, operation);
|
|
});
|
|
return init;
|
|
};
|
|
|
|
FastListManager.prototype.val = function(position) {
|
|
if (position != null) {
|
|
return (this.shortTree.find(position)).val();
|
|
} else {
|
|
return this.shortTree.map(function(operation) {
|
|
return operation.val();
|
|
});
|
|
}
|
|
};
|
|
|
|
FastListManager.prototype.ref = function(position) {
|
|
if (position != null) {
|
|
return this.shortTree.find(position);
|
|
} else {
|
|
return this.shortTree.map(function(operation) {
|
|
return operation;
|
|
});
|
|
}
|
|
};
|
|
|
|
FastListManager.prototype.push = function(content) {
|
|
return this.insertAfter(this.end.prev_cl, [content]);
|
|
};
|
|
|
|
FastListManager.prototype.insertAfter = function(left, contents) {
|
|
var nodeOnLeft, nodeOnRight, operation, right;
|
|
nodeOnLeft = left === this.beginning ? null : left.node;
|
|
nodeOnRight = nodeOnLeft ? nodeOnLeft.next() : this.shortTree.find(0);
|
|
right = nodeOnRight ? nodeOnRight.data : this.end;
|
|
left = right.prev_cl;
|
|
if (contents instanceof ops.Operation) {
|
|
operation = new ops.Insert(null, content, null, void 0, void 0, left, right);
|
|
operation.node = this.shortTree.insertAfter(nodeOnLeft, operation);
|
|
operation.execute();
|
|
} else {
|
|
contents.forEach(function(content) {
|
|
if ((content != null) && (content._name != null) && (content._getModel != null)) {
|
|
content = content._getModel(this.custom_types, this.operations);
|
|
}
|
|
operation = new ops.Insert(null, c, null, void 0, void 0, left, right);
|
|
operation.node = this.shortTree.insertAfter(nodeOnLeft, operation);
|
|
operation.execute();
|
|
left = operation;
|
|
return nodeOnLeft = operation.node;
|
|
});
|
|
}
|
|
return this;
|
|
};
|
|
|
|
FastListManager.prototype.insert = function(position, contents) {
|
|
var left;
|
|
left = (this.shortTree.find(position - 1)) || this.beginning;
|
|
return this.insertAfter(left, contents);
|
|
};
|
|
|
|
FastListManager.prototype["delete"] = function(position, length) {
|
|
var deleteOp, delete_ops, i, nextNode, operation, _i;
|
|
if (length == null) {
|
|
length = 1;
|
|
}
|
|
delete_ops = [];
|
|
operation = this.shortTree.find(position);
|
|
for (i = _i = 0; 0 <= length ? _i < length : _i > length; i = 0 <= length ? ++_i : --_i) {
|
|
if (operation instanceof ops.Delimiter) {
|
|
break;
|
|
}
|
|
deleteOp = new ops.Delete(null, void 0, operation);
|
|
deleteOp.execute();
|
|
nextNode = operation.node.next();
|
|
operation = nextNode ? nextNode.data : this.end;
|
|
this.shortTree.remove_node(operation.node);
|
|
operation.node = null;
|
|
delete_ops.push(d._encode());
|
|
}
|
|
return this;
|
|
};
|
|
|
|
FastListManager.prototype.getLength = function() {
|
|
return this.tree.size;
|
|
};
|
|
|
|
return FastListManager;
|
|
|
|
})(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.size = 0;
|
|
this._nil = new Node('nil');
|
|
this._nil.red = false;
|
|
this._nil.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) {
|
|
this.size += 1;
|
|
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.size ++;
|
|
this.insert_correction(newNode);
|
|
} else if (onRight && !onRight.left) {
|
|
onRight.left = newNode;
|
|
newNode.parent = onRight;
|
|
|
|
this.size ++;
|
|
this.insert_correction(newNode);
|
|
} else {
|
|
this.insert_node(0, newNode);
|
|
}
|
|
newNode.traverse_up(function(node, parent) {
|
|
parent.weight -= 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 --; });
|
|
this.size -= 1;
|
|
|
|
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.next(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() {
|
|
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: fals;
|
|
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) {
|
|
this.data = data;
|
|
this.left = null;
|
|
this.right = null;
|
|
this.parent = null;
|
|
this.red = true;
|
|
this.weight = 1;
|
|
}
|
|
|
|
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.next = 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.prev = 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/Documents/prog/Linagora/yjs/node_modules/gulp-browserify/node_modules/browserify/node_modules/browser-pack/_prelude.js","/home/ccc/Documents/prog/Linagora/yjs/lib/ConnectorAdapter.coffee","/home/ccc/Documents/prog/Linagora/yjs/lib/ConnectorClass.coffee","/home/ccc/Documents/prog/Linagora/yjs/lib/Engine.coffee","/home/ccc/Documents/prog/Linagora/yjs/lib/HistoryBuffer.coffee","/home/ccc/Documents/prog/Linagora/yjs/lib/ObjectType.coffee","/home/ccc/Documents/prog/Linagora/yjs/lib/Operations/Basic.coffee","/home/ccc/Documents/prog/Linagora/yjs/lib/Operations/Structured.coffee","/home/ccc/Documents/prog/Linagora/yjs/lib/y.coffee","/home/ccc/Documents/prog/Linagora/yjs/node_modules/bintrees/lib/rbtree_by_index.js","/home/ccc/Documents/prog/Linagora/yjs/node_modules/bintrees/lib/treebase.js"],"names":[],"mappings":"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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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            oDistance = o.getDistanceToOrigin()\n            if o isnt @next_cl\n              # $o happened concurrently\n              if o.getDistanceToOrigin() 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      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    getNext: (start)->\n      o = start.next_cl\n      while not ((o instanceof ops.Delimiter))\n        if o.is_deleted\n          o = o.next_cl\n        else if o instanceof ops.Delimiter\n          return false\n        else break\n\n      o\n\n\n    # Transforms the the list to an array\n    # Doesn't return left-right delimiter.\n    toArray: ()->\n      o = @beginning.next_cl\n      result = []\n      while o isnt @end\n        if not o.is_deleted\n          result.push o.val()\n        o = o.next_cl\n      result\n\n    map: (f)->\n      o = @beginning.next_cl\n      result = []\n      while o isnt @end\n        if not o.is_deleted\n          result.push f(o)\n        o = o.next_cl\n      result\n\n    fold: (init, f)->\n      o = @beginning.next_cl\n      while o isnt @end\n        if not o.is_deleted\n          init = f(init, o)\n        o = o.next_cl\n      init\n\n    val: (pos)->\n      if pos?\n        o = @getOperationByPosition(pos+1)\n        if not (o instanceof ops.Delimiter)\n          o.val()\n        else\n          throw new Error \"this position does not exist\"\n      else\n        @toArray()\n\n    ref: (pos)->\n      if pos?\n        o = @getOperationByPosition(pos+1)\n        if not (o instanceof ops.Delimiter)\n          o\n        else\n          null\n          # throw new Error \"this position does not exist\"\n      else\n        throw new Error \"you must specify a position parameter\"\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      o = @beginning\n      while true\n        # find the i-th op\n        if o instanceof ops.Delimiter and o.prev_cl?\n          # the user or you gave a position parameter that is to big\n          # for the current array. Therefore we reach a Delimiter.\n          # Then, we'll just return the last character.\n          o = o.prev_cl\n          while o.isDeleted() and o.prev_cl?\n            o = o.prev_cl\n          break\n        if position <= 0 and not o.isDeleted()\n          break\n\n        o = o.next_cl\n        if not o.isDeleted()\n          position -= 1\n      o\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      right = left.next_cl\n      while right.isDeleted()\n        right = right.next_cl # find the first character to the right, that is not deleted. In the case that position is 0, its the Delimiter.\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        (new ops.Insert null, content, null, undefined, undefined, left, right).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).execute()\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\n    deleteRef: (o, length = 1) ->\n      delete_ops = []\n      for i in [0...length]\n        if o instanceof ops.Delimiter\n          break\n        d = (new ops.Delete null, undefined, o).execute()\n        o = o.next_cl\n        while (not (o instanceof ops.Delimiter)) and o.isDeleted()\n          o = o.next_cl\n        delete_ops.push d._encode()\n      @\n\n    delete: (position, length = 1)->\n      o = @getOperationByPosition(position+1) # position 0 in this case is the deletion of the first character\n\n      @deleteRef o, length\n\n\n    callOperationSpecificInsertEvents: (op)->\n      getContentType = (content)->\n        if content instanceof ops.Operation\n          content.getCustomType()\n        else\n          content\n      @callEvent [\n        type: \"insert\"\n        reference: op\n        position: op.getPosition()\n        object: @getCustomType()\n        changedBy: op.uid.creator\n        value: getContentType op.val()\n      ]\n\n    callOperationSpecificDeleteEvents: (op, del_op)->\n      @callEvent [\n        type: \"delete\"\n        reference: op\n        position: op.getPosition()\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: op.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: (op)->\n      if @tmp_composition_ref?\n        if op.uid.creator is @tmp_composition_ref.creator and op.uid.op_number is @tmp_composition_ref.op_number\n          @composition_ref = op\n          delete @tmp_composition_ref\n          op = op.next_cl\n          if op is @end\n            return\n        else\n          return\n\n      o = @end.prev_cl\n      while o isnt op\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: op.uid.creator\n        newValue: @val()\n      ]\n\n    callOperationSpecificDeleteEvents: (op, 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: (op)->\n      if op.next_cl.type is \"Delimiter\" and op.prev_cl.type isnt \"Delimiter\"\n        # this replaces another Replaceable\n        if not op.is_deleted # When this is received from the HB, this could already be deleted!\n          old_value = op.prev_cl.val()\n          @callEventDecorator [\n            type: \"update\"\n            changedBy: op.uid.creator\n            oldValue: old_value\n          ]\n        op.prev_cl.applyDelete()\n      else if op.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        op.applyDelete()\n      else # prev _and_ next are Delimiters. This is the first created Replaceable in the RM\n        @callEventDecorator [\n          type: \"add\"\n          changedBy: op.uid.creator\n        ]\n      undefined\n\n    callOperationSpecificDeleteEvents: (op, del_op)->\n      if op.next_cl.type is \"Delimiter\"\n        @callEventDecorator [\n          type: \"delete\"\n          changedBy: del_op.uid.creator\n          oldValue: op.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\n  class ops.FastListManager extends ops.ListManager\n    constructor: () ->\n      @shortTree = new RBTreeByIndex()\n\n      super arguments...\n\n    type: 'FastListManager'\n\n    getNext: (start)->\n      start.node.next().data\n\n    getPrev: (start)->\n      start.node.prev().data\n\n    map: (fun)->\n      @shortTree.map (operation) ->\n        fun operation\n\n    fold: (init, f) ->\n      @shortTree.each (operation) ->\n        init = f(init, operation)\n      init\n\n    val: (position)->\n      if position?\n        (@shortTree.find position).val()\n      else\n        @shortTree.map (operation) ->\n          operation.val()\n\n    ref: (position)->\n      if position?\n        @shortTree.find position\n      else\n        @shortTree.map (operation) ->\n          operation\n\n    push: (content) ->\n      @insertAfter @end.prev_cl, [content]\n\n    insertAfter: (left, contents) ->\n      # the operation is inserted just before the next non deleted insertion\n      # the operation is stored in the tree just after left\n      nodeOnLeft = if left is @beginning then null else left.node\n\n      nodeOnRight = if nodeOnLeft then nodeOnLeft.next() else @shortTree.find 0\n      right = if nodeOnRight then nodeOnRight.data else @end\n      left = right.prev_cl\n\n      if contents instanceof ops.Operation\n        operation = new ops.Insert null, content, null, undefined, undefined, left, right\n        operation.node = @shortTree.insertAfter nodeOnLeft, operation\n\n        operation.execute()\n\n      else\n        contents.forEach (content) ->\n          if content? and content._name? and content._getModel?\n            content = content._getModel(@custom_types, @operations)\n          #TODO: override Insert.prototype.getDistanceToOrigin (in Insert.prototype.execute)\n          operation = new ops.Insert null, c, null, undefined, undefined, left, right\n          operation.node = @shortTree.insertAfter nodeOnLeft, operation\n\n          operation.execute()\n\n          left = operation\n          nodeOnLeft = operation.node\n      @\n\n    insert: (position, contents) ->\n      left = (@shortTree.find (position-1)) or @beginning\n      @insertAfter left, contents\n\n    delete: (position, length = 1) ->\n      delete_ops = []\n      operation = @shortTree.find position\n      for i in [0...length]\n        if operation instanceof ops.Delimiter\n          break\n        deleteOp = new ops.Delete null, undefined, operation\n\n        # execute the delete operation\n        deleteOp.execute()\n\n        # get the next operation from the shortTree\n        nextNode = operation.node.next()\n        operation = if nextNode then nextNode.data else @end\n\n        # remove the operation from the shortTree\n        @shortTree.remove_node operation.node\n        operation.node = null\n\n        delete_ops.push d._encode()\n      @\n\n    getLength: () ->\n      @tree.size\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.size = 0;\n    this._nil = new Node('nil');\n    this._nil.red = false;\n    this._nil.weight = 0;\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  this.size += 1;\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.size ++;\n    this.insert_correction(newNode);\n  } else if (onRight && !onRight.left) {\n    onRight.left = newNode;\n    newNode.parent = onRight;\n\n    this.size ++;\n    this.insert_correction(newNode);\n  } else {\n    this.insert_node(0, newNode);\n  }\n  newNode.traverse_up(function(node, parent) {\n    parent.weight -= 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  this.size -= 1;\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.next(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  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() {\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: fals;\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\nfunction Node(data) {\n    this.data = data;\n    this.left = null;\n    this.right = null;\n    this.parent = null;\n    this.red = true;\n    this.weight = 1;\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\n\nNode.prototype.next = function(fun) {\n  var node, parent;\n  if (this.get_child('right')) {\n    return this.get_child('right').min_tree(fun);\n  }\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\n    if (fun) {\n      fun(parent);\n    }\n  }\n\n  return parent;\n};\n\nNode.prototype.prev = function(fun) {\n  var node, parent;\n  if (this.get_child('left')) {\n    return this.get_child('left').max_tree(fun);\n  }\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\n    if (fun) {\n      fun(parent);\n    }\n  }\n\n  return parent;\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"]}
|