Commit 48e4ed27 authored by JC Brand's avatar JC Brand

Big refactoring.

Removed ControlBox model, and instead only create ChatBox models inside
ChatBoxes collection (the Controlbox is now also a ChatBox).

This enables us to use backbone.localStorage for saving open chatboxes (saving
and fetching currently not implemented).

createChatBox now doesn't return a ChatBoxView instance anymore. In the case
where a view was needed (to append a message), we create the message on the
ChatBox and let the view update via an event trigger. This is much more Backbonic ;)

Also created a Message model. This will most likely enable us to in the future also
store messages via backbone.localStorage.
parent 2fe7164d
...@@ -99,6 +99,12 @@ ...@@ -99,6 +99,12 @@
return String(str).indexOf(needle) !== -1; return String(str).indexOf(needle) !== -1;
}; };
xmppchat.autoLink = function (text) {
// Convert URLs into hyperlinks
var re = /((http|https|ftp):\/\/[\w?=&.\/\-;#~%\-]+(?![\w\s?&.\/;#~%"=\-]*>))/g;
return text.replace(re, '<a target="_blank" href="$1">$1</a>');
};
xmppchat.toISOString = function (date) { xmppchat.toISOString = function (date) {
var pad; var pad;
if (typeof date.toISOString !== 'undefined') { if (typeof date.toISOString !== 'undefined') {
...@@ -283,17 +289,73 @@ ...@@ -283,17 +289,73 @@
} }
}); });
xmppchat.Message = Backbone.Model.extend();
xmppchat.Messages = Backbone.Collection.extend({
model: xmppchat.Message
});
xmppchat.ChatBox = Backbone.Model.extend({ xmppchat.ChatBox = Backbone.Model.extend({
messages: new xmppchat.Messages(),
initialize: function () { initialize: function () {
this.set({ if (this.get('box_id') !== 'controlbox') {
'user_id' : Strophe.getNodeFromJid(this.get('jid')), this.set({
'box_id' : hex_sha1(this.get('jid')), 'user_id' : Strophe.getNodeFromJid(this.get('jid')),
'fullname' : this.get('fullname'), 'box_id' : hex_sha1(this.get('jid')),
'url': this.get('url'), 'fullname' : this.get('fullname'),
'image_type': this.get('image_type'), 'url': this.get('url'),
'image_src': this.get('image_src') 'image_type': this.get('image_type'),
}); 'image_src': this.get('image_src')
} });
}
},
messageReceived: function (message) {
var $message = $(message),
body = xmppchat.autoLink($message.children('body').text()),
from = Strophe.getBareJidFromJid($message.attr('from')),
composing = $message.find('composing'),
delayed = $message.find('delay').length > 0,
fullname = this.get('fullname').split(' ')[0],
stamp, time;
if (!body) {
if (composing.length) {
this.messages.add({
fullname: fullname,
sender: 'them',
delayed: delayed,
composing: composing.length
});
}
} else {
if (delayed) {
stamp = $message.find('delay').attr('stamp');
time = (new Date(stamp)).toLocaleTimeString().substring(0,5);
} else {
time = (new Date()).toLocaleTimeString().substring(0,5);
}
if (from == xmppchat.connection.bare_jid) {
this.messages.add({
fullname: 'me',
sender: 'me',
delayed: delayed,
time: time,
message: body
});
} else {
this.messages.add({
fullname: fullname,
sender: 'them',
delayed: delayed,
time: time,
message: body
});
}
}
},
}); });
xmppchat.ChatBoxView = Backbone.View.extend({ xmppchat.ChatBoxView = Backbone.View.extend({
...@@ -318,12 +380,6 @@ ...@@ -318,12 +380,6 @@
'<span class="chat-message-content">{{message}}</span>' + '<span class="chat-message-content">{{message}}</span>' +
'</div>'), '</div>'),
autoLink: function (text) {
// Convert URLs into hyperlinks
var re = /((http|https|ftp):\/\/[\w?=&.\/\-;#~%\-]+(?![\w\s?&.\/;#~%"=\-]*>))/g;
return text.replace(re, '<a target="_blank" href="$1">$1</a>');
},
appendMessage: function (message) { appendMessage: function (message) {
var now = new Date(), var now = new Date(),
time = now.toLocaleTimeString().substring(0,5), time = now.toLocaleTimeString().substring(0,5),
...@@ -338,7 +394,7 @@ ...@@ -338,7 +394,7 @@
$chat_content.append($('<div class="chat-date"></div>').text(now.toString().substring(0,15))); $chat_content.append($('<div class="chat-date"></div>').text(now.toString().substring(0,15)));
} }
} }
message = this.autoLink(message); message = xmppchat.autoLink(message);
// TODO use minutes logic or remove it // TODO use minutes logic or remove it
if (minutes.length==1) {minutes = '0'+minutes;} if (minutes.length==1) {minutes = '0'+minutes;}
$chat_content.find('div.chat-event').remove(); $chat_content.find('div.chat-event').remove();
...@@ -360,54 +416,25 @@ ...@@ -360,54 +416,25 @@
}, },
messageReceived: function (message) { messageReceived: function (message) {
/* XXX: event.mtype should be 'xhtml' for XHTML-IM messages, var $chat_content = this.$el.find('.chat-content');
but I only seem to get 'text'.
*/
var $message = $(message);
var body = this.autoLink($message.children('body').text()),
from = Strophe.getBareJidFromJid($message.attr('from')),
to = $message.attr('to'),
composing = $message.find('composing'),
$chat_content = this.$el.find('.chat-content'),
delayed = $message.find('delay').length > 0,
fullname = this.model.get('fullname'),
time, stamp, username, sender;
if (xmppchat.xmppstatus.getStatus() === 'offline') { if (xmppchat.xmppstatus.getStatus() === 'offline') {
// only update the UI if the user is not offline // only update the UI if the user is not offline
return; return;
} }
if (!body) { if (message.get('composing')) {
if (composing.length) { this.insertStatusNotification(message.get('fullname')+' '+'is typing');
this.insertStatusNotification(fullname+' '+'is typing'); return;
return;
}
} else { } else {
if (from == xmppchat.connection.bare_jid) { // xmppchat.storage.addMessage(from, body, 'from');
// I am the sender, so this must be a forwarded message... $chat_content.find('div.chat-event').remove();
$chat_content.find('div.chat-event').remove(); // TODO use toJSON here
username = 'me';
sender = 'me';
} else {
xmppchat.storage.addMessage(from, body, 'from');
$chat_content.find('div.chat-event').remove();
username = fullname.split(' ')[0];
sender = 'them';
}
if (delayed) {
// XXX: Test properly (for really old messages we somehow need to show
// their date as well)
stamp = $message.find('delay').attr('stamp');
time = (new Date(stamp)).toLocaleTimeString().substring(0,5);
} else {
time = (new Date()).toLocaleTimeString().substring(0,5);
}
$chat_content.append( $chat_content.append(
this.message_template({ this.message_template({
'sender': sender, 'sender': message.get('sender'),
'time': time, 'time': message.get('time'),
'message': body, 'message': message.get('message'),
'username': username, 'username': message.get('fullname'),
'extra_classes': delayed && 'delayed' || '' 'extra_classes': message.get('delayed') && 'delayed' || ''
})); }));
$chat_content.scrollTop($chat_content[0].scrollHeight); $chat_content.scrollTop($chat_content[0].scrollHeight);
} }
...@@ -441,7 +468,7 @@ ...@@ -441,7 +468,7 @@
$content.append($('<div class="chat-date"></div>').text(this_date.toString().substring(0,15))); $content.append($('<div class="chat-date"></div>').text(this_date.toString().substring(0,15)));
} }
} }
msg = this.autoLink(String(msg).replace(/(.*?\s.*?\s)/, '')); msg = xmppchat.autoLink(String(msg).replace(/(.*?\s.*?\s)/, ''));
if (msg_array[1] == 'to') { if (msg_array[1] == 'to') {
$content.append( $content.append(
this.message_template({ this.message_template({
...@@ -565,6 +592,10 @@ ...@@ -565,6 +592,10 @@
initialize: function (){ initialize: function (){
$('body').append(this.$el.hide()); $('body').append(this.$el.hide());
this.model.messages.on('add', function (item) {
this.messageReceived(item);
}, this);
xmppchat.roster.on('change', function (item, changed) { xmppchat.roster.on('change', function (item, changed) {
var fullname = this.model.get('fullname'), var fullname = this.model.get('fullname'),
chat_status = item.get('chat_status'); chat_status = item.get('chat_status');
...@@ -801,6 +832,7 @@ ...@@ -801,6 +832,7 @@
}); });
/*
xmppchat.ControlBox = xmppchat.ChatBox.extend({ xmppchat.ControlBox = xmppchat.ChatBox.extend({
initialize: function () { initialize: function () {
this.set({ this.set({
...@@ -808,6 +840,7 @@ ...@@ -808,6 +840,7 @@
}); });
} }
}); });
*/
xmppchat.ControlBoxView = xmppchat.ChatBoxView.extend({ xmppchat.ControlBoxView = xmppchat.ChatBoxView.extend({
// XXX: Options for the (still to be done) 'settings' tab: // XXX: Options for the (still to be done) 'settings' tab:
...@@ -822,7 +855,7 @@ ...@@ -822,7 +855,7 @@
}, },
initialize: function () { initialize: function () {
$('body').append(this.$el.hide()); // Override the one in ChatBoxView
}, },
template: _.template( template: _.template(
...@@ -854,10 +887,12 @@ ...@@ -854,10 +887,12 @@
}, },
render: function () { render: function () {
var that = this; this.$el.html(this.template(this.model.toJSON()));
this.$el.hide('fast', function () { this.$el.appendTo(xmppchat.chatboxesview.$el);
$(this).html(that.template(that.model.toJSON())); // Add login panel if the user still has to authenticate
}); if (!xmppchat.username) {
this.loginpanel = new xmppchat.LoginPanel().render();
}
return this; return this;
} }
}); });
...@@ -1040,7 +1075,7 @@ ...@@ -1040,7 +1075,7 @@
'time': (new Date()).toLocaleTimeString().substring(0,5), 'time': (new Date()).toLocaleTimeString().substring(0,5),
'message': body, 'message': body,
'username': sender, 'username': sender,
'extra_classes': ($message.find('delay').length > 0) && 'delayed' || '' 'extra_classes': ($message.find('delay').length > 0) && 'delayed' || '',
})); }));
} else { } else {
$chat_content.append( $chat_content.append(
...@@ -1147,20 +1182,14 @@ ...@@ -1147,20 +1182,14 @@
}, },
createChatBox: function (data) { createChatBox: function (data) {
var jid = data['jid']; return this.options.model.add({
var box = new xmppchat.ChatBox({ 'id': data['jid'],
'id': jid, 'jid': data['jid'],
'jid': jid, 'fullname': data['fullname'],
'fullname': data['fullname'], 'image_type': data['image_type'],
'image_type': data['image_type'], 'image': data['image'],
'image': data['image'], 'url': data['url'],
'url': data['url'], });
});
var view = new xmppchat.ChatBoxView({model: box});
this.views[jid] = view.render();
view.$el.appendTo(this.$el);
this.options.model.add(box);
return view;
}, },
closeChat: function (jid) { closeChat: function (jid) {
...@@ -1170,6 +1199,17 @@ ...@@ -1170,6 +1199,17 @@
} }
}, },
openControlBox: function () {
if (this.model.get('controlbox')) {
this.showChat('controlbox');
} else {
this.options.model.add({
id: 'controlbox',
box_id: 'controlbox'
});
}
},
openChat: function (roster_item) { openChat: function (roster_item) {
var jid = roster_item.get('jid'); var jid = roster_item.get('jid');
jid = Strophe.getBareJidFromJid(jid); jid = Strophe.getBareJidFromJid(jid);
...@@ -1214,7 +1254,7 @@ ...@@ -1214,7 +1254,7 @@
} }
var from = Strophe.getBareJidFromJid(message_from), var from = Strophe.getBareJidFromJid(message_from),
to = Strophe.getBareJidFromJid($message.attr('to')), to = Strophe.getBareJidFromJid($message.attr('to')),
view, resource; view, resource, chatboxes;
if (from == xmppchat.connection.bare_jid) { if (from == xmppchat.connection.bare_jid) {
// I am the sender, so this must be a forwarded message... // I am the sender, so this must be a forwarded message...
partner_jid = to; partner_jid = to;
...@@ -1229,31 +1269,28 @@ ...@@ -1229,31 +1269,28 @@
xmppchat.getVCard( xmppchat.getVCard(
partner_jid, partner_jid,
$.proxy(function (jid, fullname, img, img_type, url) { $.proxy(function (jid, fullname, img, img_type, url) {
view = this.createChatBox({ // FIXME: We don't get the view from createChatBox
// anymore.
// Instead, we should trigger an event on the model
chatboxes = this.createChatBox({
'jid': jid, 'jid': jid,
'fullname': fullname, 'fullname': fullname,
'image': img, 'image': img,
'image_type': img_type, 'image_type': img_type,
'url': url, 'url': url,
}) })
view.messageReceived(message); chatboxes.get(jid).messageReceived(message);
xmppchat.roster.addResource(partner_jid, resource); xmppchat.roster.addResource(partner_jid, resource);
}, this), }, this),
$.proxy(function () { $.proxy(function () {
// Error occured while fetching vcard // # TODO: call the function above
console.log("An error occured while fetching vcard"); console.log("An error occured while fetching vcard");
view = this.createChatBox({
'jid': jid,
'fullname': jid,
})
view.messageReceived(message);
xmppchat.roster.addResource(partner_jid, resource);
}, this)); }, this));
return true; return true;
} else if (!view.isVisible()) { } else if (!view.isVisible()) {
this.showChat(partner_jid); this.showChat(partner_jid);
} }
view.messageReceived(message); view.model.messageReceived(message);
xmppchat.roster.addResource(partner_jid, resource); xmppchat.roster.addResource(partner_jid, resource);
return true; return true;
}, },
...@@ -1266,39 +1303,33 @@ ...@@ -1266,39 +1303,33 @@
controlbox.roomspanel = new xmppchat.RoomsPanel().render(); controlbox.roomspanel = new xmppchat.RoomsPanel().render();
// Add the roster // Add the roster
xmppchat.roster = new xmppchat.RosterItems(); xmppchat.roster = new xmppchat.RosterItems();
xmppchat.roster.localStorage = new Backbone.LocalStorage(hex_sha1(xmppchat.connection.bare_jid)); xmppchat.roster.localStorage = new Backbone.LocalStorage(
hex_sha1('converse.rosteritems-'+xmppchat.connection.bare_jid));
xmppchat.rosterview = new xmppchat.RosterView({'model':xmppchat.roster}); xmppchat.rosterview = new xmppchat.RosterView({'model':xmppchat.roster});
xmppchat.rosterview.$el.appendTo(controlbox.contactspanel.$el); xmppchat.rosterview.$el.appendTo(controlbox.contactspanel.$el);
xmppchat.roster.fetch({add: true}); // Gets the cached roster items from localstorage xmppchat.roster.fetch({add: true}); // Gets the cached roster items from localstorage
xmppchat.rosterview.initialSort(); xmppchat.rosterview.initialSort();
// TODO: we're going to use localStorage here
// Restore previously open chatboxes // Restore previously open chatboxes
this.restoreOpenChats(); // this.restoreOpenChats();
}, },
initialize: function () { initialize: function () {
this.views = {};
this.options.model.on("add", function (item) { this.options.model.on("add", function (item) {
// The controlbox added automatically, but we don't show it var view;
// automatically (only when it was open before page load or if (item.get('box_id') === 'controlbox') {
// upon a click). view = new xmppchat.ControlBoxView({model: item});
if ((item.get('id') != 'controlbox') || (!xmppchat.username)) { } else {
this.showChat(item.get('id')); view = new xmppchat.ChatBoxView({model: item});
} }
this.views[item.get('id')] = view.render();
this.showChat(item.get('id'));
}, this); }, this);
this.views = {}; /*
// Add the controlbox view // Rebind events (necessary for click events on tabs inserted via the panels)
this.views.controlbox = new xmppchat.ControlBoxView({ this.views.controlbox.delegateEvents();
model: new xmppchat.ControlBox({'id':'controlbox', 'jid':'controlbox'}) */
}).render();
this.views.controlbox.$el.appendTo(this.$el);
// Add login panel if the user still has to authenticate
if (!xmppchat.username) {
this.views.controlbox.loginpanel = new xmppchat.LoginPanel().render();
}
// Rebind events (necessary for click events on tabs inserted via the panels)
this.views.controlbox.delegateEvents();
// Add the controlbox model to this collection (will trigger showChat)
this.options.model.add(this.views.controlbox.options.model);
} }
}); });
...@@ -2115,15 +2146,16 @@ ...@@ -2115,15 +2146,16 @@
this.fullname = chatdata.attr('fullname'); this.fullname = chatdata.attr('fullname');
this.auto_subscribe = chatdata.attr('auto_subscribe') === "True" || false; this.auto_subscribe = chatdata.attr('auto_subscribe') === "True" || false;
this.chatboxes = new this.ChatBoxes();
this.chatboxesview = new this.ChatBoxesView({ this.chatboxesview = new this.ChatBoxesView({
model: new this.ChatBoxes() model: this.chatboxes
}); });
$toggle.bind('click', $.proxy(function (e) { $toggle.bind('click', $.proxy(function (e) {
e.preventDefault(); e.preventDefault();
if ($("div#controlbox").is(':visible')) { if ($("div#controlbox").is(':visible')) {
this.chatboxesview.closeChat('controlbox'); this.chatboxesview.closeChat('controlbox');
} else { } else {
this.chatboxesview.showChat('controlbox'); this.chatboxesview.openControlBox();
} }
}, this)); }, this));
...@@ -2151,6 +2183,8 @@ ...@@ -2151,6 +2183,8 @@
this.connection.muc_domain = 'conference.' + this.connection.domain; this.connection.muc_domain = 'conference.' + this.connection.domain;
this.storage = new this.ClientStorage(hex_sha1(this.connection.bare_jid)); this.storage = new this.ClientStorage(hex_sha1(this.connection.bare_jid));
this.chatboxes.localStorage = new Backbone.LocalStorage(
hex_sha1('converse.chatboxes-'+xmppchat.connection.bare_jid));
this.chatboxesview.onConnected(); this.chatboxesview.onConnected();
this.connection.addHandler( this.connection.addHandler(
......
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