2015-03-10 20:20:29 +01:00
|
|
|
/**
|
|
|
|
* @provides conpherence-thread-manager
|
|
|
|
* @requires javelin-dom
|
|
|
|
* javelin-util
|
|
|
|
* javelin-stratcom
|
|
|
|
* javelin-install
|
2015-05-07 21:47:49 +02:00
|
|
|
* javelin-aphlict
|
2015-03-10 20:20:29 +01:00
|
|
|
* javelin-workflow
|
|
|
|
* javelin-router
|
|
|
|
* javelin-behavior-device
|
|
|
|
* javelin-vector
|
|
|
|
*/
|
|
|
|
JX.install('ConpherenceThreadManager', {
|
|
|
|
|
|
|
|
construct : function() {
|
|
|
|
if (__DEV__) {
|
|
|
|
if (JX.ConpherenceThreadManager._instance) {
|
|
|
|
JX.$E('ConpherenceThreadManager object is a singleton.');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
JX.ConpherenceThreadManager._instance = this;
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
members: {
|
|
|
|
_loadThreadURI: null,
|
|
|
|
_loadedThreadID: null,
|
|
|
|
_loadedThreadPHID: null,
|
|
|
|
_latestTransactionID: null,
|
2015-03-25 19:48:22 +01:00
|
|
|
_canEditLoadedThread: null,
|
2015-03-10 20:20:29 +01:00
|
|
|
_updating: null,
|
|
|
|
_minimalDisplay: false,
|
|
|
|
_willLoadThreadCallback: JX.bag,
|
|
|
|
_didLoadThreadCallback: JX.bag,
|
2015-03-10 21:53:30 +01:00
|
|
|
_didUpdateThreadCallback: JX.bag,
|
2015-03-10 20:20:29 +01:00
|
|
|
_willSendMessageCallback: JX.bag,
|
|
|
|
_didSendMessageCallback: JX.bag,
|
2015-03-10 21:53:30 +01:00
|
|
|
_willUpdateWorkflowCallback: JX.bag,
|
|
|
|
_didUpdateWorkflowCallback: JX.bag,
|
2015-03-10 20:20:29 +01:00
|
|
|
|
|
|
|
setLoadThreadURI: function(uri) {
|
|
|
|
this._loadThreadURI = uri;
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
getLoadThreadURI: function() {
|
|
|
|
return this._loadThreadURI;
|
|
|
|
},
|
|
|
|
|
|
|
|
isThreadLoaded: function() {
|
|
|
|
return Boolean(this._loadedThreadID);
|
|
|
|
},
|
|
|
|
|
|
|
|
isThreadIDLoaded: function(thread_id) {
|
|
|
|
return this._loadedThreadID == thread_id;
|
|
|
|
},
|
|
|
|
|
|
|
|
getLoadedThreadID: function() {
|
|
|
|
return this._loadedThreadID;
|
|
|
|
},
|
|
|
|
|
2015-03-10 21:53:30 +01:00
|
|
|
setLoadedThreadID: function(id) {
|
|
|
|
this._loadedThreadID = id;
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2015-03-10 20:20:29 +01:00
|
|
|
getLoadedThreadPHID: function() {
|
|
|
|
return this._loadedThreadPHID;
|
|
|
|
},
|
|
|
|
|
2015-03-10 21:53:30 +01:00
|
|
|
setLoadedThreadPHID: function(phid) {
|
|
|
|
this._loadedThreadPHID = phid;
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2015-03-10 20:20:29 +01:00
|
|
|
getLatestTransactionID: function() {
|
|
|
|
return this._latestTransactionID;
|
|
|
|
},
|
|
|
|
|
|
|
|
setLatestTransactionID: function(id) {
|
|
|
|
this._latestTransactionID = id;
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2015-03-25 19:48:22 +01:00
|
|
|
setCanEditLoadedThread: function(bool) {
|
|
|
|
this._canEditLoadedThread = bool;
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
getCanEditLoadedThread: function() {
|
|
|
|
if (this._canEditLoadedThread === null) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return this._canEditLoadedThread;
|
|
|
|
},
|
|
|
|
|
2015-03-10 20:20:29 +01:00
|
|
|
setMinimalDisplay: function(bool) {
|
|
|
|
this._minimalDisplay = bool;
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
setWillLoadThreadCallback: function(callback) {
|
|
|
|
this._willLoadThreadCallback = callback;
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
setDidLoadThreadCallback: function(callback) {
|
|
|
|
this._didLoadThreadCallback = callback;
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2015-03-10 21:53:30 +01:00
|
|
|
setDidUpdateThreadCallback: function(callback) {
|
|
|
|
this._didUpdateThreadCallback = callback;
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2015-03-10 20:20:29 +01:00
|
|
|
setWillSendMessageCallback: function(callback) {
|
|
|
|
this._willSendMessageCallback = callback;
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
setDidSendMessageCallback: function(callback) {
|
|
|
|
this._didSendMessageCallback = callback;
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2015-03-10 21:53:30 +01:00
|
|
|
setWillUpdateWorkflowCallback: function(callback) {
|
|
|
|
this._willUpdateWorkflowCallback = callback;
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
setDidUpdateWorkflowCallback: function(callback) {
|
|
|
|
this._didUpdateWorkflowCallback = callback;
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2015-03-10 20:20:29 +01:00
|
|
|
_getParams: function(base_params) {
|
|
|
|
if (this._minimalDisplay) {
|
|
|
|
base_params.minimal_display = true;
|
|
|
|
}
|
|
|
|
if (this._latestTransactionID) {
|
|
|
|
base_params.latest_transaction_id = this._latestTransactionID;
|
|
|
|
}
|
|
|
|
return base_params;
|
|
|
|
},
|
2015-03-10 21:53:30 +01:00
|
|
|
|
2015-03-10 20:20:29 +01:00
|
|
|
start: function() {
|
|
|
|
JX.Stratcom.listen(
|
|
|
|
'aphlict-server-message',
|
|
|
|
null,
|
|
|
|
JX.bind(this, function(e) {
|
|
|
|
var message = e.getData();
|
|
|
|
|
|
|
|
if (message.type != 'message') {
|
|
|
|
// Not a message event.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (message.threadPHID != this._loadedThreadPHID) {
|
|
|
|
// Message event for some thread other than the visible one.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (message.messageID <= this._latestTransactionID) {
|
|
|
|
// Message event for something we already know about.
|
|
|
|
return;
|
|
|
|
}
|
2015-05-07 20:26:48 +02:00
|
|
|
|
2015-03-10 20:20:29 +01:00
|
|
|
// If this notification tells us about a message which is newer than
|
2015-05-07 20:26:48 +02:00
|
|
|
// the newest one we know to exist, update our latest knownID so we
|
|
|
|
// can properly update later.
|
2015-03-10 20:20:29 +01:00
|
|
|
if (this._updating &&
|
|
|
|
this._updating.threadPHID == this._loadedThreadPHID) {
|
|
|
|
if (message.messageID > this._updating.knownID) {
|
|
|
|
this._updating.knownID = message.messageID;
|
2015-05-07 20:26:48 +02:00
|
|
|
// We're currently updating, so wait for the update to complete.
|
|
|
|
// this.syncWorkflow has us covered in this case.
|
|
|
|
if (this._updating.active) {
|
|
|
|
return;
|
|
|
|
}
|
2015-03-10 20:20:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this._updateThread();
|
|
|
|
}));
|
|
|
|
},
|
|
|
|
|
2015-03-17 00:35:05 +01:00
|
|
|
_shouldUpdateDOM: function(r) {
|
|
|
|
if (this._updating &&
|
|
|
|
this._updating.threadPHID == this._loadedThreadPHID) {
|
2015-04-10 00:07:12 +02:00
|
|
|
|
|
|
|
if (r.non_update) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-03-17 00:35:05 +01:00
|
|
|
// we have a different, more current update in progress so
|
|
|
|
// return early
|
|
|
|
if (r.latest_transaction_id < this._updating.knownID) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
2015-03-18 01:01:33 +01:00
|
|
|
_markUpdated: function(r) {
|
|
|
|
this._updating.knownID = r.latest_transaction_id;
|
|
|
|
this._latestTransactionID = r.latest_transaction_id;
|
2015-05-08 01:04:56 +02:00
|
|
|
JX.Stratcom.invoke(
|
|
|
|
'conpherence-redraw-aphlict',
|
|
|
|
null,
|
|
|
|
r.aphlictDropdownData);
|
2015-03-18 01:01:33 +01:00
|
|
|
},
|
|
|
|
|
2015-03-10 20:20:29 +01:00
|
|
|
_updateThread: function() {
|
|
|
|
var params = this._getParams({
|
|
|
|
action: 'load',
|
|
|
|
});
|
|
|
|
|
|
|
|
var uri = '/conpherence/update/' + this._loadedThreadID + '/';
|
|
|
|
|
|
|
|
var workflow = new JX.Workflow(uri)
|
|
|
|
.setData(params)
|
|
|
|
.setHandler(JX.bind(this, function(r) {
|
2015-03-18 01:01:33 +01:00
|
|
|
if (this._shouldUpdateDOM(r)) {
|
|
|
|
this._markUpdated(r);
|
2015-03-17 21:40:36 +01:00
|
|
|
|
2015-03-18 01:01:33 +01:00
|
|
|
this._didUpdateThreadCallback(r);
|
|
|
|
}
|
2015-03-10 20:20:29 +01:00
|
|
|
}));
|
|
|
|
|
|
|
|
this.syncWorkflow(workflow, 'finally');
|
|
|
|
},
|
|
|
|
|
|
|
|
syncWorkflow: function(workflow, stage) {
|
|
|
|
this._updating = {
|
|
|
|
threadPHID: this._loadedThreadPHID,
|
2015-05-07 20:26:48 +02:00
|
|
|
knownID: this._latestTransactionID,
|
|
|
|
active: true
|
2015-03-10 20:20:29 +01:00
|
|
|
};
|
|
|
|
workflow.listen(stage, JX.bind(this, function() {
|
|
|
|
// TODO - do we need to handle if we switch threads somehow?
|
2015-03-17 00:35:05 +01:00
|
|
|
var need_sync = this._updating &&
|
|
|
|
(this._updating.knownID > this._latestTransactionID);
|
2015-03-10 20:20:29 +01:00
|
|
|
if (need_sync) {
|
2015-03-17 00:35:05 +01:00
|
|
|
return this._updateThread();
|
2015-03-10 20:20:29 +01:00
|
|
|
}
|
2015-05-07 20:26:48 +02:00
|
|
|
this._updating.active = false;
|
2015-03-10 20:20:29 +01:00
|
|
|
}));
|
|
|
|
workflow.start();
|
|
|
|
},
|
|
|
|
|
|
|
|
runUpdateWorkflowFromLink: function(link, params) {
|
|
|
|
params = this._getParams(params);
|
2015-03-10 21:53:30 +01:00
|
|
|
this._willUpdateWorkflowCallback();
|
2015-03-10 20:20:29 +01:00
|
|
|
var workflow = new JX.Workflow.newFromLink(link)
|
|
|
|
.setData(params)
|
|
|
|
.setHandler(JX.bind(this, function(r) {
|
2015-03-17 00:35:05 +01:00
|
|
|
if (this._shouldUpdateDOM(r)) {
|
2015-03-18 01:01:33 +01:00
|
|
|
this._markUpdated(r);
|
2015-03-17 21:40:36 +01:00
|
|
|
|
2015-03-17 00:35:05 +01:00
|
|
|
this._didUpdateWorkflowCallback(r);
|
|
|
|
}
|
2015-03-10 20:20:29 +01:00
|
|
|
}));
|
|
|
|
this.syncWorkflow(workflow, params.stage);
|
|
|
|
},
|
|
|
|
|
2015-05-01 01:54:57 +02:00
|
|
|
loadThreadByID: function(thread_id, force_reload) {
|
2015-03-10 20:20:29 +01:00
|
|
|
if (this.isThreadLoaded() &&
|
2015-05-01 01:54:57 +02:00
|
|
|
this.isThreadIDLoaded(thread_id) &&
|
|
|
|
!force_reload) {
|
2015-03-10 20:20:29 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this._willLoadThreadCallback();
|
|
|
|
|
|
|
|
var params = {};
|
|
|
|
// We pick a thread from the server if not specified
|
|
|
|
if (thread_id) {
|
|
|
|
params.id = thread_id;
|
|
|
|
}
|
|
|
|
params = this._getParams(params);
|
|
|
|
|
|
|
|
var handler = JX.bind(this, function(r) {
|
2015-05-07 21:47:49 +02:00
|
|
|
var client = JX.Aphlict.getInstance();
|
|
|
|
if (client) {
|
|
|
|
var old_subs = client.getSubscriptions();
|
|
|
|
var new_subs = [];
|
|
|
|
for (var ii = 0; ii < old_subs.length; ii++) {
|
|
|
|
if (old_subs[ii] == this._loadedThreadPHID) {
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
new_subs.push(old_subs[ii]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
new_subs.push(r.threadPHID);
|
|
|
|
client.clearSubscriptions(client.getSubscriptions());
|
|
|
|
client.setSubscriptions(new_subs);
|
|
|
|
}
|
2015-03-10 20:20:29 +01:00
|
|
|
this._loadedThreadID = r.threadID;
|
|
|
|
this._loadedThreadPHID = r.threadPHID;
|
|
|
|
this._latestTransactionID = r.latestTransactionID;
|
2015-03-25 19:48:22 +01:00
|
|
|
this._canEditLoadedThread = r.canEdit;
|
2015-05-07 21:47:49 +02:00
|
|
|
|
2015-05-08 01:04:56 +02:00
|
|
|
JX.Stratcom.invoke(
|
|
|
|
'conpherence-redraw-aphlict',
|
|
|
|
null,
|
|
|
|
r.aphlictDropdownData);
|
2015-03-10 20:20:29 +01:00
|
|
|
|
|
|
|
this._didLoadThreadCallback(r);
|
2015-05-01 01:54:57 +02:00
|
|
|
|
|
|
|
if (force_reload) {
|
|
|
|
JX.Stratcom.invoke('hashchange');
|
|
|
|
}
|
2015-03-10 20:20:29 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
// should this be sync'd too?
|
|
|
|
new JX.Workflow(this.getLoadThreadURI())
|
|
|
|
.setData(params)
|
|
|
|
.setHandler(handler)
|
|
|
|
.start();
|
|
|
|
},
|
|
|
|
|
|
|
|
sendMessage: function(form, params) {
|
2015-04-10 00:07:12 +02:00
|
|
|
// don't bother sending up text if there is nothing to submit
|
|
|
|
var textarea = JX.DOM.find(form, 'textarea');
|
|
|
|
if (!textarea.value.length) {
|
|
|
|
return;
|
|
|
|
}
|
2015-03-10 20:20:29 +01:00
|
|
|
params = this._getParams(params);
|
|
|
|
|
2015-03-14 20:00:17 +01:00
|
|
|
var keep_enabled = true;
|
|
|
|
|
|
|
|
var workflow = JX.Workflow.newFromForm(form, params, keep_enabled)
|
2015-03-10 20:20:29 +01:00
|
|
|
.setHandler(JX.bind(this, function(r) {
|
2015-03-17 00:35:05 +01:00
|
|
|
if (this._shouldUpdateDOM(r)) {
|
2015-03-18 01:01:33 +01:00
|
|
|
this._markUpdated(r);
|
2015-03-17 21:40:36 +01:00
|
|
|
|
2015-03-17 00:35:05 +01:00
|
|
|
this._didSendMessageCallback(r);
|
2015-04-10 00:07:12 +02:00
|
|
|
} else if (r.non_update) {
|
|
|
|
this._didSendMessageCallback(r, true);
|
2015-03-17 00:35:05 +01:00
|
|
|
}
|
2015-03-10 20:20:29 +01:00
|
|
|
}));
|
|
|
|
this.syncWorkflow(workflow, 'finally');
|
2015-03-14 20:00:17 +01:00
|
|
|
|
|
|
|
this._willSendMessageCallback();
|
2015-03-11 23:54:07 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
handleDraftKeydown: function(e) {
|
|
|
|
var form = e.getNode('tag:form');
|
|
|
|
var data = e.getNodeData('tag:form');
|
|
|
|
|
|
|
|
if (!data.preview) {
|
|
|
|
var uri = '/conpherence/update/' + this._loadedThreadID + '/';
|
|
|
|
data.preview = new JX.PhabricatorShapedRequest(
|
|
|
|
uri,
|
|
|
|
JX.bag,
|
|
|
|
JX.bind(this, function () {
|
|
|
|
var data = JX.DOM.convertFormToDictionary(form);
|
|
|
|
data.action = 'draft';
|
|
|
|
data = this._getParams(data);
|
|
|
|
return data;
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
data.preview.trigger();
|
2015-03-10 20:20:29 +01:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
statics: {
|
|
|
|
_instance: null,
|
|
|
|
|
|
|
|
getInstance: function() {
|
|
|
|
var self = JX.ConpherenceThreadManager;
|
|
|
|
if (!self._instance) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return self._instance;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|