mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-19 11:11:10 +01:00
11cb2f4f6c
Summary: this is D5750 but just the conpherence part. fixes a few random conpherence bugs / quirks as well. Also messes with ApplicationTransactionEditor to expose the xactions so Conpherence doesn't over-update participation rows. Fixes T2429. Test Plan: set LIMIT to 3. verified I could scroll down all conpherences. next, picked a conpherence "in the middle" to load. verified I could page up and down. next, picked a conpherence in the middle then had another user update that conpherence. verified as I paged up the conpherence re-loaded properly selected Reviewers: epriestley Reviewed By: epriestley CC: chad, aran, Korvin, vrana Maniphest Tasks: T2429 Differential Revision: https://secure.phabricator.com/D5783
387 lines
10 KiB
JavaScript
387 lines
10 KiB
JavaScript
/**
|
|
* @provides javelin-behavior-conpherence-menu
|
|
* @requires javelin-behavior
|
|
* javelin-dom
|
|
* javelin-util
|
|
* javelin-request
|
|
* javelin-stratcom
|
|
* javelin-workflow
|
|
* javelin-behavior-device
|
|
* javelin-history
|
|
* javelin-vector
|
|
*/
|
|
|
|
JX.behavior('conpherence-menu', function(config) {
|
|
|
|
var thread = {
|
|
selected: null,
|
|
node: null,
|
|
visible: null
|
|
};
|
|
|
|
function selectthreadid(id, updatePageData) {
|
|
var threads = JX.DOM.scry(document.body, 'a', 'conpherence-menu-click');
|
|
for (var ii = 0; ii < threads.length; ii++) {
|
|
var data = JX.Stratcom.getData(threads[ii]);
|
|
if (data.id == id) {
|
|
selectthread(threads[ii], updatePageData);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
function selectthread(node, updatePageData) {
|
|
|
|
if (thread.node) {
|
|
JX.DOM.alterClass(thread.node, 'conpherence-selected', false);
|
|
// keep the unread-count hidden still. big TODO once we ajax in updates
|
|
// to threads to make this work right and move threads between read /
|
|
// unread
|
|
}
|
|
|
|
JX.DOM.alterClass(node, 'conpherence-selected', true);
|
|
JX.DOM.alterClass(node, 'hide-unread-count', true);
|
|
|
|
thread.node = node;
|
|
|
|
var data = JX.Stratcom.getData(node);
|
|
thread.selected = data.id;
|
|
|
|
if (updatePageData) {
|
|
updatepagedata(data);
|
|
}
|
|
|
|
redrawthread();
|
|
}
|
|
|
|
JX.Stratcom.listen(
|
|
'conpherence-selectthread',
|
|
null,
|
|
function (e) {
|
|
var node = JX.$(e.getData().id);
|
|
selectthread(node);
|
|
}
|
|
);
|
|
|
|
function updatepagedata(data) {
|
|
var uri_suffix = thread.selected + '/';
|
|
if (data.use_base_uri) {
|
|
uri_suffix = '';
|
|
}
|
|
JX.History.replace(config.base_uri + uri_suffix);
|
|
if (data.title) {
|
|
document.title = data.title;
|
|
} else if (thread.node) {
|
|
var threadData = JX.Stratcom.getData(thread.node);
|
|
document.title = threadData.title;
|
|
}
|
|
}
|
|
|
|
JX.Stratcom.listen(
|
|
'conpherence-update-page-data',
|
|
null,
|
|
function (e) {
|
|
updatepagedata(e.getData());
|
|
}
|
|
);
|
|
|
|
function redrawthread() {
|
|
if (!thread.node) {
|
|
return;
|
|
}
|
|
|
|
if (thread.visible == thread.selected) {
|
|
return;
|
|
}
|
|
|
|
var data = JX.Stratcom.getData(thread.node);
|
|
|
|
if (thread.visible !== null || !config.hasThread) {
|
|
var uri = config.base_uri + data.id + '/';
|
|
new JX.Workflow(uri, {})
|
|
.setHandler(onloadthreadresponse)
|
|
.start();
|
|
} else {
|
|
didredrawthread();
|
|
}
|
|
|
|
if (thread.visible !== null || !config.hasWidgets) {
|
|
var widget_uri = config.base_uri + 'widget/' + data.id + '/';
|
|
new JX.Workflow(widget_uri, {})
|
|
.setHandler(onwidgetresponse)
|
|
.start();
|
|
} else {
|
|
updatetoggledwidget();
|
|
}
|
|
|
|
thread.visible = thread.selected;
|
|
}
|
|
|
|
function onwidgetresponse(response) {
|
|
var root = JX.DOM.find(document, 'div', 'conpherence-layout');
|
|
var widgetsRoot = JX.DOM.find(root, 'div', 'conpherence-widget-pane');
|
|
JX.DOM.setContent(widgetsRoot, JX.$H(response.widgets));
|
|
updatetoggledwidget();
|
|
}
|
|
|
|
function updatetoggledwidget() {
|
|
var device = JX.Device.getDevice();
|
|
if (device != 'desktop') {
|
|
if (config.role == 'list') {
|
|
JX.Stratcom.invoke(
|
|
'conpherence-toggle-widget',
|
|
null,
|
|
{
|
|
widget : 'conpherence-menu-pane'
|
|
}
|
|
);
|
|
} else {
|
|
JX.Stratcom.invoke(
|
|
'conpherence-toggle-widget',
|
|
null,
|
|
{
|
|
widget : 'conpherence-message-pane'
|
|
}
|
|
);
|
|
}
|
|
} else {
|
|
JX.Stratcom.invoke(
|
|
'conpherence-toggle-widget',
|
|
null,
|
|
{
|
|
widget : 'widgets-files'
|
|
}
|
|
);
|
|
}
|
|
}
|
|
|
|
function onloadthreadresponse(response) {
|
|
var header = JX.$H(response.header);
|
|
var messages = JX.$H(response.messages);
|
|
var form = JX.$H(response.form);
|
|
var root = JX.DOM.find(document, 'div', 'conpherence-layout');
|
|
var headerRoot = JX.DOM.find(root, 'div', 'conpherence-header-pane');
|
|
var messagesRoot = JX.DOM.find(root, 'div', 'conpherence-messages');
|
|
var formRoot = JX.DOM.find(root, 'div', 'conpherence-form');
|
|
JX.DOM.setContent(headerRoot, header);
|
|
JX.DOM.setContent(messagesRoot, messages);
|
|
JX.DOM.setContent(formRoot, form);
|
|
|
|
didredrawthread();
|
|
}
|
|
|
|
function didredrawthread() {
|
|
var root = JX.DOM.find(document, 'div', 'conpherence-layout');
|
|
var messagesRoot = JX.DOM.find(root, 'div', 'conpherence-messages');
|
|
messagesRoot.scrollTop = messagesRoot.scrollHeight;
|
|
}
|
|
|
|
JX.Stratcom.listen(
|
|
'conpherence-redraw-thread',
|
|
null,
|
|
function (e) {
|
|
didredrawthread();
|
|
}
|
|
);
|
|
|
|
JX.Stratcom.listen(
|
|
'click',
|
|
'conpherence-menu-click',
|
|
function(e) {
|
|
if (!e.isNormalClick()) {
|
|
return;
|
|
}
|
|
|
|
// On devices, just follow the link normally.
|
|
if (JX.Device.getDevice() != 'desktop') {
|
|
return;
|
|
}
|
|
|
|
e.kill();
|
|
selectthread(e.getNode('conpherence-menu-click'), true);
|
|
});
|
|
|
|
JX.Stratcom.listen('click', 'conpherence-edit-metadata', function (e) {
|
|
e.kill();
|
|
var root = e.getNode('conpherence-layout');
|
|
var form = JX.DOM.find(root, 'form', 'conpherence-pontificate');
|
|
var data = e.getNodeData('conpherence-edit-metadata');
|
|
var header = JX.DOM.find(root, 'div', 'conpherence-header-pane');
|
|
var messages = JX.DOM.find(root, 'div', 'conpherence-messages');
|
|
|
|
new JX.Workflow.newFromForm(form, data)
|
|
.setHandler(JX.bind(this, function(r) {
|
|
JX.DOM.appendContent(messages, JX.$H(r.transactions));
|
|
messages.scrollTop = messages.scrollHeight;
|
|
|
|
JX.DOM.setContent(
|
|
header,
|
|
JX.$H(r.header)
|
|
);
|
|
|
|
try {
|
|
// update the menu entry
|
|
JX.DOM.replace(
|
|
JX.$(r.conpherence_phid + '-nav-item'),
|
|
JX.$H(r.nav_item)
|
|
);
|
|
JX.Stratcom.invoke(
|
|
'conpherence-selectthread',
|
|
null,
|
|
{ id : r.conpherence_phid + '-nav-item' }
|
|
);
|
|
} catch (ex) {
|
|
// Ignore; this view may not have a menu.
|
|
}
|
|
}))
|
|
.start();
|
|
});
|
|
|
|
JX.Stratcom.listen('click', 'show-older-messages', function(e) {
|
|
e.kill();
|
|
var data = e.getNodeData('show-older-messages');
|
|
var oldest_transaction_id = data.oldest_transaction_id;
|
|
var conf_id = thread.selected;
|
|
JX.DOM.remove(e.getNode('show-older-messages'));
|
|
var root = JX.DOM.find(document, 'div', 'conpherence-layout');
|
|
var messages_root = JX.DOM.find(root, 'div', 'conpherence-messages');
|
|
new JX.Request(config.base_uri + conf_id + '/', function(r) {
|
|
var messages = JX.$H(r.messages);
|
|
JX.DOM.prependContent(
|
|
messages_root,
|
|
JX.$H(messages));
|
|
}).setData({ oldest_transaction_id : oldest_transaction_id }).send();
|
|
});
|
|
|
|
// On mobile, we just show a thread list, so we don't want to automatically
|
|
// select or load any threads. On Desktop, we automatically select the first
|
|
// thread.
|
|
var old_device = null;
|
|
function ondevicechange() {
|
|
var new_device = JX.Device.getDevice();
|
|
if (new_device === old_device) {
|
|
return;
|
|
}
|
|
var update_toggled_widget =
|
|
new_device == 'desktop' || old_device == 'desktop';
|
|
old_device = new_device;
|
|
|
|
if (thread.visible !== null && update_toggled_widget) {
|
|
updatetoggledwidget();
|
|
}
|
|
|
|
if (!config.hasThreadList) {
|
|
loadthreads();
|
|
} else {
|
|
didloadthreads();
|
|
}
|
|
}
|
|
|
|
JX.Stratcom.listen('phabricator-device-change', null, ondevicechange);
|
|
ondevicechange();
|
|
|
|
function loadthreads() {
|
|
var uri = config.base_uri + 'thread/' + config.selectedID + '/';
|
|
new JX.Workflow(uri)
|
|
.setHandler(onloadthreadsresponse)
|
|
.start();
|
|
}
|
|
|
|
function onloadthreadsresponse(r) {
|
|
var layout = JX.$(config.layoutID);
|
|
var menu = JX.DOM.find(layout, 'div', 'conpherence-menu-pane');
|
|
JX.DOM.setContent(menu, JX.$H(r));
|
|
|
|
config.selectedID && selectthreadid(config.selectedID);
|
|
|
|
thread.node.scrollIntoView();
|
|
}
|
|
|
|
function didloadthreads() {
|
|
// If there's no thread selected yet, select the current thread or the
|
|
// first thread.
|
|
if (!thread.selected) {
|
|
if (config.selectedID) {
|
|
selectthreadid(config.selectedID, true);
|
|
} else {
|
|
var layout = JX.$(config.layoutID);
|
|
var threads = JX.DOM.scry(layout, 'a', 'conpherence-menu-click');
|
|
if (threads.length) {
|
|
selectthread(threads[0]);
|
|
} else {
|
|
var nothreads = JX.DOM.find(layout, 'div', 'conpherence-no-threads');
|
|
nothreads.style.display = 'block';
|
|
}
|
|
}
|
|
}
|
|
redrawthread();
|
|
}
|
|
|
|
var handlethreadscrollers = function (e) {
|
|
e.kill();
|
|
|
|
var data = e.getNodeData('conpherence-menu-scroller');
|
|
var scroller = e.getNode('conpherence-menu-scroller');
|
|
new JX.Workflow(scroller.href, data)
|
|
.setHandler(
|
|
JX.bind(null, threadscrollerresponse, scroller, data.direction))
|
|
.start();
|
|
};
|
|
|
|
var threadscrollerresponse = function (scroller, direction, r) {
|
|
var html = JX.$H(r.html);
|
|
|
|
var threadPhids = r.phids;
|
|
var reselectId = null;
|
|
// remove any threads that are in the list that we just got back
|
|
// in the result set; things have changed and they'll be in the
|
|
// right place soon
|
|
for (var ii = 0; ii < threadPhids.length; ii++) {
|
|
try {
|
|
var nodeId = threadPhids[ii] + '-nav-item';
|
|
var node = JX.$(nodeId);
|
|
var nodeData = JX.Stratcom.getData(node);
|
|
if (nodeData.id == thread.selected) {
|
|
reselectId = nodeId;
|
|
}
|
|
JX.DOM.remove(node);
|
|
} catch (ex) {
|
|
// ignore , just haven't seen this thread yet
|
|
}
|
|
}
|
|
|
|
var root = JX.DOM.find(document, 'div', 'conpherence-layout');
|
|
var menuRoot = JX.DOM.find(root, 'div', 'conpherence-menu-pane');
|
|
var scrollY = 0;
|
|
// we have to do some hyjinx in the up case to make the menu scroll to
|
|
// where it should
|
|
if (direction == 'up') {
|
|
var style = {
|
|
position: 'absolute',
|
|
left: '-10000px'
|
|
};
|
|
var test_size = JX.$N('div', {style: style}, html);
|
|
document.body.appendChild(test_size);
|
|
var html_size = JX.Vector.getDim(test_size);
|
|
JX.DOM.remove(test_size);
|
|
scrollY = html_size.y;
|
|
}
|
|
JX.DOM.replace(scroller, html);
|
|
menuRoot.scrollTop += scrollY;
|
|
|
|
if (reselectId) {
|
|
JX.Stratcom.invoke(
|
|
'conpherence-selectthread',
|
|
null,
|
|
{ id : reselectId }
|
|
);
|
|
}
|
|
};
|
|
|
|
JX.Stratcom.listen(
|
|
['click'],
|
|
'conpherence-menu-scroller',
|
|
handlethreadscrollers
|
|
);
|
|
|
|
});
|