2016-02-16 08:46:47 +01:00
|
|
|
// Converse.js (A browser based XMPP chat client)
|
|
|
|
// http://conversejs.org
|
|
|
|
//
|
2017-02-02 19:29:38 +01:00
|
|
|
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
|
2016-02-16 08:46:47 +01:00
|
|
|
// Licensed under the Mozilla Public License (MPLv2)
|
|
|
|
//
|
2017-01-16 21:59:30 +01:00
|
|
|
/*global Backbone, define, window, document */
|
2016-02-16 08:46:47 +01:00
|
|
|
(function (root, factory) {
|
2017-02-14 15:08:39 +01:00
|
|
|
define(["sizzle",
|
|
|
|
"jquery",
|
|
|
|
"lodash",
|
|
|
|
"polyfill",
|
|
|
|
"locales",
|
|
|
|
"utils",
|
|
|
|
"moment_with_locales",
|
|
|
|
"strophe",
|
|
|
|
"pluggable",
|
|
|
|
"strophe.disco",
|
|
|
|
"backbone.browserStorage",
|
|
|
|
"backbone.overview",
|
2016-02-23 08:13:30 +01:00
|
|
|
], factory);
|
2017-02-14 15:08:39 +01:00
|
|
|
}(this, function (sizzle, $, _, polyfill, locales, utils, moment, Strophe, pluggable) {
|
2017-02-18 10:35:16 +01:00
|
|
|
/* Cannot use this due to Safari bug.
|
2016-02-16 08:46:47 +01:00
|
|
|
* See https://github.com/jcbrand/converse.js/issues/196
|
|
|
|
*/
|
2016-02-23 08:13:30 +01:00
|
|
|
// "use strict";
|
|
|
|
|
|
|
|
// Strophe globals
|
|
|
|
var $build = Strophe.$build;
|
|
|
|
var $iq = Strophe.$iq;
|
2017-02-14 15:08:39 +01:00
|
|
|
var $msg = Strophe.$msg;
|
2016-02-23 08:13:30 +01:00
|
|
|
var $pres = Strophe.$pres;
|
|
|
|
var b64_sha1 = Strophe.SHA1.b64_sha1;
|
|
|
|
Strophe = Strophe.Strophe;
|
2016-02-16 08:46:47 +01:00
|
|
|
|
|
|
|
// Use Mustache style syntax for variable interpolation
|
2017-01-26 10:08:43 +01:00
|
|
|
/* Configuration of Lodash templates (this config is distinct to the
|
2016-02-16 08:46:47 +01:00
|
|
|
* config of requirejs-tpl in main.js). This one is for normal inline templates.
|
|
|
|
*/
|
|
|
|
_.templateSettings = {
|
|
|
|
evaluate : /\{\[([\s\S]+?)\]\}/g,
|
|
|
|
interpolate : /\{\{([\s\S]+?)\}\}/g
|
|
|
|
};
|
|
|
|
|
2016-12-20 10:30:20 +01:00
|
|
|
var _converse = {};
|
|
|
|
_converse.templates = {};
|
|
|
|
_.extend(_converse, Backbone.Events);
|
2017-02-14 10:56:30 +01:00
|
|
|
_converse.promises = {
|
|
|
|
'cachedRoster': new $.Deferred(),
|
|
|
|
'chatBoxesFetched': new $.Deferred(),
|
|
|
|
'connected': new $.Deferred(),
|
|
|
|
'pluginsInitialized': new $.Deferred(),
|
|
|
|
'roster': new $.Deferred(),
|
|
|
|
'rosterContactsFetched': new $.Deferred(),
|
|
|
|
'rosterGroupsFetched': new $.Deferred(),
|
|
|
|
'rosterInitialized': new $.Deferred(),
|
|
|
|
'statusInitialized': new $.Deferred()
|
|
|
|
};
|
|
|
|
_converse.emit = function (name) {
|
|
|
|
_converse.trigger.apply(this, arguments);
|
|
|
|
var promise = _converse.promises[name];
|
|
|
|
if (!_.isUndefined(promise)) {
|
|
|
|
promise.resolve();
|
|
|
|
}
|
|
|
|
};
|
2016-02-16 08:46:47 +01:00
|
|
|
|
2017-02-03 19:12:19 +01:00
|
|
|
_converse.core_plugins = [
|
|
|
|
'converse-bookmarks',
|
|
|
|
'converse-chatview',
|
|
|
|
'converse-controlbox',
|
|
|
|
'converse-core',
|
|
|
|
'converse-dragresize',
|
|
|
|
'converse-headline',
|
|
|
|
'converse-mam',
|
|
|
|
'converse-minimize',
|
|
|
|
'converse-muc',
|
|
|
|
'converse-notification',
|
|
|
|
'converse-otr',
|
|
|
|
'converse-ping',
|
|
|
|
'converse-register',
|
|
|
|
'converse-rosterview',
|
|
|
|
'converse-vcard'
|
|
|
|
];
|
|
|
|
|
2016-06-09 11:01:54 +02:00
|
|
|
// Make converse pluggable
|
2016-12-20 11:28:30 +01:00
|
|
|
pluggable.enable(_converse, '_converse', 'pluggable');
|
2016-06-09 11:01:54 +02:00
|
|
|
|
2016-02-20 16:06:12 +01:00
|
|
|
// Module-level constants
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.STATUS_WEIGHTS = {
|
2016-02-16 08:46:47 +01:00
|
|
|
'offline': 6,
|
|
|
|
'unavailable': 5,
|
|
|
|
'xa': 4,
|
|
|
|
'away': 3,
|
|
|
|
'dnd': 2,
|
|
|
|
'chat': 1, // We currently don't differentiate between "chat" and "online"
|
|
|
|
'online': 1
|
|
|
|
};
|
2017-02-28 06:46:21 +01:00
|
|
|
_converse.PRETTY_CHAT_STATUS = {
|
|
|
|
'offline': 'Offline',
|
|
|
|
'unavailable': 'Unavailable',
|
|
|
|
'xa': 'Extended Away',
|
|
|
|
'away': 'Away',
|
|
|
|
'dnd': 'Do not disturb',
|
|
|
|
'chat': 'Chattty',
|
|
|
|
'online': 'Online'
|
|
|
|
};
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.ANONYMOUS = "anonymous";
|
|
|
|
_converse.CLOSED = 'closed';
|
|
|
|
_converse.EXTERNAL = "external";
|
|
|
|
_converse.LOGIN = "login";
|
|
|
|
_converse.LOGOUT = "logout";
|
|
|
|
_converse.OPENED = 'opened';
|
|
|
|
_converse.PREBIND = "prebind";
|
2016-02-16 08:46:47 +01:00
|
|
|
|
2016-02-29 21:54:34 +01:00
|
|
|
var PRETTY_CONNECTION_STATUS = {
|
|
|
|
0: 'ERROR',
|
|
|
|
1: 'CONNECTING',
|
|
|
|
2: 'CONNFAIL',
|
|
|
|
3: 'AUTHENTICATING',
|
|
|
|
4: 'AUTHFAIL',
|
|
|
|
5: 'CONNECTED',
|
|
|
|
6: 'DISCONNECTED',
|
|
|
|
7: 'DISCONNECTING',
|
|
|
|
8: 'ATTACHED',
|
|
|
|
9: 'REDIRECT'
|
|
|
|
};
|
|
|
|
|
2017-01-23 11:15:07 +01:00
|
|
|
var DEFAULT_IMAGE_TYPE = 'image/png';
|
|
|
|
var DEFAULT_IMAGE = "iVBORw0KGgoAAAANSUhEUgAAAGAAAABgCAIAAABt+uBvAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH3gwHCy455JBsggAABkJJREFUeNrtnM1PE1sUwHvvTD8otWLHST/Gimi1CEgr6M6FEWuIBo2pujDVsNDEP8GN/4MbN7oxrlipG2OCgZgYlxAbkRYw1KqkIDRCSkM7nXvvW8x7vjyNeQ9m7p1p3z1LQk/v/Dhz7vkEXL161cHl9wI5Ag6IA+KAOCAOiAPigDggLhwQB2S+iNZ+PcYY/SWEEP2HAAAIoSAIoihCCP+ngDDGtVotGAz29/cfOXJEUZSOjg6n06lp2sbGRqlUWlhYyGazS0tLbrdbEASrzgksyeYJId3d3el0uqenRxRFAAAA4KdfIIRgjD9+/Pj8+fOpqSndslofEIQwHA6Pjo4mEon//qmFhYXHjx8vLi4ihBgDEnp7e9l8E0Jo165dQ0NDd+/eDYVC2/qsJElDQ0OEkKWlpa2tLZamxAhQo9EIBoOjo6MXL17csZLe3l5FUT59+lQul5l5JRaAVFWNRqN37tw5ceKEQVWRSOTw4cOFQuHbt2+iKLYCIISQLMu3b99OJpOmKAwEAgcPHszn8+vr6wzsiG6UQQhxuVyXLl0aGBgwUW0sFstkMl6v90fo1KyAMMYDAwPnzp0zXfPg4GAqlWo0Gk0MiBAiy/L58+edTqf5Aa4onj59OhaLYYybFRCEMBaL0fNxBw4cSCQStN0QRUBut3t4eJjq6U+dOiVJElVPRBFQIBDo6+ujCqirqyscDlONGykC2lYyYSR6pBoQQapHZwAoHo/TuARYAOrs7GQASFEUqn6aIiBJkhgA6ujooFpUo6iaTa7koFwnaoWadLNe81tbWwzoaJrWrICWl5cZAFpbW6OabVAEtLi4yABQsVjUNK0pAWWzWQaAcrlcswKanZ1VVZUqHYRQEwOq1Wpv3ryhCmh6erpcLjdrNl+v1ycnJ+l5UELI27dvv3//3qxxEADgy5cvExMT9Mznw4cPtFtAdAPFarU6Pj5eKpVM17yxsfHy5cvV1VXazXu62gVBKBQKT58+rdVqJqrFGL948eLdu3dU8/g/H4FBUaJYLAqC0NPTY9brMD4+PjY25mDSracOCABACJmZmXE6nUePHjWu8NWrV48ePSKEsGlAs7Agfd5nenq6Wq0mk0kjDzY2NvbkyRMIIbP2PLvhBUEQ8vl8NpuNx+M+n29bzhVjvLKycv/+/YmJCcazQuwA6YzW1tYmJyf1SY+2trZ/rRk1Go1SqfT69esHDx4UCgVmNaa/zZ/9ABUhRFXVYDB48uTJeDweiUQkSfL7/T9MA2NcqVTK5fLy8vL8/PzU1FSxWHS5XJaM4wGr9sUwxqqqer3eUCgkSZJuUBBCfTRvc3OzXC6vrKxUKhWn02nhCJ5lM4oQQo/HgxD6+vXr58+fHf8sDOp+HQDg8XgclorFU676dKLlo6yWRdItIBwQB8QBcUCtfosRQjRNQwhhjPUC4w46WXryBSHU1zgEQWBz99EFhDGu1+t+v//48ePxeFxRlD179ng8nh0Efgiher2+vr6ur3HMzMysrq7uTJVdACGEurq6Ll++nEgkPB7Pj9jPoDHqOxyqqubz+WfPnuVyuV9XPeyeagAAAoHArVu3BgcHab8CuVzu4cOHpVKJUnfA5GweY+xyuc6cOXPv3r1IJMLAR8iyPDw8XK/Xi8Wiqqqmm5KZgBBC7e3tN27cuHbtGuPVpf7+/lAoNDs7W61WzfVKpgHSSzw3b95MpVKW3MfRaDQSiczNzVUqFRMZmQOIEOL1eq9fv3727FlL1t50URRFluX5+flqtWpWEGAOIFEUU6nUlStXLKSjy759+xwOx9zcnKZpphzGHMzhcDiTydgk9r1w4YIp7RPTAAmCkMlk2FeLf/tIEKbTab/fbwtAhJBoNGrutpNx6e7uPnTokC1eMU3T0um0DZPMkZER6wERQnw+n/FFSxpy7Nix3bt3WwwIIcRgIWnHkkwmjecfRgGx7DtuV/r6+iwGhDHev3+/bQF1dnYaH6E2CkiWZdsC2rt3r8WAHA5HW1ubbQGZcjajgOwTH/4qNko1Wlg4IA6IA+KAOKBWBUQIsfNojyliKIoRRfH9+/dut9umf3wzpoUNNQ4BAJubmwz+ic+OxefzWWlBhJD29nbug7iT5sIBcUAcEAfEAXFAHBAHxOVn+QMrmWpuPZx12gAAAABJRU5ErkJggg==";
|
|
|
|
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.log = function (txt, level) {
|
2016-02-29 21:54:34 +01:00
|
|
|
var logger;
|
2017-01-26 15:49:02 +01:00
|
|
|
if (_.isUndefined(console) || _.isUndefined(console.log)) {
|
|
|
|
logger = { log: _.noop, error: _.noop };
|
2016-02-29 21:54:34 +01:00
|
|
|
} else {
|
|
|
|
logger = console;
|
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
if (_converse.debug) {
|
2016-02-29 21:54:34 +01:00
|
|
|
if (level === 'error') {
|
|
|
|
logger.log('ERROR: '+txt);
|
|
|
|
} else {
|
|
|
|
logger.log(txt);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.initialize = function (settings, callback) {
|
2016-02-16 08:46:47 +01:00
|
|
|
"use strict";
|
2017-01-26 15:49:02 +01:00
|
|
|
settings = !_.isUndefined(settings) ? settings : {};
|
2016-06-09 23:22:34 +02:00
|
|
|
var init_deferred = new $.Deferred();
|
2016-12-06 12:07:52 +01:00
|
|
|
|
2016-12-20 10:30:20 +01:00
|
|
|
if (!_.isUndefined(_converse.chatboxes)) {
|
|
|
|
// Looks like _converse.initialized was called again without logging
|
2016-12-06 12:07:52 +01:00
|
|
|
// out or disconnecting in the previous session.
|
2016-12-19 08:33:46 +01:00
|
|
|
// This happens in tests. We therefore first clean up.
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.reset();
|
|
|
|
_converse.off();
|
|
|
|
_converse.stopListening();
|
|
|
|
_converse._tearDown();
|
2016-12-06 12:07:52 +01:00
|
|
|
}
|
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
var unloadevent;
|
|
|
|
if ('onpagehide' in window) {
|
|
|
|
// Pagehide gets thrown in more cases than unload. Specifically it
|
|
|
|
// gets thrown when the page is cached and not just
|
|
|
|
// closed/destroyed. It's the only viable event on mobile Safari.
|
|
|
|
// https://www.webkit.org/blog/516/webkit-page-cache-ii-the-unload-event/
|
|
|
|
unloadevent = 'pagehide';
|
|
|
|
} else if ('onbeforeunload' in window) {
|
|
|
|
unloadevent = 'beforeunload';
|
|
|
|
} else if ('onunload' in window) {
|
|
|
|
unloadevent = 'unload';
|
|
|
|
}
|
|
|
|
|
|
|
|
// Logging
|
2016-12-20 10:30:20 +01:00
|
|
|
Strophe.log = function (level, msg) { _converse.log(level+' '+msg, level); };
|
|
|
|
Strophe.error = function (msg) { _converse.log(msg, 'error'); };
|
2016-02-16 08:46:47 +01:00
|
|
|
|
|
|
|
// Add Strophe Namespaces
|
|
|
|
Strophe.addNamespace('CARBONS', 'urn:xmpp:carbons:2');
|
|
|
|
Strophe.addNamespace('CHATSTATES', 'http://jabber.org/protocol/chatstates');
|
|
|
|
Strophe.addNamespace('CSI', 'urn:xmpp:csi:0');
|
2017-02-28 22:17:28 +01:00
|
|
|
Strophe.addNamespace('DELAY', 'urn:xmpp:delay');
|
2016-05-30 18:20:23 +02:00
|
|
|
Strophe.addNamespace('HINTS', 'urn:xmpp:hints');
|
2017-02-28 22:17:28 +01:00
|
|
|
Strophe.addNamespace('NICK', 'http://jabber.org/protocol/nick');
|
2016-06-27 23:50:38 +02:00
|
|
|
Strophe.addNamespace('PUBSUB', 'http://jabber.org/protocol/pubsub');
|
2017-02-28 22:17:28 +01:00
|
|
|
Strophe.addNamespace('ROSTERX', 'http://jabber.org/protocol/rosterx');
|
|
|
|
Strophe.addNamespace('XFORM', 'jabber:x:data');
|
2016-02-16 08:46:47 +01:00
|
|
|
|
2016-03-01 09:58:36 +01:00
|
|
|
// Instance level constants
|
2016-02-16 08:46:47 +01:00
|
|
|
this.TIMEOUTS = { // Set as module attr so that we can override in tests.
|
2016-08-18 11:28:48 +02:00
|
|
|
'PAUSED': 10000,
|
2016-02-16 08:46:47 +01:00
|
|
|
'INACTIVE': 90000
|
|
|
|
};
|
|
|
|
|
2016-03-01 09:58:36 +01:00
|
|
|
// XEP-0085 Chat states
|
|
|
|
// http://xmpp.org/extensions/xep-0085.html
|
|
|
|
this.INACTIVE = 'inactive';
|
|
|
|
this.ACTIVE = 'active';
|
|
|
|
this.COMPOSING = 'composing';
|
|
|
|
this.PAUSED = 'paused';
|
|
|
|
this.GONE = 'gone';
|
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
// Detect support for the user's locale
|
|
|
|
// ------------------------------------
|
2017-02-13 15:16:06 +01:00
|
|
|
locales = _.isUndefined(locales) ? {} : locales;
|
|
|
|
this.isConverseLocale = function (locale) {
|
|
|
|
return !_.isUndefined(locales[locale]);
|
|
|
|
};
|
2016-02-16 08:46:47 +01:00
|
|
|
this.isMomentLocale = function (locale) { return moment.locale() !== moment.locale(locale); };
|
|
|
|
if (!moment.locale) { //moment.lang is deprecated after 2.8.1, use moment.locale instead
|
|
|
|
moment.locale = moment.lang;
|
|
|
|
}
|
2016-07-18 13:41:31 +02:00
|
|
|
moment.locale(utils.detectLocale(this.isMomentLocale));
|
2017-02-13 16:16:16 +01:00
|
|
|
if (_.includes(_.keys(locales), settings.i18n)) {
|
|
|
|
settings.i18n = locales[settings.i18n];
|
|
|
|
}
|
2016-07-18 13:41:31 +02:00
|
|
|
this.i18n = settings.i18n ? settings.i18n : locales[utils.detectLocale(this.isConverseLocale)] || {};
|
2016-02-16 08:46:47 +01:00
|
|
|
|
|
|
|
// Translation machinery
|
|
|
|
// ---------------------
|
2016-12-20 11:42:20 +01:00
|
|
|
var __ = _converse.__ = utils.__.bind(_converse);
|
|
|
|
_converse.___ = utils.___;
|
2016-06-21 17:06:51 +02:00
|
|
|
var DESC_GROUP_TOGGLE = __('Click to hide these contacts');
|
2016-02-16 08:46:47 +01:00
|
|
|
|
|
|
|
// Default configuration values
|
|
|
|
// ----------------------------
|
|
|
|
this.default_settings = {
|
|
|
|
allow_contact_requests: true,
|
2017-02-01 10:14:36 +01:00
|
|
|
allow_non_roster_messaging: false,
|
2016-02-16 08:46:47 +01:00
|
|
|
animate: true,
|
2016-09-16 13:06:52 +02:00
|
|
|
authentication: 'login', // Available values are "login", "prebind", "anonymous" and "external".
|
2016-02-16 08:46:47 +01:00
|
|
|
auto_away: 0, // Seconds after which user status is set to 'away'
|
|
|
|
auto_login: false, // Currently only used in connection with anonymous login
|
|
|
|
auto_reconnect: false,
|
|
|
|
auto_subscribe: false,
|
|
|
|
auto_xa: 0, // Seconds after which user status is set to 'xa'
|
2017-02-03 19:12:19 +01:00
|
|
|
blacklisted_plugins: [],
|
|
|
|
bosh_service_url: undefined,
|
2016-10-12 13:55:47 +02:00
|
|
|
connection_options: {},
|
2016-03-31 10:37:28 +02:00
|
|
|
credentials_url: null, // URL from where login credentials can be fetched
|
2016-02-16 08:46:47 +01:00
|
|
|
csi_waiting_time: 0, // Support for XEP-0352. Seconds before client is considered idle and CSI is sent out.
|
|
|
|
debug: false,
|
2016-05-31 10:03:06 +02:00
|
|
|
default_state: 'online',
|
2016-02-16 08:46:47 +01:00
|
|
|
expose_rid_and_sid: false,
|
2016-05-28 08:35:16 +02:00
|
|
|
filter_by_resource: false,
|
2016-02-16 08:46:47 +01:00
|
|
|
forward_messages: false,
|
|
|
|
hide_offline_users: false,
|
|
|
|
include_offline_state: false,
|
|
|
|
jid: undefined,
|
2017-02-17 08:54:58 +01:00
|
|
|
keepalive: true,
|
2016-02-16 08:46:47 +01:00
|
|
|
locked_domain: undefined,
|
2017-02-27 11:09:15 +01:00
|
|
|
message_carbons: true,
|
2017-02-01 10:14:36 +01:00
|
|
|
message_storage: 'session',
|
2016-02-16 08:46:47 +01:00
|
|
|
password: undefined,
|
|
|
|
prebind_url: null,
|
2017-02-22 22:13:23 +01:00
|
|
|
priority: 0,
|
2016-02-16 08:46:47 +01:00
|
|
|
rid: undefined,
|
2017-02-17 08:54:58 +01:00
|
|
|
roster_groups: true,
|
2016-02-16 08:46:47 +01:00
|
|
|
show_only_online_users: false,
|
|
|
|
sid: undefined,
|
|
|
|
storage: 'session',
|
2016-03-14 18:03:48 +01:00
|
|
|
strict_plugin_dependencies: false,
|
2017-02-03 19:12:19 +01:00
|
|
|
synchronize_availability: true,
|
2016-02-16 08:46:47 +01:00
|
|
|
websocket_url: undefined,
|
2017-02-03 19:12:19 +01:00
|
|
|
whitelisted_plugins: [],
|
2016-02-16 08:46:47 +01:00
|
|
|
xhr_custom_status: false,
|
|
|
|
xhr_custom_status_url: '',
|
|
|
|
};
|
2017-01-26 10:08:43 +01:00
|
|
|
_.assignIn(this, this.default_settings);
|
2016-02-16 08:46:47 +01:00
|
|
|
// Allow only whitelisted configuration attributes to be overwritten
|
2017-01-26 15:49:02 +01:00
|
|
|
_.assignIn(this, _.pick(settings, _.keys(this.default_settings)));
|
2016-02-16 08:46:47 +01:00
|
|
|
|
2016-12-20 10:30:20 +01:00
|
|
|
if (this.authentication === _converse.ANONYMOUS) {
|
2016-07-24 10:54:57 +02:00
|
|
|
if (this.auto_login && !this.jid) {
|
|
|
|
throw new Error("Config Error: you need to provide the server's " +
|
|
|
|
"domain via the 'jid' option when using anonymous " +
|
|
|
|
"authentication with auto_login.");
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$.fx.off = !this.animate;
|
|
|
|
|
|
|
|
// Module-level variables
|
|
|
|
// ----------------------
|
2017-01-26 15:49:02 +01:00
|
|
|
this.callback = callback || _.noop;
|
2016-02-16 08:46:47 +01:00
|
|
|
/* When reloading the page:
|
|
|
|
* For new sessions, we need to send out a presence stanza to notify
|
|
|
|
* the server/network that we're online.
|
|
|
|
* When re-attaching to an existing session (e.g. via the keepalive
|
|
|
|
* option), we don't need to again send out a presence stanza, because
|
|
|
|
* it's as if "we never left" (see onConnectStatusChanged).
|
|
|
|
* https://github.com/jcbrand/converse.js/issues/521
|
|
|
|
*/
|
|
|
|
this.send_initial_presence = true;
|
|
|
|
this.msg_counter = 0;
|
2016-10-26 12:50:00 +02:00
|
|
|
this.user_settings = settings; // Save the user settings so that they can be used by plugins
|
2016-02-16 08:46:47 +01:00
|
|
|
|
|
|
|
// Module-level functions
|
|
|
|
// ----------------------
|
2017-02-14 15:58:29 +01:00
|
|
|
this.getViewForChatBox = function (chatbox) {
|
2016-10-26 12:50:00 +02:00
|
|
|
if (!chatbox) { return; }
|
2017-02-14 15:58:29 +01:00
|
|
|
return _converse.chatboxviews.get(chatbox.get('id'));
|
2016-10-26 12:50:00 +02:00
|
|
|
};
|
2016-02-16 08:46:47 +01:00
|
|
|
|
2016-02-20 16:06:12 +01:00
|
|
|
this.generateResource = function () {
|
2016-02-16 14:31:46 +01:00
|
|
|
return '/converse.js-' + Math.floor(Math.random()*139749825).toString();
|
|
|
|
};
|
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
this.sendCSI = function (stat) {
|
2017-01-26 15:49:02 +01:00
|
|
|
/* Send out a Chat Status Notification (XEP-0352)
|
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* (String) stat: The user's chat status
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
/* Send out a Chat Status Notification (XEP-0352) */
|
2017-01-26 15:49:02 +01:00
|
|
|
// XXX if (converse.features[Strophe.NS.CSI] || true) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.send($build(stat, {xmlns: Strophe.NS.CSI}));
|
|
|
|
_converse.inactive = (stat === _converse.INACTIVE) ? true : false;
|
2016-02-16 08:46:47 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
this.onUserActivity = function () {
|
|
|
|
/* Resets counters and flags relating to CSI and auto_away/auto_xa */
|
2016-12-20 10:30:20 +01:00
|
|
|
if (_converse.idle_seconds > 0) {
|
|
|
|
_converse.idle_seconds = 0;
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
if (!_converse.connection.authenticated) {
|
2016-02-16 08:46:47 +01:00
|
|
|
// We can't send out any stanzas when there's no authenticated connection.
|
2016-04-08 10:29:30 +02:00
|
|
|
// converse can happen when the connection reconnects.
|
2016-02-16 08:46:47 +01:00
|
|
|
return;
|
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
if (_converse.inactive) {
|
|
|
|
_converse.sendCSI(_converse.ACTIVE);
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
if (_converse.auto_changed_status === true) {
|
|
|
|
_converse.auto_changed_status = false;
|
2016-06-01 18:01:09 +02:00
|
|
|
// XXX: we should really remember the original state here, and
|
|
|
|
// then set it back to that...
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.xmppstatus.setStatus(_converse.default_state);
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
this.onEverySecond = function () {
|
|
|
|
/* An interval handler running every second.
|
2016-04-08 10:29:30 +02:00
|
|
|
* Used for CSI and the auto_away and auto_xa features.
|
2016-02-16 08:46:47 +01:00
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
if (!_converse.connection.authenticated) {
|
2016-02-16 08:46:47 +01:00
|
|
|
// We can't send out any stanzas when there's no authenticated connection.
|
|
|
|
// This can happen when the connection reconnects.
|
|
|
|
return;
|
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
var stat = _converse.xmppstatus.getStatus();
|
|
|
|
_converse.idle_seconds++;
|
|
|
|
if (_converse.csi_waiting_time > 0 &&
|
|
|
|
_converse.idle_seconds > _converse.csi_waiting_time &&
|
|
|
|
!_converse.inactive) {
|
|
|
|
_converse.sendCSI(_converse.INACTIVE);
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
if (_converse.auto_away > 0 &&
|
|
|
|
_converse.idle_seconds > _converse.auto_away &&
|
2017-02-25 10:52:31 +01:00
|
|
|
stat !== 'away' && stat !== 'xa' && stat !== 'dnd') {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.auto_changed_status = true;
|
|
|
|
_converse.xmppstatus.setStatus('away');
|
|
|
|
} else if (_converse.auto_xa > 0 &&
|
2017-02-25 10:52:31 +01:00
|
|
|
_converse.idle_seconds > _converse.auto_xa &&
|
|
|
|
stat !== 'xa' && stat !== 'dnd') {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.auto_changed_status = true;
|
|
|
|
_converse.xmppstatus.setStatus('xa');
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
this.registerIntervalHandler = function () {
|
|
|
|
/* Set an interval of one second and register a handler for it.
|
|
|
|
* Required for the auto_away, auto_xa and csi_waiting_time features.
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
if (_converse.auto_away < 1 && _converse.auto_xa < 1 && _converse.csi_waiting_time < 1) {
|
2016-02-16 08:46:47 +01:00
|
|
|
// Waiting time of less then one second means features aren't used.
|
|
|
|
return;
|
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.idle_seconds = 0;
|
|
|
|
_converse.auto_changed_status = false; // Was the user's status changed by _converse.js?
|
|
|
|
$(window).on('click mousemove keypress focus'+unloadevent, _converse.onUserActivity);
|
|
|
|
_converse.everySecondTrigger = window.setInterval(_converse.onEverySecond, 1000);
|
2016-02-16 08:46:47 +01:00
|
|
|
};
|
2016-02-29 21:54:34 +01:00
|
|
|
|
2016-08-23 10:00:41 +02:00
|
|
|
this.giveFeedback = function (subject, klass, message) {
|
2016-02-16 08:46:47 +01:00
|
|
|
$('.conn-feedback').each(function (idx, el) {
|
2016-11-24 02:07:32 +01:00
|
|
|
el.classList.add('conn-feedback');
|
|
|
|
el.textContent = subject;
|
2016-02-16 08:46:47 +01:00
|
|
|
if (klass) {
|
2016-11-24 02:07:32 +01:00
|
|
|
el.classList.add(klass);
|
2016-02-16 08:46:47 +01:00
|
|
|
} else {
|
2016-11-24 02:07:32 +01:00
|
|
|
el.classList.remove('error');
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
});
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('feedback', {
|
2016-08-23 10:00:41 +02:00
|
|
|
'klass': klass,
|
|
|
|
'message': message,
|
|
|
|
'subject': subject
|
|
|
|
});
|
2016-02-16 08:46:47 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
this.rejectPresenceSubscription = function (jid, message) {
|
|
|
|
/* Reject or cancel another user's subscription to our presence updates.
|
2017-02-01 17:01:43 +01:00
|
|
|
*
|
2016-02-16 08:46:47 +01:00
|
|
|
* Parameters:
|
|
|
|
* (String) jid - The Jabber ID of the user whose subscription
|
|
|
|
* is being canceled.
|
|
|
|
* (String) message - An optional message to the user
|
|
|
|
*/
|
|
|
|
var pres = $pres({to: jid, type: "unsubscribed"});
|
|
|
|
if (message && message !== "") { pres.c("status").t(message); }
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.send(pres);
|
2016-02-16 08:46:47 +01:00
|
|
|
};
|
|
|
|
|
2017-01-26 15:49:02 +01:00
|
|
|
this.reconnect = _.debounce(function () {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.log('RECONNECTING');
|
|
|
|
_converse.log('The connection has dropped, attempting to reconnect.');
|
|
|
|
_converse.giveFeedback(
|
2016-11-02 13:44:56 +01:00
|
|
|
__("Reconnecting"),
|
|
|
|
'warn',
|
|
|
|
__('The connection has dropped, attempting to reconnect.')
|
2016-08-23 10:00:41 +02:00
|
|
|
);
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.reconnecting = true;
|
|
|
|
_converse._tearDown();
|
|
|
|
_converse.logIn(null, true);
|
2017-02-01 17:01:43 +01:00
|
|
|
}, 3000, {'leading': true});
|
2016-02-16 08:46:47 +01:00
|
|
|
|
2016-12-04 16:00:46 +01:00
|
|
|
this.disconnect = function () {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.log('DISCONNECTED');
|
|
|
|
delete _converse.connection.reconnecting;
|
2017-02-18 10:46:33 +01:00
|
|
|
_converse.connection.reset();
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse._tearDown();
|
|
|
|
_converse.chatboxviews.closeAllChatBoxes();
|
|
|
|
_converse.emit('disconnected');
|
2016-12-04 16:00:46 +01:00
|
|
|
};
|
|
|
|
|
2017-02-01 17:01:43 +01:00
|
|
|
this.onDisconnected = function () {
|
|
|
|
/* Gets called once strophe's status reaches Strophe.Status.DISCONNECTED.
|
|
|
|
* Will either start a teardown process for converse.js or attempt
|
|
|
|
* to reconnect.
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
if (_converse.disconnection_cause === Strophe.Status.AUTHFAIL) {
|
|
|
|
if (_converse.credentials_url && _converse.auto_reconnect) {
|
2017-02-01 17:34:38 +01:00
|
|
|
/* In this case, we reconnect, because we might be receiving
|
|
|
|
* expirable tokens from the credentials_url.
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('will-reconnect');
|
|
|
|
return _converse.reconnect();
|
2017-02-01 17:34:38 +01:00
|
|
|
} else {
|
2016-12-20 10:30:20 +01:00
|
|
|
return _converse.disconnect();
|
2017-02-01 17:34:38 +01:00
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
} else if (_converse.disconnection_cause === _converse.LOGOUT ||
|
|
|
|
_converse.disconnection_reason === "host-unknown" ||
|
|
|
|
!_converse.auto_reconnect) {
|
|
|
|
return _converse.disconnect();
|
2016-08-18 21:43:18 +02:00
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('will-reconnect');
|
|
|
|
_converse.reconnect();
|
2016-08-18 21:43:18 +02:00
|
|
|
};
|
|
|
|
|
2017-01-31 20:57:19 +01:00
|
|
|
this.setDisconnectionCause = function (cause, reason, override) {
|
2017-02-01 17:01:43 +01:00
|
|
|
/* Used to keep track of why we got disconnected, so that we can
|
|
|
|
* decide on what the next appropriate action is (in onDisconnected)
|
|
|
|
*/
|
|
|
|
if (_.isUndefined(cause)) {
|
2016-12-20 10:30:20 +01:00
|
|
|
delete _converse.disconnection_cause;
|
|
|
|
delete _converse.disconnection_reason;
|
|
|
|
} else if (_.isUndefined(_converse.disconnection_cause) || override) {
|
|
|
|
_converse.disconnection_cause = cause;
|
|
|
|
_converse.disconnection_reason = reason;
|
2016-03-31 14:23:45 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-08-18 21:43:18 +02:00
|
|
|
this.onConnectStatusChanged = function (status, condition) {
|
2017-02-01 17:01:43 +01:00
|
|
|
/* Callback method called by Strophe as the Strophe.Connection goes
|
|
|
|
* through various states while establishing or tearing down a
|
|
|
|
* connection.
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.log("Status changed to: "+PRETTY_CONNECTION_STATUS[status]);
|
2016-02-16 08:46:47 +01:00
|
|
|
if (status === Strophe.Status.CONNECTED || status === Strophe.Status.ATTACHED) {
|
|
|
|
// By default we always want to send out an initial presence stanza.
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.send_initial_presence = true;
|
|
|
|
_converse.setDisconnectionCause();
|
|
|
|
if (_converse.connection.reconnecting) {
|
|
|
|
_converse.log(status === Strophe.Status.CONNECTED ? 'Reconnected' : 'Reattached');
|
|
|
|
_converse.onConnected(true);
|
2016-02-16 08:46:47 +01:00
|
|
|
} else {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.log(status === Strophe.Status.CONNECTED ? 'Connected' : 'Attached');
|
|
|
|
if (_converse.connection.restored) {
|
2016-08-18 21:43:18 +02:00
|
|
|
// No need to send an initial presence stanza when
|
|
|
|
// we're restoring an existing session.
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.send_initial_presence = false;
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.onConnected();
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
} else if (status === Strophe.Status.DISCONNECTED) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.setDisconnectionCause(status, condition);
|
|
|
|
_converse.onDisconnected();
|
2016-02-16 08:46:47 +01:00
|
|
|
} else if (status === Strophe.Status.ERROR) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.giveFeedback(
|
2016-08-23 10:00:41 +02:00
|
|
|
__('Connection error'), 'error',
|
|
|
|
__('An error occurred while connecting to the chat server.')
|
|
|
|
);
|
2016-02-16 08:46:47 +01:00
|
|
|
} else if (status === Strophe.Status.CONNECTING) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.giveFeedback(__('Connecting'));
|
2016-02-16 08:46:47 +01:00
|
|
|
} else if (status === Strophe.Status.AUTHENTICATING) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.giveFeedback(__('Authenticating'));
|
2016-02-16 08:46:47 +01:00
|
|
|
} else if (status === Strophe.Status.AUTHFAIL) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.giveFeedback(__('Authentication Failed'), 'error');
|
|
|
|
_converse.setDisconnectionCause(status, condition, true);
|
2017-02-13 16:22:56 +01:00
|
|
|
_converse.onDisconnected();
|
2016-12-04 15:14:40 +01:00
|
|
|
} else if (status === Strophe.Status.CONNFAIL) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.giveFeedback(
|
2017-01-31 21:11:49 +01:00
|
|
|
__('Connection failed'), 'error',
|
|
|
|
__('An error occurred while connecting to the chat server: '+condition)
|
|
|
|
);
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.setDisconnectionCause(status, condition);
|
2016-12-04 15:14:40 +01:00
|
|
|
} else if (status === Strophe.Status.DISCONNECTING) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.setDisconnectionCause(status, condition);
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
this.updateMsgCounter = function () {
|
|
|
|
if (this.msg_counter > 0) {
|
|
|
|
if (document.title.search(/^Messages \(\d+\) /) === -1) {
|
|
|
|
document.title = "Messages (" + this.msg_counter + ") " + document.title;
|
|
|
|
} else {
|
|
|
|
document.title = document.title.replace(/^Messages \(\d+\) /, "Messages (" + this.msg_counter + ") ");
|
|
|
|
}
|
|
|
|
} else if (document.title.search(/^Messages \(\d+\) /) !== -1) {
|
|
|
|
document.title = document.title.replace(/^Messages \(\d+\) /, "");
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
this.incrementMsgCounter = function () {
|
|
|
|
this.msg_counter += 1;
|
|
|
|
this.updateMsgCounter();
|
|
|
|
};
|
|
|
|
|
|
|
|
this.clearMsgCounter = function () {
|
|
|
|
this.msg_counter = 0;
|
|
|
|
this.updateMsgCounter();
|
|
|
|
};
|
|
|
|
|
2016-03-16 12:16:32 +01:00
|
|
|
this.initStatus = function () {
|
|
|
|
var deferred = new $.Deferred();
|
2016-02-16 08:46:47 +01:00
|
|
|
this.xmppstatus = new this.XMPPStatus();
|
2016-12-20 10:30:20 +01:00
|
|
|
var id = b64_sha1('converse.xmppstatus-'+_converse.bare_jid);
|
2016-02-16 08:46:47 +01:00
|
|
|
this.xmppstatus.id = id; // Appears to be necessary for backbone.browserStorage
|
2016-12-20 10:30:20 +01:00
|
|
|
this.xmppstatus.browserStorage = new Backbone.BrowserStorage[_converse.storage](id);
|
2016-03-16 12:16:32 +01:00
|
|
|
this.xmppstatus.fetch({
|
|
|
|
success: deferred.resolve,
|
|
|
|
error: deferred.resolve
|
|
|
|
});
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('statusInitialized');
|
2016-03-16 12:16:32 +01:00
|
|
|
return deferred.promise();
|
2016-02-16 08:46:47 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
this.initSession = function () {
|
|
|
|
this.session = new this.Session();
|
|
|
|
var id = b64_sha1('converse.bosh-session');
|
|
|
|
this.session.id = id; // Appears to be necessary for backbone.browserStorage
|
2016-12-20 10:30:20 +01:00
|
|
|
this.session.browserStorage = new Backbone.BrowserStorage[_converse.storage](id);
|
2016-02-16 08:46:47 +01:00
|
|
|
this.session.fetch();
|
|
|
|
};
|
|
|
|
|
|
|
|
this.clearSession = function () {
|
2016-10-17 10:34:09 +02:00
|
|
|
if (!_.isUndefined(this.roster)) {
|
2016-02-16 08:46:47 +01:00
|
|
|
this.roster.browserStorage._clear();
|
|
|
|
}
|
|
|
|
this.session.browserStorage._clear();
|
|
|
|
};
|
|
|
|
|
|
|
|
this.logOut = function () {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.setDisconnectionCause(_converse.LOGOUT, undefined, true);
|
|
|
|
if (!_.isUndefined(_converse.connection)) {
|
|
|
|
_converse.connection.disconnect();
|
2016-04-13 11:15:15 +02:00
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.chatboxviews.closeAllChatBoxes();
|
|
|
|
_converse.clearSession();
|
|
|
|
_converse._tearDown();
|
|
|
|
_converse.emit('logout');
|
2016-02-16 08:46:47 +01:00
|
|
|
};
|
|
|
|
|
2016-06-17 11:30:47 +02:00
|
|
|
this.saveWindowState = function (ev, hidden) {
|
|
|
|
// XXX: eventually we should be able to just use
|
|
|
|
// document.visibilityState (when we drop support for older
|
|
|
|
// browsers).
|
2016-06-16 18:14:22 +02:00
|
|
|
var state;
|
|
|
|
var v = "visible", h = "hidden",
|
|
|
|
event_map = {
|
|
|
|
'focus': v,
|
|
|
|
'focusin': v,
|
|
|
|
'pageshow': v,
|
|
|
|
'blur': h,
|
|
|
|
'focusout': h,
|
|
|
|
'pagehide': h
|
|
|
|
};
|
2016-06-17 11:30:47 +02:00
|
|
|
ev = ev || document.createEvent('Events');
|
2016-06-16 18:14:22 +02:00
|
|
|
if (ev.type in event_map) {
|
|
|
|
state = event_map[ev.type];
|
|
|
|
} else {
|
|
|
|
state = document[hidden] ? "hidden" : "visible";
|
|
|
|
}
|
|
|
|
if (state === 'visible') {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.clearMsgCounter();
|
2016-06-16 18:14:22 +02:00
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.windowState = state;
|
2016-06-16 18:14:22 +02:00
|
|
|
};
|
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
this.registerGlobalEventHandlers = function () {
|
2016-06-16 18:14:22 +02:00
|
|
|
// Taken from:
|
|
|
|
// http://stackoverflow.com/questions/1060008/is-there-a-way-to-detect-if-a-browser-window-is-not-currently-active
|
|
|
|
var hidden = "hidden";
|
|
|
|
// Standards:
|
|
|
|
if (hidden in document) {
|
2016-12-20 10:30:20 +01:00
|
|
|
document.addEventListener("visibilitychange", _.partial(_converse.saveWindowState, _, hidden));
|
2016-06-16 18:14:22 +02:00
|
|
|
} else if ((hidden = "mozHidden") in document) {
|
2016-12-20 10:30:20 +01:00
|
|
|
document.addEventListener("mozvisibilitychange", _.partial(_converse.saveWindowState, _, hidden));
|
2016-06-16 18:14:22 +02:00
|
|
|
} else if ((hidden = "webkitHidden") in document) {
|
2016-12-20 10:30:20 +01:00
|
|
|
document.addEventListener("webkitvisibilitychange", _.partial(_converse.saveWindowState, _, hidden));
|
2016-06-16 18:14:22 +02:00
|
|
|
} else if ((hidden = "msHidden") in document) {
|
2016-12-20 10:30:20 +01:00
|
|
|
document.addEventListener("msvisibilitychange", _.partial(_converse.saveWindowState, _, hidden));
|
2016-06-16 18:14:22 +02:00
|
|
|
} else if ("onfocusin" in document) {
|
|
|
|
// IE 9 and lower:
|
2016-12-20 10:30:20 +01:00
|
|
|
document.onfocusin = document.onfocusout = _.partial(_converse.saveWindowState, _, hidden);
|
2016-06-16 18:14:22 +02:00
|
|
|
} else {
|
|
|
|
// All others:
|
2016-12-20 10:30:20 +01:00
|
|
|
window.onpageshow = window.onpagehide = window.onfocus = window.onblur = _.partial(_converse.saveWindowState, _, hidden);
|
2016-06-16 18:14:22 +02:00
|
|
|
}
|
|
|
|
// set the initial state (but only if browser supports the Page Visibility API)
|
|
|
|
if( document[hidden] !== undefined ) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_.partial(_converse.saveWindowState, _, hidden)({type: document[hidden] ? "blur" : "focus"});
|
2016-06-16 18:14:22 +02:00
|
|
|
}
|
2016-02-16 08:46:47 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
this.enableCarbons = function () {
|
|
|
|
/* Ask the XMPP server to enable Message Carbons
|
|
|
|
* See XEP-0280 https://xmpp.org/extensions/xep-0280.html#enabling
|
|
|
|
*/
|
|
|
|
if (!this.message_carbons || this.session.get('carbons_enabled')) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
var carbons_iq = new Strophe.Builder('iq', {
|
|
|
|
from: this.connection.jid,
|
|
|
|
id: 'enablecarbons',
|
|
|
|
type: 'set'
|
|
|
|
})
|
|
|
|
.c('enable', {xmlns: Strophe.NS.CARBONS});
|
|
|
|
this.connection.addHandler(function (iq) {
|
2016-11-24 02:07:32 +01:00
|
|
|
if (iq.querySelectorAll('error').length > 0) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.log('ERROR: An error occured while trying to enable message carbons.');
|
2016-02-16 08:46:47 +01:00
|
|
|
} else {
|
|
|
|
this.session.save({carbons_enabled: true});
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.log('Message carbons have been enabled.');
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
}.bind(this), null, "iq", null, "enablecarbons");
|
|
|
|
this.connection.send(carbons_iq);
|
|
|
|
};
|
|
|
|
|
2016-06-22 12:21:56 +02:00
|
|
|
this.initRoster = function () {
|
2016-11-02 14:13:49 +01:00
|
|
|
/* Initialize the Bakcbone collections that represent the contats
|
|
|
|
* roster and the roster groups.
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.roster = new _converse.RosterContacts();
|
|
|
|
_converse.roster.browserStorage = new Backbone.BrowserStorage.session(
|
|
|
|
b64_sha1('converse.contacts-'+_converse.bare_jid));
|
|
|
|
_converse.rostergroups = new _converse.RosterGroups();
|
|
|
|
_converse.rostergroups.browserStorage = new Backbone.BrowserStorage.session(
|
|
|
|
b64_sha1('converse.roster.groups'+_converse.bare_jid));
|
|
|
|
_converse.emit('rosterInitialized');
|
2016-06-22 12:21:56 +02:00
|
|
|
};
|
|
|
|
|
2016-09-21 15:00:09 +02:00
|
|
|
this.populateRoster = function () {
|
|
|
|
/* Fetch all the roster groups, and then the roster contacts.
|
|
|
|
* Emit an event after fetching is done in each case.
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.rostergroups.fetchRosterGroups().then(function () {
|
|
|
|
_converse.emit('rosterGroupsFetched');
|
|
|
|
_converse.roster.fetchRosterContacts().then(function () {
|
|
|
|
_converse.emit('rosterContactsFetched');
|
|
|
|
_converse.sendInitialPresence();
|
2016-09-21 15:00:09 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2016-06-22 12:21:56 +02:00
|
|
|
this.unregisterPresenceHandler = function () {
|
2016-12-20 10:30:20 +01:00
|
|
|
if (!_.isUndefined(_converse.presence_ref)) {
|
|
|
|
_converse.connection.deleteHandler(_converse.presence_ref);
|
|
|
|
delete _converse.presence_ref;
|
2016-06-22 12:21:56 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
this.registerPresenceHandler = function () {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.unregisterPresenceHandler();
|
|
|
|
_converse.presence_ref = _converse.connection.addHandler(
|
2016-06-22 12:21:56 +02:00
|
|
|
function (presence) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.roster.presenceHandler(presence);
|
2016-06-22 12:21:56 +02:00
|
|
|
return true;
|
|
|
|
}, null, 'presence', null);
|
|
|
|
};
|
|
|
|
|
2016-09-20 12:10:28 +02:00
|
|
|
|
|
|
|
this.sendInitialPresence = function () {
|
2016-12-20 10:30:20 +01:00
|
|
|
if (_converse.send_initial_presence) {
|
|
|
|
_converse.xmppstatus.sendPresence();
|
2016-09-20 12:10:28 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-11-23 12:53:06 +01:00
|
|
|
this.onStatusInitialized = function (reconnecting) {
|
|
|
|
/* Continue with session establishment (e.g. fetching chat boxes,
|
|
|
|
* populating the roster etc.) necessary once the connection has
|
|
|
|
* been established.
|
|
|
|
*/
|
|
|
|
if (reconnecting) {
|
|
|
|
// No need to recreate the roster, otherwise we lose our
|
|
|
|
// cached data. However we still emit an event, to give
|
|
|
|
// event handlers a chance to register views for the
|
|
|
|
// roster and its groups, before we start populating.
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('rosterReadyAfterReconnection');
|
2016-11-23 12:53:06 +01:00
|
|
|
} else {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.registerIntervalHandler();
|
|
|
|
_converse.initRoster();
|
2016-11-23 12:53:06 +01:00
|
|
|
}
|
|
|
|
// First set up chat boxes, before populating the roster, so that
|
|
|
|
// the controlbox is properly set up and ready for the rosterview.
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.chatboxes.onConnected();
|
|
|
|
_converse.populateRoster();
|
|
|
|
_converse.registerPresenceHandler();
|
|
|
|
_converse.giveFeedback(__('Contacts'));
|
2016-11-23 12:53:06 +01:00
|
|
|
if (reconnecting) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.xmppstatus.sendPresence();
|
2016-11-23 12:53:06 +01:00
|
|
|
} else {
|
|
|
|
init_deferred.resolve();
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('initialized');
|
2016-03-16 12:16:32 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-11-02 13:44:56 +01:00
|
|
|
this.setUserJid = function () {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.jid = _converse.connection.jid;
|
|
|
|
_converse.bare_jid = Strophe.getBareJidFromJid(_converse.connection.jid);
|
|
|
|
_converse.resource = Strophe.getResourceFromJid(_converse.connection.jid);
|
|
|
|
_converse.domain = Strophe.getDomainFromJid(_converse.connection.jid);
|
2016-11-02 13:44:56 +01:00
|
|
|
};
|
|
|
|
|
2016-11-23 12:53:06 +01:00
|
|
|
this.onConnected = function (reconnecting) {
|
2016-11-02 13:44:56 +01:00
|
|
|
/* Called as soon as a new connection has been established, either
|
|
|
|
* by logging in or by attaching to an existing BOSH session.
|
|
|
|
*/
|
2016-11-23 12:53:06 +01:00
|
|
|
// Solves problem of returned PubSub BOSH response not received
|
|
|
|
// by browser.
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.flush();
|
2016-11-23 12:53:06 +01:00
|
|
|
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.setUserJid();
|
|
|
|
_converse.enableCarbons();
|
2016-02-16 08:46:47 +01:00
|
|
|
|
2016-11-23 12:53:06 +01:00
|
|
|
// If there's no xmppstatus obj, then we were never connected to
|
|
|
|
// begin with, so we set reconnecting to false.
|
2016-12-20 10:30:20 +01:00
|
|
|
reconnecting = _.isUndefined(_converse.xmppstatus) ? false : reconnecting;
|
2016-11-23 12:53:06 +01:00
|
|
|
|
|
|
|
if (reconnecting) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.onStatusInitialized(true);
|
|
|
|
_converse.emit('reconnected');
|
2016-11-23 11:24:33 +01:00
|
|
|
} else {
|
2016-11-23 12:53:06 +01:00
|
|
|
// There might be some open chat boxes. We don't
|
|
|
|
// know whether these boxes are of the same account or not, so we
|
|
|
|
// close them now.
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.chatboxviews.closeAllChatBoxes();
|
|
|
|
_converse.features = new _converse.Features();
|
|
|
|
_converse.initStatus().done(_.partial(_converse.onStatusInitialized, false));
|
|
|
|
_converse.emit('connected');
|
2016-11-23 11:24:33 +01:00
|
|
|
}
|
2016-11-02 14:13:49 +01:00
|
|
|
};
|
|
|
|
|
2016-02-29 22:18:28 +01:00
|
|
|
this.RosterContact = Backbone.Model.extend({
|
2016-02-29 22:59:56 +01:00
|
|
|
|
2017-01-26 15:49:02 +01:00
|
|
|
initialize: function (attributes) {
|
2016-02-29 22:18:28 +01:00
|
|
|
var jid = attributes.jid;
|
2017-03-08 15:33:00 +01:00
|
|
|
var bare_jid = Strophe.getBareJidFromJid(jid).toLowerCase();
|
2016-02-29 22:18:28 +01:00
|
|
|
var resource = Strophe.getResourceFromJid(jid);
|
|
|
|
attributes.jid = bare_jid;
|
2017-01-26 10:08:43 +01:00
|
|
|
this.set(_.assignIn({
|
2016-02-29 22:18:28 +01:00
|
|
|
'id': bare_jid,
|
|
|
|
'jid': bare_jid,
|
|
|
|
'fullname': bare_jid,
|
|
|
|
'chat_status': 'offline',
|
|
|
|
'user_id': Strophe.getNodeFromJid(jid),
|
2017-02-20 22:27:34 +01:00
|
|
|
'resources': resource ? {'resource':0} : {},
|
2016-02-29 22:18:28 +01:00
|
|
|
'groups': [],
|
2017-01-23 11:15:07 +01:00
|
|
|
'image_type': DEFAULT_IMAGE_TYPE,
|
|
|
|
'image': DEFAULT_IMAGE,
|
2016-02-29 22:18:28 +01:00
|
|
|
'status': ''
|
|
|
|
}, attributes));
|
|
|
|
|
|
|
|
this.on('destroy', function () { this.removeFromRoster(); }.bind(this));
|
2016-03-08 11:42:52 +01:00
|
|
|
this.on('change:chat_status', function (item) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('contactStatusChanged', item.attributes);
|
2016-03-08 11:42:52 +01:00
|
|
|
});
|
2016-02-29 22:18:28 +01:00
|
|
|
},
|
|
|
|
|
2016-02-29 22:59:56 +01:00
|
|
|
subscribe: function (message) {
|
2016-02-29 22:18:28 +01:00
|
|
|
/* Send a presence subscription request to this roster contact
|
2016-03-18 09:49:32 +01:00
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* (String) message - An optional message to explain the
|
|
|
|
* reason for the subscription request.
|
|
|
|
*/
|
2016-02-29 22:18:28 +01:00
|
|
|
this.save('ask', "subscribe"); // ask === 'subscribe' Means we have ask to subscribe to them.
|
|
|
|
var pres = $pres({to: this.get('jid'), type: "subscribe"});
|
|
|
|
if (message && message !== "") {
|
|
|
|
pres.c("status").t(message).up();
|
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
var nick = _converse.xmppstatus.get('fullname');
|
2016-02-29 22:18:28 +01:00
|
|
|
if (nick && nick !== "") {
|
|
|
|
pres.c('nick', {'xmlns': Strophe.NS.NICK}).t(nick).up();
|
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.send(pres);
|
2016-02-29 22:18:28 +01:00
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
ackSubscribe: function () {
|
|
|
|
/* Upon receiving the presence stanza of type "subscribed",
|
2016-03-18 09:49:32 +01:00
|
|
|
* the user SHOULD acknowledge receipt of that subscription
|
|
|
|
* state notification by sending a presence stanza of type
|
|
|
|
* "subscribe" to the contact
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.send($pres({
|
2016-02-29 22:18:28 +01:00
|
|
|
'type': 'subscribe',
|
|
|
|
'to': this.get('jid')
|
|
|
|
}));
|
|
|
|
},
|
|
|
|
|
2017-01-26 15:49:02 +01:00
|
|
|
ackUnsubscribe: function () {
|
2016-02-29 22:18:28 +01:00
|
|
|
/* Upon receiving the presence stanza of type "unsubscribed",
|
2016-03-18 09:49:32 +01:00
|
|
|
* the user SHOULD acknowledge receipt of that subscription state
|
|
|
|
* notification by sending a presence stanza of type "unsubscribe"
|
|
|
|
* this step lets the user's server know that it MUST no longer
|
|
|
|
* send notification of the subscription state change to the user.
|
|
|
|
* Parameters:
|
|
|
|
* (String) jid - The Jabber ID of the user who is unsubscribing
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.send($pres({'type': 'unsubscribe', 'to': this.get('jid')}));
|
2016-02-29 22:18:28 +01:00
|
|
|
this.destroy(); // Will cause removeFromRoster to be called.
|
|
|
|
},
|
|
|
|
|
|
|
|
unauthorize: function (message) {
|
|
|
|
/* Unauthorize this contact's presence subscription
|
2016-03-18 09:49:32 +01:00
|
|
|
* Parameters:
|
|
|
|
* (String) message - Optional message to send to the person being unauthorized
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.rejectPresenceSubscription(this.get('jid'), message);
|
2016-02-29 22:18:28 +01:00
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
authorize: function (message) {
|
|
|
|
/* Authorize presence subscription
|
2016-03-18 09:49:32 +01:00
|
|
|
* Parameters:
|
|
|
|
* (String) message - Optional message to send to the person being authorized
|
|
|
|
*/
|
2016-02-29 22:18:28 +01:00
|
|
|
var pres = $pres({to: this.get('jid'), type: "subscribed"});
|
|
|
|
if (message && message !== "") {
|
|
|
|
pres.c("status").t(message);
|
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.send(pres);
|
2016-02-29 22:18:28 +01:00
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2017-02-21 23:25:19 +01:00
|
|
|
addResource: function (presence) {
|
|
|
|
/* Adds a new resource and it's associated attributes as taken
|
|
|
|
* from the passed in presence stanza.
|
|
|
|
*
|
|
|
|
* Also updates the contact's chat_status if the presence has
|
|
|
|
* higher priority (and is newer).
|
|
|
|
*/
|
|
|
|
var jid = presence.getAttribute('from'),
|
|
|
|
chat_status = _.propertyOf(presence.querySelector('show'))('textContent') || 'online',
|
|
|
|
resource = Strophe.getResourceFromJid(jid),
|
|
|
|
priority = _.propertyOf(presence.querySelector('priority'))('textContent') || 0,
|
|
|
|
delay = presence.querySelector('delay[xmlns="'+Strophe.NS.DELAY+'"]'),
|
|
|
|
timestamp = _.isNull(delay) ? moment().format() : moment(delay.getAttribute('stamp')).format();
|
|
|
|
|
|
|
|
priority = _.isNaN(parseInt(priority, 10)) ? 0 : parseInt(priority, 10);
|
|
|
|
|
|
|
|
var resources = _.isObject(this.get('resources')) ? this.get('resources') : {};
|
2017-02-20 22:27:34 +01:00
|
|
|
resources[resource] = {
|
2017-02-21 23:25:19 +01:00
|
|
|
'priority': priority,
|
2017-02-21 22:13:49 +01:00
|
|
|
'status': chat_status,
|
2017-03-08 15:33:00 +01:00
|
|
|
'timestamp': timestamp
|
2017-02-20 22:27:34 +01:00
|
|
|
};
|
2017-02-21 23:25:19 +01:00
|
|
|
var changed = {'resources': resources};
|
|
|
|
var hpr = this.getHighestPriorityResource();
|
|
|
|
if (priority == hpr.priority && timestamp == hpr.timestamp) {
|
|
|
|
// Only set the chat status if this is the newest resource
|
|
|
|
// with the highest priority
|
|
|
|
changed.chat_status = chat_status;
|
|
|
|
}
|
|
|
|
this.save(changed);
|
2017-02-20 22:27:34 +01:00
|
|
|
return resources;
|
|
|
|
},
|
|
|
|
|
2016-02-29 22:18:28 +01:00
|
|
|
removeResource: function (resource) {
|
2017-02-21 23:25:19 +01:00
|
|
|
/* Remove the passed in resource from the contact's resources map.
|
|
|
|
*
|
|
|
|
* Also recomputes the chat_status given that there's one less
|
|
|
|
* resource.
|
2017-02-20 22:27:34 +01:00
|
|
|
*/
|
|
|
|
var resources = this.get('resources');
|
|
|
|
if (!_.isObject(resources)) {
|
|
|
|
resources = {};
|
|
|
|
} else {
|
|
|
|
delete resources[resource];
|
2016-02-29 22:18:28 +01:00
|
|
|
}
|
2017-02-21 23:25:19 +01:00
|
|
|
this.save({
|
|
|
|
'resources': resources,
|
|
|
|
'chat_status': _.propertyOf(
|
|
|
|
this.getHighestPriorityResource())('status') || 'offline'
|
|
|
|
});
|
2017-02-20 22:27:34 +01:00
|
|
|
},
|
|
|
|
|
2017-02-21 23:25:19 +01:00
|
|
|
getHighestPriorityResource: function () {
|
|
|
|
/* Return the resource with the highest priority.
|
2017-02-21 22:13:49 +01:00
|
|
|
*
|
|
|
|
* If multiple resources have the same priority, take the
|
|
|
|
* newest one.
|
2017-02-20 22:27:34 +01:00
|
|
|
*/
|
|
|
|
var resources = this.get('resources');
|
|
|
|
if (_.isObject(resources) && _.size(resources)) {
|
2017-02-21 22:13:49 +01:00
|
|
|
var val = _.flow(
|
|
|
|
_.values,
|
|
|
|
_.partial(_.sortBy, _, ['priority', 'timestamp']),
|
|
|
|
_.reverse
|
|
|
|
)(resources)[0];
|
2017-02-20 22:27:34 +01:00
|
|
|
if (!_.isUndefined(val)) {
|
2017-02-21 23:25:19 +01:00
|
|
|
return val;
|
2017-02-20 22:27:34 +01:00
|
|
|
}
|
2016-02-29 22:18:28 +01:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
removeFromRoster: function (callback) {
|
|
|
|
/* Instruct the XMPP server to remove this contact from our roster
|
2016-03-18 09:49:32 +01:00
|
|
|
* Parameters:
|
|
|
|
* (Function) callback
|
|
|
|
*/
|
2016-02-29 22:18:28 +01:00
|
|
|
var iq = $iq({type: 'set'})
|
|
|
|
.c('query', {xmlns: Strophe.NS.ROSTER})
|
|
|
|
.c('item', {jid: this.get('jid'), subscription: "remove"});
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.sendIQ(iq, callback, callback);
|
2016-02-29 22:18:28 +01:00
|
|
|
return this;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
this.RosterContacts = Backbone.Collection.extend({
|
2016-12-20 10:30:20 +01:00
|
|
|
model: _converse.RosterContact,
|
2016-02-29 22:59:56 +01:00
|
|
|
|
2016-02-29 22:18:28 +01:00
|
|
|
comparator: function (contact1, contact2) {
|
|
|
|
var name1, name2;
|
|
|
|
var status1 = contact1.get('chat_status') || 'offline';
|
|
|
|
var status2 = contact2.get('chat_status') || 'offline';
|
2016-12-20 10:30:20 +01:00
|
|
|
if (_converse.STATUS_WEIGHTS[status1] === _converse.STATUS_WEIGHTS[status2]) {
|
2016-02-29 22:18:28 +01:00
|
|
|
name1 = contact1.get('fullname').toLowerCase();
|
|
|
|
name2 = contact2.get('fullname').toLowerCase();
|
|
|
|
return name1 < name2 ? -1 : (name1 > name2? 1 : 0);
|
|
|
|
} else {
|
2016-12-20 10:30:20 +01:00
|
|
|
return _converse.STATUS_WEIGHTS[status1] < _converse.STATUS_WEIGHTS[status2] ? -1 : 1;
|
2016-02-29 22:18:28 +01:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2016-09-20 12:10:28 +02:00
|
|
|
fetchRosterContacts: function () {
|
|
|
|
/* Fetches the roster contacts, first by trying the
|
|
|
|
* sessionStorage cache, and if that's empty, then by querying
|
|
|
|
* the XMPP server.
|
|
|
|
*
|
|
|
|
* Returns a promise which resolves once the contacts have been
|
|
|
|
* fetched.
|
|
|
|
*/
|
|
|
|
var deferred = new $.Deferred();
|
|
|
|
this.fetch({
|
|
|
|
add: true,
|
|
|
|
success: function (collection) {
|
|
|
|
if (collection.length === 0) {
|
|
|
|
/* We don't have any roster contacts stored in sessionStorage,
|
|
|
|
* so lets fetch the roster from the XMPP server. We pass in
|
|
|
|
* 'sendPresence' as callback method, because after initially
|
|
|
|
* fetching the roster we are ready to receive presence
|
|
|
|
* updates from our contacts.
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.send_initial_presence = true;
|
|
|
|
_converse.roster.fetchFromServer(deferred.resolve);
|
2016-09-20 12:10:28 +02:00
|
|
|
} else {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('cachedRoster', collection);
|
2016-09-20 12:10:28 +02:00
|
|
|
deferred.resolve();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return deferred.promise();
|
|
|
|
},
|
|
|
|
|
2016-02-29 22:18:28 +01:00
|
|
|
subscribeToSuggestedItems: function (msg) {
|
2016-11-24 02:07:32 +01:00
|
|
|
_.each(msg.querySelectorAll('item'), function (item) {
|
|
|
|
if (item.getAttribute('action') === 'add') {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.roster.addAndSubscribe(
|
2016-11-24 02:07:32 +01:00
|
|
|
item.getAttribute('jid'),
|
|
|
|
null,
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.xmppstatus.get('fullname')
|
2016-11-24 02:07:32 +01:00
|
|
|
);
|
2016-02-29 22:18:28 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
|
|
|
isSelf: function (jid) {
|
2016-12-20 10:30:20 +01:00
|
|
|
return (Strophe.getBareJidFromJid(jid) === Strophe.getBareJidFromJid(_converse.connection.jid));
|
2016-02-29 22:18:28 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
addAndSubscribe: function (jid, name, groups, message, attributes) {
|
|
|
|
/* Add a roster contact and then once we have confirmation from
|
2016-03-18 09:49:32 +01:00
|
|
|
* the XMPP server we subscribe to that contact's presence updates.
|
|
|
|
* Parameters:
|
|
|
|
* (String) jid - The Jabber ID of the user being added and subscribed to.
|
|
|
|
* (String) name - The name of that user
|
|
|
|
* (Array of Strings) groups - Any roster groups the user might belong to
|
|
|
|
* (String) message - An optional message to explain the
|
|
|
|
* reason for the subscription request.
|
|
|
|
* (Object) attributes - Any additional attributes to be stored on the user's model.
|
|
|
|
*/
|
2016-02-29 22:18:28 +01:00
|
|
|
this.addContact(jid, name, groups, attributes).done(function (contact) {
|
2016-12-20 10:30:20 +01:00
|
|
|
if (contact instanceof _converse.RosterContact) {
|
2016-02-29 22:18:28 +01:00
|
|
|
contact.subscribe(message);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
sendContactAddIQ: function (jid, name, groups, callback, errback) {
|
|
|
|
/* Send an IQ stanza to the XMPP server to add a new roster contact.
|
2016-03-01 22:57:27 +01:00
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* (String) jid - The Jabber ID of the user being added
|
|
|
|
* (String) name - The name of that user
|
|
|
|
* (Array of Strings) groups - Any roster groups the user might belong to
|
2016-03-16 12:16:32 +01:00
|
|
|
* (Function) callback - A function to call once the IQ is returned
|
2016-03-01 22:57:27 +01:00
|
|
|
* (Function) errback - A function to call if an error occured
|
|
|
|
*/
|
2016-02-29 22:18:28 +01:00
|
|
|
name = _.isEmpty(name)? jid: name;
|
|
|
|
var iq = $iq({type: 'set'})
|
|
|
|
.c('query', {xmlns: Strophe.NS.ROSTER})
|
|
|
|
.c('item', { jid: jid, name: name });
|
2017-01-26 15:49:02 +01:00
|
|
|
_.each(groups, function (group) { iq.c('group').t(group).up(); });
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.sendIQ(iq, callback, errback);
|
2016-02-29 22:18:28 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
addContact: function (jid, name, groups, attributes) {
|
2016-12-20 10:30:20 +01:00
|
|
|
/* Adds a RosterContact instance to _converse.roster and
|
2016-03-01 22:57:27 +01:00
|
|
|
* registers the contact on the XMPP server.
|
|
|
|
* Returns a promise which is resolved once the XMPP server has
|
|
|
|
* responded.
|
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* (String) jid - The Jabber ID of the user being added and subscribed to.
|
|
|
|
* (String) name - The name of that user
|
|
|
|
* (Array of Strings) groups - Any roster groups the user might belong to
|
|
|
|
* (Object) attributes - Any additional attributes to be stored on the user's model.
|
|
|
|
*/
|
2016-02-29 22:18:28 +01:00
|
|
|
var deferred = new $.Deferred();
|
|
|
|
groups = groups || [];
|
|
|
|
name = _.isEmpty(name)? jid: name;
|
|
|
|
this.sendContactAddIQ(jid, name, groups,
|
2017-01-26 15:49:02 +01:00
|
|
|
function () {
|
2017-01-26 10:08:43 +01:00
|
|
|
var contact = this.create(_.assignIn({
|
2016-02-29 22:18:28 +01:00
|
|
|
ask: undefined,
|
|
|
|
fullname: name,
|
|
|
|
groups: groups,
|
|
|
|
jid: jid,
|
|
|
|
requesting: false,
|
|
|
|
subscription: 'none'
|
|
|
|
}, attributes), {sort: false});
|
|
|
|
deferred.resolve(contact);
|
|
|
|
}.bind(this),
|
|
|
|
function (err) {
|
|
|
|
alert(__("Sorry, there was an error while trying to add "+name+" as a contact."));
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.log(err);
|
2016-02-29 22:18:28 +01:00
|
|
|
deferred.resolve(err);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
return deferred.promise();
|
|
|
|
},
|
|
|
|
|
|
|
|
subscribeBack: function (bare_jid) {
|
|
|
|
var contact = this.get(bare_jid);
|
2016-12-20 10:30:20 +01:00
|
|
|
if (contact instanceof _converse.RosterContact) {
|
2016-02-29 22:18:28 +01:00
|
|
|
contact.authorize().subscribe();
|
|
|
|
} else {
|
|
|
|
// Can happen when a subscription is retried or roster was deleted
|
|
|
|
this.addContact(bare_jid, '', [], { 'subscription': 'from' }).done(function (contact) {
|
2016-12-20 10:30:20 +01:00
|
|
|
if (contact instanceof _converse.RosterContact) {
|
2016-02-29 22:18:28 +01:00
|
|
|
contact.authorize().subscribe();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
getNumOnlineContacts: function () {
|
|
|
|
var count = 0,
|
|
|
|
ignored = ['offline', 'unavailable'],
|
|
|
|
models = this.models,
|
|
|
|
models_length = models.length,
|
|
|
|
i;
|
2016-12-20 10:30:20 +01:00
|
|
|
if (_converse.show_only_online_users) {
|
2016-02-29 22:18:28 +01:00
|
|
|
ignored = _.union(ignored, ['dnd', 'xa', 'away']);
|
|
|
|
}
|
|
|
|
for (i=0; i<models_length; i++) {
|
2017-01-26 15:49:02 +01:00
|
|
|
if (!_.includes(ignored, models[i].get('chat_status'))) {
|
2016-02-29 22:18:28 +01:00
|
|
|
count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
},
|
|
|
|
|
|
|
|
onRosterPush: function (iq) {
|
|
|
|
/* Handle roster updates from the XMPP server.
|
2016-03-18 09:49:32 +01:00
|
|
|
* See: https://xmpp.org/rfcs/rfc6121.html#roster-syntax-actions-push
|
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* (XMLElement) IQ - The IQ stanza received from the XMPP server.
|
|
|
|
*/
|
2016-02-29 22:18:28 +01:00
|
|
|
var id = iq.getAttribute('id');
|
|
|
|
var from = iq.getAttribute('from');
|
2016-12-20 10:30:20 +01:00
|
|
|
if (from && from !== "" && Strophe.getBareJidFromJid(from) !== _converse.bare_jid) {
|
2016-02-29 22:18:28 +01:00
|
|
|
// Receiving client MUST ignore stanza unless it has no from or from = user's bare JID.
|
|
|
|
// XXX: Some naughty servers apparently send from a full
|
|
|
|
// JID so we need to explicitly compare bare jids here.
|
|
|
|
// https://github.com/jcbrand/converse.js/issues/493
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.send(
|
|
|
|
$iq({type: 'error', id: id, from: _converse.connection.jid})
|
2016-02-29 22:18:28 +01:00
|
|
|
.c('error', {'type': 'cancel'})
|
|
|
|
.c('service-unavailable', {'xmlns': Strophe.NS.ROSTER })
|
|
|
|
);
|
|
|
|
return true;
|
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.send($iq({type: 'result', id: id, from: _converse.connection.jid}));
|
2016-11-24 02:07:32 +01:00
|
|
|
// var items = iq.querySelectorAll('query[xmlns="'+Strophe.NS.ROSTER+'"] item');
|
2016-12-07 20:45:29 +01:00
|
|
|
var items = sizzle('query[xmlns="'+Strophe.NS.ROSTER+'"] item', iq);
|
2016-11-24 02:07:32 +01:00
|
|
|
_.each(items, this.updateContact.bind(this));
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('rosterPush', iq);
|
2016-02-29 22:18:28 +01:00
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
|
|
|
fetchFromServer: function (callback) {
|
|
|
|
/* Get the roster from the XMPP server */
|
2016-12-20 10:30:20 +01:00
|
|
|
var iq = $iq({type: 'get', 'id': _converse.connection.getUniqueId('roster')})
|
2016-02-29 22:18:28 +01:00
|
|
|
.c('query', {xmlns: Strophe.NS.ROSTER});
|
2016-12-20 10:30:20 +01:00
|
|
|
return _converse.connection.sendIQ(iq, function () {
|
2016-02-29 22:18:28 +01:00
|
|
|
this.onReceivedFromServer.apply(this, arguments);
|
|
|
|
callback.apply(this, arguments);
|
|
|
|
}.bind(this));
|
|
|
|
},
|
|
|
|
|
|
|
|
onReceivedFromServer: function (iq) {
|
|
|
|
/* An IQ stanza containing the roster has been received from
|
2016-03-18 09:49:32 +01:00
|
|
|
* the XMPP server.
|
|
|
|
*/
|
2016-11-24 02:07:32 +01:00
|
|
|
// var items = iq.querySelectorAll('query[xmlns="'+Strophe.NS.ROSTER+'"] item');
|
2016-12-07 20:45:29 +01:00
|
|
|
var items = sizzle('query[xmlns="'+Strophe.NS.ROSTER+'"] item', iq);
|
2016-11-24 02:07:32 +01:00
|
|
|
_.each(items, this.updateContact.bind(this));
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('roster', iq);
|
2016-02-29 22:18:28 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
updateContact: function (item) {
|
|
|
|
/* Update or create RosterContact models based on items
|
2016-03-18 09:49:32 +01:00
|
|
|
* received in the IQ from the server.
|
|
|
|
*/
|
2016-02-29 22:18:28 +01:00
|
|
|
var jid = item.getAttribute('jid');
|
|
|
|
if (this.isSelf(jid)) { return; }
|
2017-01-26 15:49:02 +01:00
|
|
|
var groups = _.map(item.getElementsByTagName('group'), Strophe.getText),
|
2016-02-29 22:18:28 +01:00
|
|
|
contact = this.get(jid),
|
|
|
|
ask = item.getAttribute("ask"),
|
|
|
|
subscription = item.getAttribute("subscription");
|
|
|
|
if (!contact) {
|
|
|
|
if ((subscription === "none" && ask === null) || (subscription === "remove")) {
|
|
|
|
return; // We're lazy when adding contacts.
|
|
|
|
}
|
|
|
|
this.create({
|
|
|
|
ask: ask,
|
|
|
|
fullname: item.getAttribute("name") || jid,
|
|
|
|
groups: groups,
|
|
|
|
jid: jid,
|
|
|
|
subscription: subscription
|
|
|
|
}, {sort: false});
|
|
|
|
} else {
|
|
|
|
if (subscription === "remove") {
|
|
|
|
return contact.destroy(); // will trigger removeFromRoster
|
|
|
|
}
|
|
|
|
// We only find out about requesting contacts via the
|
|
|
|
// presence handler, so if we receive a contact
|
|
|
|
// here, we know they aren't requesting anymore.
|
|
|
|
// see docs/DEVELOPER.rst
|
|
|
|
contact.save({
|
|
|
|
subscription: subscription,
|
|
|
|
ask: ask,
|
|
|
|
requesting: null,
|
|
|
|
groups: groups
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2016-03-16 12:16:32 +01:00
|
|
|
createRequestingContact: function (presence) {
|
|
|
|
/* Creates a Requesting Contact.
|
|
|
|
*
|
|
|
|
* Note: this method gets completely overridden by converse-vcard.js
|
2016-03-01 22:57:27 +01:00
|
|
|
*/
|
2016-03-16 12:16:32 +01:00
|
|
|
var bare_jid = Strophe.getBareJidFromJid(presence.getAttribute('from'));
|
2016-12-20 11:42:20 +01:00
|
|
|
var nick_el = presence.querySelector('nick[xmlns="'+Strophe.NS.NICK+'"]');
|
2016-03-01 22:57:27 +01:00
|
|
|
var user_data = {
|
2016-02-29 22:18:28 +01:00
|
|
|
jid: bare_jid,
|
|
|
|
subscription: 'none',
|
|
|
|
ask: null,
|
|
|
|
requesting: true,
|
2016-11-24 02:07:32 +01:00
|
|
|
fullname: nick_el && nick_el.textContent || bare_jid,
|
2016-03-01 22:57:27 +01:00
|
|
|
};
|
|
|
|
this.create(user_data);
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('contactRequest', user_data);
|
2016-02-29 22:18:28 +01:00
|
|
|
},
|
|
|
|
|
2016-03-16 12:16:32 +01:00
|
|
|
handleIncomingSubscription: function (presence) {
|
|
|
|
var jid = presence.getAttribute('from');
|
2016-02-29 22:18:28 +01:00
|
|
|
var bare_jid = Strophe.getBareJidFromJid(jid);
|
|
|
|
var contact = this.get(bare_jid);
|
2016-12-20 10:30:20 +01:00
|
|
|
if (!_converse.allow_contact_requests) {
|
|
|
|
_converse.rejectPresenceSubscription(
|
2016-03-16 12:16:32 +01:00
|
|
|
jid,
|
|
|
|
__("This client does not allow presence subscriptions")
|
|
|
|
);
|
2016-02-29 22:18:28 +01:00
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
if (_converse.auto_subscribe) {
|
2016-02-29 22:18:28 +01:00
|
|
|
if ((!contact) || (contact.get('subscription') !== 'to')) {
|
|
|
|
this.subscribeBack(bare_jid);
|
|
|
|
} else {
|
|
|
|
contact.authorize();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (contact) {
|
|
|
|
if (contact.get('subscription') !== 'none') {
|
|
|
|
contact.authorize();
|
|
|
|
} else if (contact.get('ask') === "subscribe") {
|
|
|
|
contact.authorize();
|
|
|
|
}
|
|
|
|
} else if (!contact) {
|
2016-03-16 12:16:32 +01:00
|
|
|
this.createRequestingContact(presence);
|
2016-02-29 22:18:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
presenceHandler: function (presence) {
|
2016-11-24 02:07:32 +01:00
|
|
|
var presence_type = presence.getAttribute('type');
|
2016-02-29 22:18:28 +01:00
|
|
|
if (presence_type === 'error') { return true; }
|
|
|
|
var jid = presence.getAttribute('from'),
|
|
|
|
bare_jid = Strophe.getBareJidFromJid(jid),
|
|
|
|
resource = Strophe.getResourceFromJid(jid),
|
2016-11-24 02:07:32 +01:00
|
|
|
chat_status = _.propertyOf(presence.querySelector('show'))('textContent') || 'online',
|
|
|
|
status_message = _.propertyOf(presence.querySelector('status'))('textContent'),
|
2016-02-29 22:18:28 +01:00
|
|
|
contact = this.get(bare_jid);
|
2017-02-20 22:27:34 +01:00
|
|
|
|
2016-02-29 22:18:28 +01:00
|
|
|
if (this.isSelf(bare_jid)) {
|
2016-12-20 10:30:20 +01:00
|
|
|
if ((_converse.connection.jid !== jid) &&
|
2016-03-16 16:49:44 +01:00
|
|
|
(presence_type !== 'unavailable') &&
|
2016-12-20 10:30:20 +01:00
|
|
|
(_converse.synchronize_availability === true ||
|
|
|
|
_converse.synchronize_availability === resource)) {
|
2016-03-16 16:49:44 +01:00
|
|
|
// Another resource has changed its status and
|
|
|
|
// synchronize_availability option set to update,
|
|
|
|
// we'll update ours as well.
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.xmppstatus.save({'status': chat_status});
|
2016-11-24 02:07:32 +01:00
|
|
|
if (status_message) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.xmppstatus.save({'status_message': status_message});
|
2016-03-16 16:49:44 +01:00
|
|
|
}
|
2016-02-29 22:18:28 +01:00
|
|
|
}
|
|
|
|
return;
|
2016-12-07 20:45:29 +01:00
|
|
|
} else if (sizzle('query[xmlns="'+Strophe.NS.MUC+'"]', presence).length) {
|
2016-02-29 22:18:28 +01:00
|
|
|
return; // Ignore MUC
|
|
|
|
}
|
2016-11-24 02:07:32 +01:00
|
|
|
if (contact && (status_message !== contact.get('status'))) {
|
|
|
|
contact.save({'status': status_message});
|
2016-02-29 22:18:28 +01:00
|
|
|
}
|
|
|
|
if (presence_type === 'subscribed' && contact) {
|
|
|
|
contact.ackSubscribe();
|
|
|
|
} else if (presence_type === 'unsubscribed' && contact) {
|
|
|
|
contact.ackUnsubscribe();
|
|
|
|
} else if (presence_type === 'unsubscribe') {
|
|
|
|
return;
|
|
|
|
} else if (presence_type === 'subscribe') {
|
2016-03-16 12:16:32 +01:00
|
|
|
this.handleIncomingSubscription(presence);
|
2016-02-29 22:18:28 +01:00
|
|
|
} else if (presence_type === 'unavailable' && contact) {
|
2017-02-20 22:27:34 +01:00
|
|
|
contact.removeResource(resource);
|
2017-02-21 23:25:19 +01:00
|
|
|
} else if (contact) {
|
|
|
|
// presence_type is undefined
|
|
|
|
contact.addResource(presence);
|
2016-02-29 22:18:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2016-06-21 17:06:51 +02:00
|
|
|
this.RosterGroup = Backbone.Model.extend({
|
2017-01-26 15:49:02 +01:00
|
|
|
initialize: function (attributes) {
|
2017-01-26 10:08:43 +01:00
|
|
|
this.set(_.assignIn({
|
2016-06-21 17:06:51 +02:00
|
|
|
description: DESC_GROUP_TOGGLE,
|
2016-12-20 10:30:20 +01:00
|
|
|
state: _converse.OPENED
|
2016-06-21 17:06:51 +02:00
|
|
|
}, attributes));
|
|
|
|
// Collection of contacts belonging to this group.
|
2016-12-20 10:30:20 +01:00
|
|
|
this.contacts = new _converse.RosterContacts();
|
2016-06-21 17:06:51 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
this.RosterGroups = Backbone.Collection.extend({
|
2016-12-20 10:30:20 +01:00
|
|
|
model: _converse.RosterGroup,
|
2016-09-20 12:10:28 +02:00
|
|
|
|
|
|
|
fetchRosterGroups: function () {
|
|
|
|
/* Fetches all the roster groups from sessionStorage.
|
|
|
|
*
|
|
|
|
* Returns a promise which resolves once the groups have been
|
|
|
|
* returned.
|
|
|
|
*/
|
|
|
|
var deferred = new $.Deferred();
|
|
|
|
this.fetch({
|
|
|
|
silent: true, // We need to first have all groups before
|
|
|
|
// we can start positioning them, so we set
|
|
|
|
// 'silent' to true.
|
|
|
|
success: deferred.resolve
|
|
|
|
});
|
|
|
|
return deferred.promise();
|
|
|
|
}
|
2016-06-21 17:06:51 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
this.Message = Backbone.Model.extend({
|
|
|
|
defaults: function(){
|
|
|
|
return {
|
2016-12-20 10:30:20 +01:00
|
|
|
msgid: _converse.connection.getUniqueId()
|
2016-02-16 08:46:47 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
});
|
2016-02-29 22:18:28 +01:00
|
|
|
|
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
this.Messages = Backbone.Collection.extend({
|
2016-12-20 10:30:20 +01:00
|
|
|
model: _converse.Message,
|
2016-02-16 08:46:47 +01:00
|
|
|
comparator: 'time'
|
|
|
|
});
|
|
|
|
|
2016-02-29 22:18:28 +01:00
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
this.ChatBox = Backbone.Model.extend({
|
|
|
|
|
|
|
|
initialize: function () {
|
2016-12-20 10:30:20 +01:00
|
|
|
this.messages = new _converse.Messages();
|
|
|
|
this.messages.browserStorage = new Backbone.BrowserStorage[_converse.message_storage](
|
|
|
|
b64_sha1('converse.messages'+this.get('jid')+_converse.bare_jid));
|
2016-04-01 14:46:19 +02:00
|
|
|
this.save({
|
2016-02-28 20:24:06 +01:00
|
|
|
// The chat_state will be set to ACTIVE once the chat box is opened
|
|
|
|
// and we listen for change:chat_state, so shouldn't set it to ACTIVE here.
|
|
|
|
'box_id' : b64_sha1(this.get('jid')),
|
2016-04-01 14:46:19 +02:00
|
|
|
'chat_state': undefined,
|
|
|
|
'num_unread': this.get('num_unread') || 0,
|
2016-02-28 20:24:06 +01:00
|
|
|
'time_opened': this.get('time_opened') || moment().valueOf(),
|
|
|
|
'url': '',
|
|
|
|
'user_id' : Strophe.getNodeFromJid(this.get('jid'))
|
2016-04-01 14:46:19 +02:00
|
|
|
});
|
2016-02-16 08:46:47 +01:00
|
|
|
},
|
|
|
|
|
2016-11-24 02:07:32 +01:00
|
|
|
getMessageAttributes: function (message, delay, original_stanza) {
|
|
|
|
delay = delay || message.querySelector('delay');
|
|
|
|
var type = message.getAttribute('type'),
|
2017-03-09 12:38:54 +01:00
|
|
|
body, stamp, time, sender, from, fullname;
|
2016-06-22 18:47:07 +02:00
|
|
|
|
|
|
|
if (type === 'error') {
|
2016-11-24 02:07:32 +01:00
|
|
|
body = _.propertyOf(message.querySelector('error text'))('textContent');
|
2016-06-22 18:47:07 +02:00
|
|
|
} else {
|
2016-11-24 02:07:32 +01:00
|
|
|
body = _.propertyOf(message.querySelector('body'))('textContent');
|
2016-06-22 18:47:07 +02:00
|
|
|
}
|
2016-11-24 02:07:32 +01:00
|
|
|
var delayed = !_.isNull(delay),
|
2016-06-22 18:47:07 +02:00
|
|
|
is_groupchat = type === 'groupchat',
|
2016-12-20 10:30:20 +01:00
|
|
|
chat_state = message.getElementsByTagName(_converse.COMPOSING).length && _converse.COMPOSING ||
|
|
|
|
message.getElementsByTagName(_converse.PAUSED).length && _converse.PAUSED ||
|
|
|
|
message.getElementsByTagName(_converse.INACTIVE).length && _converse.INACTIVE ||
|
|
|
|
message.getElementsByTagName(_converse.ACTIVE).length && _converse.ACTIVE ||
|
|
|
|
message.getElementsByTagName(_converse.GONE).length && _converse.GONE;
|
2016-02-16 08:46:47 +01:00
|
|
|
|
|
|
|
if (is_groupchat) {
|
2016-11-24 02:07:32 +01:00
|
|
|
from = Strophe.unescapeNode(Strophe.getResourceFromJid(message.getAttribute('from')));
|
2016-02-16 08:46:47 +01:00
|
|
|
} else {
|
2016-11-24 02:07:32 +01:00
|
|
|
from = Strophe.getBareJidFromJid(message.getAttribute('from'));
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
if (delayed) {
|
2016-11-24 02:07:32 +01:00
|
|
|
stamp = delay.getAttribute('stamp');
|
2016-02-16 08:46:47 +01:00
|
|
|
time = stamp;
|
|
|
|
} else {
|
|
|
|
time = moment().format();
|
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
if ((is_groupchat && from === this.get('nick')) || (!is_groupchat && from === _converse.bare_jid)) {
|
2016-02-16 08:46:47 +01:00
|
|
|
sender = 'me';
|
2017-03-09 12:38:54 +01:00
|
|
|
fullname = _converse.xmppstatus.get('fullname') || from;
|
2016-02-16 08:46:47 +01:00
|
|
|
} else {
|
|
|
|
sender = 'them';
|
2017-03-09 12:38:54 +01:00
|
|
|
fullname = this.get('fullname') || from;
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
2016-05-30 19:07:59 +02:00
|
|
|
return {
|
2016-06-22 18:47:07 +02:00
|
|
|
'type': type,
|
|
|
|
'chat_state': chat_state,
|
|
|
|
'delayed': delayed,
|
|
|
|
'fullname': fullname,
|
|
|
|
'message': body || undefined,
|
2016-11-24 02:07:32 +01:00
|
|
|
'msgid': message.getAttribute('id'),
|
2016-06-22 18:47:07 +02:00
|
|
|
'sender': sender,
|
|
|
|
'time': time
|
2016-05-30 19:07:59 +02:00
|
|
|
};
|
|
|
|
},
|
|
|
|
|
2016-11-24 02:07:32 +01:00
|
|
|
createMessage: function (message, delay, original_stanza) {
|
2016-05-30 19:07:59 +02:00
|
|
|
return this.messages.create(this.getMessageAttributes.apply(this, arguments));
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
});
|
2016-05-25 09:52:57 +02:00
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
this.ChatBoxes = Backbone.Collection.extend({
|
2016-12-20 10:30:20 +01:00
|
|
|
model: _converse.ChatBox,
|
2016-02-16 08:46:47 +01:00
|
|
|
comparator: 'time_opened',
|
|
|
|
|
|
|
|
registerMessageHandler: function () {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.addHandler(this.onMessage.bind(this), null, 'message', 'chat');
|
|
|
|
_converse.connection.addHandler(this.onErrorMessage.bind(this), null, 'message', 'error');
|
2016-02-16 08:46:47 +01:00
|
|
|
},
|
|
|
|
|
2016-04-05 13:23:16 +02:00
|
|
|
chatBoxMayBeShown: function (chatbox) {
|
2016-03-14 17:04:27 +01:00
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
|
|
|
onChatBoxesFetched: function (collection) {
|
2016-02-28 20:24:06 +01:00
|
|
|
/* Show chat boxes upon receiving them from sessionStorage
|
|
|
|
*
|
|
|
|
* This method gets overridden entirely in src/converse-controlbox.js
|
|
|
|
* if the controlbox plugin is active.
|
|
|
|
*/
|
2016-11-02 13:44:56 +01:00
|
|
|
var that = this;
|
2016-02-28 20:24:06 +01:00
|
|
|
collection.each(function (chatbox) {
|
2016-11-02 13:44:56 +01:00
|
|
|
if (that.chatBoxMayBeShown(chatbox)) {
|
2016-02-28 20:24:06 +01:00
|
|
|
chatbox.trigger('show');
|
|
|
|
}
|
2016-11-02 13:44:56 +01:00
|
|
|
});
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('chatBoxesFetched');
|
2016-02-28 20:24:06 +01:00
|
|
|
},
|
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
onConnected: function () {
|
2016-12-20 10:30:20 +01:00
|
|
|
this.browserStorage = new Backbone.BrowserStorage[_converse.storage](
|
|
|
|
b64_sha1('converse.chatboxes-'+_converse.bare_jid));
|
2016-02-16 08:46:47 +01:00
|
|
|
this.registerMessageHandler();
|
|
|
|
this.fetch({
|
|
|
|
add: true,
|
2016-03-09 11:54:50 +01:00
|
|
|
success: this.onChatBoxesFetched.bind(this)
|
2016-02-16 08:46:47 +01:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2016-06-22 18:47:07 +02:00
|
|
|
onErrorMessage: function (message) {
|
|
|
|
/* Handler method for all incoming error message stanzas
|
|
|
|
*/
|
|
|
|
// TODO: we can likely just reuse "onMessage" below
|
2016-11-24 02:07:32 +01:00
|
|
|
var from_jid = Strophe.getBareJidFromJid(message.getAttribute('from'));
|
2016-12-20 10:30:20 +01:00
|
|
|
if (from_jid === _converse.bare_jid) {
|
2016-06-22 18:47:07 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// Get chat box, but only create a new one when the message has a body.
|
|
|
|
var chatbox = this.getChatBox(from_jid);
|
|
|
|
if (!chatbox) {
|
|
|
|
return true;
|
|
|
|
}
|
2016-11-24 02:07:32 +01:00
|
|
|
chatbox.createMessage(message, null, message);
|
2016-06-22 18:47:07 +02:00
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
onMessage: function (message) {
|
2016-03-28 12:49:52 +02:00
|
|
|
/* Handler method for all incoming single-user chat "message"
|
|
|
|
* stanzas.
|
2016-02-16 08:46:47 +01:00
|
|
|
*/
|
2016-11-24 02:07:32 +01:00
|
|
|
var original_stanza = message,
|
|
|
|
contact_jid, forwarded, delay, from_bare_jid,
|
2016-03-28 12:49:52 +02:00
|
|
|
from_resource, is_me, msgid,
|
2016-02-16 08:46:47 +01:00
|
|
|
chatbox, resource,
|
2016-11-24 02:07:32 +01:00
|
|
|
from_jid = message.getAttribute('from'),
|
|
|
|
to_jid = message.getAttribute('to'),
|
2017-02-25 09:53:36 +01:00
|
|
|
to_resource = Strophe.getResourceFromJid(to_jid),
|
|
|
|
is_carbon = !_.isNull(message.querySelector('received[xmlns="'+Strophe.NS.CARBONS+'"]'));
|
2016-02-16 08:46:47 +01:00
|
|
|
|
2016-12-20 10:30:20 +01:00
|
|
|
if (_converse.filter_by_resource && (to_resource && to_resource !== _converse.resource)) {
|
|
|
|
_converse.log(
|
2016-03-28 12:49:52 +02:00
|
|
|
'onMessage: Ignoring incoming message intended for a different resource: '+to_jid,
|
|
|
|
'info'
|
|
|
|
);
|
2016-02-16 08:46:47 +01:00
|
|
|
return true;
|
2016-03-28 12:49:52 +02:00
|
|
|
} else if (utils.isHeadlineMessage(message)) {
|
|
|
|
// XXX: Ideally we wouldn't have to check for headline
|
|
|
|
// messages, but Prosody sends headline messages with the
|
|
|
|
// wrong type ('chat'), so we need to filter them out here.
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.log(
|
2016-03-28 12:49:52 +02:00
|
|
|
"onMessage: Ignoring incoming headline message sent with type 'chat' from JID: "+from_jid,
|
|
|
|
'info'
|
|
|
|
);
|
2016-02-16 08:46:47 +01:00
|
|
|
return true;
|
|
|
|
}
|
2016-11-24 02:07:32 +01:00
|
|
|
forwarded = message.querySelector('forwarded');
|
|
|
|
if (!_.isNull(forwarded)) {
|
|
|
|
var forwarded_message = forwarded.querySelector('message');
|
|
|
|
var forwarded_from = forwarded_message.getAttribute('from');
|
2017-02-25 09:53:36 +01:00
|
|
|
if (is_carbon && Strophe.getBareJidFromJid(forwarded_from) !== from_jid) {
|
2017-01-31 23:18:26 +01:00
|
|
|
// Prevent message forging via carbons
|
|
|
|
//
|
|
|
|
// https://xmpp.org/extensions/xep-0280.html#security
|
|
|
|
return true;
|
|
|
|
}
|
2016-11-24 02:07:32 +01:00
|
|
|
message = forwarded_message;
|
|
|
|
delay = forwarded.querySelector('delay');
|
|
|
|
from_jid = message.getAttribute('from');
|
|
|
|
to_jid = message.getAttribute('to');
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
from_bare_jid = Strophe.getBareJidFromJid(from_jid);
|
|
|
|
from_resource = Strophe.getResourceFromJid(from_jid);
|
2016-12-20 10:30:20 +01:00
|
|
|
is_me = from_bare_jid === _converse.bare_jid;
|
2016-11-24 02:07:32 +01:00
|
|
|
msgid = message.getAttribute('id');
|
2016-02-16 08:46:47 +01:00
|
|
|
if (is_me) {
|
|
|
|
// I am the sender, so this must be a forwarded message...
|
|
|
|
contact_jid = Strophe.getBareJidFromJid(to_jid);
|
|
|
|
resource = Strophe.getResourceFromJid(to_jid);
|
|
|
|
} else {
|
|
|
|
contact_jid = from_bare_jid;
|
|
|
|
resource = from_resource;
|
|
|
|
}
|
2017-03-01 11:37:00 +01:00
|
|
|
_converse.emit('message', original_stanza);
|
2016-02-16 08:46:47 +01:00
|
|
|
// Get chat box, but only create a new one when the message has a body.
|
2016-11-24 02:07:32 +01:00
|
|
|
chatbox = this.getChatBox(contact_jid, !_.isNull(message.querySelector('body')));
|
2016-02-16 08:46:47 +01:00
|
|
|
if (!chatbox) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (msgid && chatbox.messages.findWhere({msgid: msgid})) {
|
|
|
|
return true; // We already have this message stored.
|
|
|
|
}
|
2016-11-24 02:07:32 +01:00
|
|
|
chatbox.createMessage(message, delay, original_stanza);
|
2016-02-16 08:46:47 +01:00
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
2016-08-19 17:16:36 +02:00
|
|
|
getChatBox: function (jid, create, attrs) {
|
2016-02-16 08:46:47 +01:00
|
|
|
/* Returns a chat box or optionally return a newly
|
|
|
|
* created one if one doesn't exist.
|
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* (String) jid - The JID of the user whose chat box we want
|
|
|
|
* (Boolean) create - Should a new chat box be created if none exists?
|
2017-02-01 10:14:36 +01:00
|
|
|
* (Object) attrs - Optional chat box atributes.
|
2016-02-16 08:46:47 +01:00
|
|
|
*/
|
|
|
|
jid = jid.toLowerCase();
|
|
|
|
var bare_jid = Strophe.getBareJidFromJid(jid);
|
|
|
|
var chatbox = this.get(bare_jid);
|
|
|
|
if (!chatbox && create) {
|
2017-02-01 10:14:36 +01:00
|
|
|
var roster_info = {};
|
2016-12-20 10:30:20 +01:00
|
|
|
var roster_item = _converse.roster.get(bare_jid);
|
2017-02-01 10:14:36 +01:00
|
|
|
if (! _.isUndefined(roster_item)) {
|
|
|
|
roster_info = {
|
|
|
|
'fullname': _.isEmpty(roster_item.get('fullname'))? jid: roster_item.get('fullname'),
|
|
|
|
'image_type': roster_item.get('image_type'),
|
|
|
|
'image': roster_item.get('image'),
|
|
|
|
'url': roster_item.get('url'),
|
|
|
|
};
|
2016-12-20 10:30:20 +01:00
|
|
|
} else if (!_converse.allow_non_roster_messaging) {
|
|
|
|
_converse.log('Could not get roster item for JID '+bare_jid+
|
2017-02-01 10:14:36 +01:00
|
|
|
' and allow_non_roster_messaging is set to false', 'error');
|
|
|
|
return;
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
2017-01-26 10:08:43 +01:00
|
|
|
chatbox = this.create(_.assignIn({
|
2016-02-16 08:46:47 +01:00
|
|
|
'id': bare_jid,
|
|
|
|
'jid': bare_jid,
|
2017-01-23 11:15:07 +01:00
|
|
|
'fullname': jid,
|
|
|
|
'image_type': DEFAULT_IMAGE_TYPE,
|
|
|
|
'image': DEFAULT_IMAGE,
|
|
|
|
'url': '',
|
2017-02-01 10:14:36 +01:00
|
|
|
}, roster_info, attrs || {}));
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
return chatbox;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
this.ChatBoxViews = Backbone.Overview.extend({
|
|
|
|
|
|
|
|
initialize: function () {
|
|
|
|
this.model.on("add", this.onChatBoxAdded, this);
|
2016-03-28 13:42:33 +02:00
|
|
|
this.model.on("destroy", this.removeChat, this);
|
2016-02-16 08:46:47 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
_ensureElement: function () {
|
|
|
|
/* Override method from backbone.js
|
|
|
|
* If the #conversejs element doesn't exist, create it.
|
|
|
|
*/
|
|
|
|
if (!this.el) {
|
|
|
|
var $el = $('#conversejs');
|
|
|
|
if (!$el.length) {
|
|
|
|
$el = $('<div id="conversejs">');
|
|
|
|
$('body').append($el);
|
|
|
|
}
|
2016-11-22 17:36:39 +01:00
|
|
|
$el.html('');
|
2016-02-16 08:46:47 +01:00
|
|
|
this.setElement($el, false);
|
|
|
|
} else {
|
|
|
|
this.setElement(_.result(this, 'el'), false);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
onChatBoxAdded: function (item) {
|
2016-03-29 13:50:50 +02:00
|
|
|
// Views aren't created here, since the core code doesn't
|
2016-03-28 13:42:33 +02:00
|
|
|
// contain any views. Instead, they're created in overrides in
|
2016-03-29 13:50:50 +02:00
|
|
|
// plugins, such as in converse-chatview.js and converse-muc.js
|
|
|
|
return this.get(item.get('id'));
|
2016-02-16 08:46:47 +01:00
|
|
|
},
|
|
|
|
|
2016-03-28 13:42:33 +02:00
|
|
|
removeChat: function (item) {
|
|
|
|
this.remove(item.get('id'));
|
|
|
|
},
|
|
|
|
|
2016-02-28 20:24:06 +01:00
|
|
|
closeAllChatBoxes: function () {
|
|
|
|
/* This method gets overridden in src/converse-controlbox.js if
|
|
|
|
* the controlbox plugin is active.
|
|
|
|
*/
|
|
|
|
this.each(function (view) { view.close(); });
|
2016-02-16 08:46:47 +01:00
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2016-04-05 13:23:16 +02:00
|
|
|
chatBoxMayBeShown: function (chatbox) {
|
|
|
|
return this.model.chatBoxMayBeShown(chatbox);
|
|
|
|
},
|
|
|
|
|
2016-06-17 10:37:40 +02:00
|
|
|
getChatBox: function (attrs, create) {
|
2016-02-16 08:46:47 +01:00
|
|
|
var chatbox = this.model.get(attrs.jid);
|
2016-06-17 10:37:40 +02:00
|
|
|
if (!chatbox && create) {
|
2016-02-16 08:46:47 +01:00
|
|
|
chatbox = this.model.create(attrs, {
|
|
|
|
'error': function (model, response) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.log(response.responseText);
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2016-06-06 09:49:44 +02:00
|
|
|
return chatbox;
|
|
|
|
},
|
|
|
|
|
|
|
|
showChat: function (attrs) {
|
|
|
|
/* Find the chat box and show it (if it may be shown).
|
|
|
|
* If it doesn't exist, create it.
|
|
|
|
*/
|
2016-06-17 10:37:40 +02:00
|
|
|
var chatbox = this.getChatBox(attrs, true);
|
2016-04-05 13:23:16 +02:00
|
|
|
if (this.chatBoxMayBeShown(chatbox)) {
|
|
|
|
chatbox.trigger('show', true);
|
|
|
|
}
|
2016-02-16 08:46:47 +01:00
|
|
|
return chatbox;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2016-02-28 20:24:06 +01:00
|
|
|
this.XMPPStatus = Backbone.Model.extend({
|
|
|
|
initialize: function () {
|
|
|
|
this.set({
|
|
|
|
'status' : this.getStatus()
|
|
|
|
});
|
|
|
|
this.on('change', function (item) {
|
|
|
|
if (_.has(item.changed, 'status')) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('statusChanged', this.get('status'));
|
2016-02-28 20:24:06 +01:00
|
|
|
}
|
|
|
|
if (_.has(item.changed, 'status_message')) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('statusMessageChanged', this.get('status_message'));
|
2016-02-28 20:24:06 +01:00
|
|
|
}
|
|
|
|
}.bind(this));
|
2016-02-16 08:46:47 +01:00
|
|
|
},
|
|
|
|
|
2016-02-28 20:24:06 +01:00
|
|
|
constructPresence: function (type, status_message) {
|
|
|
|
var presence;
|
2016-12-20 10:30:20 +01:00
|
|
|
type = _.isString(type) ? type : (this.get('status') || _converse.default_state);
|
2017-01-26 15:49:02 +01:00
|
|
|
status_message = _.isString(status_message) ? status_message : undefined;
|
2016-02-28 20:24:06 +01:00
|
|
|
// Most of these presence types are actually not explicitly sent,
|
2016-05-31 11:39:14 +02:00
|
|
|
// but I add all of them here for reference and future proofing.
|
2016-02-28 20:24:06 +01:00
|
|
|
if ((type === 'unavailable') ||
|
|
|
|
(type === 'probe') ||
|
|
|
|
(type === 'error') ||
|
|
|
|
(type === 'unsubscribe') ||
|
|
|
|
(type === 'unsubscribed') ||
|
|
|
|
(type === 'subscribe') ||
|
|
|
|
(type === 'subscribed')) {
|
|
|
|
presence = $pres({'type': type});
|
|
|
|
} else if (type === 'offline') {
|
|
|
|
presence = $pres({'type': 'unavailable'});
|
2016-05-31 11:39:14 +02:00
|
|
|
} else if (type === 'online') {
|
|
|
|
presence = $pres();
|
2016-02-28 20:24:06 +01:00
|
|
|
} else {
|
2016-05-31 11:39:14 +02:00
|
|
|
presence = $pres().c('show').t(type).up();
|
|
|
|
}
|
|
|
|
if (status_message) {
|
2017-02-22 22:13:23 +01:00
|
|
|
presence.c('status').t(status_message).up();
|
2016-02-28 20:24:06 +01:00
|
|
|
}
|
2017-02-22 22:13:23 +01:00
|
|
|
presence.c('priority').t(
|
|
|
|
_.isNaN(Number(_converse.priority)) ? 0 : _converse.priority
|
|
|
|
);
|
2016-02-28 20:24:06 +01:00
|
|
|
return presence;
|
2016-02-16 08:46:47 +01:00
|
|
|
},
|
|
|
|
|
2016-02-28 20:24:06 +01:00
|
|
|
sendPresence: function (type, status_message) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.send(this.constructPresence(type, status_message));
|
2016-02-16 08:46:47 +01:00
|
|
|
},
|
|
|
|
|
2016-02-28 20:24:06 +01:00
|
|
|
setStatus: function (value) {
|
|
|
|
this.sendPresence(value);
|
|
|
|
this.save({'status': value});
|
2016-02-16 08:46:47 +01:00
|
|
|
},
|
|
|
|
|
2016-02-28 20:24:06 +01:00
|
|
|
getStatus: function () {
|
2016-12-20 10:30:20 +01:00
|
|
|
return this.get('status') || _converse.default_state;
|
2016-02-16 08:46:47 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
setStatusMessage: function (status_message) {
|
|
|
|
this.sendPresence(this.getStatus(), status_message);
|
|
|
|
var prev_status = this.get('status_message');
|
|
|
|
this.save({'status_message': status_message});
|
|
|
|
if (this.xhr_custom_status) {
|
|
|
|
$.ajax({
|
|
|
|
url: this.xhr_custom_status_url,
|
|
|
|
type: 'POST',
|
|
|
|
data: {'msg': status_message}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (prev_status === status_message) {
|
|
|
|
this.trigger("update-status-ui", this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
this.Session = Backbone.Model; // General session settings to be saved to sessionStorage.
|
|
|
|
this.Feature = Backbone.Model;
|
|
|
|
this.Features = Backbone.Collection.extend({
|
|
|
|
/* Service Discovery
|
2016-03-18 09:49:32 +01:00
|
|
|
* -----------------
|
|
|
|
* This collection stores Feature Models, representing features
|
|
|
|
* provided by available XMPP entities (e.g. servers)
|
|
|
|
* See XEP-0030 for more details: http://xmpp.org/extensions/xep-0030.html
|
|
|
|
* All features are shown here: http://xmpp.org/registrar/disco-features.html
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
model: _converse.Feature,
|
2016-02-16 08:46:47 +01:00
|
|
|
initialize: function () {
|
|
|
|
this.addClientIdentities().addClientFeatures();
|
2016-12-20 10:30:20 +01:00
|
|
|
this.browserStorage = new Backbone.BrowserStorage[_converse.storage](
|
|
|
|
b64_sha1('converse.features'+_converse.bare_jid)
|
2016-09-22 14:03:42 +02:00
|
|
|
);
|
2016-02-16 08:46:47 +01:00
|
|
|
this.on('add', this.onFeatureAdded, this);
|
|
|
|
if (this.browserStorage.records.length === 0) {
|
|
|
|
// browserStorage is empty, so we've likely never queried this
|
|
|
|
// domain for features yet
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.disco.info(_converse.domain, null, this.onInfo.bind(this));
|
|
|
|
_converse.connection.disco.items(_converse.domain, null, this.onItems.bind(this));
|
2016-02-16 08:46:47 +01:00
|
|
|
} else {
|
|
|
|
this.fetch({add:true});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
onFeatureAdded: function (feature) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.emit('serviceDiscovered', feature);
|
2016-02-16 08:46:47 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
addClientIdentities: function () {
|
|
|
|
/* See http://xmpp.org/registrar/disco-categories.html
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.disco.addIdentity('client', 'web', 'Converse.js');
|
2016-02-16 08:46:47 +01:00
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
addClientFeatures: function () {
|
|
|
|
/* The strophe.disco.js plugin keeps a list of features which
|
|
|
|
* it will advertise to any #info queries made to it.
|
|
|
|
*
|
|
|
|
* See: http://xmpp.org/extensions/xep-0030.html#info
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.disco.addFeature(Strophe.NS.BOSH);
|
|
|
|
_converse.connection.disco.addFeature(Strophe.NS.CHATSTATES);
|
|
|
|
_converse.connection.disco.addFeature(Strophe.NS.DISCO_INFO);
|
|
|
|
_converse.connection.disco.addFeature(Strophe.NS.ROSTERX); // Limited support
|
|
|
|
if (_converse.message_carbons) {
|
|
|
|
_converse.connection.disco.addFeature(Strophe.NS.CARBONS);
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
onItems: function (stanza) {
|
2016-11-24 02:07:32 +01:00
|
|
|
var that = this;
|
|
|
|
_.each(stanza.querySelectorAll('query item'), function (item) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.connection.disco.info(
|
2016-11-24 02:07:32 +01:00
|
|
|
item.getAttribute('jid'),
|
2016-02-16 08:46:47 +01:00
|
|
|
null,
|
2016-11-24 02:07:32 +01:00
|
|
|
that.onInfo.bind(that));
|
|
|
|
});
|
2016-02-16 08:46:47 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
onInfo: function (stanza) {
|
|
|
|
var $stanza = $(stanza);
|
|
|
|
if (($stanza.find('identity[category=server][type=im]').length === 0) &&
|
|
|
|
($stanza.find('identity[category=conference][type=text]').length === 0)) {
|
|
|
|
// This isn't an IM server component
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
$stanza.find('feature').each(function (idx, feature) {
|
2016-11-24 02:07:32 +01:00
|
|
|
var namespace = feature.getAttribute('var');
|
2016-02-16 08:46:47 +01:00
|
|
|
this[namespace] = true;
|
|
|
|
this.create({
|
|
|
|
'var': namespace,
|
2016-11-24 02:07:32 +01:00
|
|
|
'from': stanza.getAttribute('from')
|
2016-02-16 08:46:47 +01:00
|
|
|
});
|
|
|
|
}.bind(this));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
this.setUpXMLLogging = function () {
|
2016-05-28 10:55:03 +02:00
|
|
|
Strophe.log = function (level, msg) {
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.log(msg, level);
|
2016-05-28 10:55:03 +02:00
|
|
|
};
|
2016-02-16 08:46:47 +01:00
|
|
|
if (this.debug) {
|
2016-12-20 10:30:20 +01:00
|
|
|
this.connection.xmlInput = function (body) { _converse.log(body.outerHTML); };
|
|
|
|
this.connection.xmlOutput = function (body) { _converse.log(body.outerHTML); };
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-03-31 10:37:28 +02:00
|
|
|
this.fetchLoginCredentials = function () {
|
|
|
|
var deferred = new $.Deferred();
|
|
|
|
$.ajax({
|
2016-12-20 10:30:20 +01:00
|
|
|
url: _converse.credentials_url,
|
2016-03-31 10:37:28 +02:00
|
|
|
type: 'GET',
|
|
|
|
dataType: "json",
|
|
|
|
success: function (response) {
|
|
|
|
deferred.resolve({
|
|
|
|
'jid': response.jid,
|
|
|
|
'password': response.password
|
|
|
|
});
|
|
|
|
},
|
|
|
|
error: function (response) {
|
2016-12-20 10:30:20 +01:00
|
|
|
delete _converse.connection;
|
|
|
|
_converse.emit('noResumeableSession');
|
2016-03-31 10:37:28 +02:00
|
|
|
deferred.reject(response);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return deferred.promise();
|
|
|
|
};
|
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
this.startNewBOSHSession = function () {
|
2016-11-07 18:30:10 +01:00
|
|
|
var that = this;
|
2016-02-16 08:46:47 +01:00
|
|
|
$.ajax({
|
|
|
|
url: this.prebind_url,
|
|
|
|
type: 'GET',
|
2016-03-16 13:56:11 +01:00
|
|
|
dataType: "json",
|
2016-02-16 08:46:47 +01:00
|
|
|
success: function (response) {
|
2016-11-07 18:30:10 +01:00
|
|
|
that.connection.attach(
|
2016-02-16 08:46:47 +01:00
|
|
|
response.jid,
|
|
|
|
response.sid,
|
|
|
|
response.rid,
|
2016-11-07 18:30:10 +01:00
|
|
|
that.onConnectStatusChanged
|
2016-02-16 08:46:47 +01:00
|
|
|
);
|
2016-11-07 18:30:10 +01:00
|
|
|
},
|
2016-02-16 08:46:47 +01:00
|
|
|
error: function (response) {
|
2016-11-07 18:30:10 +01:00
|
|
|
delete that.connection;
|
|
|
|
that.emit('noResumeableSession');
|
|
|
|
}
|
2016-02-16 08:46:47 +01:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2016-11-23 12:53:06 +01:00
|
|
|
this.attemptPreboundSession = function (reconnecting) {
|
2016-02-16 08:46:47 +01:00
|
|
|
/* Handle session resumption or initialization when prebind is being used.
|
|
|
|
*/
|
2016-11-23 12:53:06 +01:00
|
|
|
if (!reconnecting && this.keepalive) {
|
2016-02-16 08:46:47 +01:00
|
|
|
if (!this.jid) {
|
2016-04-13 17:11:04 +02:00
|
|
|
throw new Error("attemptPreboundSession: when using 'keepalive' with 'prebind, "+
|
|
|
|
"you must supply the JID of the current user.");
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
try {
|
|
|
|
return this.connection.restore(this.jid, this.onConnectStatusChanged);
|
|
|
|
} catch (e) {
|
|
|
|
this.log("Could not restore session for jid: "+this.jid+" Error message: "+e.message);
|
|
|
|
this.clearSession(); // If there's a roster, we want to clear it (see #555)
|
|
|
|
}
|
|
|
|
}
|
2016-11-07 18:30:10 +01:00
|
|
|
|
|
|
|
// No keepalive, or session resumption has failed.
|
2016-11-23 12:53:06 +01:00
|
|
|
if (!reconnecting && this.jid && this.sid && this.rid) {
|
2016-11-07 18:30:10 +01:00
|
|
|
return this.connection.attach(this.jid, this.sid, this.rid, this.onConnectStatusChanged);
|
|
|
|
} else if (this.prebind_url) {
|
|
|
|
return this.startNewBOSHSession();
|
2016-02-16 08:46:47 +01:00
|
|
|
} else {
|
2016-11-07 18:30:10 +01:00
|
|
|
throw new Error("attemptPreboundSession: If you use prebind and not keepalive, "+
|
|
|
|
"then you MUST supply JID, RID and SID values or a prebind_url.");
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-03-31 10:37:28 +02:00
|
|
|
this.autoLogin = function (credentials) {
|
|
|
|
if (credentials) {
|
2016-04-13 13:59:09 +02:00
|
|
|
// If passed in, then they come from credentials_url, so we
|
2016-12-20 10:30:20 +01:00
|
|
|
// set them on the _converse object.
|
2016-03-31 10:37:28 +02:00
|
|
|
this.jid = credentials.jid;
|
|
|
|
this.password = credentials.password;
|
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
if (this.authentication === _converse.ANONYMOUS) {
|
2016-07-24 10:54:57 +02:00
|
|
|
if (!this.jid) {
|
|
|
|
throw new Error("Config Error: when using anonymous login " +
|
|
|
|
"you need to provide the server's domain via the 'jid' option. " +
|
|
|
|
"Either when calling converse.initialize, or when calling " +
|
2016-12-20 10:30:20 +01:00
|
|
|
"_converse.api.user.login.");
|
2016-07-24 10:54:57 +02:00
|
|
|
}
|
2016-03-31 10:37:28 +02:00
|
|
|
this.connection.connect(this.jid.toLowerCase(), null, this.onConnectStatusChanged);
|
2016-12-20 10:30:20 +01:00
|
|
|
} else if (this.authentication === _converse.LOGIN) {
|
|
|
|
var password = _converse.connection.pass || this.password;
|
2016-08-18 21:43:18 +02:00
|
|
|
if (!password) {
|
2016-08-23 20:10:08 +02:00
|
|
|
if (this.auto_login && !this.password) {
|
|
|
|
throw new Error("initConnection: If you use auto_login and "+
|
|
|
|
"authentication='login' then you also need to provide a password.");
|
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.setDisconnectionCause(Strophe.Status.AUTHFAIL, undefined, true);
|
|
|
|
_converse.disconnect();
|
2016-10-27 12:58:51 +02:00
|
|
|
return;
|
2016-03-31 10:37:28 +02:00
|
|
|
}
|
|
|
|
var resource = Strophe.getResourceFromJid(this.jid);
|
|
|
|
if (!resource) {
|
2016-12-20 10:30:20 +01:00
|
|
|
this.jid = this.jid.toLowerCase() + _converse.generateResource();
|
2016-03-31 10:37:28 +02:00
|
|
|
} else {
|
|
|
|
this.jid = Strophe.getBareJidFromJid(this.jid).toLowerCase()+'/'+resource;
|
|
|
|
}
|
2016-08-18 21:43:18 +02:00
|
|
|
this.connection.connect(this.jid, password, this.onConnectStatusChanged);
|
2016-03-31 10:37:28 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-11-23 12:53:06 +01:00
|
|
|
this.attemptNonPreboundSession = function (credentials, reconnecting) {
|
2016-02-16 08:46:47 +01:00
|
|
|
/* Handle session resumption or initialization when prebind is not being used.
|
|
|
|
*
|
|
|
|
* Two potential options exist and are handled in this method:
|
|
|
|
* 1. keepalive
|
|
|
|
* 2. auto_login
|
|
|
|
*/
|
2016-11-23 12:53:06 +01:00
|
|
|
if (this.keepalive && !reconnecting) {
|
2016-02-16 08:46:47 +01:00
|
|
|
try {
|
2016-05-31 12:24:36 +02:00
|
|
|
return this.connection.restore(this.jid, this.onConnectStatusChanged);
|
2016-02-16 08:46:47 +01:00
|
|
|
} catch (e) {
|
|
|
|
this.log("Could not restore session. Error message: "+e.message);
|
|
|
|
this.clearSession(); // If there's a roster, we want to clear it (see #555)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (this.auto_login) {
|
2016-11-07 18:30:10 +01:00
|
|
|
if (credentials) {
|
|
|
|
// When credentials are passed in, they override prebinding
|
|
|
|
// or credentials fetching via HTTP
|
|
|
|
this.autoLogin(credentials);
|
|
|
|
} else if (this.credentials_url) {
|
2016-03-31 10:37:28 +02:00
|
|
|
this.fetchLoginCredentials().done(this.autoLogin.bind(this));
|
|
|
|
} else if (!this.jid) {
|
|
|
|
throw new Error(
|
|
|
|
"initConnection: If you use auto_login, you also need"+
|
|
|
|
"to give either a jid value (and if applicable a "+
|
|
|
|
"password) or you need to pass in a URL from where the "+
|
|
|
|
"username and password can be fetched (via credentials_url)."
|
|
|
|
);
|
|
|
|
} else {
|
2016-11-07 18:30:10 +01:00
|
|
|
// Probably ANONYMOUS login
|
2016-03-31 10:37:28 +02:00
|
|
|
this.autoLogin();
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
2016-12-04 15:14:40 +01:00
|
|
|
} else if (reconnecting) {
|
|
|
|
this.autoLogin();
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-11-23 12:53:06 +01:00
|
|
|
this.logIn = function (credentials, reconnecting) {
|
2016-11-07 18:30:10 +01:00
|
|
|
// We now try to resume or automatically set up a new session.
|
|
|
|
// Otherwise the user will be shown a login form.
|
2016-12-20 10:30:20 +01:00
|
|
|
if (this.authentication === _converse.PREBIND) {
|
2016-11-23 12:53:06 +01:00
|
|
|
this.attemptPreboundSession(reconnecting);
|
2016-04-13 13:52:28 +02:00
|
|
|
} else {
|
2016-11-23 12:53:06 +01:00
|
|
|
this.attemptNonPreboundSession(credentials, reconnecting);
|
2016-04-13 13:52:28 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
this.initConnection = function () {
|
2016-04-13 17:11:04 +02:00
|
|
|
if (this.connection) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!this.bosh_service_url && ! this.websocket_url) {
|
|
|
|
throw new Error("initConnection: you must supply a value for either the bosh_service_url or websocket_url or both.");
|
|
|
|
}
|
|
|
|
if (('WebSocket' in window || 'MozWebSocket' in window) && this.websocket_url) {
|
2016-10-12 13:55:47 +02:00
|
|
|
this.connection = new Strophe.Connection(this.websocket_url, this.connection_options);
|
2016-04-13 17:11:04 +02:00
|
|
|
} else if (this.bosh_service_url) {
|
2016-10-12 13:55:47 +02:00
|
|
|
this.connection = new Strophe.Connection(
|
|
|
|
this.bosh_service_url,
|
2017-01-26 10:08:43 +01:00
|
|
|
_.assignIn(this.connection_options, {'keepalive': this.keepalive})
|
2016-10-12 13:55:47 +02:00
|
|
|
);
|
2016-02-16 08:46:47 +01:00
|
|
|
} else {
|
2016-04-13 17:11:04 +02:00
|
|
|
throw new Error("initConnection: this browser does not support websockets and bosh_service_url wasn't specified.");
|
2016-02-16 08:46:47 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
this._tearDown = function () {
|
|
|
|
/* Remove those views which are only allowed with a valid
|
|
|
|
* connection.
|
|
|
|
*/
|
2016-06-22 12:21:56 +02:00
|
|
|
this.unregisterPresenceHandler();
|
2016-02-16 08:46:47 +01:00
|
|
|
if (this.roster) {
|
|
|
|
this.roster.off().reset(); // Removes roster contacts
|
|
|
|
}
|
|
|
|
this.chatboxes.remove(); // Don't call off(), events won't get re-registered upon reconnect.
|
|
|
|
if (this.features) {
|
|
|
|
this.features.reset();
|
|
|
|
}
|
2016-12-20 10:30:20 +01:00
|
|
|
$(window).off('click mousemove keypress focus'+unloadevent, _converse.onUserActivity);
|
|
|
|
window.clearInterval(_converse.everySecondTrigger);
|
2016-02-16 08:46:47 +01:00
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
2016-11-22 17:36:39 +01:00
|
|
|
this.initChatBoxes = function () {
|
2016-02-16 08:46:47 +01:00
|
|
|
this.chatboxes = new this.ChatBoxes();
|
|
|
|
this.chatboxviews = new this.ChatBoxViews({model: this.chatboxes});
|
2016-11-22 17:36:39 +01:00
|
|
|
};
|
|
|
|
|
2016-06-09 11:01:54 +02:00
|
|
|
var updateSettings = function (settings) {
|
|
|
|
/* Helper method which gets put on the plugin and allows it to
|
|
|
|
* add more user-facing config settings to converse.js.
|
|
|
|
*/
|
2016-12-20 10:30:20 +01:00
|
|
|
utils.merge(_converse.default_settings, settings);
|
|
|
|
utils.merge(_converse, settings);
|
|
|
|
utils.applyUserSettings(_converse, settings, _converse.user_settings);
|
2016-06-09 11:01:54 +02:00
|
|
|
};
|
2016-11-02 21:19:17 +01:00
|
|
|
|
2017-03-07 12:02:40 +01:00
|
|
|
this.initPlugins = function () {
|
|
|
|
// If initialize gets called a second time (e.g. during tests), then we
|
|
|
|
// need to re-apply all plugins (for a new converse instance), and we
|
|
|
|
// therefore need to clear this array that prevents plugins from being
|
|
|
|
// initialized twice.
|
|
|
|
// If initialize is called for the first time, then this array is empty
|
|
|
|
// in any case.
|
|
|
|
_converse.pluggable.initialized_plugins = [];
|
|
|
|
var whitelist = _converse.core_plugins.concat(
|
|
|
|
_converse.whitelisted_plugins);
|
|
|
|
|
|
|
|
_converse.pluggable.initializePlugins({
|
|
|
|
'updateSettings': updateSettings,
|
|
|
|
'_converse': _converse
|
|
|
|
}, whitelist, _converse.blacklisted_plugins);
|
|
|
|
_converse.emit('pluginsInitialized');
|
|
|
|
};
|
2017-02-03 19:12:19 +01:00
|
|
|
|
2017-03-07 12:02:40 +01:00
|
|
|
// Initialization
|
|
|
|
// --------------
|
|
|
|
// This is the end of the initialize method.
|
|
|
|
if (settings.connection) {
|
|
|
|
this.connection = settings.connection;
|
|
|
|
}
|
|
|
|
_converse.initPlugins();
|
|
|
|
_converse.initChatBoxes();
|
|
|
|
_converse.initSession();
|
|
|
|
_converse.initConnection();
|
|
|
|
_converse.setUpXMLLogging();
|
|
|
|
_converse.logIn();
|
2016-12-20 10:30:20 +01:00
|
|
|
_converse.registerGlobalEventHandlers();
|
2016-11-02 23:08:20 +01:00
|
|
|
|
2016-12-20 10:30:20 +01:00
|
|
|
if (!_.isUndefined(_converse.connection) &&
|
|
|
|
_converse.connection.service === 'jasmine tests') {
|
|
|
|
return _converse;
|
2016-11-02 23:08:20 +01:00
|
|
|
} else {
|
|
|
|
return init_deferred.promise();
|
|
|
|
}
|
2016-02-16 08:46:47 +01:00
|
|
|
};
|
2017-02-14 15:08:39 +01:00
|
|
|
|
|
|
|
// API methods only available to plugins
|
|
|
|
_converse.api = {
|
|
|
|
'connection': {
|
|
|
|
'connected': function () {
|
|
|
|
return _converse.connection && _converse.connection.connected || false;
|
|
|
|
},
|
|
|
|
'disconnect': function () {
|
|
|
|
_converse.connection.disconnect();
|
|
|
|
},
|
|
|
|
},
|
|
|
|
'user': {
|
|
|
|
'jid': function () {
|
|
|
|
return _converse.connection.jid;
|
|
|
|
},
|
|
|
|
'login': function (credentials) {
|
|
|
|
_converse.initConnection();
|
|
|
|
_converse.logIn(credentials);
|
|
|
|
},
|
|
|
|
'logout': function () {
|
|
|
|
_converse.logOut();
|
|
|
|
},
|
|
|
|
'status': {
|
|
|
|
'get': function () {
|
|
|
|
return _converse.xmppstatus.get('status');
|
|
|
|
},
|
|
|
|
'set': function (value, message) {
|
|
|
|
var data = {'status': value};
|
|
|
|
if (!_.includes(_.keys(_converse.STATUS_WEIGHTS), value)) {
|
|
|
|
throw new Error('Invalid availability value. See https://xmpp.org/rfcs/rfc3921.html#rfc.section.2.2.2.1');
|
|
|
|
}
|
|
|
|
if (_.isString(message)) {
|
|
|
|
data.status_message = message;
|
|
|
|
}
|
|
|
|
_converse.xmppstatus.sendPresence(value);
|
|
|
|
_converse.xmppstatus.save(data);
|
|
|
|
},
|
|
|
|
'message': {
|
|
|
|
'get': function () {
|
|
|
|
return _converse.xmppstatus.get('status_message');
|
|
|
|
},
|
|
|
|
'set': function (stat) {
|
|
|
|
_converse.xmppstatus.save({'status_message': stat});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
'settings': {
|
|
|
|
'get': function (key) {
|
|
|
|
if (_.includes(_.keys(_converse.default_settings), key)) {
|
|
|
|
return _converse[key];
|
|
|
|
}
|
|
|
|
},
|
|
|
|
'set': function (key, val) {
|
|
|
|
var o = {};
|
|
|
|
if (_.isObject(key)) {
|
|
|
|
_.assignIn(_converse, _.pick(key, _.keys(_converse.default_settings)));
|
|
|
|
} else if (_.isString("string")) {
|
|
|
|
o[key] = val;
|
|
|
|
_.assignIn(_converse, _.pick(o, _.keys(_converse.default_settings)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
'contacts': {
|
|
|
|
'get': function (jids) {
|
|
|
|
var _transform = function (jid) {
|
|
|
|
var contact = _converse.roster.get(Strophe.getBareJidFromJid(jid));
|
|
|
|
if (contact) {
|
|
|
|
return contact.attributes;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
};
|
|
|
|
if (_.isUndefined(jids)) {
|
|
|
|
jids = _converse.roster.pluck('jid');
|
|
|
|
} else if (_.isString(jids)) {
|
|
|
|
return _transform(jids);
|
|
|
|
}
|
|
|
|
return _.map(jids, _transform);
|
|
|
|
},
|
|
|
|
'add': function (jid, name) {
|
|
|
|
if (!_.isString(jid) || !_.includes(jid, '@')) {
|
|
|
|
throw new TypeError('contacts.add: invalid jid');
|
|
|
|
}
|
|
|
|
_converse.roster.addAndSubscribe(jid, _.isEmpty(name)? jid: name);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
'chats': {
|
|
|
|
'open': function (jids, attrs) {
|
|
|
|
var chatbox;
|
|
|
|
if (_.isUndefined(jids)) {
|
|
|
|
_converse.log("chats.open: You need to provide at least one JID", "error");
|
|
|
|
return null;
|
|
|
|
} else if (_.isString(jids)) {
|
2017-02-14 15:58:29 +01:00
|
|
|
chatbox = _converse.getViewForChatBox(
|
2017-02-14 15:08:39 +01:00
|
|
|
_converse.chatboxes.getChatBox(jids, true, attrs).trigger('show')
|
|
|
|
);
|
|
|
|
return chatbox;
|
|
|
|
}
|
|
|
|
return _.map(jids, function (jid) {
|
2017-02-14 15:58:29 +01:00
|
|
|
chatbox = _converse.getViewForChatBox(
|
2017-02-14 15:08:39 +01:00
|
|
|
_converse.chatboxes.getChatBox(jid, true, attrs).trigger('show')
|
|
|
|
);
|
|
|
|
return chatbox;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
'get': function (jids) {
|
|
|
|
if (_.isUndefined(jids)) {
|
|
|
|
var result = [];
|
|
|
|
_converse.chatboxes.each(function (chatbox) {
|
|
|
|
// FIXME: Leaky abstraction from MUC. We need to add a
|
|
|
|
// base type for chat boxes, and check for that.
|
|
|
|
if (chatbox.get('type') !== 'chatroom') {
|
2017-02-14 15:58:29 +01:00
|
|
|
result.push(_converse.getViewForChatBox(chatbox));
|
2017-02-14 15:08:39 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
return result;
|
|
|
|
} else if (_.isString(jids)) {
|
2017-02-14 15:58:29 +01:00
|
|
|
return _converse.getViewForChatBox(_converse.chatboxes.getChatBox(jids));
|
2017-02-14 15:08:39 +01:00
|
|
|
}
|
|
|
|
return _.map(jids,
|
|
|
|
_.partial(
|
|
|
|
_.flow(
|
|
|
|
_converse.chatboxes.getChatBox.bind(_converse.chatboxes),
|
2017-02-14 15:58:29 +01:00
|
|
|
_converse.getViewForChatBox.bind(_converse)
|
2017-02-14 15:08:39 +01:00
|
|
|
), _, true
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
'tokens': {
|
|
|
|
'get': function (id) {
|
|
|
|
if (!_converse.expose_rid_and_sid || _.isUndefined(_converse.connection)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (id.toLowerCase() === 'rid') {
|
|
|
|
return _converse.connection.rid || _converse.connection._proto.rid;
|
|
|
|
} else if (id.toLowerCase() === 'sid') {
|
|
|
|
return _converse.connection.sid || _converse.connection._proto.sid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
'listen': {
|
2017-03-08 12:52:41 +01:00
|
|
|
'once': _converse.once.bind(_converse),
|
|
|
|
'on': _converse.on.bind(_converse),
|
|
|
|
'not': _converse.off.bind(_converse),
|
2017-02-14 15:08:39 +01:00
|
|
|
'stanza': function (name, options, handler) {
|
|
|
|
if (_.isFunction(options)) {
|
|
|
|
handler = options;
|
|
|
|
options = {};
|
|
|
|
} else {
|
|
|
|
options = options || {};
|
|
|
|
}
|
|
|
|
_converse.connection.addHandler(
|
|
|
|
handler,
|
|
|
|
options.ns,
|
|
|
|
name,
|
|
|
|
options.type,
|
|
|
|
options.id,
|
|
|
|
options.from,
|
|
|
|
options
|
|
|
|
);
|
|
|
|
},
|
|
|
|
},
|
|
|
|
'waitUntil': function (name) {
|
|
|
|
var promise = _converse.promises[name];
|
|
|
|
if (_.isUndefined(promise)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return _converse.promises[name].promise();
|
|
|
|
},
|
|
|
|
'send': function (stanza) {
|
|
|
|
_converse.connection.send(stanza);
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
// The public API
|
|
|
|
return {
|
|
|
|
'initialize': function (settings, callback) {
|
|
|
|
return _converse.initialize(settings, callback);
|
|
|
|
},
|
|
|
|
'plugins': {
|
|
|
|
'add': function (name, plugin) {
|
|
|
|
plugin.__name__ = name;
|
|
|
|
if (!_.isUndefined(_converse.pluggable.plugins[name])) {
|
|
|
|
throw new TypeError(
|
|
|
|
'Error: plugin with name "'+name+'" has already been '+
|
|
|
|
'registered!');
|
|
|
|
} else {
|
|
|
|
_converse.pluggable.plugins[name] = plugin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
'env': {
|
|
|
|
'$build': $build,
|
|
|
|
'$iq': $iq,
|
|
|
|
'$msg': $msg,
|
|
|
|
'$pres': $pres,
|
|
|
|
'Strophe': Strophe,
|
|
|
|
'b64_sha1': b64_sha1,
|
|
|
|
'_': _,
|
|
|
|
'jQuery': $,
|
2017-02-24 11:31:05 +01:00
|
|
|
'sizzle': sizzle,
|
2017-02-14 15:08:39 +01:00
|
|
|
'moment': moment,
|
|
|
|
'utils': utils
|
|
|
|
}
|
|
|
|
};
|
2016-02-16 08:46:47 +01:00
|
|
|
}));
|