2019-07-11 10:48:52 +02:00
|
|
|
|
/**
|
2020-01-26 16:21:20 +01:00
|
|
|
|
* @copyright The Converse.js contributors
|
2019-09-19 16:54:55 +02:00
|
|
|
|
* @license Mozilla Public License (MPLv2)
|
2019-07-11 10:48:52 +02:00
|
|
|
|
*/
|
2021-07-05 17:15:48 +02:00
|
|
|
|
import URI from 'urijs';
|
2021-03-09 12:55:44 +01:00
|
|
|
|
import _converse from '@converse/headless/shared/_converse';
|
2019-05-21 16:18:14 +02:00
|
|
|
|
import advancedFormat from 'dayjs/plugin/advancedFormat';
|
2019-06-05 06:57:20 +02:00
|
|
|
|
import dayjs from 'dayjs';
|
2021-03-09 12:55:44 +01:00
|
|
|
|
import i18n from '@converse/headless/shared/i18n';
|
2021-05-12 12:14:28 +02:00
|
|
|
|
import invoke from 'lodash-es/invoke';
|
|
|
|
|
import isFunction from 'lodash-es/isFunction';
|
2021-07-19 15:50:50 +02:00
|
|
|
|
import log from '@converse/headless/log.js';
|
2021-05-12 11:07:38 +02:00
|
|
|
|
import pluggable from 'pluggable.js/src/pluggable.js';
|
2021-09-15 12:20:12 +02:00
|
|
|
|
import { settings_api, user_settings_api } from '@converse/headless/shared/settings/api.js';
|
2019-06-05 06:57:20 +02:00
|
|
|
|
import sizzle from 'sizzle';
|
2021-07-15 18:45:16 +02:00
|
|
|
|
import u, { setUnloadEvent, replacePromise } from '@converse/headless/utils/core.js';
|
2020-06-08 16:08:50 +02:00
|
|
|
|
import { Collection } from "@converse/skeletor/src/collection";
|
2020-12-03 13:40:30 +01:00
|
|
|
|
import { Connection, MockConnection } from '@converse/headless/shared/connection.js';
|
2020-06-08 16:08:50 +02:00
|
|
|
|
import { Events } from '@converse/skeletor/src/events.js';
|
|
|
|
|
import { Model } from '@converse/skeletor/src/model.js';
|
2020-09-10 07:33:00 +02:00
|
|
|
|
import { Strophe, $build, $iq, $msg, $pres } from 'strophe.js/src/strophe';
|
2021-03-09 12:55:44 +01:00
|
|
|
|
import { TimeoutError } from '@converse/headless/shared/errors';
|
2021-04-28 18:35:08 +02:00
|
|
|
|
import { getOpenPromise } from '@converse/openpromise';
|
2021-04-14 22:56:59 +02:00
|
|
|
|
import { html } from 'lit';
|
2021-09-15 12:20:12 +02:00
|
|
|
|
import { initAppSettings, } from '@converse/headless/shared/settings/utils.js';
|
2020-09-10 07:33:00 +02:00
|
|
|
|
import { sprintf } from 'sprintf-js';
|
2018-10-23 03:41:38 +02:00
|
|
|
|
|
2021-03-09 12:55:44 +01:00
|
|
|
|
export { _converse };
|
|
|
|
|
export { i18n };
|
2019-05-29 17:40:16 +02:00
|
|
|
|
|
2021-07-15 18:45:16 +02:00
|
|
|
|
import {
|
|
|
|
|
attemptNonPreboundSession,
|
|
|
|
|
cleanup,
|
|
|
|
|
initClientConfig,
|
|
|
|
|
initPlugins,
|
|
|
|
|
initSession,
|
|
|
|
|
initSessionStorage,
|
|
|
|
|
registerGlobalEventHandlers
|
|
|
|
|
} from './utils/init.js';
|
|
|
|
|
|
2019-05-21 16:18:14 +02:00
|
|
|
|
dayjs.extend(advancedFormat);
|
2019-05-06 11:16:56 +02:00
|
|
|
|
|
2018-10-23 03:41:38 +02:00
|
|
|
|
// Add Strophe Namespaces
|
2021-08-13 10:18:24 +02:00
|
|
|
|
Strophe.addNamespace('ACTIVITY', 'http://jabber.org/protocol/activity');
|
2018-10-23 03:41:38 +02:00
|
|
|
|
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');
|
2021-07-05 14:56:20 +02:00
|
|
|
|
Strophe.addNamespace('EME', 'urn:xmpp:eme:0');
|
2019-09-24 15:33:41 +02:00
|
|
|
|
Strophe.addNamespace('FASTEN', 'urn:xmpp:fasten:0');
|
2018-10-23 03:41:38 +02:00
|
|
|
|
Strophe.addNamespace('FORWARD', 'urn:xmpp:forward:0');
|
|
|
|
|
Strophe.addNamespace('HINTS', 'urn:xmpp:hints');
|
|
|
|
|
Strophe.addNamespace('HTTPUPLOAD', 'urn:xmpp:http:upload:0');
|
2018-11-12 18:46:03 +01:00
|
|
|
|
Strophe.addNamespace('IDLE', 'urn:xmpp:idle:1');
|
2018-10-23 03:41:38 +02:00
|
|
|
|
Strophe.addNamespace('MAM', 'urn:xmpp:mam:2');
|
2021-02-22 21:07:54 +01:00
|
|
|
|
Strophe.addNamespace('MARKERS', 'urn:xmpp:chat-markers:0');
|
2020-12-21 13:45:04 +01:00
|
|
|
|
Strophe.addNamespace('MENTIONS', 'urn:xmpp:mmn:0');
|
2021-02-22 21:07:54 +01:00
|
|
|
|
Strophe.addNamespace('MESSAGE_CORRECT', 'urn:xmpp:message-correct:0');
|
2019-09-24 15:33:41 +02:00
|
|
|
|
Strophe.addNamespace('MODERATE', 'urn:xmpp:message-moderate:0');
|
2018-10-23 03:41:38 +02:00
|
|
|
|
Strophe.addNamespace('NICK', 'http://jabber.org/protocol/nick');
|
2019-06-05 06:57:20 +02:00
|
|
|
|
Strophe.addNamespace('OMEMO', 'eu.siacs.conversations.axolotl');
|
2018-10-23 03:41:38 +02:00
|
|
|
|
Strophe.addNamespace('OUTOFBAND', 'jabber:x:oob');
|
|
|
|
|
Strophe.addNamespace('PUBSUB', 'http://jabber.org/protocol/pubsub');
|
2020-12-08 14:51:04 +01:00
|
|
|
|
Strophe.addNamespace('RAI', 'urn:xmpp:rai:0');
|
2021-02-22 21:07:54 +01:00
|
|
|
|
Strophe.addNamespace('RECEIPTS', 'urn:xmpp:receipts');
|
|
|
|
|
Strophe.addNamespace('REFERENCE', 'urn:xmpp:reference:0');
|
2018-10-23 03:41:38 +02:00
|
|
|
|
Strophe.addNamespace('REGISTER', 'jabber:iq:register');
|
2019-09-24 15:33:41 +02:00
|
|
|
|
Strophe.addNamespace('RETRACT', 'urn:xmpp:message-retract:0');
|
2018-10-23 03:41:38 +02:00
|
|
|
|
Strophe.addNamespace('ROSTERX', 'http://jabber.org/protocol/rosterx');
|
|
|
|
|
Strophe.addNamespace('RSM', 'http://jabber.org/protocol/rsm');
|
|
|
|
|
Strophe.addNamespace('SID', 'urn:xmpp:sid:0');
|
|
|
|
|
Strophe.addNamespace('SPOILER', 'urn:xmpp:spoiler:0');
|
2019-04-10 22:03:00 +02:00
|
|
|
|
Strophe.addNamespace('STANZAS', 'urn:ietf:params:xml:ns:xmpp-stanzas');
|
2020-10-16 09:02:12 +02:00
|
|
|
|
Strophe.addNamespace('STYLING', 'urn:xmpp:styling:0');
|
2018-10-23 03:41:38 +02:00
|
|
|
|
Strophe.addNamespace('VCARD', 'vcard-temp');
|
|
|
|
|
Strophe.addNamespace('VCARDUPDATE', 'vcard-temp:x:update');
|
|
|
|
|
Strophe.addNamespace('XFORM', 'jabber:x:data');
|
2021-02-17 15:59:44 +01:00
|
|
|
|
Strophe.addNamespace('XHTML', 'http://www.w3.org/1999/xhtml');
|
2018-10-23 03:41:38 +02:00
|
|
|
|
|
2021-09-09 22:43:37 +02:00
|
|
|
|
_converse.VERSION_NAME = "v8.0.2dev";
|
2018-11-15 22:59:55 +01:00
|
|
|
|
|
2020-03-31 12:51:30 +02:00
|
|
|
|
Object.assign(_converse, Events);
|
2018-11-15 22:59:55 +01:00
|
|
|
|
|
2020-03-31 12:51:30 +02:00
|
|
|
|
// Make converse pluggable
|
|
|
|
|
pluggable.enable(_converse, '_converse', 'pluggable');
|
2019-05-21 11:12:34 +02:00
|
|
|
|
|
2018-11-15 22:59:55 +01:00
|
|
|
|
|
2020-03-31 12:51:30 +02:00
|
|
|
|
/**
|
|
|
|
|
* ### The private API
|
|
|
|
|
*
|
|
|
|
|
* The private API methods are only accessible via the closured {@link _converse}
|
|
|
|
|
* 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
|
|
|
|
|
*/
|
2020-04-15 13:59:55 +02:00
|
|
|
|
export const api = _converse.api = {
|
2019-03-22 08:15:35 +01:00
|
|
|
|
/**
|
2020-03-31 12:51:30 +02:00
|
|
|
|
* This grouping collects API functions related to the XMPP connection.
|
2019-03-22 08:15:35 +01:00
|
|
|
|
*
|
2020-03-31 12:51:30 +02:00
|
|
|
|
* @namespace _converse.api.connection
|
|
|
|
|
* @memberOf _converse.api
|
2019-03-22 08:15:35 +01:00
|
|
|
|
*/
|
2020-03-31 12:51:30 +02:00
|
|
|
|
connection: {
|
|
|
|
|
/**
|
|
|
|
|
* @method _converse.api.connection.connected
|
|
|
|
|
* @memberOf _converse.api.connection
|
|
|
|
|
* @returns {boolean} Whether there is an established connection or not.
|
|
|
|
|
*/
|
|
|
|
|
connected () {
|
|
|
|
|
return _converse?.connection?.connected && true;
|
|
|
|
|
},
|
2018-05-29 12:00:23 +02:00
|
|
|
|
|
2020-03-31 12:51:30 +02:00
|
|
|
|
/**
|
|
|
|
|
* Terminates the connection.
|
|
|
|
|
*
|
|
|
|
|
* @method _converse.api.connection.disconnectkjjjkk
|
|
|
|
|
* @memberOf _converse.api.connection
|
|
|
|
|
*/
|
|
|
|
|
disconnect () {
|
|
|
|
|
if (_converse.connection) {
|
|
|
|
|
_converse.connection.disconnect();
|
|
|
|
|
}
|
|
|
|
|
},
|
2019-06-01 15:37:18 +02:00
|
|
|
|
|
2020-03-31 12:51:30 +02:00
|
|
|
|
/**
|
|
|
|
|
* Can be called once the XMPP connection has dropped and we want
|
|
|
|
|
* to attempt reconnection.
|
|
|
|
|
* Only needs to be called once, if reconnect fails Converse will
|
|
|
|
|
* attempt to reconnect every two seconds, alternating between BOSH and
|
|
|
|
|
* Websocket if URLs for both were provided.
|
|
|
|
|
* @method reconnect
|
|
|
|
|
* @memberOf _converse.api.connection
|
|
|
|
|
*/
|
|
|
|
|
async reconnect () {
|
|
|
|
|
const conn_status = _converse.connfeedback.get('connection_status');
|
2019-06-01 15:37:18 +02:00
|
|
|
|
|
2020-03-31 12:51:30 +02:00
|
|
|
|
if (api.settings.get("authentication") === _converse.ANONYMOUS) {
|
|
|
|
|
await tearDown();
|
|
|
|
|
await clearSession();
|
|
|
|
|
}
|
|
|
|
|
if (conn_status === Strophe.Status.CONNFAIL) {
|
|
|
|
|
// When reconnecting with a new transport, we call setUserJID
|
|
|
|
|
// so that a new resource is generated, to avoid multiple
|
|
|
|
|
// server-side sessions with the same resource.
|
|
|
|
|
//
|
|
|
|
|
// We also call `_proto._doDisconnect` so that connection event handlers
|
|
|
|
|
// for the old transport are removed.
|
|
|
|
|
if (api.connection.isType('websocket') && api.settings.get('bosh_service_url')) {
|
|
|
|
|
await _converse.setUserJID(_converse.bare_jid);
|
|
|
|
|
_converse.connection._proto._doDisconnect();
|
|
|
|
|
_converse.connection._proto = new Strophe.Bosh(_converse.connection);
|
|
|
|
|
_converse.connection.service = api.settings.get('bosh_service_url');
|
|
|
|
|
} else if (api.connection.isType('bosh') && api.settings.get("websocket_url")) {
|
|
|
|
|
if (api.settings.get("authentication") === _converse.ANONYMOUS) {
|
|
|
|
|
// When reconnecting anonymously, we need to connect with only
|
|
|
|
|
// the domain, not the full JID that we had in our previous
|
|
|
|
|
// (now failed) session.
|
|
|
|
|
await _converse.setUserJID(api.settings.get("jid"));
|
|
|
|
|
} else {
|
|
|
|
|
await _converse.setUserJID(_converse.bare_jid);
|
|
|
|
|
}
|
|
|
|
|
_converse.connection._proto._doDisconnect();
|
|
|
|
|
_converse.connection._proto = new Strophe.Websocket(_converse.connection);
|
|
|
|
|
_converse.connection.service = api.settings.get("websocket_url");
|
|
|
|
|
}
|
2020-05-12 11:22:35 +02:00
|
|
|
|
} else if (conn_status === Strophe.Status.AUTHFAIL && api.settings.get("authentication") === _converse.ANONYMOUS) {
|
2020-03-31 12:51:30 +02:00
|
|
|
|
// When reconnecting anonymously, we need to connect with only
|
|
|
|
|
// the domain, not the full JID that we had in our previous
|
|
|
|
|
// (now failed) session.
|
|
|
|
|
await _converse.setUserJID(api.settings.get("jid"));
|
|
|
|
|
}
|
2019-07-23 10:18:02 +02:00
|
|
|
|
|
2021-05-11 15:46:57 +02:00
|
|
|
|
if (_converse.connection?.reconnecting) {
|
2020-06-03 17:59:41 +02:00
|
|
|
|
_converse.connection.debouncedReconnect();
|
2020-03-31 12:51:30 +02:00
|
|
|
|
} else {
|
2020-06-03 17:59:41 +02:00
|
|
|
|
return _converse.connection.reconnect();
|
2020-03-31 12:51:30 +02:00
|
|
|
|
}
|
|
|
|
|
},
|
2019-07-23 10:18:02 +02:00
|
|
|
|
|
2020-03-31 12:51:30 +02:00
|
|
|
|
/**
|
|
|
|
|
* Utility method to determine the type of connection we have
|
|
|
|
|
* @method isType
|
|
|
|
|
* @memberOf _converse.api.connection
|
|
|
|
|
* @returns {boolean}
|
|
|
|
|
*/
|
|
|
|
|
isType (type) {
|
2020-06-03 17:59:41 +02:00
|
|
|
|
return _converse.connection.isType(type);
|
2019-07-23 10:18:02 +02:00
|
|
|
|
}
|
2020-03-31 12:51:30 +02:00
|
|
|
|
},
|
2019-07-23 10:18:02 +02:00
|
|
|
|
|
2020-03-31 12:51:30 +02:00
|
|
|
|
/**
|
|
|
|
|
* Lets you trigger events, which can be listened to via
|
|
|
|
|
* {@link _converse.api.listen.on} or {@link _converse.api.listen.once}
|
|
|
|
|
* (see [_converse.api.listen](http://localhost:8000/docs/html/api/-_converse.api.listen.html)).
|
|
|
|
|
*
|
|
|
|
|
* Some events also double as promises and can be waited on via {@link _converse.api.waitUntil}.
|
|
|
|
|
*
|
|
|
|
|
* @method _converse.api.trigger
|
|
|
|
|
* @param {string} name - The event name
|
|
|
|
|
* @param {...any} [argument] - Argument to be passed to the event handler
|
|
|
|
|
* @param {object} [options]
|
|
|
|
|
* @param {boolean} [options.synchronous] - Whether the event is synchronous or not.
|
|
|
|
|
* When a synchronous event is fired, a promise will be returned
|
|
|
|
|
* by {@link _converse.api.trigger} which resolves once all the
|
|
|
|
|
* event handlers' promises have been resolved.
|
|
|
|
|
*/
|
|
|
|
|
async trigger (name) {
|
2020-07-30 11:25:53 +02:00
|
|
|
|
if (!_converse._events) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2020-03-31 12:51:30 +02:00
|
|
|
|
const args = Array.from(arguments);
|
|
|
|
|
const options = args.pop();
|
|
|
|
|
if (options && options.synchronous) {
|
|
|
|
|
const events = _converse._events[name] || [];
|
2021-01-08 09:39:35 +01:00
|
|
|
|
const event_args = args.splice(1);
|
|
|
|
|
await Promise.all(events.map(e => e.callback.apply(e.ctx, event_args)));
|
2020-03-31 12:51:30 +02:00
|
|
|
|
} else {
|
|
|
|
|
_converse.trigger.apply(_converse, arguments);
|
|
|
|
|
}
|
|
|
|
|
const promise = _converse.promises[name];
|
|
|
|
|
if (promise !== undefined) {
|
|
|
|
|
promise.resolve();
|
|
|
|
|
}
|
|
|
|
|
},
|
2019-07-23 10:18:02 +02:00
|
|
|
|
|
2020-04-06 16:04:31 +02:00
|
|
|
|
/**
|
|
|
|
|
* Triggers a hook which can be intercepted by registered listeners via
|
|
|
|
|
* {@link _converse.api.listen.on} or {@link _converse.api.listen.once}.
|
|
|
|
|
* (see [_converse.api.listen](http://localhost:8000/docs/html/api/-_converse.api.listen.html)).
|
|
|
|
|
* A hook is a special kind of event which allows you to intercept a data
|
|
|
|
|
* structure in order to modify it, before passing it back.
|
|
|
|
|
* @async
|
|
|
|
|
* @param {string} name - The hook name
|
|
|
|
|
* @param {...any} context - The context to which the hook applies (could be for example, a {@link _converse.ChatBox)).
|
2020-09-16 13:28:59 +02:00
|
|
|
|
* @param {...any} data - The data structure to be intercepted and modified by the hook listeners.
|
2020-09-25 13:17:56 +02:00
|
|
|
|
* @returns {Promise<any>} - A promise that resolves with the modified data structure.
|
2020-04-06 16:04:31 +02:00
|
|
|
|
*/
|
|
|
|
|
hook (name, context, data) {
|
|
|
|
|
const events = _converse._events[name] || [];
|
|
|
|
|
if (events.length) {
|
|
|
|
|
// Create a chain of promises, with each one feeding its output to
|
|
|
|
|
// the next. The first input is a promise with the original data
|
|
|
|
|
// sent to this hook.
|
|
|
|
|
const o = events.reduce((o, e) => o.then(d => e.callback(context, d)), Promise.resolve(data));
|
|
|
|
|
o.catch(e => {
|
|
|
|
|
log.error(e)
|
|
|
|
|
throw e;
|
|
|
|
|
});
|
|
|
|
|
return o;
|
|
|
|
|
} else {
|
|
|
|
|
return data;
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
2019-06-01 19:36:23 +02:00
|
|
|
|
/**
|
2020-03-31 12:51:30 +02:00
|
|
|
|
* This grouping collects API functions related to the current logged in user.
|
2019-06-01 19:36:23 +02:00
|
|
|
|
*
|
2020-03-31 12:51:30 +02:00
|
|
|
|
* @namespace _converse.api.user
|
|
|
|
|
* @memberOf _converse.api
|
2019-06-01 19:36:23 +02:00
|
|
|
|
*/
|
2020-03-31 12:51:30 +02:00
|
|
|
|
user: {
|
2021-09-15 12:20:12 +02:00
|
|
|
|
settings: user_settings_api,
|
|
|
|
|
|
2020-03-31 12:51:30 +02:00
|
|
|
|
/**
|
|
|
|
|
* @method _converse.api.user.jid
|
|
|
|
|
* @returns {string} The current user's full JID (Jabber ID)
|
|
|
|
|
* @example _converse.api.user.jid())
|
|
|
|
|
*/
|
|
|
|
|
jid () {
|
|
|
|
|
return _converse.connection.jid;
|
|
|
|
|
},
|
2019-06-01 19:36:23 +02:00
|
|
|
|
|
2020-03-31 12:51:30 +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 {string} [jid]
|
|
|
|
|
* @param {string} [password]
|
|
|
|
|
* @param {boolean} [automatic=false] - An internally used flag that indicates whether
|
|
|
|
|
* this method was called automatically once the connection has been
|
|
|
|
|
* initialized. It's used together with the `auto_login` configuration flag
|
|
|
|
|
* to determine whether Converse should try to log the user in if it
|
|
|
|
|
* fails to restore a previous auth'd session.
|
2020-09-25 13:17:56 +02:00
|
|
|
|
* @returns {void}
|
2020-03-31 12:51:30 +02:00
|
|
|
|
*/
|
|
|
|
|
async login (jid, password, automatic=false) {
|
2021-09-23 22:27:48 +02:00
|
|
|
|
jid = jid || api.settings.get('jid');
|
2020-06-03 17:59:41 +02:00
|
|
|
|
if (!_converse.connection?.jid || (jid && !u.isSameDomain(_converse.connection.jid, jid))) {
|
|
|
|
|
await _converse.initConnection();
|
|
|
|
|
}
|
|
|
|
|
if (api.settings.get("connection_options")?.worker && (await _converse.connection.restoreWorkerSession())) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (jid) {
|
|
|
|
|
jid = await _converse.setUserJID(jid);
|
2020-03-31 12:51:30 +02:00
|
|
|
|
}
|
2019-06-01 19:36:23 +02:00
|
|
|
|
|
2020-03-31 12:51:30 +02:00
|
|
|
|
// See whether there is a BOSH session to re-attach to
|
|
|
|
|
const bosh_plugin = _converse.pluggable.plugins['converse-bosh'];
|
|
|
|
|
if (bosh_plugin && bosh_plugin.enabled()) {
|
|
|
|
|
if (await _converse.restoreBOSHSession()) {
|
|
|
|
|
return;
|
|
|
|
|
} else if (api.settings.get("authentication") === _converse.PREBIND && (!automatic || api.settings.get("auto_login"))) {
|
|
|
|
|
return _converse.startNewPreboundBOSHSession();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
password = password || api.settings.get("password");
|
|
|
|
|
const credentials = (jid && password) ? { jid, password } : null;
|
|
|
|
|
attemptNonPreboundSession(credentials, automatic);
|
|
|
|
|
},
|
2019-06-01 19:36:23 +02:00
|
|
|
|
|
2020-03-31 12:51:30 +02:00
|
|
|
|
/**
|
|
|
|
|
* Logs the user out of the current XMPP session.
|
|
|
|
|
* @method _converse.api.user.logout
|
|
|
|
|
* @example _converse.api.user.logout();
|
|
|
|
|
*/
|
2021-03-23 14:52:08 +01:00
|
|
|
|
async logout () {
|
|
|
|
|
/**
|
|
|
|
|
* Triggered before the user is logged out
|
|
|
|
|
* @event _converse#beforeLogout
|
|
|
|
|
*/
|
|
|
|
|
await api.trigger('beforeLogout', {'synchronous': true});
|
|
|
|
|
|
2021-04-28 18:35:08 +02:00
|
|
|
|
const promise = getOpenPromise();
|
2020-03-31 12:51:30 +02:00
|
|
|
|
const complete = () => {
|
|
|
|
|
// Recreate all the promises
|
|
|
|
|
Object.keys(_converse.promises).forEach(replacePromise);
|
|
|
|
|
delete _converse.jid
|
|
|
|
|
/**
|
|
|
|
|
* Triggered once the user has logged out.
|
|
|
|
|
* @event _converse#logout
|
|
|
|
|
*/
|
|
|
|
|
api.trigger('logout');
|
|
|
|
|
promise.resolve();
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-03 17:59:41 +02:00
|
|
|
|
_converse.connection.setDisconnectionCause(_converse.LOGOUT, undefined, true);
|
2020-03-31 12:51:30 +02:00
|
|
|
|
if (_converse.connection !== undefined) {
|
|
|
|
|
api.listen.once('disconnected', () => complete());
|
|
|
|
|
_converse.connection.disconnect();
|
|
|
|
|
} else {
|
|
|
|
|
complete();
|
|
|
|
|
}
|
|
|
|
|
return promise;
|
|
|
|
|
}
|
|
|
|
|
},
|
2019-08-01 10:26:35 +02:00
|
|
|
|
|
2021-07-19 15:50:50 +02:00
|
|
|
|
settings: settings_api,
|
2020-03-31 12:51:30 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Converse and its plugins trigger various events which you can listen to via the
|
|
|
|
|
* {@link _converse.api.listen} namespace.
|
|
|
|
|
*
|
|
|
|
|
* 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)
|
|
|
|
|
*
|
|
|
|
|
* 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
|
|
|
|
|
*/
|
|
|
|
|
promises: {
|
|
|
|
|
/**
|
|
|
|
|
* 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
|
|
|
|
|
* {@link _converse.api.waitUntil} method.
|
|
|
|
|
*
|
|
|
|
|
* Generally, it's the responsibility of the plugin which adds the promise to
|
|
|
|
|
* also resolve it.
|
|
|
|
|
*
|
|
|
|
|
* This is done by calling {@link _converse.api.trigger}, which not only resolves the
|
|
|
|
|
* promise, but also emits an event with the same name (which can be listened to
|
|
|
|
|
* via {@link _converse.api.listen}).
|
|
|
|
|
*
|
|
|
|
|
* @method _converse.api.promises.add
|
|
|
|
|
* @param {string|array} [name|names] The name or an array of names for the promise(s) to be added
|
|
|
|
|
* @param {boolean} [replace=true] Whether this promise should be replaced with a new one when the user logs out.
|
|
|
|
|
* @example _converse.api.promises.add('foo-completed');
|
|
|
|
|
*/
|
|
|
|
|
add (promises, replace=true) {
|
|
|
|
|
promises = Array.isArray(promises) ? promises : [promises];
|
|
|
|
|
promises.forEach(name => {
|
2021-04-28 18:35:08 +02:00
|
|
|
|
const promise = getOpenPromise();
|
2020-03-31 12:51:30 +02:00
|
|
|
|
promise.replace = replace;
|
|
|
|
|
_converse.promises[name] = promise;
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 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
|
|
|
|
|
*/
|
|
|
|
|
listen: {
|
|
|
|
|
/**
|
|
|
|
|
* Lets you listen to an event exactly once.
|
|
|
|
|
* @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) { ... });
|
|
|
|
|
*/
|
|
|
|
|
once: _converse.once.bind(_converse),
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Lets you subscribe to an event.
|
|
|
|
|
* Every time the event fires, the callback method specified by `callback` will be called.
|
|
|
|
|
* @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) { ... });
|
|
|
|
|
*/
|
|
|
|
|
on: _converse.on.bind(_converse),
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* To stop listening to an event, you can use the `not` method.
|
|
|
|
|
* @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);
|
|
|
|
|
*/
|
|
|
|
|
not: _converse.off.bind(_converse),
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Subscribe to an incoming stanza
|
|
|
|
|
* Every a matched stanza is received, the callback method specified by
|
|
|
|
|
* `callback` will be called.
|
|
|
|
|
* @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
|
|
|
|
|
*/
|
|
|
|
|
stanza (name, options, handler) {
|
|
|
|
|
if (isFunction(options)) {
|
|
|
|
|
handler = options;
|
|
|
|
|
options = {};
|
|
|
|
|
} else {
|
|
|
|
|
options = options || {};
|
|
|
|
|
}
|
|
|
|
|
_converse.connection.addHandler(
|
|
|
|
|
handler,
|
|
|
|
|
options.ns,
|
|
|
|
|
name,
|
|
|
|
|
options.type,
|
|
|
|
|
options.id,
|
|
|
|
|
options.from,
|
|
|
|
|
options
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Wait until a promise is resolved or until the passed in function returns
|
|
|
|
|
* a truthy value.
|
|
|
|
|
* @method _converse.api.waitUntil
|
|
|
|
|
* @param {string|function} condition - The name of the promise to wait for,
|
|
|
|
|
* or a function which should eventually return a truthy value.
|
|
|
|
|
* @returns {Promise}
|
|
|
|
|
*/
|
|
|
|
|
waitUntil (condition) {
|
|
|
|
|
if (isFunction(condition)) {
|
|
|
|
|
return u.waitUntil(condition);
|
|
|
|
|
} else {
|
|
|
|
|
const promise = _converse.promises[condition];
|
|
|
|
|
if (promise === undefined) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
return promise;
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Allows you to send XML stanzas.
|
|
|
|
|
* @method _converse.api.send
|
2020-09-25 13:17:56 +02:00
|
|
|
|
* @param {XMLElement} stanza
|
|
|
|
|
* @return {void}
|
2020-03-31 12:51:30 +02:00
|
|
|
|
* @example
|
|
|
|
|
* const msg = converse.env.$msg({
|
|
|
|
|
* 'from': 'juliet@example.com/balcony',
|
|
|
|
|
* 'to': 'romeo@example.net',
|
|
|
|
|
* 'type':'chat'
|
|
|
|
|
* });
|
|
|
|
|
* _converse.api.send(msg);
|
|
|
|
|
*/
|
|
|
|
|
send (stanza) {
|
|
|
|
|
if (!api.connection.connected()) {
|
|
|
|
|
log.warn("Not sending stanza because we're not connected!");
|
|
|
|
|
log.warn(Strophe.serialize(stanza));
|
|
|
|
|
return;
|
|
|
|
|
}
|
2020-10-01 11:13:13 +02:00
|
|
|
|
if (typeof stanza === 'string') {
|
2020-03-31 12:51:30 +02:00
|
|
|
|
stanza = u.toStanza(stanza);
|
2021-05-13 12:22:06 +02:00
|
|
|
|
} else if (stanza?.nodeTree) {
|
|
|
|
|
stanza = stanza.nodeTree;
|
2020-03-31 12:51:30 +02:00
|
|
|
|
}
|
2021-05-13 12:22:06 +02:00
|
|
|
|
|
2020-03-31 12:51:30 +02:00
|
|
|
|
if (stanza.tagName === 'iq') {
|
|
|
|
|
return api.sendIQ(stanza);
|
|
|
|
|
} else {
|
|
|
|
|
_converse.connection.send(stanza);
|
|
|
|
|
api.trigger('send', stanza);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
/**
|
2020-09-25 13:17:56 +02:00
|
|
|
|
* Send an IQ stanza
|
2020-03-31 12:51:30 +02:00
|
|
|
|
* @method _converse.api.sendIQ
|
2020-09-25 13:17:56 +02:00
|
|
|
|
* @param {XMLElement} stanza
|
|
|
|
|
* @param {Integer} [timeout=_converse.STANZA_TIMEOUT]
|
|
|
|
|
* @param {Boolean} [reject=true] - Whether an error IQ should cause the promise
|
2020-03-31 12:51:30 +02:00
|
|
|
|
* to be rejected. If `false`, the promise will resolve instead of being rejected.
|
2020-09-25 13:17:56 +02:00
|
|
|
|
* @returns {Promise} A promise which resolves (or potentially rejected) once we
|
|
|
|
|
* receive a `result` or `error` stanza or once a timeout is reached.
|
|
|
|
|
* If the IQ stanza being sent is of type `result` or `error`, there's
|
|
|
|
|
* nothing to wait for, so an already resolved promise is returned.
|
2020-03-31 12:51:30 +02:00
|
|
|
|
*/
|
2020-09-25 13:17:56 +02:00
|
|
|
|
sendIQ (stanza, timeout=_converse.STANZA_TIMEOUT, reject=true) {
|
2020-03-31 12:51:30 +02:00
|
|
|
|
let promise;
|
2020-09-25 13:17:56 +02:00
|
|
|
|
stanza = stanza?.nodeTree ?? stanza;
|
|
|
|
|
if (['get', 'set'].includes(stanza.getAttribute('type'))) {
|
|
|
|
|
timeout = timeout || _converse.STANZA_TIMEOUT;
|
|
|
|
|
if (reject) {
|
|
|
|
|
promise = new Promise((resolve, reject) => _converse.connection.sendIQ(stanza, resolve, reject, timeout));
|
|
|
|
|
promise.catch(e => {
|
|
|
|
|
if (e === null) {
|
|
|
|
|
throw new TimeoutError(
|
|
|
|
|
`Timeout error after ${timeout}ms for the following IQ stanza: ${Strophe.serialize(stanza)}`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
} else {
|
|
|
|
|
promise = new Promise(resolve => _converse.connection.sendIQ(stanza, resolve, resolve, timeout));
|
|
|
|
|
}
|
2020-03-31 12:51:30 +02:00
|
|
|
|
} else {
|
2020-09-25 13:17:56 +02:00
|
|
|
|
_converse.connection.sendIQ(stanza);
|
2020-09-25 19:29:23 +02:00
|
|
|
|
promise = Promise.resolve();
|
2020-03-31 12:51:30 +02:00
|
|
|
|
}
|
|
|
|
|
api.trigger('send', stanza);
|
|
|
|
|
return promise;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2020-06-03 17:59:41 +02:00
|
|
|
|
export async function tearDown () {
|
2020-03-31 12:51:30 +02:00
|
|
|
|
await _converse.api.trigger('beforeTearDown', {'synchronous': true});
|
|
|
|
|
window.removeEventListener('click', _converse.onUserActivity);
|
|
|
|
|
window.removeEventListener('focus', _converse.onUserActivity);
|
|
|
|
|
window.removeEventListener('keypress', _converse.onUserActivity);
|
|
|
|
|
window.removeEventListener('mousemove', _converse.onUserActivity);
|
|
|
|
|
window.removeEventListener(_converse.unloadevent, _converse.onUserActivity);
|
|
|
|
|
window.clearInterval(_converse.everySecondTrigger);
|
|
|
|
|
_converse.api.trigger('afterTearDown');
|
|
|
|
|
return _converse;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-10-20 06:45:35 +02:00
|
|
|
|
_converse.shouldClearCache = () => (
|
|
|
|
|
!_converse.config.get('trusted') ||
|
|
|
|
|
api.settings.get('clear_cache_on_logout') ||
|
|
|
|
|
_converse.isTestEnv()
|
|
|
|
|
);
|
2020-03-31 12:51:30 +02:00
|
|
|
|
|
|
|
|
|
|
2021-06-29 11:20:50 +02:00
|
|
|
|
export function clearSession () {
|
2020-06-03 17:59:41 +02:00
|
|
|
|
_converse.session?.destroy();
|
|
|
|
|
delete _converse.session;
|
|
|
|
|
_converse.shouldClearCache() && _converse.api.user.settings.clear();
|
2018-05-29 12:00:23 +02:00
|
|
|
|
/**
|
2020-03-02 16:29:21 +01:00
|
|
|
|
* Synchronouse event triggered once the user session has been cleared,
|
2018-05-29 12:00:23 +02:00
|
|
|
|
* for example when the user has logged out or when Converse has
|
|
|
|
|
* disconnected for some other reason.
|
|
|
|
|
* @event _converse#clearSession
|
|
|
|
|
*/
|
2020-03-02 16:29:21 +01:00
|
|
|
|
return _converse.api.trigger('clearSession', {'synchronous': true});
|
2018-05-29 12:00:23 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-06-05 10:01:55 +02:00
|
|
|
|
|
2020-06-03 17:59:41 +02:00
|
|
|
|
_converse.initConnection = function () {
|
|
|
|
|
const api = _converse.api;
|
2018-11-18 19:14:22 +01:00
|
|
|
|
|
2020-06-03 17:59:41 +02:00
|
|
|
|
if (! api.settings.get('bosh_service_url')) {
|
|
|
|
|
if (api.settings.get("authentication") === _converse.PREBIND) {
|
2018-11-18 19:14:22 +01:00
|
|
|
|
throw new Error("authentication is set to 'prebind' but we don't have a BOSH connection");
|
2018-11-20 17:31:53 +01:00
|
|
|
|
}
|
2020-06-03 17:59:41 +02:00
|
|
|
|
if (! api.settings.get("websocket_url")) {
|
2018-11-18 19:14:22 +01:00
|
|
|
|
throw new Error("initConnection: you must supply a value for either the bosh_service_url or websocket_url or both.");
|
2019-07-26 16:42:13 +02:00
|
|
|
|
}
|
2018-11-20 17:31:53 +01:00
|
|
|
|
}
|
2018-11-18 19:14:22 +01:00
|
|
|
|
|
2020-06-03 17:59:41 +02:00
|
|
|
|
const XMPPConnection = _converse.isTestEnv() ? MockConnection : Connection;
|
|
|
|
|
if (('WebSocket' in window || 'MozWebSocket' in window) && api.settings.get("websocket_url")) {
|
|
|
|
|
_converse.connection = new XMPPConnection(
|
|
|
|
|
api.settings.get("websocket_url"),
|
|
|
|
|
Object.assign(_converse.default_connection_options, api.settings.get("connection_options"))
|
2018-11-18 19:14:22 +01:00
|
|
|
|
);
|
2020-06-03 17:59:41 +02:00
|
|
|
|
} else if (api.settings.get('bosh_service_url')) {
|
|
|
|
|
_converse.connection = new XMPPConnection(
|
|
|
|
|
api.settings.get('bosh_service_url'),
|
2018-11-18 19:14:22 +01:00
|
|
|
|
Object.assign(
|
|
|
|
|
_converse.default_connection_options,
|
2020-06-03 17:59:41 +02:00
|
|
|
|
api.settings.get("connection_options"),
|
|
|
|
|
{'keepalive': api.settings.get("keepalive")}
|
2018-11-18 19:14:22 +01:00
|
|
|
|
)
|
|
|
|
|
);
|
|
|
|
|
} else {
|
|
|
|
|
throw new Error("initConnection: this browser does not support "+
|
|
|
|
|
"websockets and bosh_service_url wasn't specified.");
|
|
|
|
|
}
|
2019-03-14 21:05:39 +01:00
|
|
|
|
setUpXMLLogging();
|
2019-03-22 08:15:35 +01:00
|
|
|
|
/**
|
2020-06-03 17:59:41 +02:00
|
|
|
|
* Triggered once the `Connection` constructor has been initialized, which
|
2019-03-22 08:15:35 +01:00
|
|
|
|
* will be responsible for managing the connection to the XMPP server.
|
|
|
|
|
*
|
|
|
|
|
* @event _converse#connectionInitialized
|
|
|
|
|
*/
|
2020-06-03 17:59:41 +02:00
|
|
|
|
api.trigger('connectionInitialized');
|
2018-11-18 19:14:22 +01:00
|
|
|
|
}
|
2018-11-15 22:59:55 +01:00
|
|
|
|
|
2018-05-29 12:00:23 +02:00
|
|
|
|
|
2018-11-18 19:14:22 +01:00
|
|
|
|
/**
|
|
|
|
|
* Stores the passed in JID for the current user, potentially creating a
|
|
|
|
|
* resource if the JID is bare.
|
|
|
|
|
*
|
|
|
|
|
* Given that we can only create an XMPP connection if we know the domain of
|
|
|
|
|
* the server connect to and we only know this once we know the JID, we also
|
|
|
|
|
* call {@link _converse.initConnection } (if necessary) to make sure that the
|
|
|
|
|
* connection is set up.
|
|
|
|
|
*
|
|
|
|
|
* @method _converse#setUserJID
|
|
|
|
|
* @emits _converse#setUserJID
|
|
|
|
|
* @params { String } jid
|
|
|
|
|
*/
|
|
|
|
|
_converse.setUserJID = async function (jid) {
|
2021-07-15 18:45:16 +02:00
|
|
|
|
await initSession(_converse, jid);
|
2018-11-18 19:14:22 +01:00
|
|
|
|
/**
|
|
|
|
|
* Triggered whenever the user's JID has been updated
|
|
|
|
|
* @event _converse#setUserJID
|
|
|
|
|
*/
|
|
|
|
|
_converse.api.trigger('setUserJID');
|
|
|
|
|
return jid;
|
2018-05-29 12:00:23 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2018-11-15 22:59:55 +01:00
|
|
|
|
function setUpXMLLogging () {
|
2019-11-22 14:23:19 +01:00
|
|
|
|
const lmap = {}
|
|
|
|
|
lmap[Strophe.LogLevel.DEBUG] = 'debug';
|
|
|
|
|
lmap[Strophe.LogLevel.INFO] = 'info';
|
|
|
|
|
lmap[Strophe.LogLevel.WARN] = 'warn';
|
|
|
|
|
lmap[Strophe.LogLevel.ERROR] = 'error';
|
|
|
|
|
lmap[Strophe.LogLevel.FATAL] = 'fatal';
|
|
|
|
|
|
|
|
|
|
Strophe.log = (level, msg) => log.log(msg, lmap[level]);
|
|
|
|
|
Strophe.error = (msg) => log.error(msg);
|
|
|
|
|
|
|
|
|
|
_converse.connection.xmlInput = body => log.debug(body.outerHTML, 'color: darkgoldenrod');
|
|
|
|
|
_converse.connection.xmlOutput = body => log.debug(body.outerHTML, 'color: darkcyan');
|
2018-11-15 22:59:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2020-03-31 12:51:30 +02:00
|
|
|
|
_converse.saveWindowState = function (ev) {
|
|
|
|
|
// XXX: eventually we should be able to just use
|
|
|
|
|
// document.visibilityState (when we drop support for older
|
|
|
|
|
// browsers).
|
|
|
|
|
let state;
|
|
|
|
|
const event_map = {
|
|
|
|
|
'focus': "visible",
|
|
|
|
|
'focusin': "visible",
|
|
|
|
|
'pageshow': "visible",
|
|
|
|
|
'blur': "hidden",
|
|
|
|
|
'focusout': "hidden",
|
|
|
|
|
'pagehide': "hidden"
|
|
|
|
|
};
|
|
|
|
|
ev = ev || document.createEvent('Events');
|
|
|
|
|
if (ev.type in event_map) {
|
|
|
|
|
state = event_map[ev.type];
|
|
|
|
|
} else {
|
|
|
|
|
state = document.hidden ? "hidden" : "visible";
|
|
|
|
|
}
|
|
|
|
|
_converse.windowState = state;
|
|
|
|
|
/**
|
|
|
|
|
* Triggered when window state has changed.
|
|
|
|
|
* Used to determine when a user left the page and when came back.
|
|
|
|
|
* @event _converse#windowStateChanged
|
|
|
|
|
* @type { object }
|
|
|
|
|
* @property{ string } state - Either "hidden" or "visible"
|
|
|
|
|
* @example _converse.api.listen.on('windowStateChanged', obj => { ... });
|
|
|
|
|
*/
|
|
|
|
|
api.trigger('windowStateChanged', {state});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_converse.ConnectionFeedback = Model.extend({
|
|
|
|
|
defaults: {
|
|
|
|
|
'connection_status': Strophe.Status.DISCONNECTED,
|
|
|
|
|
'message': ''
|
|
|
|
|
},
|
|
|
|
|
initialize () {
|
|
|
|
|
this.on('change', () => api.trigger('connfeedback', _converse.connfeedback));
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
2020-04-30 13:27:01 +02:00
|
|
|
|
export const converse = window.converse || {};
|
2020-03-31 12:28:11 +02:00
|
|
|
|
|
2019-10-08 16:47:30 +02:00
|
|
|
|
|
2018-10-23 03:41:38 +02:00
|
|
|
|
/**
|
|
|
|
|
* ### The Public API
|
|
|
|
|
*
|
|
|
|
|
* This namespace contains public API methods which are are
|
|
|
|
|
* accessible on the global `converse` object.
|
|
|
|
|
* 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.
|
|
|
|
|
*
|
2019-07-20 21:55:42 +02:00
|
|
|
|
* @global
|
2018-10-23 03:41:38 +02:00
|
|
|
|
* @namespace converse
|
|
|
|
|
*/
|
2020-04-23 13:22:31 +02:00
|
|
|
|
Object.assign(converse, {
|
2020-03-18 19:32:03 +01:00
|
|
|
|
|
|
|
|
|
CHAT_STATES: ['active', 'composing', 'gone', 'inactive', 'paused'],
|
|
|
|
|
|
2019-09-11 15:23:58 +02:00
|
|
|
|
keycodes: {
|
|
|
|
|
TAB: 9,
|
|
|
|
|
ENTER: 13,
|
|
|
|
|
SHIFT: 16,
|
|
|
|
|
CTRL: 17,
|
|
|
|
|
ALT: 18,
|
|
|
|
|
ESCAPE: 27,
|
|
|
|
|
LEFT_ARROW: 37,
|
|
|
|
|
UP_ARROW: 38,
|
|
|
|
|
RIGHT_ARROW: 39,
|
|
|
|
|
DOWN_ARROW: 40,
|
|
|
|
|
FORWARD_SLASH: 47,
|
|
|
|
|
AT: 50,
|
|
|
|
|
META: 91,
|
|
|
|
|
META_RIGHT: 93
|
|
|
|
|
},
|
|
|
|
|
|
2018-10-23 03:41:38 +02:00
|
|
|
|
/**
|
|
|
|
|
* Public API method which initializes Converse.
|
|
|
|
|
* This method must always be called when using Converse.
|
2020-03-31 12:28:11 +02:00
|
|
|
|
* @async
|
2018-10-23 03:41:38 +02:00
|
|
|
|
* @memberOf converse
|
|
|
|
|
* @method initialize
|
|
|
|
|
* @param {object} config A map of [configuration-settings](https://conversejs.org/docs/html/configuration.html#configuration-settings).
|
|
|
|
|
* @example
|
|
|
|
|
* converse.initialize({
|
|
|
|
|
* auto_list_rooms: false,
|
|
|
|
|
* auto_subscribe: false,
|
|
|
|
|
* bosh_service_url: 'https://bind.example.com',
|
|
|
|
|
* hide_muc_server: false,
|
2019-08-07 10:28:49 +02:00
|
|
|
|
* i18n: 'en',
|
2018-10-23 03:41:38 +02:00
|
|
|
|
* play_sounds: true,
|
|
|
|
|
* show_controlbox_by_default: true,
|
|
|
|
|
* debug: false,
|
|
|
|
|
* roster_groups: true
|
|
|
|
|
* });
|
|
|
|
|
*/
|
2020-03-31 12:28:11 +02:00
|
|
|
|
async initialize (settings) {
|
2021-07-15 18:45:16 +02:00
|
|
|
|
await cleanup(_converse);
|
2020-09-03 16:05:56 +02:00
|
|
|
|
|
2020-03-31 12:28:11 +02:00
|
|
|
|
setUnloadEvent();
|
2021-03-09 12:55:44 +01:00
|
|
|
|
initAppSettings(settings);
|
2020-03-31 12:28:11 +02:00
|
|
|
|
_converse.strict_plugin_dependencies = settings.strict_plugin_dependencies; // Needed by pluggable.js
|
|
|
|
|
log.setLogLevel(api.settings.get("loglevel"));
|
|
|
|
|
|
|
|
|
|
if (api.settings.get("authentication") === _converse.ANONYMOUS) {
|
|
|
|
|
if (api.settings.get("auto_login") && !api.settings.get('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.");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
_converse.router.route(
|
|
|
|
|
/^converse\?loglevel=(debug|info|warn|error|fatal)$/, 'loglevel',
|
|
|
|
|
l => log.setLogLevel(l)
|
|
|
|
|
);
|
|
|
|
|
_converse.connfeedback = new _converse.ConnectionFeedback();
|
|
|
|
|
|
|
|
|
|
/* 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 we don't need to again send out a presence stanza,
|
|
|
|
|
* because it's as if "we never left" (see onConnectStatusChanged).
|
|
|
|
|
* https://github.com/conversejs/converse.js/issues/521
|
|
|
|
|
*/
|
|
|
|
|
_converse.send_initial_presence = true;
|
|
|
|
|
|
2021-07-15 18:45:16 +02:00
|
|
|
|
await initSessionStorage(_converse);
|
|
|
|
|
await initClientConfig(_converse);
|
2020-09-28 12:55:06 +02:00
|
|
|
|
await i18n.initialize();
|
2021-07-15 18:45:16 +02:00
|
|
|
|
initPlugins(_converse);
|
2021-09-24 10:32:33 +02:00
|
|
|
|
|
|
|
|
|
// Register all custom elements
|
|
|
|
|
api.elements.register();
|
|
|
|
|
|
2021-07-15 18:45:16 +02:00
|
|
|
|
registerGlobalEventHandlers(_converse);
|
2020-03-31 12:51:30 +02:00
|
|
|
|
|
2021-02-24 15:48:11 +01:00
|
|
|
|
try {
|
|
|
|
|
!History.started && _converse.router.history.start();
|
|
|
|
|
} catch (e) {
|
|
|
|
|
log.error(e);
|
|
|
|
|
}
|
2020-03-31 12:51:30 +02:00
|
|
|
|
|
|
|
|
|
if (api.settings.get("idle_presence_timeout") > 0) {
|
|
|
|
|
api.listen.on('addClientFeatures', () => api.disco.own.features.add(Strophe.NS.IDLE));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const plugins = _converse.pluggable.plugins
|
|
|
|
|
if (api.settings.get("auto_login") || api.settings.get("keepalive") && invoke(plugins['converse-bosh'], 'enabled')) {
|
|
|
|
|
await api.user.login(null, null, true);
|
|
|
|
|
}
|
|
|
|
|
|
2020-08-26 16:07:59 +02:00
|
|
|
|
/**
|
|
|
|
|
* Triggered once converse.initialize has finished.
|
|
|
|
|
* @event _converse#initialized
|
|
|
|
|
*/
|
|
|
|
|
api.trigger('initialized');
|
|
|
|
|
|
2020-03-31 12:28:11 +02:00
|
|
|
|
if (_converse.isTestEnv()) {
|
|
|
|
|
return _converse;
|
|
|
|
|
}
|
2018-10-23 03:41:38 +02:00
|
|
|
|
},
|
2020-03-31 12:28:11 +02:00
|
|
|
|
|
2018-10-23 03:41:38 +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.
|
|
|
|
|
*
|
|
|
|
|
* For more information on plugins, read the documentation on [writing a plugin](/docs/html/plugin_development.html).
|
|
|
|
|
* @namespace plugins
|
|
|
|
|
* @memberOf converse
|
|
|
|
|
*/
|
2019-06-05 06:57:20 +02:00
|
|
|
|
plugins: {
|
2019-08-16 15:44:58 +02:00
|
|
|
|
/**
|
|
|
|
|
* Registers a new plugin.
|
2018-10-23 03:41:38 +02:00
|
|
|
|
* @method converse.plugins.add
|
|
|
|
|
* @param {string} name The name of the plugin
|
|
|
|
|
* @param {object} plugin The plugin object
|
2018-09-02 10:11:37 +02:00
|
|
|
|
* @example
|
2018-10-23 03:41:38 +02:00
|
|
|
|
* const plugin = {
|
|
|
|
|
* initialize: function () {
|
|
|
|
|
* // Gets called as soon as the plugin has been loaded.
|
2018-09-02 10:11:37 +02:00
|
|
|
|
*
|
2018-10-23 03:41:38 +02:00
|
|
|
|
* // Inside this method, you have access to the private
|
|
|
|
|
* // API via `_covnerse.api`.
|
2018-09-02 10:11:37 +02:00
|
|
|
|
*
|
2018-10-23 03:41:38 +02:00
|
|
|
|
* // The private _converse object contains the core logic
|
|
|
|
|
* // and data-structures of Converse.
|
|
|
|
|
* }
|
|
|
|
|
* }
|
|
|
|
|
* converse.plugins.add('myplugin', plugin);
|
2018-09-02 10:11:37 +02:00
|
|
|
|
*/
|
2019-06-05 06:57:20 +02:00
|
|
|
|
add (name, plugin) {
|
2018-10-23 03:41:38 +02:00
|
|
|
|
plugin.__name__ = name;
|
2019-07-29 10:19:05 +02:00
|
|
|
|
if (_converse.pluggable.plugins[name] !== undefined) {
|
2018-10-23 03:41:38 +02:00
|
|
|
|
throw new TypeError(
|
2019-06-05 06:57:20 +02:00
|
|
|
|
`Error: plugin with name "${name}" has already been ` + 'registered!'
|
|
|
|
|
);
|
2018-10-23 03:41:38 +02:00
|
|
|
|
} else {
|
|
|
|
|
_converse.pluggable.plugins[name] = plugin;
|
|
|
|
|
}
|
2017-02-14 15:08:39 +01:00
|
|
|
|
}
|
2018-10-23 03:41:38 +02:00
|
|
|
|
|
|
|
|
|
},
|
|
|
|
|
/**
|
|
|
|
|
* Utility methods and globals from bundled 3rd party libraries.
|
2020-09-10 14:08:43 +02:00
|
|
|
|
* @typedef ConverseEnv
|
2018-10-23 03:41:38 +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 {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 {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.sizzle - [Sizzle](https://sizzlejs.com) CSS selector engine.
|
2020-09-10 14:08:43 +02:00
|
|
|
|
* @property {function} converse.env.sprintf
|
|
|
|
|
* @property {object} converse.env._ - The instance of [lodash-es](http://lodash.com) used by Converse.
|
|
|
|
|
* @property {object} converse.env.dayjs - [DayJS](https://github.com/iamkun/dayjs) date manipulation library.
|
2018-10-23 03:41:38 +02:00
|
|
|
|
* @property {object} converse.env.utils - Module containing common utility methods used by Converse.
|
2020-09-10 14:08:43 +02:00
|
|
|
|
* @memberOf converse
|
2018-10-23 03:41:38 +02:00
|
|
|
|
*/
|
2020-09-10 14:08:43 +02:00
|
|
|
|
'env': {
|
|
|
|
|
$build,
|
|
|
|
|
$iq,
|
|
|
|
|
$msg,
|
|
|
|
|
$pres,
|
|
|
|
|
'utils': u,
|
|
|
|
|
Collection,
|
|
|
|
|
Model,
|
|
|
|
|
Promise,
|
|
|
|
|
Strophe,
|
2021-07-05 17:15:48 +02:00
|
|
|
|
URI,
|
2020-09-10 14:08:43 +02:00
|
|
|
|
dayjs,
|
|
|
|
|
html,
|
|
|
|
|
log,
|
|
|
|
|
sizzle,
|
|
|
|
|
sprintf,
|
|
|
|
|
u,
|
|
|
|
|
}
|
2019-10-06 12:48:05 +02:00
|
|
|
|
});
|