/* vim: ts=4:sw=4:expandtab * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program. If not, see . */ (function () { 'use strict'; window.Whisper = window.Whisper || {}; // TODO: Factor out private and group subclasses of Conversation Whisper.Conversation = Backbone.Model.extend({ database: Whisper.Database, storeName: 'conversations', defaults: function() { var timestamp = new Date().getTime(); return { unreadCount : 0, timestamp : timestamp, active_at : timestamp }; }, initialize: function() { this.messageCollection = new Whisper.MessageCollection(); this.contactCollection = new Whisper.ConversationCollection(); }, validate: function(attributes, options) { var required = ['id', 'type', 'timestamp']; var missing = _.filter(required, function(attr) { return !attributes[attr]; }); if (missing.length) { return "Conversation must have " + missing; } if (attributes.type !== 'private' && attributes.type !== 'group') { return "Invalid conversation type: " + attributes.type; } // hack if (this.get('type') === 'private') { try { this.id = libphonenumber.util.verifyNumber(this.id); var number = libphonenumber.util.splitCountryCode(this.id); this.set({ e164_number: this.id, national_number: '' + number.national_number, international_number: '' + number.country_code + number.national_number }); } catch(ex) { return ex; } } }, sendMessage: function(body, attachments) { var now = Date.now(); var message = this.messageCollection.add({ body : body, conversationId : this.id, type : 'outgoing', attachments : attachments, sent_at : now, received_at : now }); message.save(); this.save({ unreadCount : 0, active_at : now, timestamp : now, lastMessage : body }); var sendFunc; if (this.get('type') == 'private') { sendFunc = textsecure.messaging.sendMessageToNumber; } else { sendFunc = textsecure.messaging.sendMessageToGroup; } sendFunc(this.get('id'), body, attachments, now).catch(function(errors) { var keyErrors = []; _.each(errors, function(e) { if (e.error.name === 'OutgoingIdentityKeyError') { e.error.args.push(message.id); keyErrors.push(e.error); } }); if (keyErrors.length) { message.save({ errors : keyErrors }).then(function() { extension.trigger('message', message); // notify frontend listeners }); } else { throw errors; } }); }, endSession: function() { if (this.get('type') === 'private') { textsecure.messaging.closeSession(this.id); } }, leaveGroup: function() { if (this.get('type') === 'group') { textsecure.messaging.leaveGroup(this.id); } }, receiveMessage: function(decrypted) { var conversation = this; var timestamp = decrypted.pushMessage.timestamp.toNumber(); var m = this.messageCollection.add({ body : decrypted.message.body, timestamp : timestamp, conversationId : this.id, attachments : decrypted.message.attachments, type : 'incoming', sender : decrypted.pushMessage.source }); if (timestamp > this.get('timestamp')) { this.set('timestamp', timestamp); } this.save({unreadCount: this.get('unreadCount') + 1, active: true}); return new Promise(function (resolve) { m.save().then(resolve(m)); }); }, fetchMessages: function(options) { return this.messageCollection.fetchConversation(this.id, options); }, fetchContacts: function(options) { if (this.isPrivate()) { this.contactCollection.reset([this]); } else { this.contactCollection.reset( this.get('members').map(function(number) { var c = this.collection.add({id: number, type: 'private'}); c.fetch(); return c; }.bind(this)) ); } }, archive: function() { this.unset('active_at'); }, destroyMessages: function() { var models = this.messageCollection.models; this.messageCollection.reset([]); _.each(models, function(message) { message.destroy(); }); this.archive(); return this.save(); }, getTitle: function() { return this.get('name') || this.get('members') || this.id; }, getNumber: function() { if (this.get('type') === 'private') { return this.id; } else { return ''; } }, isPrivate: function() { return this.get('type') === 'private'; } }); Whisper.ConversationCollection = Backbone.Collection.extend({ database: Whisper.Database, storeName: 'conversations', model: Whisper.Conversation, comparator: function(m) { return -m.get('timestamp'); }, createGroup: function(recipients, name, avatar) { var attributes = {}; attributes = { name : name, members : recipients, type : 'group', avatar : avatar }; var conversation = this.add(attributes, {merge: true}); return textsecure.messaging.createGroup(recipients, name, avatar).then(function(groupId) { conversation.save({ id : getString(groupId), groupId : getString(groupId) }); return conversation; }); }, findOrCreateForRecipient: function(recipient) { var attributes = {}; attributes = { id : recipient, name : recipient, type : 'private', }; var conversation = this.add(attributes, {merge: true}); conversation.save(); return conversation; }, addIncomingMessage: function(decrypted) { var attributes = {}; if (decrypted.message.group) { attributes = { id : decrypted.message.group.id, groupId : decrypted.message.group.id, name : decrypted.message.group.name || 'New group', type : 'group', }; } else { attributes = { id : decrypted.pushMessage.source, name : decrypted.pushMessage.source, type : 'private' }; } var conversation = this.add(attributes, {merge: true}); return conversation.receiveMessage(decrypted); }, destroyAll: function () { return Promise.all(this.models.map(function(m) { return new Promise(function(resolve, reject) { m.destroy().then(resolve).fail(reject); }); })); }, fetchGroups: function(number) { return this.fetch({ index: { name: 'group', only: number } }); } }); })();