mirror of
https://github.com/moparisthebest/kaiwa
synced 2024-11-22 09:12:19 -05:00
358 lines
12 KiB
JavaScript
358 lines
12 KiB
JavaScript
/*global app, me, client, URL*/
|
|
"use strict";
|
|
|
|
var _ = require('underscore');
|
|
var crypto = require('crypto');
|
|
var async = require('async');
|
|
var uuid = require('node-uuid');
|
|
var HumanModel = require('human-model');
|
|
var Resources = require('./resources');
|
|
var Messages = require('./messages');
|
|
var Message = require('./message');
|
|
var logger = require('andlog');
|
|
var fetchAvatar = require('../helpers/fetchAvatar');
|
|
|
|
|
|
module.exports = HumanModel.define({
|
|
initialize: function (attrs) {
|
|
if (attrs.jid) {
|
|
this.id = attrs.jid;
|
|
}
|
|
this.setAvatar(attrs.avatarID);
|
|
|
|
this.resources.bind('add remove reset', this.onResourceListChange, this);
|
|
this.resources.bind('change', this.onResourceChange, this);
|
|
|
|
this.bind('change:topResource change:lockedResource change:_forceUpdate', this.summarizeResources, this);
|
|
},
|
|
type: 'contact',
|
|
props: {
|
|
id: ['string', true, false],
|
|
avatarID: ['string', false, ''],
|
|
groups: ['array', false, []],
|
|
inRoster: ['bool', true, false],
|
|
jid: ['string', true],
|
|
name: ['string', false, ''],
|
|
owner: ['string', true, ''],
|
|
storageId: ['string', true, ''],
|
|
subscription: ['string', false, 'none']
|
|
},
|
|
session: {
|
|
activeContact: ['bool', false, false],
|
|
avatar: 'string',
|
|
avatarSource: 'string',
|
|
lastInteraction: 'date',
|
|
lastHistoryFetch: 'date',
|
|
lastSentMessage: 'object',
|
|
lockedResource: 'string',
|
|
offlineStatus: ['string', false, ''],
|
|
topResource: 'string',
|
|
unreadCount: ['number', false, 0],
|
|
_forceUpdate: ['number', false, 0],
|
|
onCall: ['boolean', false, false],
|
|
stream: 'object'
|
|
},
|
|
derived: {
|
|
streamUrl: {
|
|
deps: ['stream'],
|
|
cache: true,
|
|
fn: function () {
|
|
if (!this.stream) return '';
|
|
return URL.createObjectURL(this.stream);
|
|
}
|
|
},
|
|
displayName: {
|
|
deps: ['name', 'jid'],
|
|
fn: function () {
|
|
return this.name || this.jid;
|
|
}
|
|
},
|
|
displayUnreadCount: {
|
|
deps: ['unreadCount'],
|
|
fn: function () {
|
|
if (this.unreadCount > 0) {
|
|
return this.unreadCount.toString();
|
|
}
|
|
return '';
|
|
}
|
|
},
|
|
formattedTZO: {
|
|
deps: ['timezoneOffset'],
|
|
fn: function () {
|
|
if (!this.timezoneOffset) return '';
|
|
|
|
var localTime = new Date();
|
|
var localTZO = localTime.getTimezoneOffset();
|
|
var diff = Math.abs(localTZO % (24 * 60) - this.timezoneOffset % (24 * 60));
|
|
var remoteTime = new Date(Date.now() + diff * 60000);
|
|
|
|
|
|
var day = remoteTime.getDate();
|
|
var hour = remoteTime.getHours();
|
|
var minutes = remoteTime.getMinutes();
|
|
|
|
var days = ['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa'];
|
|
|
|
var dow = days[remoteTime.getDay()];
|
|
var localDow = days[localTime.getDay()];
|
|
|
|
var m = (hour >= 12) ? ' PM' : ' AM';
|
|
|
|
hour = hour % 12;
|
|
if (hour === 0) {
|
|
hour = 12;
|
|
}
|
|
|
|
var strDay = (day < 10) ? '0' + day : day;
|
|
var strHour = (hour < 10) ? '0' + hour : hour;
|
|
var strMin = (minutes < 10) ? '0' + minutes: minutes;
|
|
|
|
if (localDow == dow) {
|
|
return strHour + ':' + strMin + m;
|
|
} else {
|
|
return dow + ' ' + strHour + ':' + strMin + m;
|
|
}
|
|
}
|
|
},
|
|
status: {
|
|
deps: ['topResource', 'lockedResource', '_forceUpdate'],
|
|
fn: function () {
|
|
var resource = this.resources.get(this.lockedResource) || this.resources.get(this.topResource) || {};
|
|
return resource.status || '';
|
|
}
|
|
},
|
|
show: {
|
|
deps: ['topResource', 'lockedResource', '_forceUpdate'],
|
|
fn: function () {
|
|
if (this.resources.length === 0) {
|
|
return 'offline';
|
|
}
|
|
var resource = this.resources.get(this.lockedResource) || this.resources.get(this.topResource) || {};
|
|
return resource.show || 'online';
|
|
}
|
|
},
|
|
timezoneOffset: {
|
|
deps: ['topResource', 'lockedResource', '_forceUpdate'],
|
|
fn: function () {
|
|
var resource = this.resources.get(this.lockedResource) || this.resources.get(this.topResource) || {};
|
|
return resource.timezoneOffset || undefined;
|
|
}
|
|
},
|
|
idleSince: {
|
|
deps: ['topResource', 'lockedResource', '_forceUpdate'],
|
|
fn: function () {
|
|
var resource = this.resources.get(this.lockedResource) || this.resources.get(this.topResource) || {};
|
|
return resource.idleSince || undefined;
|
|
}
|
|
},
|
|
idle: {
|
|
deps: ['idleSince'],
|
|
fn: function () {
|
|
return this.idleSince && !isNaN(this.idleSince.valueOf());
|
|
}
|
|
},
|
|
chatState: {
|
|
deps: ['topResource', 'lockedResource', '_forceUpdate'],
|
|
fn: function () {
|
|
var states = {};
|
|
this.resources.models.forEach(function (resource) {
|
|
states[resource.chatState] = true;
|
|
});
|
|
|
|
if (states.composing) return 'composing';
|
|
if (states.paused) return 'paused';
|
|
if (states.active) return 'active';
|
|
if (states.inactive) return 'inactive';
|
|
return 'gone';
|
|
}
|
|
},
|
|
supportsReceipts: {
|
|
deps: ['lockedResource', '_forceUpdate'],
|
|
fn: function () {
|
|
if (!this.lockedResource) return false;
|
|
var res = this.resources.get(this.lockedResource);
|
|
return res.supportsReceipts;
|
|
}
|
|
},
|
|
supportsChatStates: {
|
|
deps: ['lockedResource', '_forceUpdate'],
|
|
fn: function () {
|
|
if (!this.lockedResource) return false;
|
|
var res = this.resources.get(this.lockedResource);
|
|
return res.supportsChatStates;
|
|
}
|
|
},
|
|
hasUnread: {
|
|
deps: ['unreadCount'],
|
|
fn: function () {
|
|
return this.unreadCount > 0;
|
|
}
|
|
},
|
|
jingleResources: {
|
|
deps: ['_forceUpdate'],
|
|
fn: function () {
|
|
return this.resources.filter(function (res) {
|
|
return res.supportsJingleMedia;
|
|
});
|
|
}
|
|
},
|
|
callable: {
|
|
deps: ['jingleResources'],
|
|
fn: function () {
|
|
return !!this.jingleResources.length;
|
|
}
|
|
},
|
|
callObject: {
|
|
fn: function () {
|
|
return app.calls.where('contact', this);
|
|
}
|
|
}
|
|
},
|
|
collections: {
|
|
resources: Resources,
|
|
messages: Messages
|
|
},
|
|
call: function () {
|
|
if (this.jingleResources.length) {
|
|
var peer = this.jingleResources[0];
|
|
this.callState = 'starting';
|
|
app.api.call(peer.id);
|
|
} else {
|
|
logger.error('no jingle resources for this user');
|
|
}
|
|
},
|
|
setAvatar: function (id, type, source) {
|
|
var self = this;
|
|
fetchAvatar(this.jid, id, type, source, function (avatar) {
|
|
if (source == 'vcard' && self.avatarSource == 'pubsub') return;
|
|
self.avatarID = avatar.id;
|
|
self.avatar = avatar.uri;
|
|
self.avatarSource = source;
|
|
self.save();
|
|
});
|
|
},
|
|
onResourceChange: function () {
|
|
this.resources.sort();
|
|
this.topResource = (this.resources.first() || {}).id;
|
|
this._forceUpdate++;
|
|
},
|
|
onResourceListChange: function () {
|
|
// Manually propagate change events for properties that
|
|
// depend on the resources collection.
|
|
this.resources.sort();
|
|
|
|
var res = this.resources.first();
|
|
if (res) {
|
|
this.offlineStatus = '';
|
|
this.topResource = res.id;
|
|
} else {
|
|
this.topResource = undefined;
|
|
}
|
|
|
|
this.lockedResource = undefined;
|
|
},
|
|
addMessage: function (message, notify) {
|
|
message.owner = me.jid.bare;
|
|
|
|
if (notify && (!this.activeContact || (this.activeContact && !app.state.focused)) && message.from.bare === this.jid) {
|
|
this.unreadCount++;
|
|
app.notifications.create(this.displayName, {
|
|
body: message.body,
|
|
icon: this.avatar,
|
|
tag: this.jid,
|
|
onclick: _.bind(app.navigate, app, '/chat/' + this.jid)
|
|
});
|
|
}
|
|
|
|
var existing = Message.idLookup(message.from[message.type == 'groupchat' ? 'full' : 'bare'], message.mid);
|
|
if (existing) {
|
|
existing.set(message);
|
|
existing.save();
|
|
} else {
|
|
this.messages.add(message);
|
|
message.save();
|
|
}
|
|
|
|
var newInteraction = new Date(message.created);
|
|
if (!this.lastInteraction || this.lastInteraction < newInteraction) {
|
|
this.lastInteraction = newInteraction;
|
|
}
|
|
},
|
|
fetchHistory: function () {
|
|
var self = this;
|
|
app.whenConnected(function () {
|
|
var filter = {
|
|
'with': self.jid,
|
|
rsm: {
|
|
count: 20,
|
|
before: true
|
|
}
|
|
};
|
|
|
|
var lastMessage = self.messages.last();
|
|
if (lastMessage && lastMessage.archivedId) {
|
|
filter.rsm.after = lastMessage.archivedId;
|
|
}
|
|
|
|
if (self.lastHistoryFetch && !isNaN(self.lastHistoryFetch.valueOf())) {
|
|
if (self.lastInteraction > self.lastHistoryFetch) {
|
|
filter.start = self.lastInteraction;
|
|
} else {
|
|
filter.start = self.lastHistoryFetch;
|
|
}
|
|
} else {
|
|
filter.end = new Date(Date.now());
|
|
}
|
|
|
|
client.getHistory(filter, function (err, res) {
|
|
if (err) return;
|
|
|
|
self.lastHistoryFetch = new Date(Date.now());
|
|
|
|
var results = res.mamQuery.results || [];
|
|
results.reverse();
|
|
results.forEach(function (result) {
|
|
result = result.toJSON();
|
|
var msg = result.mam.forwarded.message;
|
|
|
|
msg.mid = msg.id;
|
|
delete msg.id;
|
|
|
|
if (!msg.delay) {
|
|
msg.delay = result.mam.forwarded.delay;
|
|
}
|
|
|
|
if (msg.replace) {
|
|
var original = Message.idLookup(msg.from[msg.type == 'groupchat' ? 'full' : 'bare'], msg.replace);
|
|
// Drop the message if editing a previous, but
|
|
// keep it if it didn't actually change an
|
|
// existing message.
|
|
if (original && original.correct(msg)) return;
|
|
}
|
|
|
|
var message = new Message(msg);
|
|
message.archivedId = result.mam.id;
|
|
message.acked = true;
|
|
|
|
self.addMessage(message, false);
|
|
});
|
|
});
|
|
});
|
|
},
|
|
save: function () {
|
|
if (!this.inRoster) return;
|
|
|
|
var storageId = crypto.createHash('sha1').update(this.owner + '/' + this.id).digest('hex');
|
|
var data = {
|
|
storageId: storageId,
|
|
owner: this.owner,
|
|
jid: this.jid,
|
|
name: this.name,
|
|
groups: this.groups,
|
|
subscription: this.subscription,
|
|
avatarID: this.avatarID
|
|
};
|
|
app.storage.roster.add(data);
|
|
}
|
|
});
|