2018-04-16 18:08:00 +02:00
|
|
|
|
// Converse.js
|
|
|
|
|
// https://conversejs.org
|
2016-02-16 08:46:47 +01:00
|
|
|
|
//
|
2018-05-29 12:03:49 +02:00
|
|
|
|
// Copyright (c) 2013-2018, the Converse.js developers
|
2016-02-16 08:46:47 +01:00
|
|
|
|
// Licensed under the Mozilla Public License (MPLv2)
|
2018-04-16 18:08:00 +02:00
|
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
|
(function (root, factory) {
|
2017-02-14 15:08:39 +01:00
|
|
|
|
define(["sizzle",
|
2017-07-10 21:14:48 +02:00
|
|
|
|
"es6-promise",
|
2017-04-21 15:52:16 +02:00
|
|
|
|
"lodash.noconflict",
|
2018-02-08 17:06:53 +01:00
|
|
|
|
"lodash.fp",
|
2017-02-14 15:08:39 +01:00
|
|
|
|
"polyfill",
|
2017-10-24 12:56:35 +02:00
|
|
|
|
"i18n",
|
2018-06-06 11:04:23 +02:00
|
|
|
|
"utils/core",
|
2017-10-30 10:25:30 +01:00
|
|
|
|
"moment",
|
2017-02-14 15:08:39 +01:00
|
|
|
|
"strophe",
|
|
|
|
|
"pluggable",
|
2017-04-21 15:52:16 +02:00
|
|
|
|
"backbone.noconflict",
|
2018-01-03 20:02:05 +01:00
|
|
|
|
"backbone.nativeview",
|
2017-12-20 20:29:57 +01:00
|
|
|
|
"backbone.browserStorage"
|
2016-02-23 08:13:30 +01:00
|
|
|
|
], factory);
|
2018-02-19 10:34:52 +01:00
|
|
|
|
}(this, function (sizzle, Promise, _, f, polyfill, i18n, u, moment, Strophe, pluggable, Backbone) {
|
2018-06-06 11:07:59 +02:00
|
|
|
|
"use strict";
|
2016-02-23 08:13:30 +01:00
|
|
|
|
|
|
|
|
|
// Strophe globals
|
2017-07-10 17:46:22 +02:00
|
|
|
|
const { $build, $iq, $msg, $pres } = Strophe;
|
2017-07-11 15:22:05 +02:00
|
|
|
|
const b64_sha1 = Strophe.SHA1.b64_sha1;
|
|
|
|
|
Strophe = Strophe.Strophe;
|
2016-02-16 08:46:47 +01:00
|
|
|
|
|
2017-09-24 20:36:39 +02: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');
|
|
|
|
|
Strophe.addNamespace('DELAY', 'urn:xmpp:delay');
|
2018-01-04 14:05:35 +01:00
|
|
|
|
Strophe.addNamespace('FORWARD', 'urn:xmpp:forward:0');
|
2017-09-24 20:36:39 +02:00
|
|
|
|
Strophe.addNamespace('HINTS', 'urn:xmpp:hints');
|
2018-04-16 21:58:11 +02:00
|
|
|
|
Strophe.addNamespace('HTTPUPLOAD', 'urn:xmpp:http:upload:0');
|
2018-05-02 14:30:14 +02:00
|
|
|
|
Strophe.addNamespace('MAM', 'urn:xmpp:mam:2');
|
2017-09-24 20:36:39 +02:00
|
|
|
|
Strophe.addNamespace('NICK', 'http://jabber.org/protocol/nick');
|
2018-04-18 16:55:26 +02:00
|
|
|
|
Strophe.addNamespace('OUTOFBAND', 'jabber:x:oob');
|
2017-09-24 20:36:39 +02:00
|
|
|
|
Strophe.addNamespace('PUBSUB', 'http://jabber.org/protocol/pubsub');
|
|
|
|
|
Strophe.addNamespace('ROSTERX', 'http://jabber.org/protocol/rosterx');
|
|
|
|
|
Strophe.addNamespace('RSM', 'http://jabber.org/protocol/rsm');
|
2018-01-04 14:05:35 +01:00
|
|
|
|
Strophe.addNamespace('SID', 'urn:xmpp:sid:0');
|
2018-02-05 21:42:29 +01:00
|
|
|
|
Strophe.addNamespace('SPOILER', 'urn:xmpp:spoiler:0');
|
2018-05-08 14:46:00 +02:00
|
|
|
|
Strophe.addNamespace('VCARD', 'vcard-temp');
|
2018-05-03 13:47:31 +02:00
|
|
|
|
Strophe.addNamespace('VCARDUPDATE', 'vcard-temp:x:update');
|
2018-05-08 14:46:00 +02:00
|
|
|
|
Strophe.addNamespace('XFORM', 'jabber:x:data');
|
2017-09-24 20:36:39 +02:00
|
|
|
|
|
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 = {
|
2017-05-09 09:45:12 +02:00
|
|
|
|
'escape': /\{\{\{([\s\S]+?)\}\}\}/g,
|
|
|
|
|
'evaluate': /\{\[([\s\S]+?)\]\}/g,
|
2017-09-26 18:27:41 +02:00
|
|
|
|
'interpolate': /\{\{([\s\S]+?)\}\}/g,
|
|
|
|
|
'imports': { '_': _ }
|
2016-02-16 08:46:47 +01:00
|
|
|
|
};
|
|
|
|
|
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* A private, closured object containing the private api (via `_converse.api`)
|
|
|
|
|
* as well as private methods and internal data-structures.
|
2018-09-02 12:15:26 +02:00
|
|
|
|
*
|
|
|
|
|
* @namespace _converse
|
|
|
|
|
*/
|
2017-07-10 21:14:48 +02:00
|
|
|
|
const _converse = {
|
|
|
|
|
'templates': {},
|
|
|
|
|
'promises': {}
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-11 15:22:05 +02:00
|
|
|
|
_.extend(_converse, Backbone.Events);
|
2016-02-16 08:46:47 +01:00
|
|
|
|
|
2018-03-14 18:25:34 +01:00
|
|
|
|
// Core plugins are whitelisted automatically
|
2017-02-03 19:12:19 +01:00
|
|
|
|
_converse.core_plugins = [
|
2018-08-10 14:09:21 +02:00
|
|
|
|
'converse-autocomplete',
|
2017-02-03 19:12:19 +01:00
|
|
|
|
'converse-bookmarks',
|
2018-06-06 11:07:59 +02:00
|
|
|
|
'converse-caps',
|
2017-08-16 11:16:22 +02:00
|
|
|
|
'converse-chatboxes',
|
2018-09-06 15:22:14 +02:00
|
|
|
|
'converse-chatboxviews',
|
2017-02-03 19:12:19 +01:00
|
|
|
|
'converse-chatview',
|
|
|
|
|
'converse-controlbox',
|
|
|
|
|
'converse-core',
|
2017-07-21 12:41:16 +02:00
|
|
|
|
'converse-disco',
|
2017-02-03 19:12:19 +01:00
|
|
|
|
'converse-dragresize',
|
2018-05-03 17:01:17 +02:00
|
|
|
|
'converse-embedded',
|
2017-11-02 15:30:24 +01:00
|
|
|
|
'converse-fullscreen',
|
2017-02-03 19:12:19 +01:00
|
|
|
|
'converse-headline',
|
|
|
|
|
'converse-mam',
|
2018-04-16 18:08:00 +02:00
|
|
|
|
'converse-message-view',
|
2017-02-03 19:12:19 +01:00
|
|
|
|
'converse-minimize',
|
2018-02-22 18:41:01 +01:00
|
|
|
|
'converse-modal',
|
2017-02-03 19:12:19 +01:00
|
|
|
|
'converse-muc',
|
2018-02-21 22:29:21 +01:00
|
|
|
|
'converse-muc-views',
|
2017-02-03 19:12:19 +01:00
|
|
|
|
'converse-notification',
|
2018-05-11 22:05:45 +02:00
|
|
|
|
'converse-omemo',
|
2017-02-03 19:12:19 +01:00
|
|
|
|
'converse-ping',
|
2017-11-17 13:32:03 +01:00
|
|
|
|
'converse-profile',
|
2018-06-06 11:07:59 +02:00
|
|
|
|
'converse-push',
|
2017-02-03 19:12:19 +01:00
|
|
|
|
'converse-register',
|
2017-06-14 10:47:57 +02:00
|
|
|
|
'converse-roomslist',
|
2018-05-04 23:27:08 +02:00
|
|
|
|
'converse-roster',
|
2017-02-03 19:12:19 +01:00
|
|
|
|
'converse-rosterview',
|
2017-11-01 10:37:24 +01:00
|
|
|
|
'converse-singleton',
|
2018-02-09 16:33:48 +01:00
|
|
|
|
'converse-spoilers',
|
|
|
|
|
'converse-vcard'
|
2017-02-03 19:12:19 +01:00
|
|
|
|
];
|
|
|
|
|
|
2018-07-28 17:13:13 +02:00
|
|
|
|
// Setting wait to 59 instead of 60 to avoid timing conflicts with the
|
|
|
|
|
// webserver, which is often also set to 60 and might therefore sometimes
|
|
|
|
|
// return a 504 error page instead of passing through to the BOSH proxy.
|
|
|
|
|
const BOSH_WAIT = 59;
|
|
|
|
|
|
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
|
|
|
|
|
2018-08-10 14:09:21 +02:00
|
|
|
|
_converse.keycodes = {
|
|
|
|
|
TAB: 9,
|
|
|
|
|
ENTER: 13,
|
|
|
|
|
SHIFT: 16,
|
|
|
|
|
CTRL: 17,
|
|
|
|
|
ALT: 18,
|
|
|
|
|
ESCAPE: 27,
|
|
|
|
|
UP_ARROW: 38,
|
|
|
|
|
DOWN_ARROW: 40,
|
|
|
|
|
FORWARD_SLASH: 47,
|
2018-08-14 18:44:08 +02:00
|
|
|
|
AT: 50,
|
2018-08-10 14:09:21 +02:00
|
|
|
|
META: 91,
|
|
|
|
|
META_RIGHT: 93
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
2018-05-12 19:37:44 +02:00
|
|
|
|
_converse.IQ_TIMEOUT = 20000;
|
|
|
|
|
|
2017-09-22 18:55:02 +02:00
|
|
|
|
_converse.CONNECTION_STATUS = {
|
2016-02-29 21:54:34 +01:00
|
|
|
|
0: 'ERROR',
|
|
|
|
|
1: 'CONNECTING',
|
|
|
|
|
2: 'CONNFAIL',
|
|
|
|
|
3: 'AUTHENTICATING',
|
|
|
|
|
4: 'AUTHFAIL',
|
|
|
|
|
5: 'CONNECTED',
|
|
|
|
|
6: 'DISCONNECTED',
|
|
|
|
|
7: 'DISCONNECTING',
|
|
|
|
|
8: 'ATTACHED',
|
2017-09-22 18:55:02 +02:00
|
|
|
|
9: 'REDIRECT',
|
2017-12-13 23:04:49 +01:00
|
|
|
|
10: 'RECONNECTING',
|
2016-02-29 21:54:34 +01:00
|
|
|
|
};
|
|
|
|
|
|
2018-04-17 15:17:39 +02:00
|
|
|
|
_converse.SUCCESS = 'success';
|
|
|
|
|
_converse.FAILURE = 'failure';
|
|
|
|
|
|
2017-08-16 11:16:22 +02:00
|
|
|
|
_converse.DEFAULT_IMAGE_TYPE = 'image/png';
|
|
|
|
|
_converse.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==";
|
2017-01-23 11:15:07 +01:00
|
|
|
|
|
2018-06-01 12:21:41 +02:00
|
|
|
|
_converse.TIMEOUTS = { // Set as module attr so that we can override in tests.
|
|
|
|
|
'PAUSED': 10000,
|
|
|
|
|
'INACTIVE': 90000
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// XEP-0085 Chat states
|
|
|
|
|
// http://xmpp.org/extensions/xep-0085.html
|
|
|
|
|
_converse.INACTIVE = 'inactive';
|
|
|
|
|
_converse.ACTIVE = 'active';
|
|
|
|
|
_converse.COMPOSING = 'composing';
|
|
|
|
|
_converse.PAUSED = 'paused';
|
|
|
|
|
_converse.GONE = 'gone';
|
|
|
|
|
|
2018-08-28 14:58:33 +02:00
|
|
|
|
|
|
|
|
|
// Chat types
|
|
|
|
|
_converse.PRIVATE_CHAT_TYPE = 'chatbox';
|
|
|
|
|
_converse.CHATROOMS_TYPE = 'chatroom';
|
|
|
|
|
_converse.HEADLINES_TYPE = 'headline';
|
|
|
|
|
_converse.CONTROLBOX_TYPE = 'controlbox';
|
|
|
|
|
|
|
|
|
|
|
2018-06-01 12:21:41 +02:00
|
|
|
|
// Default configuration values
|
|
|
|
|
// ----------------------------
|
|
|
|
|
_converse.default_settings = {
|
|
|
|
|
allow_non_roster_messaging: false,
|
|
|
|
|
animate: true,
|
|
|
|
|
authentication: 'login', // Available values are "login", "prebind", "anonymous" and "external".
|
|
|
|
|
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: true,
|
|
|
|
|
auto_xa: 0, // Seconds after which user status is set to 'xa'
|
|
|
|
|
blacklisted_plugins: [],
|
|
|
|
|
bosh_service_url: undefined,
|
|
|
|
|
connection_options: {},
|
|
|
|
|
credentials_url: null, // URL from where login credentials can be fetched
|
|
|
|
|
csi_waiting_time: 0, // Support for XEP-0352. Seconds before client is considered idle and CSI is sent out.
|
|
|
|
|
debug: false,
|
|
|
|
|
default_state: 'online',
|
|
|
|
|
expose_rid_and_sid: false,
|
|
|
|
|
geouri_regex: /https:\/\/www.openstreetmap.org\/.*#map=[0-9]+\/([\-0-9.]+)\/([\-0-9.]+)\S*/g,
|
|
|
|
|
geouri_replacement: 'https://www.openstreetmap.org/?mlat=$1&mlon=$2#map=18/$1/$2',
|
|
|
|
|
jid: undefined,
|
|
|
|
|
keepalive: true,
|
|
|
|
|
locales_url: 'locale/{{{locale}}}/LC_MESSAGES/converse.json',
|
|
|
|
|
locales: [
|
|
|
|
|
'af', 'ar', 'bg', 'ca', 'de', 'es', 'eu', 'en', 'fr', 'he',
|
|
|
|
|
'hu', 'id', 'it', 'ja', 'nb', 'nl',
|
|
|
|
|
'pl', 'pt_BR', 'ru', 'tr', 'uk', 'zh_CN', 'zh_TW'
|
|
|
|
|
],
|
|
|
|
|
message_carbons: true,
|
|
|
|
|
nickname: undefined,
|
|
|
|
|
password: undefined,
|
|
|
|
|
prebind_url: null,
|
|
|
|
|
priority: 0,
|
|
|
|
|
rid: undefined,
|
|
|
|
|
root: window.document,
|
|
|
|
|
sid: undefined,
|
|
|
|
|
strict_plugin_dependencies: false,
|
|
|
|
|
trusted: true,
|
|
|
|
|
view_mode: 'overlayed', // Choices are 'overlayed', 'fullscreen', 'mobile'
|
|
|
|
|
websocket_url: undefined,
|
|
|
|
|
whitelisted_plugins: []
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2017-10-26 15:47:49 +02:00
|
|
|
|
_converse.log = function (message, level, style='') {
|
2017-09-26 18:27:41 +02:00
|
|
|
|
/* Logs messages to the browser's developer console.
|
|
|
|
|
*
|
|
|
|
|
* Parameters:
|
|
|
|
|
* (String) message - The message to be logged.
|
|
|
|
|
* (Integer) level - The loglevel which allows for filtering of log
|
|
|
|
|
* messages.
|
2017-10-24 12:56:35 +02:00
|
|
|
|
*
|
2017-09-26 18:27:41 +02:00
|
|
|
|
* Available loglevels are 0 for 'debug', 1 for 'info', 2 for 'warn',
|
|
|
|
|
* 3 for 'error' and 4 for 'fatal'.
|
|
|
|
|
*
|
2017-10-24 12:56:35 +02:00
|
|
|
|
* When using the 'error' or 'warn' loglevels, a full stacktrace will be
|
2017-09-26 18:27:41 +02:00
|
|
|
|
* logged as well.
|
|
|
|
|
*/
|
2017-10-26 15:47:49 +02:00
|
|
|
|
if (level === Strophe.LogLevel.ERROR || level === Strophe.LogLevel.FATAL) {
|
|
|
|
|
style = style || 'color: maroon';
|
|
|
|
|
}
|
2017-07-21 12:38:16 +02:00
|
|
|
|
if (message instanceof Error) {
|
|
|
|
|
message = message.stack;
|
2018-07-02 12:52:19 +02:00
|
|
|
|
} else if (_.isElement(message)) {
|
|
|
|
|
message = message.outerHTML;
|
2017-07-21 12:38:16 +02:00
|
|
|
|
}
|
2017-10-26 15:47:49 +02:00
|
|
|
|
const prefix = style ? '%c' : '';
|
2017-08-21 12:52:18 +02:00
|
|
|
|
const logger = _.assign({
|
|
|
|
|
'debug': _.get(console, 'log') ? console.log.bind(console) : _.noop,
|
|
|
|
|
'error': _.get(console, 'log') ? console.log.bind(console) : _.noop,
|
|
|
|
|
'info': _.get(console, 'log') ? console.log.bind(console) : _.noop,
|
|
|
|
|
'warn': _.get(console, 'log') ? console.log.bind(console) : _.noop
|
2017-07-10 10:43:13 +02:00
|
|
|
|
}, console);
|
2017-07-05 11:33:55 +02:00
|
|
|
|
if (level === Strophe.LogLevel.ERROR) {
|
2018-01-29 11:16:18 +01:00
|
|
|
|
logger.error(`${prefix} ERROR: ${message}`, style);
|
2017-07-05 11:33:55 +02:00
|
|
|
|
} else if (level === Strophe.LogLevel.WARN) {
|
2017-10-25 17:09:35 +02:00
|
|
|
|
if (_converse.debug) {
|
2017-10-26 15:47:49 +02:00
|
|
|
|
logger.warn(`${prefix} ${moment().format()} WARNING: ${message}`, style);
|
2017-10-25 17:09:35 +02:00
|
|
|
|
}
|
2017-07-05 11:33:55 +02:00
|
|
|
|
} else if (level === Strophe.LogLevel.FATAL) {
|
2018-01-29 11:16:18 +01:00
|
|
|
|
logger.error(`${prefix} FATAL: ${message}`, style);
|
2017-07-05 11:33:55 +02:00
|
|
|
|
} else if (_converse.debug) {
|
|
|
|
|
if (level === Strophe.LogLevel.DEBUG) {
|
2017-10-26 15:47:49 +02:00
|
|
|
|
logger.debug(`${prefix} ${moment().format()} DEBUG: ${message}`, style);
|
2017-07-10 10:43:13 +02:00
|
|
|
|
} else {
|
2017-10-26 15:47:49 +02:00
|
|
|
|
logger.info(`${prefix} ${moment().format()} INFO: ${message}`, style);
|
2016-02-29 21:54:34 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2017-10-25 17:09:35 +02:00
|
|
|
|
Strophe.log = function (level, msg) { _converse.log(level+' '+msg, level); };
|
|
|
|
|
Strophe.error = function (msg) { _converse.log(msg, Strophe.LogLevel.ERROR); };
|
|
|
|
|
|
|
|
|
|
|
2017-09-26 18:27:41 +02:00
|
|
|
|
_converse.__ = function (str) {
|
|
|
|
|
/* Translate the given string based on the current locale.
|
|
|
|
|
*
|
|
|
|
|
* Parameters:
|
|
|
|
|
* (String) str - The string to translate.
|
|
|
|
|
*/
|
2017-10-24 12:56:35 +02:00
|
|
|
|
if (_.isUndefined(i18n)) {
|
2017-09-26 18:27:41 +02:00
|
|
|
|
return str;
|
|
|
|
|
}
|
2017-10-24 12:56:35 +02:00
|
|
|
|
return i18n.translate.apply(i18n, arguments);
|
2017-09-27 00:33:58 +02:00
|
|
|
|
}
|
2017-09-26 18:27:41 +02:00
|
|
|
|
|
2017-10-24 12:56:35 +02:00
|
|
|
|
const __ = _converse.__;
|
2017-09-26 18:27:41 +02:00
|
|
|
|
|
2017-07-11 15:22:05 +02:00
|
|
|
|
const PROMISES = [
|
2017-08-16 12:29:31 +02:00
|
|
|
|
'initialized',
|
2017-08-16 12:31:17 +02:00
|
|
|
|
'connectionInitialized',
|
2017-07-11 15:22:05 +02:00
|
|
|
|
'pluginsInitialized',
|
|
|
|
|
'statusInitialized'
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
function addPromise (promise) {
|
|
|
|
|
/* Private function, used to add a new promise to the ones already
|
|
|
|
|
* available via the `waitUntil` api method.
|
|
|
|
|
*/
|
2018-02-19 10:34:52 +01:00
|
|
|
|
_converse.promises[promise] = u.getResolveablePromise();
|
2017-07-11 15:22:05 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_converse.emit = function (name) {
|
|
|
|
|
/* Event emitter and promise resolver */
|
|
|
|
|
_converse.trigger.apply(this, arguments);
|
|
|
|
|
const promise = _converse.promises[name];
|
|
|
|
|
if (!_.isUndefined(promise)) {
|
|
|
|
|
promise.resolve();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2018-07-02 23:29:57 +02:00
|
|
|
|
_converse.isSingleton = function () {
|
|
|
|
|
return _.includes(['mobile', 'fullscreen', 'embedded'], _converse.view_mode);
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-31 21:47:27 +01:00
|
|
|
|
_converse.router = new Backbone.Router();
|
|
|
|
|
|
|
|
|
|
|
2016-12-20 10:30:20 +01:00
|
|
|
|
_converse.initialize = function (settings, callback) {
|
2017-01-26 15:49:02 +01:00
|
|
|
|
settings = !_.isUndefined(settings) ? settings : {};
|
2018-02-19 10:34:52 +01:00
|
|
|
|
const init_promise = u.getResolveablePromise();
|
2016-12-06 12:07:52 +01:00
|
|
|
|
|
2017-07-11 15:22:05 +02:00
|
|
|
|
_.each(PROMISES, addPromise);
|
|
|
|
|
|
2017-08-16 11:16:22 +02:00
|
|
|
|
if (!_.isUndefined(_converse.connection)) {
|
2016-12-20 10:30:20 +01:00
|
|
|
|
// 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.
|
2017-09-17 21:50:59 +02:00
|
|
|
|
Backbone.history.stop();
|
2017-12-19 20:17:38 +01:00
|
|
|
|
_converse.chatboxviews.closeAllChatBoxes();
|
2018-05-22 21:50:28 +02:00
|
|
|
|
if (_converse.bookmarks) {
|
|
|
|
|
_converse.bookmarks.reset();
|
|
|
|
|
}
|
2017-08-29 12:29:16 +02:00
|
|
|
|
delete _converse.controlboxtoggle;
|
2017-12-19 20:17:38 +01:00
|
|
|
|
delete _converse.chatboxviews;
|
2016-12-20 10:30:20 +01:00
|
|
|
|
_converse.connection.reset();
|
|
|
|
|
_converse.stopListening();
|
2018-06-07 14:30:10 +02:00
|
|
|
|
_converse.tearDown();
|
2018-08-23 09:41:39 +02:00
|
|
|
|
delete _converse.config;
|
|
|
|
|
_converse.initClientConfig();
|
2018-08-22 14:31:23 +02:00
|
|
|
|
_converse.off();
|
2016-12-06 12:07:52 +01:00
|
|
|
|
}
|
|
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
|
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/
|
2018-06-06 09:29:32 +02:00
|
|
|
|
_converse.unloadevent = 'pagehide';
|
2016-02-16 08:46:47 +01:00
|
|
|
|
} else if ('onbeforeunload' in window) {
|
2018-06-06 09:29:32 +02:00
|
|
|
|
_converse.unloadevent = 'beforeunload';
|
2016-02-16 08:46:47 +01:00
|
|
|
|
} else if ('onunload' in window) {
|
2018-06-06 09:29:32 +02:00
|
|
|
|
_converse.unloadevent = 'unload';
|
2016-02-16 08:46:47 +01:00
|
|
|
|
}
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-24 12:56:35 +02:00
|
|
|
|
/* Localisation */
|
|
|
|
|
if (!_.isUndefined(i18n)) {
|
|
|
|
|
i18n.setLocales(settings.i18n, _converse);
|
|
|
|
|
} else {
|
|
|
|
|
_converse.locale = 'en';
|
|
|
|
|
}
|
2017-09-27 00:33:58 +02:00
|
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
|
// 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
|
|
|
|
|
// ----------------------
|
|
|
|
|
|
2018-01-22 09:53:35 +01:00
|
|
|
|
this.generateResource = () => `/converse.js-${Math.floor(Math.random()*139749528).toString()}`;
|
2016-02-16 14:31:46 +01:00
|
|
|
|
|
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...
|
2018-02-21 14:40:41 +01:00
|
|
|
|
_converse.xmppstatus.set('status', _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;
|
|
|
|
|
}
|
2018-02-21 14:40:41 +01:00
|
|
|
|
const stat = _converse.xmppstatus.get('status');
|
2016-12-20 10:30:20 +01:00
|
|
|
|
_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;
|
2018-02-21 14:40:41 +01:00
|
|
|
|
_converse.xmppstatus.set('status', 'away');
|
2016-12-20 10:30:20 +01:00
|
|
|
|
} 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;
|
2018-02-21 14:40:41 +01:00
|
|
|
|
_converse.xmppstatus.set('status', '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?
|
2017-04-11 11:14:26 +02:00
|
|
|
|
window.addEventListener('click', _converse.onUserActivity);
|
|
|
|
|
window.addEventListener('focus', _converse.onUserActivity);
|
|
|
|
|
window.addEventListener('keypress', _converse.onUserActivity);
|
|
|
|
|
window.addEventListener('mousemove', _converse.onUserActivity);
|
2018-06-06 09:29:32 +02:00
|
|
|
|
const options = {'once': true, 'passive': true};
|
|
|
|
|
window.addEventListener(_converse.unloadevent, _converse.onUserActivity, options);
|
2016-12-20 10:30:20 +01:00
|
|
|
|
_converse.everySecondTrigger = window.setInterval(_converse.onEverySecond, 1000);
|
2016-02-16 08:46:47 +01:00
|
|
|
|
};
|
2016-02-29 21:54:34 +01:00
|
|
|
|
|
2017-09-22 18:55:02 +02:00
|
|
|
|
this.setConnectionStatus = function (connection_status, message) {
|
2017-09-08 14:24:38 +02:00
|
|
|
|
_converse.connfeedback.set({
|
2017-09-22 18:55:02 +02:00
|
|
|
|
'connection_status': connection_status,
|
2017-09-08 14:24:38 +02:00
|
|
|
|
'message': message
|
2016-08-23 10:00:41 +02:00
|
|
|
|
});
|
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
|
|
|
|
|
*/
|
2017-06-12 20:30:58 +02:00
|
|
|
|
const pres = $pres({to: jid, type: "unsubscribed"});
|
2016-02-16 08:46:47 +01:00
|
|
|
|
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.');
|
2017-09-22 18:55:02 +02:00
|
|
|
|
_converse.setConnectionStatus(
|
|
|
|
|
Strophe.Status.RECONNECTING,
|
2016-11-02 13:44:56 +01:00
|
|
|
|
__('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;
|
2018-06-07 14:30:10 +02:00
|
|
|
|
_converse.tearDown();
|
2016-12-20 10:30:20 +01:00
|
|
|
|
_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();
|
2018-06-07 14:30:10 +02:00
|
|
|
|
_converse.tearDown();
|
2018-08-31 10:17:24 +02:00
|
|
|
|
_converse.clearSession();
|
2016-12-20 10:30:20 +01:00
|
|
|
|
_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.
|
|
|
|
|
*/
|
2017-09-08 14:24:38 +02:00
|
|
|
|
const reason = _converse.disconnection_reason;
|
|
|
|
|
|
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 ||
|
2017-09-08 14:24:38 +02:00
|
|
|
|
(!_.isUndefined(reason) && reason === _.get(Strophe, 'ErrorCondition.NO_AUTH_MECH')) ||
|
|
|
|
|
reason === "host-unknown" ||
|
|
|
|
|
reason === "remote-connection-failed" ||
|
2016-12-20 10:30:20 +01:00
|
|
|
|
!_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
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2017-09-08 14:24:38 +02:00
|
|
|
|
this.onConnectStatusChanged = function (status, message) {
|
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.
|
|
|
|
|
*/
|
2017-09-22 18:55:02 +02:00
|
|
|
|
_converse.log(`Status changed to: ${_converse.CONNECTION_STATUS[status]}`);
|
2016-02-16 08:46:47 +01:00
|
|
|
|
if (status === Strophe.Status.CONNECTED || status === Strophe.Status.ATTACHED) {
|
2017-09-22 18:55:02 +02:00
|
|
|
|
_converse.setConnectionStatus(status);
|
2016-02-16 08:46:47 +01:00
|
|
|
|
// 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) {
|
2017-09-08 14:24:38 +02:00
|
|
|
|
_converse.setDisconnectionCause(status, message);
|
2016-12-20 10:30:20 +01:00
|
|
|
|
_converse.onDisconnected();
|
2016-02-16 08:46:47 +01:00
|
|
|
|
} else if (status === Strophe.Status.ERROR) {
|
2017-09-22 18:55:02 +02:00
|
|
|
|
_converse.setConnectionStatus(
|
|
|
|
|
status,
|
2016-08-23 10:00:41 +02:00
|
|
|
|
__('An error occurred while connecting to the chat server.')
|
|
|
|
|
);
|
2016-02-16 08:46:47 +01:00
|
|
|
|
} else if (status === Strophe.Status.CONNECTING) {
|
2017-09-22 18:55:02 +02:00
|
|
|
|
_converse.setConnectionStatus(status);
|
2016-02-16 08:46:47 +01:00
|
|
|
|
} else if (status === Strophe.Status.AUTHENTICATING) {
|
2017-09-22 18:55:02 +02:00
|
|
|
|
_converse.setConnectionStatus(status);
|
2016-02-16 08:46:47 +01:00
|
|
|
|
} else if (status === Strophe.Status.AUTHFAIL) {
|
2017-09-22 10:45:03 +02:00
|
|
|
|
if (!message) {
|
|
|
|
|
message = __('Your Jabber ID and/or password is incorrect. Please try again.');
|
|
|
|
|
}
|
2017-09-22 18:55:02 +02:00
|
|
|
|
_converse.setConnectionStatus(status, message);
|
2017-09-08 14:24:38 +02:00
|
|
|
|
_converse.setDisconnectionCause(status, message, 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) {
|
2017-09-08 14:24:38 +02:00
|
|
|
|
let feedback = message;
|
|
|
|
|
if (message === "host-unknown" || message == "remote-connection-failed") {
|
2018-01-17 17:02:51 +01:00
|
|
|
|
feedback = __("Sorry, we could not connect to the XMPP host with domain: %1$s",
|
|
|
|
|
`\"${Strophe.getDomainFromJid(_converse.connection.jid)}\"`);
|
2017-09-08 14:24:38 +02:00
|
|
|
|
} else if (!_.isUndefined(message) && message === _.get(Strophe, 'ErrorCondition.NO_AUTH_MECH')) {
|
|
|
|
|
feedback = __("The XMPP server did not offer a supported authentication mechanism");
|
|
|
|
|
}
|
2017-09-22 18:55:02 +02:00
|
|
|
|
_converse.setConnectionStatus(status, feedback);
|
2017-09-08 14:24:38 +02:00
|
|
|
|
_converse.setDisconnectionCause(status, message);
|
2016-12-04 15:14:40 +01:00
|
|
|
|
} else if (status === Strophe.Status.DISCONNECTING) {
|
2017-09-08 14:24:38 +02:00
|
|
|
|
_converse.setDisconnectionCause(status, message);
|
2016-02-16 08:46:47 +01:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
this.incrementMsgCounter = function () {
|
|
|
|
|
this.msg_counter += 1;
|
2017-06-12 20:30:58 +02:00
|
|
|
|
const unreadMsgCount = this.msg_counter;
|
2018-02-09 16:02:56 +01:00
|
|
|
|
let title = document.title;
|
|
|
|
|
if (_.isNil(title)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (title.search(/^Messages \(\d+\) /) === -1) {
|
|
|
|
|
title = `Messages (${unreadMsgCount}) ${title}`;
|
2017-05-03 09:06:28 +02:00
|
|
|
|
} else {
|
2018-02-09 16:02:56 +01:00
|
|
|
|
title = title.replace(/^Messages \(\d+\) /, `Messages (${unreadMsgCount})`);
|
2017-05-03 09:06:28 +02:00
|
|
|
|
}
|
2016-02-16 08:46:47 +01:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
this.clearMsgCounter = function () {
|
|
|
|
|
this.msg_counter = 0;
|
2018-02-09 16:02:56 +01:00
|
|
|
|
let title = document.title;
|
|
|
|
|
if (_.isNil(title)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (title.search(/^Messages \(\d+\) /) !== -1) {
|
|
|
|
|
title = title.replace(/^Messages \(\d+\) /, "");
|
2017-05-03 09:06:28 +02:00
|
|
|
|
}
|
2016-02-16 08:46:47 +01:00
|
|
|
|
};
|
|
|
|
|
|
2018-02-22 11:06:30 +01:00
|
|
|
|
this.initStatus = (reconnecting) => {
|
|
|
|
|
// If there's no xmppstatus obj, then we were never connected to
|
|
|
|
|
// begin with, so we set reconnecting to false.
|
|
|
|
|
reconnecting = _.isUndefined(_converse.xmppstatus) ? false : reconnecting;
|
|
|
|
|
if (reconnecting) {
|
|
|
|
|
_converse.onStatusInitialized(reconnecting);
|
|
|
|
|
} else {
|
2018-08-22 22:20:15 +02:00
|
|
|
|
const id = `converse.xmppstatus-${_converse.bare_jid}`;
|
|
|
|
|
this.xmppstatus = new this.XMPPStatus({'id': id});
|
|
|
|
|
this.xmppstatus.browserStorage = new Backbone.BrowserStorage.session(id);
|
2017-07-10 21:14:48 +02:00
|
|
|
|
this.xmppstatus.fetch({
|
2018-08-22 22:20:15 +02:00
|
|
|
|
'success': _.partial(_converse.onStatusInitialized, reconnecting),
|
|
|
|
|
'error': _.partial(_converse.onStatusInitialized, reconnecting)
|
2017-07-10 21:14:48 +02:00
|
|
|
|
});
|
2018-02-22 11:06:30 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2016-02-16 08:46:47 +01:00
|
|
|
|
|
2018-08-23 09:41:39 +02:00
|
|
|
|
this.initClientConfig = function () {
|
|
|
|
|
/* The client config refers to configuration of the client which is
|
|
|
|
|
* independent of any particular user.
|
|
|
|
|
* What this means is that config values need to persist across
|
|
|
|
|
* user sessions.
|
|
|
|
|
*/
|
|
|
|
|
const id = b64_sha1('converse.client-config');
|
|
|
|
|
_converse.config = new Backbone.Model({
|
2018-08-22 22:20:15 +02:00
|
|
|
|
'id': id,
|
|
|
|
|
'trusted': _converse.trusted && true || false,
|
|
|
|
|
'storage': _converse.trusted ? 'local' : 'session'
|
|
|
|
|
});
|
2018-08-23 09:41:39 +02:00
|
|
|
|
_converse.config.browserStorage = new Backbone.BrowserStorage.session(id);
|
|
|
|
|
_converse.config.fetch();
|
|
|
|
|
_converse.emit('clientConfigInitialized');
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
this.initSession = function () {
|
|
|
|
|
const id = b64_sha1('converse.bosh-session');
|
|
|
|
|
_converse.session = new Backbone.Model({'id': id});
|
2018-06-06 11:07:59 +02:00
|
|
|
|
_converse.session.browserStorage = new Backbone.BrowserStorage.session(id);
|
2017-08-16 12:31:17 +02:00
|
|
|
|
_converse.session.fetch();
|
2018-05-28 11:41:02 +02:00
|
|
|
|
_converse.emit('sessionInitialized');
|
2016-02-16 08:46:47 +01:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
this.clearSession = function () {
|
2018-08-23 09:41:39 +02:00
|
|
|
|
if (!_converse.config.get('trusted')) {
|
2018-05-18 12:21:02 +02:00
|
|
|
|
window.localStorage.clear();
|
|
|
|
|
window.sessionStorage.clear();
|
2018-05-18 12:44:27 +02:00
|
|
|
|
} else if (!_.isUndefined(this.session) && this.session.browserStorage) {
|
|
|
|
|
this.session.browserStorage._clear();
|
2018-05-18 12:21:02 +02:00
|
|
|
|
}
|
2018-05-18 12:44:27 +02:00
|
|
|
|
_converse.emit('clearSession');
|
2016-02-16 08:46:47 +01:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
this.logOut = function () {
|
2017-06-23 18:21:00 +02:00
|
|
|
|
_converse.clearSession();
|
2016-12-20 10:30:20 +01:00
|
|
|
|
_converse.setDisconnectionCause(_converse.LOGOUT, undefined, true);
|
|
|
|
|
if (!_.isUndefined(_converse.connection)) {
|
|
|
|
|
_converse.connection.disconnect();
|
2017-06-23 18:21:00 +02:00
|
|
|
|
} else {
|
2018-06-07 14:30:10 +02:00
|
|
|
|
_converse.tearDown();
|
2016-04-13 11:15:15 +02:00
|
|
|
|
}
|
2018-02-22 13:26:06 +01:00
|
|
|
|
// Recreate all the promises
|
|
|
|
|
_.each(_.keys(_converse.promises), addPromise);
|
|
|
|
|
|
2016-12-20 10:30:20 +01:00
|
|
|
|
_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).
|
2017-06-12 20:30:58 +02:00
|
|
|
|
let state;
|
|
|
|
|
const event_map = {
|
|
|
|
|
'focus': "visible",
|
|
|
|
|
'focusin': "visible",
|
|
|
|
|
'pageshow': "visible",
|
|
|
|
|
'blur': "hidden",
|
|
|
|
|
'focusout': "hidden",
|
|
|
|
|
'pagehide': "hidden"
|
|
|
|
|
};
|
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;
|
2017-07-10 17:46:22 +02:00
|
|
|
|
_converse.emit('windowStateChanged', {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
|
2017-06-12 20:30:58 +02:00
|
|
|
|
let hidden = "hidden";
|
2016-06-16 18:14:22 +02:00
|
|
|
|
// 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
|
|
|
|
}
|
2018-05-30 17:16:26 +02:00
|
|
|
|
_converse.emit('registeredGlobalEventHandlers');
|
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;
|
|
|
|
|
}
|
2017-06-12 20:30:58 +02:00
|
|
|
|
const carbons_iq = new Strophe.Builder('iq', {
|
2018-03-27 12:14:18 +02:00
|
|
|
|
'from': this.connection.jid,
|
|
|
|
|
'id': 'enablecarbons',
|
|
|
|
|
'type': 'set'
|
2016-02-16 08:46:47 +01:00
|
|
|
|
})
|
|
|
|
|
.c('enable', {xmlns: Strophe.NS.CARBONS});
|
2017-07-10 17:46:22 +02:00
|
|
|
|
this.connection.addHandler((iq) => {
|
2016-11-24 02:07:32 +01:00
|
|
|
|
if (iq.querySelectorAll('error').length > 0) {
|
2017-07-05 11:33:55 +02:00
|
|
|
|
_converse.log(
|
2018-07-03 21:26:18 +02:00
|
|
|
|
'An error occurred while trying to enable message carbons.',
|
|
|
|
|
Strophe.LogLevel.WARN);
|
2016-02-16 08:46:47 +01:00
|
|
|
|
} else {
|
2018-06-06 11:07:59 +02:00
|
|
|
|
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
|
|
|
|
}
|
2017-07-10 17:46:22 +02:00
|
|
|
|
}, null, "iq", null, "enablecarbons");
|
2016-02-16 08:46:47 +01:00
|
|
|
|
this.connection.send(carbons_iq);
|
|
|
|
|
};
|
|
|
|
|
|
2016-09-21 15:00:09 +02:00
|
|
|
|
|
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) {
|
2018-05-04 23:27:08 +02:00
|
|
|
|
_converse.emit('statusInitialized', reconnecting);
|
2018-02-22 11:06:30 +01:00
|
|
|
|
if (reconnecting) {
|
|
|
|
|
_converse.emit('reconnected');
|
|
|
|
|
} else {
|
2017-07-10 21:14:48 +02:00
|
|
|
|
init_promise.resolve();
|
2016-12-20 10:30:20 +01:00
|
|
|
|
_converse.emit('initialized');
|
2018-02-22 11:06:30 +01:00
|
|
|
|
_converse.emit('connected');
|
2016-03-16 12:16:32 +01:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2018-08-22 22:20:15 +02: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);
|
2018-08-22 22:20:15 +02:00
|
|
|
|
_converse.emit('setUserJID');
|
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.
|
|
|
|
|
*/
|
2018-02-22 11:06:30 +01:00
|
|
|
|
_converse.connection.flush(); // Solves problem of returned PubSub BOSH response not received by browser
|
2018-08-22 22:20:15 +02:00
|
|
|
|
_converse.setUserJID();
|
2018-08-23 09:41:39 +02:00
|
|
|
|
_converse.initSession();
|
2016-12-20 10:30:20 +01:00
|
|
|
|
_converse.enableCarbons();
|
2018-02-22 11:06:30 +01:00
|
|
|
|
_converse.initStatus(reconnecting)
|
2016-11-02 14:13:49 +01:00
|
|
|
|
};
|
|
|
|
|
|
2016-06-21 17:06:51 +02:00
|
|
|
|
|
2017-09-08 14:24:38 +02:00
|
|
|
|
this.ConnectionFeedback = Backbone.Model.extend({
|
2017-09-22 18:55:02 +02:00
|
|
|
|
defaults: {
|
2017-12-13 23:04:49 +01:00
|
|
|
|
'connection_status': Strophe.Status.DISCONNECTED,
|
2017-09-22 18:55:02 +02:00
|
|
|
|
'message': ''
|
2017-09-08 14:24:38 +02:00
|
|
|
|
},
|
|
|
|
|
|
2017-09-22 18:55:02 +02:00
|
|
|
|
initialize () {
|
|
|
|
|
this.on('change', () => {
|
|
|
|
|
_converse.emit('connfeedback', _converse.connfeedback);
|
2017-09-08 14:24:38 +02:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
this.connfeedback = new this.ConnectionFeedback();
|
|
|
|
|
|
2018-04-30 10:38:02 +02:00
|
|
|
|
|
2018-05-15 09:47:52 +02:00
|
|
|
|
this.XMPPStatus = Backbone.Model.extend({
|
2018-02-23 22:15:04 +01:00
|
|
|
|
|
|
|
|
|
defaults () {
|
|
|
|
|
return {
|
|
|
|
|
"jid": _converse.bare_jid,
|
2018-08-07 08:52:28 +02:00
|
|
|
|
"status": _converse.default_state
|
2018-02-23 22:15:04 +01:00
|
|
|
|
}
|
2018-02-21 14:40:41 +01:00
|
|
|
|
},
|
2018-02-07 17:30:44 +01:00
|
|
|
|
|
2017-07-10 17:46:22 +02:00
|
|
|
|
initialize () {
|
2018-05-08 19:24:50 +02:00
|
|
|
|
this.vcard = _converse.vcards.findWhere({'jid': this.get('jid')});
|
|
|
|
|
if (_.isNil(this.vcard)) {
|
|
|
|
|
this.vcard = _converse.vcards.create({'jid': this.get('jid')});
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-21 14:40:41 +01:00
|
|
|
|
this.on('change:status', (item) => {
|
|
|
|
|
const status = this.get('status');
|
|
|
|
|
this.sendPresence(status);
|
|
|
|
|
_converse.emit('statusChanged', status);
|
2016-02-28 20:24:06 +01:00
|
|
|
|
});
|
2018-02-21 14:40:41 +01:00
|
|
|
|
|
|
|
|
|
this.on('change:status_message', () => {
|
|
|
|
|
const status_message = this.get('status_message');
|
|
|
|
|
this.sendPresence(this.get('status'), status_message);
|
|
|
|
|
_converse.emit('statusMessageChanged', status_message);
|
2017-07-10 17:46:22 +02:00
|
|
|
|
});
|
2016-02-16 08:46:47 +01:00
|
|
|
|
},
|
|
|
|
|
|
2017-07-10 17:46:22 +02:00
|
|
|
|
constructPresence (type, status_message) {
|
2017-06-12 20:30:58 +02:00
|
|
|
|
let presence;
|
2016-12-20 10:30:20 +01:00
|
|
|
|
type = _.isString(type) ? type : (this.get('status') || _converse.default_state);
|
2018-03-02 19:35:06 +01:00
|
|
|
|
status_message = _.isString(status_message) ? status_message : this.get('status_message');
|
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
|
|
|
|
},
|
|
|
|
|
|
2017-07-10 17:46:22 +02:00
|
|
|
|
sendPresence (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
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
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) {
|
2017-07-10 10:43:13 +02:00
|
|
|
|
this.connection.xmlInput = function (body) {
|
2017-10-26 15:47:49 +02:00
|
|
|
|
_converse.log(body.outerHTML, Strophe.LogLevel.DEBUG, 'color: darkgoldenrod');
|
2017-07-10 10:43:13 +02:00
|
|
|
|
};
|
|
|
|
|
this.connection.xmlOutput = function (body) {
|
2017-10-26 15:47:49 +02:00
|
|
|
|
_converse.log(body.outerHTML, Strophe.LogLevel.DEBUG, 'color: darkcyan');
|
2017-07-10 10:43:13 +02:00
|
|
|
|
};
|
2016-02-16 08:46:47 +01:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2017-07-10 21:14:48 +02:00
|
|
|
|
this.fetchLoginCredentials = () =>
|
|
|
|
|
new Promise((resolve, reject) => {
|
|
|
|
|
const xhr = new XMLHttpRequest();
|
|
|
|
|
xhr.open('GET', _converse.credentials_url, true);
|
|
|
|
|
xhr.setRequestHeader('Accept', "application/json, text/javascript");
|
|
|
|
|
xhr.onload = function() {
|
|
|
|
|
if (xhr.status >= 200 && xhr.status < 400) {
|
|
|
|
|
const data = JSON.parse(xhr.responseText);
|
|
|
|
|
resolve({
|
|
|
|
|
'jid': data.jid,
|
|
|
|
|
'password': data.password
|
|
|
|
|
});
|
|
|
|
|
} else {
|
|
|
|
|
xhr.onerror();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
xhr.onerror = function () {
|
|
|
|
|
delete _converse.connection;
|
2017-08-16 11:16:22 +02:00
|
|
|
|
_converse.emit('noResumeableSession', this);
|
2017-07-10 21:14:48 +02:00
|
|
|
|
reject(xhr.responseText);
|
|
|
|
|
};
|
|
|
|
|
xhr.send();
|
|
|
|
|
});
|
2016-03-31 10:37:28 +02:00
|
|
|
|
|
2016-02-16 08:46:47 +01:00
|
|
|
|
this.startNewBOSHSession = function () {
|
2017-06-12 20:30:58 +02:00
|
|
|
|
const xhr = new XMLHttpRequest();
|
2017-02-25 23:26:14 +01:00
|
|
|
|
xhr.open('GET', _converse.prebind_url, true);
|
|
|
|
|
xhr.setRequestHeader('Accept', "application/json, text/javascript");
|
|
|
|
|
xhr.onload = function() {
|
|
|
|
|
if (xhr.status >= 200 && xhr.status < 400) {
|
2017-06-12 20:30:58 +02:00
|
|
|
|
const data = JSON.parse(xhr.responseText);
|
2017-02-25 23:26:14 +01:00
|
|
|
|
_converse.connection.attach(
|
|
|
|
|
data.jid, data.sid, data.rid,
|
|
|
|
|
_converse.onConnectStatusChanged);
|
|
|
|
|
} else {
|
|
|
|
|
xhr.onerror();
|
2016-11-07 18:30:10 +01:00
|
|
|
|
}
|
2017-02-25 23:26:14 +01:00
|
|
|
|
};
|
|
|
|
|
xhr.onerror = function () {
|
|
|
|
|
delete _converse.connection;
|
2017-08-16 11:16:22 +02:00
|
|
|
|
_converse.emit('noResumeableSession', this);
|
2017-02-25 23:26:14 +01:00
|
|
|
|
};
|
|
|
|
|
xhr.send();
|
2016-02-16 08:46:47 +01:00
|
|
|
|
};
|
|
|
|
|
|
2017-05-15 16:47:10 +02:00
|
|
|
|
this.restoreBOSHSession = function (jid_is_required) {
|
2017-05-11 10:23:29 +02:00
|
|
|
|
/* Tries to restore a cached BOSH session. */
|
|
|
|
|
if (!this.jid) {
|
2017-06-12 20:30:58 +02:00
|
|
|
|
const msg = "restoreBOSHSession: tried to restore a \"keepalive\" session "+
|
|
|
|
|
"but we don't have the JID for the user!";
|
2017-05-15 16:47:10 +02:00
|
|
|
|
if (jid_is_required) {
|
|
|
|
|
throw new Error(msg);
|
|
|
|
|
} else {
|
|
|
|
|
_converse.log(msg);
|
|
|
|
|
}
|
2017-05-11 10:23:29 +02:00
|
|
|
|
}
|
|
|
|
|
try {
|
|
|
|
|
this.connection.restore(this.jid, this.onConnectStatusChanged);
|
|
|
|
|
return true;
|
|
|
|
|
} catch (e) {
|
2017-08-21 12:52:18 +02:00
|
|
|
|
_converse.log(
|
2017-05-11 10:23:29 +02:00
|
|
|
|
"Could not restore session for jid: "+
|
2017-12-18 13:19:14 +01:00
|
|
|
|
this.jid+" Error message: "+e.message, Strophe.LogLevel.WARN);
|
2018-05-23 04:34:38 +02:00
|
|
|
|
this.clearSession(); // We want to clear presences (see #555)
|
2017-05-11 10:23:29 +02:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2016-11-23 12:53:06 +01:00
|
|
|
|
this.attemptPreboundSession = function (reconnecting) {
|
2017-05-11 10:23:29 +02:00
|
|
|
|
/* Handle session resumption or initialization when prebind is
|
|
|
|
|
* being used.
|
2016-02-16 08:46:47 +01:00
|
|
|
|
*/
|
2017-05-11 10:23:29 +02:00
|
|
|
|
if (!reconnecting) {
|
2017-05-15 16:47:10 +02:00
|
|
|
|
if (this.keepalive && this.restoreBOSHSession(true)) {
|
2017-05-11 10:23:29 +02:00
|
|
|
|
return;
|
2016-02-16 08:46:47 +01:00
|
|
|
|
}
|
2017-05-11 10:23:29 +02:00
|
|
|
|
// No keepalive, or session resumption has failed.
|
|
|
|
|
if (this.jid && this.sid && this.rid) {
|
|
|
|
|
return this.connection.attach(
|
|
|
|
|
this.jid, this.sid, this.rid,
|
|
|
|
|
this.onConnectStatusChanged
|
|
|
|
|
);
|
2016-02-16 08:46:47 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2017-05-11 10:23:29 +02:00
|
|
|
|
if (this.prebind_url) {
|
2016-11-07 18:30:10 +01:00
|
|
|
|
return this.startNewBOSHSession();
|
2016-02-16 08:46:47 +01:00
|
|
|
|
} else {
|
2017-05-11 10:23:29 +02:00
|
|
|
|
throw new Error(
|
|
|
|
|
"attemptPreboundSession: If you use prebind and not keepalive, "+
|
2016-11-07 18:30:10 +01:00
|
|
|
|
"then you MUST supply JID, RID and SID values or a prebind_url.");
|
2016-02-16 08:46:47 +01:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2017-05-11 10:35:45 +02:00
|
|
|
|
this.attemptNonPreboundSession = function (credentials, reconnecting) {
|
|
|
|
|
/* 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
|
|
|
|
|
*/
|
|
|
|
|
if (!reconnecting && this.keepalive && this.restoreBOSHSession()) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2017-09-16 20:02:47 +02:00
|
|
|
|
|
|
|
|
|
if (credentials) {
|
|
|
|
|
// When credentials are passed in, they override prebinding
|
|
|
|
|
// or credentials fetching via HTTP
|
|
|
|
|
this.autoLogin(credentials);
|
|
|
|
|
} else if (this.auto_login) {
|
|
|
|
|
if (this.credentials_url) {
|
2017-07-10 21:14:48 +02:00
|
|
|
|
this.fetchLoginCredentials().then(
|
|
|
|
|
this.autoLogin.bind(this),
|
|
|
|
|
this.autoLogin.bind(this)
|
|
|
|
|
);
|
2017-05-11 10:35:45 +02:00
|
|
|
|
} else if (!this.jid) {
|
|
|
|
|
throw new Error(
|
|
|
|
|
"attemptNonPreboundSession: 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 {
|
2018-05-19 08:22:24 +02:00
|
|
|
|
this.autoLogin(); // Could be ANONYMOUS or EXTERNAL
|
2017-05-11 10:35:45 +02:00
|
|
|
|
}
|
|
|
|
|
} else if (reconnecting) {
|
|
|
|
|
this.autoLogin();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2016-03-31 10:37:28 +02:00
|
|
|
|
this.autoLogin = function (credentials) {
|
|
|
|
|
if (credentials) {
|
2017-05-11 14:33:41 +02:00
|
|
|
|
// If passed in, the credentials come from credentials_url,
|
|
|
|
|
// so we set them on the converse object.
|
2016-03-31 10:37:28 +02:00
|
|
|
|
this.jid = credentials.jid;
|
|
|
|
|
}
|
2018-05-19 08:22:24 +02:00
|
|
|
|
if (this.authentication === _converse.ANONYMOUS || this.authentication === _converse.EXTERNAL) {
|
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
|
|
|
|
}
|
2017-06-23 20:19:38 +02:00
|
|
|
|
if (!this.connection.reconnecting) {
|
|
|
|
|
this.connection.reset();
|
|
|
|
|
}
|
2018-07-28 17:13:13 +02:00
|
|
|
|
this.connection.connect(this.jid.toLowerCase(), null, this.onConnectStatusChanged, BOSH_WAIT);
|
2016-12-20 10:30:20 +01:00
|
|
|
|
} else if (this.authentication === _converse.LOGIN) {
|
2017-06-12 20:30:58 +02:00
|
|
|
|
const password = _.isNil(credentials) ? (_converse.connection.pass || this.password) : credentials.password;
|
2016-08-18 21:43:18 +02:00
|
|
|
|
if (!password) {
|
2017-05-11 14:33:41 +02:00
|
|
|
|
if (this.auto_login) {
|
2016-08-23 20:10:08 +02:00
|
|
|
|
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
|
|
|
|
}
|
2017-06-12 20:30:58 +02:00
|
|
|
|
const resource = Strophe.getResourceFromJid(this.jid);
|
2016-03-31 10:37:28 +02:00
|
|
|
|
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;
|
|
|
|
|
}
|
2017-06-23 20:19:38 +02:00
|
|
|
|
if (!this.connection.reconnecting) {
|
|
|
|
|
this.connection.reset();
|
|
|
|
|
}
|
2018-07-28 17:13:13 +02:00
|
|
|
|
this.connection.connect(this.jid, password, this.onConnectStatusChanged, BOSH_WAIT);
|
2016-03-31 10:37:28 +02: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 () {
|
2018-05-03 11:12:38 +02:00
|
|
|
|
/* Creates a new Strophe.Connection instance if we don't already have one.
|
|
|
|
|
*/
|
2017-08-16 12:31:17 +02:00
|
|
|
|
if (!this.connection) {
|
|
|
|
|
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) {
|
|
|
|
|
this.connection = new Strophe.Connection(this.websocket_url, this.connection_options);
|
|
|
|
|
} else if (this.bosh_service_url) {
|
|
|
|
|
this.connection = new Strophe.Connection(
|
|
|
|
|
this.bosh_service_url,
|
|
|
|
|
_.assignIn(this.connection_options, {'keepalive': this.keepalive})
|
|
|
|
|
);
|
|
|
|
|
} else {
|
|
|
|
|
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
|
|
|
|
}
|
2017-08-16 12:31:17 +02:00
|
|
|
|
_converse.emit('connectionInitialized');
|
2016-02-16 08:46:47 +01:00
|
|
|
|
};
|
|
|
|
|
|
2018-05-03 18:24:24 +02:00
|
|
|
|
|
2018-06-07 14:30:10 +02:00
|
|
|
|
this.tearDown = function () {
|
2016-02-16 08:46:47 +01:00
|
|
|
|
/* Remove those views which are only allowed with a valid
|
|
|
|
|
* connection.
|
|
|
|
|
*/
|
2017-07-21 12:41:16 +02:00
|
|
|
|
_converse.emit('beforeTearDown');
|
2017-08-16 12:31:17 +02:00
|
|
|
|
if (!_.isUndefined(_converse.session)) {
|
|
|
|
|
_converse.session.destroy();
|
2016-02-16 08:46:47 +01:00
|
|
|
|
}
|
2017-04-11 11:14:26 +02:00
|
|
|
|
window.removeEventListener('click', _converse.onUserActivity);
|
|
|
|
|
window.removeEventListener('focus', _converse.onUserActivity);
|
|
|
|
|
window.removeEventListener('keypress', _converse.onUserActivity);
|
|
|
|
|
window.removeEventListener('mousemove', _converse.onUserActivity);
|
2018-06-06 09:29:32 +02:00
|
|
|
|
window.removeEventListener(_converse.unloadevent, _converse.onUserActivity);
|
2016-12-20 10:30:20 +01:00
|
|
|
|
window.clearInterval(_converse.everySecondTrigger);
|
2017-07-21 12:41:16 +02:00
|
|
|
|
_converse.emit('afterTearDown');
|
2017-08-16 12:31:17 +02:00
|
|
|
|
return _converse;
|
2016-02-16 08:46:47 +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 = [];
|
2017-06-12 20:30:58 +02:00
|
|
|
|
const whitelist = _converse.core_plugins.concat(
|
2017-03-07 12:02:40 +01:00
|
|
|
|
_converse.whitelisted_plugins);
|
|
|
|
|
|
2018-02-09 16:33:48 +01:00
|
|
|
|
if (_converse.view_mode === 'embedded') {
|
|
|
|
|
_.forEach([ // eslint-disable-line lodash/prefer-map
|
|
|
|
|
"converse-bookmarks",
|
|
|
|
|
"converse-controlbox",
|
|
|
|
|
"converse-headline",
|
2018-05-03 17:01:17 +02:00
|
|
|
|
"converse-register"
|
2018-02-09 16:33:48 +01:00
|
|
|
|
], (name) => {
|
|
|
|
|
_converse.blacklisted_plugins.push(name)
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2017-03-07 12:02:40 +01:00
|
|
|
|
_converse.pluggable.initializePlugins({
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'updateSettings' () {
|
2017-07-05 11:33:55 +02:00
|
|
|
|
_converse.log(
|
|
|
|
|
"(DEPRECATION) "+
|
|
|
|
|
"The `updateSettings` method has been deprecated. "+
|
|
|
|
|
"Please use `_converse.api.settings.update` instead.",
|
|
|
|
|
Strophe.LogLevel.WARN
|
|
|
|
|
)
|
|
|
|
|
_converse.api.settings.update.apply(_converse, arguments);
|
|
|
|
|
},
|
2017-03-07 12:02:40 +01:00
|
|
|
|
'_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;
|
|
|
|
|
}
|
2016-11-02 23:08:20 +01:00
|
|
|
|
|
2017-09-24 20:36:39 +02:00
|
|
|
|
function finishInitialization () {
|
|
|
|
|
_converse.initPlugins();
|
2018-08-23 09:41:39 +02:00
|
|
|
|
_converse.initClientConfig();
|
2017-09-24 20:36:39 +02:00
|
|
|
|
_converse.initConnection();
|
|
|
|
|
_converse.setUpXMLLogging();
|
|
|
|
|
_converse.logIn();
|
|
|
|
|
_converse.registerGlobalEventHandlers();
|
2017-09-28 23:29:37 +02:00
|
|
|
|
|
|
|
|
|
if (!Backbone.history.started) {
|
|
|
|
|
Backbone.history.start();
|
|
|
|
|
}
|
2017-09-24 20:36:39 +02:00
|
|
|
|
}
|
|
|
|
|
|
2016-12-20 10:30:20 +01:00
|
|
|
|
if (!_.isUndefined(_converse.connection) &&
|
2017-10-24 12:56:35 +02:00
|
|
|
|
_converse.connection.service === 'jasmine tests') {
|
2017-09-24 20:36:39 +02:00
|
|
|
|
finishInitialization();
|
2016-12-20 10:30:20 +01:00
|
|
|
|
return _converse;
|
2017-10-24 12:56:35 +02:00
|
|
|
|
} else if (_.isUndefined(i18n)) {
|
|
|
|
|
finishInitialization();
|
2016-11-02 23:08:20 +01:00
|
|
|
|
} else {
|
2017-10-24 12:56:35 +02:00
|
|
|
|
i18n.fetchTranslations(
|
2017-09-24 20:36:39 +02:00
|
|
|
|
_converse.locale,
|
2017-10-24 12:56:35 +02:00
|
|
|
|
_converse.locales,
|
2018-02-19 10:35:58 +01:00
|
|
|
|
u.interpolate(_converse.locales_url, {'locale': _converse.locale}))
|
2018-08-17 10:05:36 +02:00
|
|
|
|
.catch(e => _converse.log(e.message, Strophe.LogLevel.FATAL))
|
2018-02-09 16:33:48 +01:00
|
|
|
|
.then(finishInitialization)
|
|
|
|
|
.catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
|
2016-11-02 23:08:20 +01:00
|
|
|
|
}
|
2017-11-10 15:52:50 +01:00
|
|
|
|
return init_promise;
|
2016-02-16 08:46:47 +01:00
|
|
|
|
};
|
2017-02-14 15:08:39 +01:00
|
|
|
|
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* ### The private API
|
|
|
|
|
*
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* The private API methods are only accessible via the closured {@link _converse}
|
2018-09-02 12:15:26 +02:00
|
|
|
|
* object, which is only available to plugins.
|
|
|
|
|
*
|
|
|
|
|
* These methods are kept private (i.e. not global) because they may return
|
|
|
|
|
* sensitive data which should be kept off-limits to other 3rd-party scripts
|
|
|
|
|
* that might be running in the page.
|
|
|
|
|
*
|
|
|
|
|
* @namespace _converse.api
|
|
|
|
|
* @memberOf _converse
|
|
|
|
|
*/
|
2017-02-14 15:08:39 +01:00
|
|
|
|
_converse.api = {
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* This grouping collects API functions related to the XMPP connection.
|
|
|
|
|
*
|
|
|
|
|
* @namespace _converse.api.connection
|
|
|
|
|
* @memberOf _converse.api
|
|
|
|
|
*/
|
2017-02-14 15:08:39 +01:00
|
|
|
|
'connection': {
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* @method _converse.api.connection.connected
|
|
|
|
|
* @memberOf _converse.api.connection
|
|
|
|
|
* @returns {boolean} Whether there is an established connection or not.
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'connected' () {
|
2017-02-14 15:08:39 +01:00
|
|
|
|
return _converse.connection && _converse.connection.connected || false;
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* Terminates the connection.
|
|
|
|
|
*
|
|
|
|
|
* @method _converse.api.connection.disconnect
|
|
|
|
|
* @memberOf _converse.api.connection
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'disconnect' () {
|
2017-02-14 15:08:39 +01:00
|
|
|
|
_converse.connection.disconnect();
|
|
|
|
|
},
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Lets you emit (i.e. trigger) events, which can be listened to via
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* {@link _converse.api.listen.on} or {@link _converse.api.listen.once}
|
2018-09-02 12:15:26 +02:00
|
|
|
|
* (see [_converse.api.listen](http://localhost:8000/docs/html/api/-_converse.api.listen.html)).
|
|
|
|
|
*
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* @method _converse.api.emit
|
2018-09-02 12:15:26 +02:00
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'emit' () {
|
2017-07-05 11:03:13 +02:00
|
|
|
|
_converse.emit.apply(_converse, arguments);
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* This grouping collects API functions related to the current logged in user.
|
|
|
|
|
*
|
|
|
|
|
* @namespace _converse.api.user
|
|
|
|
|
* @memberOf _converse.api
|
|
|
|
|
*/
|
2017-02-14 15:08:39 +01:00
|
|
|
|
'user': {
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* @method _converse.api.user.jid
|
|
|
|
|
* @returns {string} The current user's full JID (Jabber ID)
|
|
|
|
|
* @example _converse.api.user.jid())
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'jid' () {
|
2017-02-14 15:08:39 +01:00
|
|
|
|
return _converse.connection.jid;
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* Logs the user in.
|
|
|
|
|
*
|
|
|
|
|
* If called without any parameters, Converse will try
|
|
|
|
|
* to log the user in by calling the `prebind_url` or `credentials_url` depending
|
|
|
|
|
* on whether prebinding is used or not.
|
|
|
|
|
*
|
|
|
|
|
* @method _converse.api.user.login
|
|
|
|
|
* @param {object} [credentials] An object with the credentials.
|
|
|
|
|
* @example
|
|
|
|
|
* converse.plugins.add('myplugin', {
|
|
|
|
|
* initialize: function () {
|
|
|
|
|
*
|
|
|
|
|
* this._converse.api.user.login({
|
|
|
|
|
* 'jid': 'dummy@example.com',
|
|
|
|
|
* 'password': 'secret'
|
|
|
|
|
* });
|
|
|
|
|
*
|
|
|
|
|
* }
|
|
|
|
|
* });
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'login' (credentials) {
|
2017-02-14 15:08:39 +01:00
|
|
|
|
_converse.logIn(credentials);
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* Logs the user out of the current XMPP session.
|
|
|
|
|
*
|
|
|
|
|
* @method _converse.api.user.logout
|
|
|
|
|
* @example _converse.api.user.logout();
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'logout' () {
|
2017-02-14 15:08:39 +01:00
|
|
|
|
_converse.logOut();
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* Set and get the user's chat status, also called their *availability*.
|
|
|
|
|
*
|
|
|
|
|
* @namespace _converse.api.user.status
|
|
|
|
|
* @memberOf _converse.api.user
|
|
|
|
|
*/
|
2017-02-14 15:08:39 +01:00
|
|
|
|
'status': {
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/** Return the current user's availability status.
|
|
|
|
|
*
|
|
|
|
|
* @method _converse.api.user.status.get
|
|
|
|
|
* @example _converse.api.user.status.get();
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'get' () {
|
2017-02-14 15:08:39 +01:00
|
|
|
|
return _converse.xmppstatus.get('status');
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* The user's status can be set to one of the following values:
|
|
|
|
|
*
|
|
|
|
|
* @method _converse.api.user.status.set
|
|
|
|
|
* @param {string} value The user's chat status (e.g. 'away', 'dnd', 'offline', 'online', 'unavailable' or 'xa')
|
|
|
|
|
* @param {string} [message] A custom status message
|
|
|
|
|
*
|
|
|
|
|
* @example this._converse.api.user.status.set('dnd');
|
|
|
|
|
* @example this._converse.api.user.status.set('dnd', 'In a meeting');
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'set' (value, message) {
|
2017-06-12 20:30:58 +02:00
|
|
|
|
const data = {'status': value};
|
2017-02-14 15:08:39 +01:00
|
|
|
|
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);
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Set and retrieve the user's custom status message.
|
|
|
|
|
*
|
|
|
|
|
* @namespace _converse.api.user.status.message
|
|
|
|
|
* @memberOf _converse.api.user.status
|
|
|
|
|
*/
|
2017-02-14 15:08:39 +01:00
|
|
|
|
'message': {
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* @method _converse.api.user.status.message.get
|
|
|
|
|
* @returns {string} The status message
|
|
|
|
|
* @example const message = _converse.api.user.status.message.get()
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'get' () {
|
2017-02-14 15:08:39 +01:00
|
|
|
|
return _converse.xmppstatus.get('status_message');
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* @method _converse.api.user.status.message.set
|
|
|
|
|
* @param {string} status The status message
|
|
|
|
|
* @example _converse.api.user.status.message.set('In a meeting');
|
|
|
|
|
*/
|
|
|
|
|
'set' (status) {
|
|
|
|
|
_converse.xmppstatus.save({'status_message': status});
|
2017-02-14 15:08:39 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
|
|
|
|
|
/**
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* This grouping allows access to the
|
|
|
|
|
* [configuration settings](/docs/html/configuration.html#configuration-settings)
|
|
|
|
|
* of Converse.
|
2018-09-02 12:15:26 +02:00
|
|
|
|
*
|
|
|
|
|
* @namespace _converse.api.settings
|
|
|
|
|
* @memberOf _converse.api
|
|
|
|
|
*/
|
2017-02-14 15:08:39 +01:00
|
|
|
|
'settings': {
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* Allows new configuration settings to be specified, or new default values for
|
|
|
|
|
* existing configuration settings to be specified.
|
|
|
|
|
*
|
|
|
|
|
* @method _converse.api.settings.update
|
|
|
|
|
* @param {object} settings The configuration settings
|
|
|
|
|
* @example
|
|
|
|
|
* _converse.api.settings.update({
|
|
|
|
|
* 'enable_foo': true
|
|
|
|
|
* });
|
|
|
|
|
*
|
|
|
|
|
* // The user can then override the default value of the configuration setting when
|
|
|
|
|
* // calling `converse.initialize`.
|
|
|
|
|
* converse.initialize({
|
|
|
|
|
* 'enable_foo': false
|
|
|
|
|
* });
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'update' (settings) {
|
2018-02-19 10:34:52 +01:00
|
|
|
|
u.merge(_converse.default_settings, settings);
|
|
|
|
|
u.merge(_converse, settings);
|
|
|
|
|
u.applyUserSettings(_converse, settings, _converse.user_settings);
|
2017-07-05 11:03:13 +02:00
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* @method _converse.api.settings.get
|
|
|
|
|
* @returns {*} Value of the particular configuration setting.
|
|
|
|
|
* @example _converse.api.settings.get("play_sounds");
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'get' (key) {
|
2017-02-14 15:08:39 +01:00
|
|
|
|
if (_.includes(_.keys(_converse.default_settings), key)) {
|
|
|
|
|
return _converse[key];
|
|
|
|
|
}
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* Set one or many configuration settings.
|
|
|
|
|
*
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* Note, this is not an alternative to calling {@link converse.initialize}, which still needs
|
2018-09-02 12:15:26 +02:00
|
|
|
|
* to be called. Generally, you'd use this method after Converse is already
|
|
|
|
|
* running and you want to change the configuration on-the-fly.
|
|
|
|
|
*
|
|
|
|
|
* @method _converse.api.settings.set
|
|
|
|
|
* @param {Object} [settings] An object containing configuration settings.
|
|
|
|
|
* @param {string} [key] Alternatively to passing in an object, you can pass in a key and a value.
|
|
|
|
|
* @param {string} [value]
|
|
|
|
|
* @example _converse.api.settings.set("play_sounds", true);
|
|
|
|
|
* @example
|
|
|
|
|
* _converse.api.settings.set({
|
|
|
|
|
* "play_sounds", true,
|
|
|
|
|
* "hide_offline_users" true
|
|
|
|
|
* });
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'set' (key, val) {
|
2017-06-12 20:30:58 +02:00
|
|
|
|
const o = {};
|
2017-02-14 15:08:39 +01:00
|
|
|
|
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)));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Converse and its plugins emit various events which you can listen to via the
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* {@link _converse.api.listen} namespace.
|
2018-09-02 12:15:26 +02:00
|
|
|
|
*
|
|
|
|
|
* Some of these events are also available as [ES2015 Promises](http://es6-features.org/#PromiseUsage)
|
|
|
|
|
* although not all of them could logically act as promises, since some events
|
|
|
|
|
* might be fired multpile times whereas promises are to be resolved (or
|
|
|
|
|
* rejected) only once.
|
|
|
|
|
*
|
|
|
|
|
* Events which are also promises include:
|
|
|
|
|
*
|
|
|
|
|
* * [cachedRoster](/docs/html/events.html#cachedroster)
|
|
|
|
|
* * [chatBoxesFetched](/docs/html/events.html#chatBoxesFetched)
|
|
|
|
|
* * [pluginsInitialized](/docs/html/events.html#pluginsInitialized)
|
|
|
|
|
* * [roster](/docs/html/events.html#roster)
|
|
|
|
|
* * [rosterContactsFetched](/docs/html/events.html#rosterContactsFetched)
|
|
|
|
|
* * [rosterGroupsFetched](/docs/html/events.html#rosterGroupsFetched)
|
|
|
|
|
* * [rosterInitialized](/docs/html/events.html#rosterInitialized)
|
|
|
|
|
* * [statusInitialized](/docs/html/events.html#statusInitialized)
|
|
|
|
|
* * [roomsPanelRendered](/docs/html/events.html#roomsPanelRendered)
|
|
|
|
|
*
|
|
|
|
|
* The various plugins might also provide promises, and they do this by using the
|
|
|
|
|
* `promises.add` api method.
|
|
|
|
|
*
|
|
|
|
|
* @namespace _converse.api.promises
|
|
|
|
|
* @memberOf _converse.api
|
|
|
|
|
*/
|
2017-07-05 11:03:13 +02:00
|
|
|
|
'promises': {
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* By calling `promises.add`, a new [Promise](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise)
|
|
|
|
|
* is made available for other code or plugins to depend on via the
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* {@link _converse.api.waitUntil} method.
|
2018-09-02 12:15:26 +02:00
|
|
|
|
*
|
|
|
|
|
* Generally, it's the responsibility of the plugin which adds the promise to
|
|
|
|
|
* also resolve it.
|
|
|
|
|
*
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* This is done by calling {@link _converse.api.emit}, which not only resolves the
|
2018-09-02 12:15:26 +02:00
|
|
|
|
* promise, but also emits an event with the same name (which can be listened to
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* via {@link _converse.api.listen}).
|
2018-09-02 12:15:26 +02:00
|
|
|
|
*
|
|
|
|
|
* @method _converse.api.promises.add
|
|
|
|
|
* @param {string|array} [name|names] The name or an array of names for the promise(s) to be added
|
|
|
|
|
* @example _converse.api.promises.add('foo-completed');
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'add' (promises) {
|
2017-07-05 11:03:13 +02:00
|
|
|
|
promises = _.isArray(promises) ? promises : [promises]
|
2017-07-10 21:14:48 +02:00
|
|
|
|
_.each(promises, addPromise);
|
2017-07-05 11:03:13 +02:00
|
|
|
|
}
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* This namespace lets you access the BOSH tokens
|
|
|
|
|
*
|
|
|
|
|
* @namespace _converse.api.tokens
|
|
|
|
|
* @memberOf _converse.api
|
|
|
|
|
*/
|
2017-02-14 15:08:39 +01:00
|
|
|
|
'tokens': {
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* @method _converse.api.tokens.get
|
|
|
|
|
* @param {string} [id] The type of token to return ('rid' or 'sid').
|
|
|
|
|
* @returns 'string' A token, either the RID or SID token depending on what's asked for.
|
|
|
|
|
* @example _converse.api.tokens.get('rid');
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'get' (id) {
|
2017-02-14 15:08:39 +01:00
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Converse emits events to which you can subscribe to.
|
|
|
|
|
*
|
|
|
|
|
* The `listen` namespace exposes methods for creating event listeners
|
|
|
|
|
* (aka handlers) for these events.
|
|
|
|
|
*
|
|
|
|
|
* @namespace _converse.api.listen
|
|
|
|
|
* @memberOf _converse
|
|
|
|
|
*/
|
2017-02-14 15:08:39 +01:00
|
|
|
|
'listen': {
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* Lets you listen to an event exactly once.
|
2018-09-02 15:07:14 +02:00
|
|
|
|
*
|
2018-09-02 12:15:26 +02:00
|
|
|
|
* @method _converse.api.listen.once
|
|
|
|
|
* @param {string} name The event's name
|
|
|
|
|
* @param {function} callback The callback method to be called when the event is emitted.
|
|
|
|
|
* @param {object} [context] The value of the `this` parameter for the callback.
|
|
|
|
|
* @example _converse.api.listen.once('message', function (messageXML) { ... });
|
|
|
|
|
*/
|
2017-03-08 12:52:41 +01:00
|
|
|
|
'once': _converse.once.bind(_converse),
|
2018-09-02 12:15:26 +02:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Lets you subscribe to an event.
|
|
|
|
|
*
|
|
|
|
|
* Every time the event fires, the callback method specified by `callback` will be called.
|
2018-09-02 15:07:14 +02:00
|
|
|
|
*
|
2018-09-02 12:15:26 +02:00
|
|
|
|
* @method _converse.api.listen.on
|
|
|
|
|
* @param {string} name The event's name
|
|
|
|
|
* @param {function} callback The callback method to be called when the event is emitted.
|
|
|
|
|
* @param {object} [context] The value of the `this` parameter for the callback.
|
|
|
|
|
* @example _converse.api.listen.on('message', function (messageXML) { ... });
|
|
|
|
|
*/
|
2017-03-08 12:52:41 +01:00
|
|
|
|
'on': _converse.on.bind(_converse),
|
2018-09-02 12:15:26 +02:00
|
|
|
|
|
|
|
|
|
/**
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* To stop listening to an event, you can use the `not` method.
|
2018-09-02 12:15:26 +02:00
|
|
|
|
*
|
|
|
|
|
* Every time the event fires, the callback method specified by `callback` will be called.
|
2018-09-02 15:07:14 +02:00
|
|
|
|
*
|
2018-09-02 12:15:26 +02:00
|
|
|
|
* @method _converse.api.listen.not
|
|
|
|
|
* @param {string} name The event's name
|
|
|
|
|
* @param {function} callback The callback method that is to no longer be called when the event fires
|
|
|
|
|
* @example _converse.api.listen.not('message', function (messageXML);
|
|
|
|
|
*/
|
2017-03-08 12:52:41 +01:00
|
|
|
|
'not': _converse.off.bind(_converse),
|
2018-09-02 12:15:26 +02:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Subscribe to an incoming stanza
|
|
|
|
|
*
|
|
|
|
|
* Every a matched stanza is received, the callback method specified by `callback` will be called.
|
2018-09-02 15:07:14 +02:00
|
|
|
|
*
|
2018-09-02 12:15:26 +02:00
|
|
|
|
* @method _converse.api.listen.stanza
|
|
|
|
|
* @param {string} name The stanza's name
|
|
|
|
|
* @param {object} options Matching options
|
|
|
|
|
* (e.g. 'ns' for namespace, 'type' for stanza type, also 'id' and 'from');
|
|
|
|
|
* @param {function} handler The callback method to be called when the stanza appears
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'stanza' (name, options, handler) {
|
2017-02-14 15:08:39 +01:00
|
|
|
|
if (_.isFunction(options)) {
|
|
|
|
|
handler = options;
|
|
|
|
|
options = {};
|
|
|
|
|
} else {
|
|
|
|
|
options = options || {};
|
|
|
|
|
}
|
|
|
|
|
_converse.connection.addHandler(
|
|
|
|
|
handler,
|
|
|
|
|
options.ns,
|
|
|
|
|
name,
|
|
|
|
|
options.type,
|
|
|
|
|
options.id,
|
|
|
|
|
options.from,
|
|
|
|
|
options
|
|
|
|
|
);
|
|
|
|
|
},
|
|
|
|
|
},
|
2018-09-02 12:15:26 +02:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Wait until a promise is resolved
|
|
|
|
|
*
|
|
|
|
|
* @method _converse.api.waitUntil
|
|
|
|
|
* @param {string} name The name of the promise
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* @returns {Promise}
|
2018-09-02 12:15:26 +02:00
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'waitUntil' (name) {
|
2017-06-12 20:30:58 +02:00
|
|
|
|
const promise = _converse.promises[name];
|
2017-02-14 15:08:39 +01:00
|
|
|
|
if (_.isUndefined(promise)) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
2017-11-10 15:52:50 +01:00
|
|
|
|
return promise;
|
2017-02-14 15:08:39 +01:00
|
|
|
|
},
|
2018-09-02 15:07:14 +02:00
|
|
|
|
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* Allows you to send XML stanzas.
|
|
|
|
|
*
|
2018-09-02 12:15:26 +02:00
|
|
|
|
* @method _converse.api.send
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* @example
|
|
|
|
|
* const msg = converse.env.$msg({
|
|
|
|
|
* 'from': 'juliet@example.com/balcony',
|
|
|
|
|
* 'to': 'romeo@example.net',
|
|
|
|
|
* 'type':'chat'
|
|
|
|
|
* });
|
|
|
|
|
* _converse.api.send(msg);
|
2018-09-02 12:15:26 +02:00
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'send' (stanza) {
|
2017-02-14 15:08:39 +01:00
|
|
|
|
_converse.connection.send(stanza);
|
|
|
|
|
},
|
2018-09-02 15:07:14 +02:00
|
|
|
|
|
2018-09-02 12:15:26 +02:00
|
|
|
|
/**
|
|
|
|
|
* Send an IQ stanza and receive a promise
|
|
|
|
|
*
|
|
|
|
|
* @method _converse.api.sendIQ
|
|
|
|
|
* @returns {Promise} A promise which resolves when we receive a `result` stanza
|
|
|
|
|
* or is rejected when we receive an `error` stanza.
|
|
|
|
|
*/
|
2018-06-06 11:07:59 +02:00
|
|
|
|
'sendIQ' (stanza) {
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
|
_converse.connection.sendIQ(stanza, resolve, reject, _converse.IQ_TIMEOUT);
|
|
|
|
|
});
|
|
|
|
|
}
|
2017-02-14 15:08:39 +01:00
|
|
|
|
};
|
|
|
|
|
|
2018-09-02 10:11:37 +02:00
|
|
|
|
/**
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* ### The Public API
|
|
|
|
|
*
|
2018-09-02 10:11:37 +02:00
|
|
|
|
* This namespace contains public API methods which are are
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* accessible on the global `converse` object.
|
2018-09-02 10:11:37 +02:00
|
|
|
|
* They are public, because any JavaScript in the
|
|
|
|
|
* page can call them. Public methods therefore don’t expose any sensitive
|
|
|
|
|
* or closured data. To do that, you’ll need to create a plugin, which has
|
|
|
|
|
* access to the private API method.
|
|
|
|
|
*
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* @namespace converse
|
2018-09-02 10:11:37 +02:00
|
|
|
|
*/
|
2018-09-02 15:07:14 +02:00
|
|
|
|
const converse = {
|
2018-09-02 10:11:37 +02:00
|
|
|
|
/**
|
|
|
|
|
* Public API method which initializes Converse.
|
|
|
|
|
* This method must always be called when using Converse.
|
|
|
|
|
*
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* @memberOf converse
|
2018-09-02 10:11:37 +02:00
|
|
|
|
* @method initialize
|
|
|
|
|
* @param {object} config A map of [configuration-settings](https://conversejs.org/docs/html/configuration.html#configuration-settings).
|
|
|
|
|
*
|
|
|
|
|
* @example
|
|
|
|
|
* converse.initialize({
|
|
|
|
|
* allow_otr: true,
|
|
|
|
|
* auto_list_rooms: false,
|
|
|
|
|
* auto_subscribe: false,
|
|
|
|
|
* bosh_service_url: 'https://bind.example.com',
|
|
|
|
|
* hide_muc_server: false,
|
|
|
|
|
* i18n: locales['en'],
|
|
|
|
|
* keepalive: true,
|
|
|
|
|
* play_sounds: true,
|
|
|
|
|
* prebind: false,
|
|
|
|
|
* show_controlbox_by_default: true,
|
|
|
|
|
* debug: false,
|
|
|
|
|
* roster_groups: true
|
|
|
|
|
* });
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'initialize' (settings, callback) {
|
2017-02-14 15:08:39 +01:00
|
|
|
|
return _converse.initialize(settings, callback);
|
|
|
|
|
},
|
2018-09-02 10:11:37 +02:00
|
|
|
|
/**
|
|
|
|
|
* Exposes methods for adding and removing plugins. You'll need to write a plugin
|
|
|
|
|
* if you want to have access to the private API methods defined further down below.
|
|
|
|
|
*
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* For more information on plugins, read the documentation on [writing a plugin](/docs/html/plugin_development.html).
|
2018-09-02 10:11:37 +02:00
|
|
|
|
*
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* @namespace plugins
|
|
|
|
|
* @memberOf converse
|
2018-09-02 10:11:37 +02:00
|
|
|
|
*/
|
2017-02-14 15:08:39 +01:00
|
|
|
|
'plugins': {
|
2018-09-02 10:11:37 +02:00
|
|
|
|
/** Registers a new plugin.
|
|
|
|
|
*
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* @method converse.plugins.add
|
2018-09-02 10:11:37 +02:00
|
|
|
|
* @param {string} name The name of the plugin
|
|
|
|
|
* @param {object} plugin The plugin object
|
|
|
|
|
*
|
|
|
|
|
* @example
|
|
|
|
|
*
|
|
|
|
|
* const plugin = {
|
|
|
|
|
* initialize: function () {
|
|
|
|
|
* // Gets called as soon as the plugin has been loaded.
|
|
|
|
|
*
|
|
|
|
|
* // Inside this method, you have access to the private
|
|
|
|
|
* // API via `_covnerse.api`.
|
|
|
|
|
*
|
|
|
|
|
* // The private _converse object contains the core logic
|
|
|
|
|
* // and data-structures of Converse.
|
|
|
|
|
* }
|
|
|
|
|
* }
|
|
|
|
|
* converse.plugins.add('myplugin', plugin);
|
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
|
'add' (name, plugin) {
|
2017-02-14 15:08:39 +01:00
|
|
|
|
plugin.__name__ = name;
|
|
|
|
|
if (!_.isUndefined(_converse.pluggable.plugins[name])) {
|
|
|
|
|
throw new TypeError(
|
2017-07-10 17:46:22 +02:00
|
|
|
|
`Error: plugin with name "${name}" has already been `+
|
2017-02-14 15:08:39 +01:00
|
|
|
|
'registered!');
|
|
|
|
|
} else {
|
|
|
|
|
_converse.pluggable.plugins[name] = plugin;
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-09-02 12:15:26 +02:00
|
|
|
|
|
2017-02-14 15:08:39 +01:00
|
|
|
|
},
|
2018-09-02 10:11:37 +02:00
|
|
|
|
/**
|
|
|
|
|
* Utility methods and globals from bundled 3rd party libraries.
|
2018-09-02 15:07:14 +02:00
|
|
|
|
* @memberOf converse
|
2018-09-02 10:11:37 +02:00
|
|
|
|
*
|
2018-09-02 12:15:26 +02:00
|
|
|
|
* @property {function} converse.env.$build - Creates a Strophe.Builder, for creating stanza objects.
|
|
|
|
|
* @property {function} converse.env.$iq - Creates a Strophe.Builder with an <iq/> element as the root.
|
|
|
|
|
* @property {function} converse.env.$msg - Creates a Strophe.Builder with an <message/> element as the root.
|
|
|
|
|
* @property {function} converse.env.$pres - Creates a Strophe.Builder with an <presence/> element as the root.
|
|
|
|
|
* @property {object} converse.env.Backbone - The [Backbone](http://backbonejs.org) object used by Converse to create models and views.
|
|
|
|
|
* @property {function} converse.env.Promise - The Promise implementation used by Converse.
|
|
|
|
|
* @property {function} converse.env.Strophe - The [Strophe](http://strophe.im/strophejs) XMPP library used by Converse.
|
|
|
|
|
* @property {object} converse.env._ - The instance of [lodash](http://lodash.com) used by Converse.
|
|
|
|
|
* @property {function} converse.env.f - And instance of Lodash with its methods wrapped to produce immutable auto-curried iteratee-first data-last methods.
|
|
|
|
|
* @property {function} converse.env.b64_sha1 - Utility method from Strophe for creating base64 encoded sha1 hashes.
|
|
|
|
|
* @property {object} converse.env.moment - [Moment](https://momentjs.com) date manipulation library.
|
|
|
|
|
* @property {function} converse.env.sizzle - [Sizzle](https://sizzlejs.com) CSS selector engine.
|
|
|
|
|
* @property {object} converse.env.utils - Module containing common utility methods used by Converse.
|
2018-09-02 10:11:37 +02:00
|
|
|
|
*/
|
2017-02-14 15:08:39 +01:00
|
|
|
|
'env': {
|
|
|
|
|
'$build': $build,
|
|
|
|
|
'$iq': $iq,
|
|
|
|
|
'$msg': $msg,
|
|
|
|
|
'$pres': $pres,
|
2017-04-21 15:52:16 +02:00
|
|
|
|
'Backbone': Backbone,
|
2017-07-10 21:14:48 +02:00
|
|
|
|
'Promise': Promise,
|
2017-02-14 15:08:39 +01:00
|
|
|
|
'Strophe': Strophe,
|
|
|
|
|
'_': _,
|
2018-02-08 17:06:53 +01:00
|
|
|
|
'f': f,
|
2017-04-21 15:52:16 +02:00
|
|
|
|
'b64_sha1': b64_sha1,
|
2017-02-14 15:08:39 +01:00
|
|
|
|
'moment': moment,
|
2017-04-21 15:52:16 +02:00
|
|
|
|
'sizzle': sizzle,
|
2018-02-19 10:34:52 +01:00
|
|
|
|
'utils': u
|
2017-02-14 15:08:39 +01:00
|
|
|
|
}
|
|
|
|
|
};
|
2018-09-02 15:07:14 +02:00
|
|
|
|
window.converse = converse;
|
2018-03-02 15:51:04 +01:00
|
|
|
|
window.dispatchEvent(new CustomEvent('converse-loaded'));
|
2018-09-02 15:07:14 +02:00
|
|
|
|
return converse;
|
2016-02-16 08:46:47 +01:00
|
|
|
|
}));
|