a569e34b33
Create a cleaner seperation between generating notifications and updating frontend conversation views. The former is now handled by `conversation.notify` while the latter is achieved by triggering an event on the conversation model, which will only be acted on if there are any views listening for it. Additionally, instead of re-fetching the entire message history, which is overkill, just add or update the new/modified message. This will help speed up the newmessage event handler and also help avoid unnecessary re-rendering when resolving key conflicts. // FREEBIE
267 lines
9.2 KiB
JavaScript
267 lines
9.2 KiB
JavaScript
/*
|
|
* vim: ts=4:sw=4:expandtab
|
|
*/
|
|
(function () {
|
|
'use strict';
|
|
window.Whisper = window.Whisper || {};
|
|
emoji.init_colons();
|
|
|
|
Whisper.ConversationView = Whisper.View.extend({
|
|
className: function() {
|
|
return [ 'conversation', this.model.get('type') ].join(' ');
|
|
},
|
|
id: function() {
|
|
return 'conversation-' + this.model.cid;
|
|
},
|
|
template: $('#conversation').html(),
|
|
render_attributes: function() {
|
|
return {
|
|
group: this.model.get('type') === 'group',
|
|
title: this.model.getTitle()
|
|
};
|
|
},
|
|
initialize: function(options) {
|
|
this.listenTo(this.model, 'destroy', this.stopListening);
|
|
this.listenTo(this.model, 'change:name', this.updateTitle);
|
|
this.listenTo(this.model, 'newmessage', this.addMessage);
|
|
this.listenTo(this.model, 'change:unreadCount', this.onUnread);
|
|
this.listenTo(this.model, 'opened', this.focusMessageField);
|
|
|
|
this.render();
|
|
twemoji.parse(this.el, { base: '/images/twemoji/', size: 16 });
|
|
|
|
this.appWindow = options.appWindow;
|
|
this.fileInput = new Whisper.FileInputView({
|
|
el: this.$('form.send'),
|
|
window: this.appWindow.contentWindow
|
|
});
|
|
|
|
this.view = new Whisper.MessageListView({
|
|
collection: this.model.messageCollection,
|
|
window: this.appWindow.contentWindow
|
|
});
|
|
this.$('.discussion-container').append(this.view.el);
|
|
this.view.render();
|
|
|
|
this.$messageField = this.$('.send-message');
|
|
|
|
var onResize = this.forceUpdateMessageFieldSize.bind(this);
|
|
this.appWindow.contentWindow.addEventListener('resize', onResize);
|
|
|
|
var onFocus = function() {
|
|
if (this.$el.css('display') !== 'none') {
|
|
this.markRead();
|
|
}
|
|
}.bind(this);
|
|
this.appWindow.contentWindow.addEventListener('focus', onFocus);
|
|
|
|
this.appWindow.onClosed.addListener(function () {
|
|
this.appWindow.contentWindow.removeEventListener('resize', onResize);
|
|
this.appWindow.contentWindow.removeEventListener('focus', onFocus);
|
|
window.autosize.destroy(this.$messageField);
|
|
this.remove();
|
|
}.bind(this));
|
|
|
|
setTimeout(function() {
|
|
this.view.scrollToBottom();
|
|
}.bind(this), 10);
|
|
},
|
|
|
|
events: {
|
|
'submit .send': 'sendMessage',
|
|
'input .send-message': 'updateMessageFieldSize',
|
|
'keydown .send-message': 'updateMessageFieldSize',
|
|
'click .destroy': 'destroyMessages',
|
|
'click .end-session': 'endSession',
|
|
'click .leave-group': 'leaveGroup',
|
|
'click .update-group': 'newGroupUpdate',
|
|
'click .verify-identity': 'verifyIdentity',
|
|
'click .view-members': 'viewMembers',
|
|
'click .drop-down': 'toggleMenu',
|
|
'click .openInbox' : 'openInbox',
|
|
'click' : 'onClick',
|
|
'select .message-list .entry': 'messageDetail',
|
|
'force-resize': 'forceUpdateMessageFieldSize',
|
|
'click .choose-file': 'focusMessageField'
|
|
},
|
|
focusMessageField: function() {
|
|
this.$messageField.focus();
|
|
},
|
|
|
|
addMessage: function(message) {
|
|
this.model.messageCollection.add(message, {merge: true});
|
|
},
|
|
|
|
viewMembers: function() {
|
|
var view = new Whisper.GroupMemberList({ model: this.model });
|
|
this.listenBack(view);
|
|
},
|
|
|
|
openInbox: function() {
|
|
openInbox();
|
|
},
|
|
|
|
onClick: function(e) {
|
|
this.closeMenu(e);
|
|
this.markRead(e);
|
|
},
|
|
|
|
markRead: function(e) {
|
|
this.model.markRead();
|
|
},
|
|
|
|
onUnread: function(model, previous) {
|
|
if (!this.isHidden()) {
|
|
this.markRead();
|
|
}
|
|
},
|
|
|
|
verifyIdentity: function() {
|
|
if (this.model.isPrivate()) {
|
|
var their_number = this.model.id;
|
|
var our_number = textsecure.storage.user.getNumber();
|
|
textsecure.storage.axolotl.getIdentityKey(their_number).then(function(their_key) {
|
|
textsecure.storage.axolotl.getIdentityKey(our_number).then(function(our_key) {
|
|
var view = new Whisper.KeyVerificationView({
|
|
model: { their_key: their_key, your_key: our_key }
|
|
}).render();
|
|
this.listenBack(view);
|
|
}.bind(this));
|
|
}.bind(this));
|
|
}
|
|
},
|
|
|
|
messageDetail: function(e, data) {
|
|
var view = new Whisper.MessageDetailView({
|
|
model: data.message,
|
|
conversation: this.model
|
|
});
|
|
this.listenBack(view);
|
|
view.render();
|
|
},
|
|
|
|
listenBack: function(view) {
|
|
this.$('.panel').hide();
|
|
view.$el.prependTo(this.el);
|
|
this.listenToOnce(view, 'back', function(e) {
|
|
view.remove();
|
|
this.$('.panel').show();
|
|
this.$el.trigger('force-resize');
|
|
}.bind(this));
|
|
},
|
|
|
|
closeMenu: function(e) {
|
|
if (e && !$(e.target).hasClass('drop-down')) {
|
|
this.$('.menu-list').hide();
|
|
}
|
|
},
|
|
|
|
endSession: function() {
|
|
this.model.endSession();
|
|
this.$('.menu-list').hide();
|
|
},
|
|
|
|
leaveGroup: function() {
|
|
this.model.leaveGroup();
|
|
this.$('.menu-list').hide();
|
|
},
|
|
|
|
toggleMenu: function() {
|
|
this.$('.menu-list').toggle();
|
|
},
|
|
|
|
newGroupUpdate: function() {
|
|
this.newGroupUpdateView = new Whisper.NewGroupUpdateView({
|
|
model: this.model,
|
|
window: this.appWindow.contentWindow
|
|
});
|
|
this.listenBack(this.newGroupUpdateView);
|
|
},
|
|
|
|
destroyMessages: function(e) {
|
|
this.confirm("Permanently delete this conversation?").then(function() {
|
|
this.model.destroyMessages();
|
|
this.remove();
|
|
}.bind(this)).catch(function() {
|
|
// clicked cancel, nothing to do.
|
|
});
|
|
this.$('.menu-list').hide();
|
|
},
|
|
|
|
sendMessage: function(e) {
|
|
e.preventDefault();
|
|
var input = this.$messageField;
|
|
var message = this.replace_colons(input.val()).trim();
|
|
var convo = this.model;
|
|
|
|
if (message.length > 0 || this.fileInput.hasFiles()) {
|
|
this.fileInput.getFiles().then(function(attachments) {
|
|
convo.sendMessage(message, attachments);
|
|
});
|
|
input.val("");
|
|
this.forceUpdateMessageFieldSize(e);
|
|
this.fileInput.deleteFiles();
|
|
}
|
|
},
|
|
|
|
replace_colons: function(str) {
|
|
return str.replace(emoji.rx_colons, function(m){
|
|
var idx = m.substr(1, m.length-2);
|
|
var val = emoji.map.colons[idx];
|
|
if (val) {
|
|
return emoji.data[val][0][0];
|
|
} else {
|
|
return m;
|
|
}
|
|
});
|
|
},
|
|
|
|
updateTitle: function() {
|
|
this.$('.conversation-title').text(this.model.getTitle());
|
|
},
|
|
|
|
updateMessageFieldSize: function (event) {
|
|
var keyCode = event.which || event.keyCode;
|
|
|
|
if (keyCode === 13 && !event.altKey && !event.shiftKey && !event.ctrlKey) {
|
|
// enter pressed - submit the form now
|
|
event.preventDefault();
|
|
return this.$('.bottom-bar form').submit();
|
|
}
|
|
|
|
this.view.measureScrollPosition();
|
|
window.autosize(this.$messageField);
|
|
|
|
var $discussionContainer = this.$('.discussion-container'),
|
|
$conversationHeader = this.$('.conversation-header'),
|
|
$attachmentPreviews = this.$('.attachment-previews'),
|
|
$bottomBar = this.$('.bottom-bar');
|
|
|
|
$bottomBar.outerHeight(
|
|
this.$messageField.outerHeight() +
|
|
$attachmentPreviews.outerHeight() + 1);
|
|
var $bottomBarNewHeight = $bottomBar.outerHeight();
|
|
|
|
//TODO: revisit this logic for new layout.
|
|
$discussionContainer.outerHeight(
|
|
this.$el.outerHeight() -
|
|
$bottomBarNewHeight -
|
|
$conversationHeader.outerHeight() - 20);
|
|
|
|
this.view.scrollToBottomIfNeeded();
|
|
},
|
|
|
|
forceUpdateMessageFieldSize: function (event) {
|
|
if (this.isHidden()) {
|
|
return;
|
|
}
|
|
this.view.scrollToBottomIfNeeded();
|
|
window.autosize.update(this.$messageField);
|
|
this.updateMessageFieldSize(event);
|
|
},
|
|
|
|
isHidden: function() {
|
|
return (this.$el.css('display') === 'none') || this.$('.panel').css('display') === 'none';
|
|
}
|
|
});
|
|
})();
|