Commit 434e21d0 authored by JC Brand's avatar JC Brand

Some sorting optimizations

* Differentiate between adding new roster items and rendering existing ones.
* Also, only sort pending and requesting contacts once they've all been added
  (similar to what was already being done with existing contacts)
parent b1b63b02
...@@ -122,6 +122,14 @@ ...@@ -122,6 +122,14 @@
var KEY = { var KEY = {
ENTER: 13 ENTER: 13
}; };
var STATUS_WEIGHTS = {
'offline': 6,
'unavailable': 5,
'xa': 4,
'away': 3,
'dnd': 2,
'online': 1
};
var HAS_CSPRNG = ((typeof crypto !== 'undefined') && var HAS_CSPRNG = ((typeof crypto !== 'undefined') &&
((typeof crypto.randomBytes === 'function') || ((typeof crypto.randomBytes === 'function') ||
(typeof crypto.getRandomValues === 'function') (typeof crypto.getRandomValues === 'function')
...@@ -3231,7 +3239,7 @@ ...@@ -3231,7 +3239,7 @@
initialize: function () { initialize: function () {
this.model.on("add", function (item) { this.model.on("add", function (item) {
this.addRosterItemView(item).renderRosterItem(item).updateRoster(); this.addRosterItemView(item).addRosterItem(item).updateRoster();
if (item.get('is_last')) { if (item.get('is_last')) {
this.sortRoster().showRoster(); this.sortRoster().showRoster();
} }
...@@ -3317,33 +3325,40 @@ ...@@ -3317,33 +3325,40 @@
}, },
renderRosterItem: function (item) { renderRosterItem: function (item) {
var $contact_requests = this.$('#xmpp-contact-requests'),
$pending_contacts = this.$('#pending-xmpp-contacts'),
$current_contacts = this.$el.find('.roster-group'),
crit = {order:'asc'};
var view = this.get(item.id); var view = this.get(item.id);
if (!view) { if (!view) {
// This method should only be called for items with views. converse.log("renderRosterItem called with item that doesn't have a view", "error");
return; return this;
} }
if ((converse.show_only_online_users) && (item.get('chat_status') !== 'online')) { if ((converse.show_only_online_users) && (item.get('chat_status') !== 'online')) {
view.$el.remove(); view.$el.remove();
view.delegateEvents(); view.delegateEvents();
} else {
view.render()
}
return this;
},
addRosterItem: function (item) {
if ((converse.show_only_online_users) && (item.get('chat_status') !== 'online')) {
return this;
}
var view = this.get(item.id);
if (!view) {
converse.log("renderRosterItem called with item that doesn't have a view", "error");
return this; return this;
} }
// TODO: need to choose proper group var $contact_requests = this.$('#xmpp-contact-requests'),
// FIXME: DON'T SORT CONTINUOUSLY, SUPER SLOW $pending_contacts = this.$('#pending-xmpp-contacts'),
// TODO: Insert the item in the correct order $current_contacts = this.$el.find('.roster-group');
view.render() view.render()
// TODO: need to add group support
if (view.$el.hasClass('current-xmpp-contact')) { if (view.$el.hasClass('current-xmpp-contact')) {
$current_contacts.after(view.el); $current_contacts.after(view.el);
$current_contacts.after($current_contacts.siblings('dd.current-xmpp-contact').tsort(crit));
} else if (view.$el.hasClass('pending-xmpp-contact')) { } else if (view.$el.hasClass('pending-xmpp-contact')) {
$pending_contacts.after(view.el); $pending_contacts.after(view.el);
$pending_contacts.after($pending_contacts.siblings('dd.pending-xmpp-contact').tsort(crit));
} else if (view.$el.hasClass('requesting-xmpp-contact')) { } else if (view.$el.hasClass('requesting-xmpp-contact')) {
$contact_requests.after(view.render().el); $contact_requests.after(view.render().el);
$contact_requests.after($contact_requests.siblings('dd.requesting-xmpp-contact').tsort(crit));
} }
return this; return this;
}, },
...@@ -3399,31 +3414,34 @@ ...@@ -3399,31 +3414,34 @@
} }
}, },
sortRoster: function (chat_status) { sortFunction: function (a, b) {
var sortFunction = function (a, b) {
var a_status = a.s[0], var a_status = a.s[0],
a_name =a.s[1], a_name =a.s[1],
b_status = b.s[0], b_status = b.s[0],
b_name =b.s[1], b_name =b.s[1];
comp = { if (STATUS_WEIGHTS[a_status] === STATUS_WEIGHTS[b_status]) {
'offline': 6,
'unavailable': 5,
'xa': 4,
'away': 3,
'dnd': 2,
'online': 1
};
if (comp[a_status] === comp[b_status]) {
return a_name < b_name ? -1 : (a_name > b_name ? 1 : 0); return a_name < b_name ? -1 : (a_name > b_name ? 1 : 0);
} else { } else {
return comp[a_status] < comp[b_status] ? -1 : 1; return STATUS_WEIGHTS[a_status] < STATUS_WEIGHTS[b_status] ? -1 : 1;
} }
}; },
this.$el.find('.roster-group').each(function (idx, group) {
sortRoster: function (chat_status) {
/* XXX
* 1). See if the jquery detach method can be somehow used to avoid DOM reflows.
* 2). Likewise see if documentFragment can be used.
*/
this.$el.find('.roster-group').each($.proxy(function (idx, group) {
var $group = $(group); var $group = $(group);
var $contacts = $group.nextUntil('dt', 'dd.current-xmpp-contact'); var $contacts = $group.nextUntil('dt', 'dd.current-xmpp-contact');
$group.after($contacts.tsort({sortFunction: sortFunction, data: 'status'}, 'a')); $group.after($contacts.tsort({sortFunction: this.sortFunction, data: 'status'}, 'a'));
}); },this));
// Also sort pending and requesting contacts
var crit = {order:'asc'},
$contact_requests = this.$('#xmpp-contact-requests'),
$pending_contacts = this.$('#pending-xmpp-contacts');
$pending_contacts.after($pending_contacts.siblings('dd.pending-xmpp-contact').tsort(crit));
$contact_requests.after($contact_requests.siblings('dd.requesting-xmpp-contact').tsort(crit));
return this; return this;
}, },
......
...@@ -215,10 +215,10 @@ ...@@ -215,10 +215,10 @@
}); });
expect(this.rosterview.updateRoster).toHaveBeenCalled(); expect(this.rosterview.updateRoster).toHaveBeenCalled();
expect(converse.emit).toHaveBeenCalledWith('rosterViewUpdated'); expect(converse.emit).toHaveBeenCalledWith('rosterViewUpdated');
}
// Check that they are sorted alphabetically // Check that they are sorted alphabetically
t = this.rosterview.$el.find('dt#pending-xmpp-contacts').siblings('dd.pending-xmpp-contact').find('span').text(); t = this.rosterview.$el.find('dt#pending-xmpp-contacts').siblings('dd.pending-xmpp-contact').find('span').text();
expect(t).toEqual(mock.pend_names.slice(0,i+1).sort().join('')); expect(t).toEqual(mock.pend_names.slice(0,i+1).sort().join(''));
}
}, converse)); }, converse));
}, converse)); }, converse));
...@@ -478,16 +478,16 @@ ...@@ -478,16 +478,16 @@
is_last: i===(mock.req_names.length-1) is_last: i===(mock.req_names.length-1)
}); });
expect(this.rosterview.updateRoster).toHaveBeenCalled(); expect(this.rosterview.updateRoster).toHaveBeenCalled();
// Check that they are sorted alphabetically
children = this.rosterview.$el.find('dt#xmpp-contact-requests').siblings('dd.requesting-xmpp-contact').children('span');
names = [];
children.each(addName);
expect(names.join('')).toEqual(mock.req_names.slice(0,i+1).sort().join(''));
// When a requesting contact is added, the controlbox must // When a requesting contact is added, the controlbox must
// be opened. // be opened.
expect(this.controlboxtoggle.showControlBox).toHaveBeenCalled(); expect(this.controlboxtoggle.showControlBox).toHaveBeenCalled();
expect(converse.emit).toHaveBeenCalledWith('rosterViewUpdated'); expect(converse.emit).toHaveBeenCalledWith('rosterViewUpdated');
} }
// Check that they are sorted alphabetically
children = this.rosterview.$el.find('dt#xmpp-contact-requests').siblings('dd.requesting-xmpp-contact').children('span');
names = [];
children.each(addName);
expect(names.join('')).toEqual(mock.req_names.slice(0,i+1).sort().join(''));
}, converse)); }, converse));
it("can be collapsed under their own header", $.proxy(function () { it("can be collapsed under their own header", $.proxy(function () {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment