Commit a45bd8d1 authored by JC Brand's avatar JC Brand

Convert older docstrings to JSDoc syntax

parent 7ed99092
...@@ -261,4 +261,4 @@ html: dev docsdev apidoc ...@@ -261,4 +261,4 @@ html: dev docsdev apidoc
PHONY: apidoc PHONY: apidoc
apidoc: apidoc:
$(JSDOC) --readme docs/source/jsdoc_intro.md -c docs/source/conf.json -d docs/html/api src/*.js src/utils/*.js src/headless/*.js src/headless/utils/*.js $(JSDOC) --private --readme docs/source/jsdoc_intro.md -c docs/source/conf.json -d docs/html/api src/*.js src/utils/*.js src/headless/*.js src/headless/utils/*.js
...@@ -49525,18 +49525,18 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -49525,18 +49525,18 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
_.each(this.content.querySelectorAll('span.spinner'), el => el.parentNode.removeChild(el)); _.each(this.content.querySelectorAll('span.spinner'), el => el.parentNode.removeChild(el));
}, },
insertDayIndicator(next_msg_el) { /**
/* Inserts an indicator into the chat area, showing the * Inserts an indicator into the chat area, showing the
* day as given by the passed in date. * day as given by the passed in date.
*
* The indicator is only inserted if necessary. * The indicator is only inserted if necessary.
* * @private
* Parameters: * @method _converse.ChatBoxView#insertDayIndicator
* (HTMLElement) next_msg_el - The message element before * @param { HTMLElement } next_msg_el - The message element before
* which the day indicator element must be inserted. * which the day indicator element must be inserted.
* This element must have a "data-isodate" attribute * This element must have a "data-isodate" attribute
* which specifies its creation date. * which specifies its creation date.
*/ */
insertDayIndicator(next_msg_el) {
const prev_msg_el = _converse_headless_utils_emoji__WEBPACK_IMPORTED_MODULE_21__["default"].getPreviousElement(next_msg_el, ".message:not(.chat-state-notification)"), const prev_msg_el = _converse_headless_utils_emoji__WEBPACK_IMPORTED_MODULE_21__["default"].getPreviousElement(next_msg_el, ".message:not(.chat-state-notification)"),
prev_msg_date = _.isNull(prev_msg_el) ? null : prev_msg_el.getAttribute('data-isodate'), prev_msg_date = _.isNull(prev_msg_el) ? null : prev_msg_el.getAttribute('data-isodate'),
next_msg_date = next_msg_el.getAttribute('data-isodate'); next_msg_date = next_msg_el.getAttribute('data-isodate');
...@@ -49554,13 +49554,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -49554,13 +49554,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
} }
}, },
getLastMessageDate(cutoff) { /**
/* Return the ISO8601 format date of the latest message. * Return the ISO8601 format date of the latest message.
* * @private
* Parameters: * @method _converse.ChatBoxView#getLastMessageDate
* (Object) cutoff: Moment Date cutoff date. The last * @param { object } cutoff - Moment Date cutoff date. The last
* message received cutoff this date will be returned. * message received cutoff this date will be returned.
*/ */
getLastMessageDate(cutoff) {
const first_msg = _converse_headless_utils_emoji__WEBPACK_IMPORTED_MODULE_21__["default"].getFirstChildElement(this.content, '.message:not(.chat-state-notification)'), const first_msg = _converse_headless_utils_emoji__WEBPACK_IMPORTED_MODULE_21__["default"].getFirstChildElement(this.content, '.message:not(.chat-state-notification)'),
oldest_date = first_msg ? first_msg.getAttribute('data-isodate') : null; oldest_date = first_msg ? first_msg.getAttribute('data-isodate') : null;
...@@ -49656,13 +49657,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -49656,13 +49657,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
return !_converse_headless_utils_emoji__WEBPACK_IMPORTED_MODULE_21__["default"].isVisible(this.el); return !_converse_headless_utils_emoji__WEBPACK_IMPORTED_MODULE_21__["default"].isVisible(this.el);
}, },
insertMessage(view) { /**
/* Given a view representing a message, insert it into the * Given a view representing a message, insert it into the
* content area of the chat box. * content area of the chat box.
* * @private
* Parameters: * @method _converse.ChatBoxView#insertMessage
* (Backbone.View) message: The message Backbone.View * @param { Backbone.View } message - The message Backbone.View
*/ */
insertMessage(view) {
if (view.model.get('type') === 'error') { if (view.model.get('type') === 'error') {
const previous_msg_el = this.content.querySelector(`[data-msgid="${view.model.get('msgid')}"]`); const previous_msg_el = this.content.querySelector(`[data-msgid="${view.model.get('msgid')}"]`);
...@@ -49692,8 +49694,8 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -49692,8 +49694,8 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
return this.trigger('messageInserted', view.el); return this.trigger('messageInserted', view.el);
}, },
markFollowups(el) { /**
/* Given a message element, determine wether it should be * Given a message element, determine wether it should be
* marked as a followup message to the previous element. * marked as a followup message to the previous element.
* *
* Also determine whether the element following it is a * Also determine whether the element following it is a
...@@ -49703,9 +49705,11 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -49703,9 +49705,11 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
* author with no other conversation elements inbetween and * author with no other conversation elements inbetween and
* posted within 10 minutes of one another. * posted within 10 minutes of one another.
* *
* Parameters: * @private
* (HTMLElement) el - The message element. * @method _converse.ChatBoxView#markFollowups
* @param { HTMLElement } el - The message element
*/ */
markFollowups(el) {
const from = el.getAttribute('data-from'), const from = el.getAttribute('data-from'),
previous_el = el.previousElementSibling, previous_el = el.previousElementSibling,
date = moment(el.getAttribute('data-isodate')), date = moment(el.getAttribute('data-isodate')),
...@@ -49726,15 +49730,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -49726,15 +49730,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
} }
}, },
async showMessage(message) { /**
/* Inserts a chat message into the content area of the chat box. * Inserts a chat message into the content area of the chat box.
* * Will also insert a new day indicator if the message is on a different day.
* Will also insert a new day indicator if the message is on a * @private
* different day. * @method _converse.ChatBoxView#showMessage
* * @param { _converse.Message } message - The message object
* Parameters:
* (Backbone.Model) message: The message object
*/ */
async showMessage(message) {
if (!_converse_headless_utils_emoji__WEBPACK_IMPORTED_MODULE_21__["default"].isNewMessage(message) && _converse_headless_utils_emoji__WEBPACK_IMPORTED_MODULE_21__["default"].isEmptyMessage(message)) { if (!_converse_headless_utils_emoji__WEBPACK_IMPORTED_MODULE_21__["default"].isNewMessage(message) && _converse_headless_utils_emoji__WEBPACK_IMPORTED_MODULE_21__["default"].isEmptyMessage(message)) {
// Handle archived or delayed messages without any message // Handle archived or delayed messages without any message
// text to show. // text to show.
...@@ -49769,12 +49772,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -49769,12 +49772,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
} }
}, },
onMessageAdded(message) { /**
/* Handler that gets called when a new message object is created. * Handler that gets called when a new message object is created.
* * @private
* Parameters: * @method _converse.ChatBoxView#onMessageAdded
* (Object) message - The message Backbone object that was added. * @param { object } message - The message Backbone object that was added.
*/ */
onMessageAdded(message) {
this.showMessage(message); this.showMessage(message);
if (message.get('correcting')) { if (message.get('correcting')) {
...@@ -49811,17 +49815,18 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -49811,17 +49815,18 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
} }
}, },
setChatState(state, options) { /**
/* Mutator for setting the chat state of this chat session. * Mutator for setting the chat state of this chat session.
* Handles clearing of any chat state notification timeouts and * Handles clearing of any chat state notification timeouts and
* setting new ones if necessary. * setting new ones if necessary.
* Timeouts are set when the state being set is COMPOSING or PAUSED. * Timeouts are set when the state being set is COMPOSING or PAUSED.
* After the timeout, COMPOSING will become PAUSED and PAUSED will become INACTIVE. * After the timeout, COMPOSING will become PAUSED and PAUSED will become INACTIVE.
* See XEP-0085 Chat State Notifications. * See XEP-0085 Chat State Notifications.
* * @private
* Parameters: * @method _converse.ChatBoxView#setChatState
* (string) state - The chat state (consts ACTIVE, COMPOSING, PAUSED, INACTIVE, GONE) * @param { string } state - The chat state (consts ACTIVE, COMPOSING, PAUSED, INACTIVE, GONE)
*/ */
setChatState(state, options) {
if (!_.isUndefined(this.chat_state_timeout)) { if (!_.isUndefined(this.chat_state_timeout)) {
window.clearTimeout(this.chat_state_timeout); window.clearTimeout(this.chat_state_timeout);
delete this.chat_state_timeout; delete this.chat_state_timeout;
...@@ -50872,11 +50877,6 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -50872,11 +50877,6 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
}, },
showHelpMessages() { showHelpMessages() {
/* Override showHelpMessages in ChatBoxView, for now do nothing.
*
* Parameters:
* (Array) msgs: Array of messages
*/
return; return;
} }
...@@ -53538,16 +53538,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -53538,16 +53538,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
303: ___('Your nickname has been changed to %1$s') 303: ___('Your nickname has been changed to %1$s')
} }
}; };
function insertRoomInfo(el, stanza) {
/* Insert groupchat info (based on returned #disco IQ stanza) /* Insert groupchat info (based on returned #disco IQ stanza)
* * @function insertRoomInfo
* Parameters: * @param { HTMLElement } el - The HTML DOM element that contains the info.
* (HTMLElement) el: The HTML DOM element that should * @param { XMLElement } stanza - The IQ stanza containing the groupchat info.
* contain the info.
* (XMLElement) stanza: The IQ stanza containing the groupchat
* info.
*/ */
function insertRoomInfo(el, stanza) {
// All MUC features found here: https://xmpp.org/registrar/disco-features.html // All MUC features found here: https://xmpp.org/registrar/disco-features.html
el.querySelector('span.spinner').remove(); el.querySelector('span.spinner').remove();
el.querySelector('a.room-info').classList.add('selected'); el.querySelector('a.room-info').classList.add('selected');
...@@ -54528,12 +54525,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -54528,12 +54525,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
this.model.addHandler('message', 'ChatRoomView.showStatusMessages', this.showStatusMessages.bind(this)); this.model.addHandler('message', 'ChatRoomView.showStatusMessages', this.showStatusMessages.bind(this));
}, },
onPresence(pres) { /**
/* Handles all MUC presence stanzas. * Handles all MUC presence stanzas.
* * @private
* Parameters: * @method _converse.ChatRoomView#onPresence
* (XMLElement) pres: The stanza * @param { XMLElement } pres - The stanza
*/ */
onPresence(pres) {
// XXX: Current thinking is that excessive stanza // XXX: Current thinking is that excessive stanza
// processing inside a view is a "code smell". // processing inside a view is a "code smell".
// Instead stanza processing should happen inside the // Instead stanza processing should happen inside the
...@@ -54556,14 +54554,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -54556,14 +54554,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
this.fetchMessages(); this.fetchMessages();
}, },
join(nick, password) { /**
/* Join the groupchat. * Join the groupchat.
* * @private
* Parameters: * @method _converse.ChatRoomView#join
* (String) nick: The user's nickname * @param { String } nick - The user's nickname
* (String) password: Optional password, if required by * @param { String } password - Optional password, if required by the groupchat
* the groupchat.
*/ */
join(nick, password) {
if (!nick && !this.model.get('nick')) { if (!nick && !this.model.get('nick')) {
this.checkForReservedNick(); this.checkForReservedNick();
return this; return this;
...@@ -54573,17 +54571,16 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -54573,17 +54571,16 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
return this; return this;
}, },
renderConfigurationForm(stanza) { /**
/* Renders a form given an IQ stanza containing the current * Renders a form given an IQ stanza containing the current
* groupchat configuration. * groupchat configuration.
*
* Returns a promise which resolves once the user has * Returns a promise which resolves once the user has
* either submitted the form, or canceled it. * either submitted the form, or canceled it.
* * @private
* Parameters: * @method _converse.ChatRoomView#renderConfigurationForm
* (XMLElement) stanza: The IQ stanza containing the groupchat * @param { XMLElement } stanza: The IQ stanza containing the groupchat config.
* config.
*/ */
renderConfigurationForm(stanza) {
const container_el = this.el.querySelector('.chatroom-body'); const container_el = this.el.querySelector('.chatroom-body');
_.each(container_el.querySelectorAll('.chatroom-form-container'), u.removeElement); _.each(container_el.querySelectorAll('.chatroom-form-container'), u.removeElement);
...@@ -54834,13 +54831,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -54834,13 +54831,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
u.showElement(container); u.showElement(container);
}, },
getMessageFromStatus(stat, stanza, is_self) { /**
/* Parameters: * @private
* (XMLElement) stat: A <status> element. * @method _converse.ChatRoomView#getMessageFromStatus
* (Boolean) is_self: Whether the element refers to the * @param { XMLElement } stat: A <status> element
* current user. * @param { Boolean } is_self: Whether the element refers to the current user
* (XMLElement) stanza: The original stanza received. * @param { XMLElement } stanza: The original stanza received
*/ */
getMessageFromStatus(stat, stanza, is_self) {
const code = stat.getAttribute('code'); const code = stat.getAttribute('code');
if (code === '110' || code === '100' && !is_self) { if (code === '110' || code === '100' && !is_self) {
...@@ -55150,14 +55148,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins ...@@ -55150,14 +55148,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_5__["default"].plugins
this.scrollDown(); this.scrollDown();
}, },
showStatusMessages(stanza) { /**
/* Check for status codes and communicate their purpose to the user. * Check for status codes and communicate their purpose to the user.
* See: https://xmpp.org/registrar/mucstatus.html * See: https://xmpp.org/registrar/mucstatus.html
* * @private
* Parameters: * @method _converse.ChatRoomView#showStatusMessages
* (XMLElement) stanza: The message or presence stanza * @param { XMLElement } stanza - The message or presence stanza containing the status codes
* containing the status codes.
*/ */
showStatusMessages(stanza) {
const elements = sizzle(`x[xmlns="${Strophe.NS.MUC_USER}"]`, stanza); const elements = sizzle(`x[xmlns="${Strophe.NS.MUC_USER}"]`, stanza);
const is_self = stanza.querySelectorAll("status[code='110']").length; const is_self = stanza.querySelectorAll("status[code='110']").length;
...@@ -56634,13 +56632,11 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -56634,13 +56632,11 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
} else if (chatbox.get('type') === _converse.PRIVATE_CHAT_TYPE) { } else if (chatbox.get('type') === _converse.PRIVATE_CHAT_TYPE) {
const their_devices = await getDevicesForContact(chatbox.get('jid')), const their_devices = await getDevicesForContact(chatbox.get('jid')),
devicelist = _converse.devicelists.get(_converse.bare_jid), devicelist = _converse.devicelists.get(_converse.bare_jid),
own_devices = devicelist.devices.filter(d => d.get('id') !== id); own_devices = devicelist.devices;
devices = _.concat(own_devices, their_devices.models); devices = _.concat(own_devices, their_devices.models);
} // Filter out our own device }
devices = devices.filter(d => d.get('id') !== id);
await Promise.all(devices.map(d => d.getBundle())); await Promise.all(devices.map(d => d.getBundle()));
await Promise.all(devices.map(d => getSession(d))); await Promise.all(devices.map(d => getSession(d)));
return devices; return devices;
...@@ -58158,6 +58154,12 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins ...@@ -58158,6 +58154,12 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins
} }
}); });
/**
* @class
* @namespace _converse.RegisterPanel
* @memberOf _converse
*/
_converse.RegisterPanel = Backbone.NativeView.extend({ _converse.RegisterPanel = Backbone.NativeView.extend({
tagName: 'div', tagName: 'div',
id: "converse-register-panel", id: "converse-register-panel",
...@@ -58203,20 +58205,21 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins ...@@ -58203,20 +58205,21 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins
if (!this._registering) { if (!this._registering) {
connect_cb(req, callback, raw); connect_cb(req, callback, raw);
} else { } else {
if (this.getRegistrationFields(req, callback, raw)) { if (this.getRegistrationFields(req, callback)) {
this._registering = false; this._registering = false;
} }
} }
}; };
}, },
getRegistrationFields(req, _callback, raw) { /**
/* Send an IQ stanza to the XMPP server asking for the * Send an IQ stanza to the XMPP server asking for the registration fields.
* registration fields. * @private
* Parameters: * @method _converse.RegisterPanel#getRegistrationFields
* (Strophe.Request) req - The current request * @param { Strophe.Request } req - The current request
* (Function) callback * @param { Function } callback - The callback function
*/ */
getRegistrationFields(req, _callback) {
const conn = _converse.connection; const conn = _converse.connection;
conn.connected = true; conn.connected = true;
...@@ -58261,12 +58264,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins ...@@ -58261,12 +58264,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins
return true; return true;
}, },
onRegistrationFields(stanza) { /**
/* Handler for Registration Fields Request. * Handler for {@link _converse.RegisterPanel#getRegistrationFields}
* * @private
* Parameters: * @method _converse.RegisterPanel#onRegistrationFields
* (XMLElement) elem - The query stanza. * @param { XMLElement } stanza - The query stanza.
*/ */
onRegistrationFields(stanza) {
if (stanza.getAttribute("type") === "error") { if (stanza.getAttribute("type") === "error") {
_converse.connection._changeConnectStatus(Strophe.Status.REGIFAIL, __('Something went wrong while establishing a connection with "%1$s". ' + 'Are you sure it exists?', this.domain)); _converse.connection._changeConnectStatus(Strophe.Status.REGIFAIL, __('Something went wrong while establishing a connection with "%1$s". ' + 'Are you sure it exists?', this.domain));
...@@ -58325,13 +58329,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins ...@@ -58325,13 +58329,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins
} }
}, },
onProviderChosen(form) { /**
/* Callback method that gets called when the user has chosen an * Callback method that gets called when the user has chosen an XMPP provider
* XMPP provider. * @private
* * @method _converse.RegisterPanel#onProviderChosen
* Parameters: * @param { HTMLElement } form - The form that was submitted
* (HTMLElement) form - The form that was submitted
*/ */
onProviderChosen(form) {
const domain_input = form.querySelector('input[name=domain]'), const domain_input = form.querySelector('input[name=domain]'),
domain = _.get(domain_input, 'value'); domain = _.get(domain_input, 'value');
...@@ -58345,13 +58349,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins ...@@ -58345,13 +58349,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins
this.fetchRegistrationForm(domain.trim()); this.fetchRegistrationForm(domain.trim());
}, },
fetchRegistrationForm(domain_name) { /**
/* This is called with a domain name based on which, it fetches a * Fetch a registration form from the requested domain
* registration form from the requested domain. * @private
* * @method _converse.RegisterPanel#fetchRegistrationForm
* Parameters: * @param { String } domain_name - XMPP server domain
* (String) domain_name - XMPP server domain
*/ */
fetchRegistrationForm(domain_name) {
if (!this.model.get('registration_form_rendered')) { if (!this.model.get('registration_form_rendered')) {
this.renderRegistrationRequest(); this.renderRegistrationRequest();
} }
...@@ -58407,16 +58411,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins ...@@ -58407,16 +58411,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins
return this; return this;
}, },
onConnectStatusChanged(status_code) { /**
/* Callback function called by Strophe whenever the * Callback function called by Strophe whenever the connection status changes.
* connection status changes. * Passed to Strophe specifically during a registration attempt.
* * @private
* Passed to Strophe specifically during a registration * @method _converse.RegisterPanel#onConnectStatusChanged
* attempt. * @param { integer } status_code - The Strophe.Status status code
*
* Parameters:
* (Integer) status_code - The Stroph.Status status code
*/ */
onConnectStatusChanged(status_code) {
_converse.log('converse-register: onConnectStatusChanged'); _converse.log('converse-register: onConnectStatusChanged');
if (_.includes([Strophe.Status.DISCONNECTED, Strophe.Status.CONNFAIL, Strophe.Status.REGIFAIL, Strophe.Status.NOTACCEPTABLE, Strophe.Status.CONFLICT], status_code)) { if (_.includes([Strophe.Status.DISCONNECTED, Strophe.Status.CONNFAIL, Strophe.Status.REGIFAIL, Strophe.Status.NOTACCEPTABLE, Strophe.Status.CONFLICT], status_code)) {
...@@ -58480,13 +58482,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins ...@@ -58480,13 +58482,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins
}); });
}, },
renderRegistrationForm(stanza) { /**
/* Renders the registration form based on the XForm fields * Renders the registration form based on the XForm fields
* received from the XMPP server. * received from the XMPP server.
* * @private
* Parameters: * @method _converse.RegisterPanel#renderRegistrationForm
* (XMLElement) stanza - The IQ stanza received from the XMPP server. * @param { XMLElement } stanza - The IQ stanza received from the XMPP server.
*/ */
renderRegistrationForm(stanza) {
const form = this.el.querySelector('form'); const form = this.el.querySelector('form');
form.innerHTML = templates_registration_form_html__WEBPACK_IMPORTED_MODULE_6___default()({ form.innerHTML = templates_registration_form_html__WEBPACK_IMPORTED_MODULE_6___default()({
'__': _converse.__, '__': _converse.__,
...@@ -58536,14 +58539,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins ...@@ -58536,14 +58539,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins
flash.classList.remove('hidden'); flash.classList.remove('hidden');
}, },
reportErrors(stanza) { /**
/* Report back to the user any error messages received from the * Report back to the user any error messages received from the
* XMPP server after attempted registration. * XMPP server after attempted registration.
* * @private
* Parameters: * @method _converse.RegisterPanel#reportErrors
* (XMLElement) stanza - The IQ stanza received from the * @param { XMLElement } stanza - The IQ stanza received from the XMPP server
* XMPP server.
*/ */
reportErrors(stanza) {
const errors = stanza.querySelectorAll('error'); const errors = stanza.querySelectorAll('error');
_.each(errors, error => { _.each(errors, error => {
...@@ -58583,14 +58586,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins ...@@ -58583,14 +58586,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins
} }
}, },
submitRegistrationForm(form) { /**
/* Handler, when the user submits the registration form. * Handler, when the user submits the registration form.
* Provides form error feedback or starts the registration * Provides form error feedback or starts the registration process.
* process. * @private
* * @method _converse.RegisterPanel#submitRegistrationForm
* Parameters: * @param { HTMLElement } form - The HTML form that was submitted
* (HTMLElement) form - The HTML form that was submitted
*/ */
submitRegistrationForm(form) {
const has_empty_inputs = _.reduce(this.el.querySelectorAll('input.required'), function (result, input) { const has_empty_inputs = _.reduce(this.el.querySelectorAll('input.required'), function (result, input) {
if (input.value === '') { if (input.value === '') {
input.classList.add('error'); input.classList.add('error');
...@@ -58634,13 +58637,12 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins ...@@ -58634,13 +58637,12 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins
this.setFields(iq.tree()); this.setFields(iq.tree());
}, },
setFields(stanza) { /* Stores the values that will be sent to the XMPP server during attempted registration.
/* Stores the values that will be sent to the XMPP server * @private
* during attempted registration. * @method _converse.RegisterPanel#setFields
* * @param { XMLElement } stanza - the IQ stanza that will be sent to the XMPP server.
* Parameters:
* (XMLElement) stanza - the IQ stanza that will be sent to the XMPP server.
*/ */
setFields(stanza) {
const query = stanza.querySelector('query'); const query = stanza.querySelector('query');
const xform = sizzle(`x[xmlns="${Strophe.NS.XFORM}"]`, query); const xform = sizzle(`x[xmlns="${Strophe.NS.XFORM}"]`, query);
...@@ -58688,14 +58690,15 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins ...@@ -58688,14 +58690,15 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_1__["default"].plugins
this.form_type = 'xform'; this.form_type = 'xform';
}, },
_onRegisterIQ(stanza) { /**
/* Callback method that gets called when a return IQ stanza * Callback method that gets called when a return IQ stanza
* is received from the XMPP server, after attempting to * is received from the XMPP server, after attempting to
* register a new user. * register a new user.
* * @private
* Parameters: * @method _converse.RegisterPanel#reportErrors
* (XMLElement) stanza - The IQ stanza. * @param { XMLElement } stanza - The IQ stanza.
*/ */
_onRegisterIQ(stanza) {
if (stanza.getAttribute("type") === "error") { if (stanza.getAttribute("type") === "error") {
_converse.log("Registration failed.", Strophe.LogLevel.ERROR); _converse.log("Registration failed.", Strophe.LogLevel.ERROR);
...@@ -62162,13 +62165,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha ...@@ -62162,13 +62165,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha
return false; return false;
}, },
createMessageStanza(message) { /**
/* Given a _converse.Message Backbone.Model, return the XML * Given a {@link _converse.Message} return the XML stanza that represents it.
* stanza that represents it. * @private
* * @method _converse.ChatBox#createMessageStanza
* Parameters: * @param { _converse.Message } message - The message object
* (Object) message - The Backbone.Model representing the message
*/ */
createMessageStanza(message) {
const stanza = $msg({ const stanza = $msg({
'from': _converse.connection.jid, 'from': _converse.connection.jid,
'to': this.get('jid'), 'to': this.get('jid'),
...@@ -62378,13 +62381,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha ...@@ -62378,13 +62381,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha
}); });
}, },
getStanzaIDs(stanza) { /**
/* Extract the XEP-0359 stanza IDs from the passed in stanza * Extract the XEP-0359 stanza IDs from the passed in stanza
* and return a map containing them. * and return a map containing them.
* * @private
* Parameters: * @method _converse.ChatBox#getStanzaIDs
* (XMLElement) stanza - The message stanza * @param { XMLElement } stanza - The message stanza
*/ */
getStanzaIDs(stanza) {
const attrs = {}; const attrs = {};
const stanza_ids = sizzle(`stanza-id[xmlns="${Strophe.NS.SID}"]`, stanza); const stanza_ids = sizzle(`stanza-id[xmlns="${Strophe.NS.SID}"]`, stanza);
...@@ -62412,18 +62416,17 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha ...@@ -62412,18 +62416,17 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha
return !_.isNil(sizzle(`result[xmlns="${Strophe.NS.MAM}"]`, original_stanza).pop()); return !_.isNil(sizzle(`result[xmlns="${Strophe.NS.MAM}"]`, original_stanza).pop());
}, },
getMessageAttributesFromStanza(stanza, original_stanza) { /**
/* Parses a passed in message stanza and returns an object * Parses a passed in message stanza and returns an object
* of attributes. * of attributes.
* * @private
* Parameters: * @method _converse.ChatBox#getMessageAttributesFromStanza
* (XMLElement) stanza - The message stanza * @param { XMLElement } stanza - The message stanza
* (XMLElement) delay - The <delay> node from the * @param { XMLElement } delay - The <delay> node from the stanza, if there was one.
* stanza, if there was one. * @param { XMLElement } original_stanza - The original stanza, that contains the
* (XMLElement) original_stanza - The original stanza, * message stanza, if it was contained, otherwise it's the message stanza itself.
* that contains the message stanza, if it was
* contained, otherwise it's the message stanza itself.
*/ */
getMessageAttributesFromStanza(stanza, original_stanza) {
const spoiler = sizzle(`spoiler[xmlns="${Strophe.NS.SPOILER}"]`, original_stanza).pop(), const spoiler = sizzle(`spoiler[xmlns="${Strophe.NS.SPOILER}"]`, original_stanza).pop(),
delay = sizzle(`delay[xmlns="${Strophe.NS.DELAY}"]`, original_stanza).pop(), delay = sizzle(`delay[xmlns="${Strophe.NS.DELAY}"]`, original_stanza).pop(),
text = _converse.chatboxes.getMessageBody(stanza) || undefined, text = _converse.chatboxes.getMessageBody(stanza) || undefined,
...@@ -62634,13 +62637,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha ...@@ -62634,13 +62637,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha
} }
}, },
async onMessage(stanza) { /**
/* Handler method for all incoming single-user chat "message" * Handler method for all incoming single-user chat "message" stanzas.
* stanzas. * @private
* * @method _converse.ChatBox#onMessage
* Parameters: * @param { XMLElement } stanza - The incoming message stanza
* (XMLElement) stanza - The incoming message stanza
*/ */
async onMessage(stanza) {
let to_jid = stanza.getAttribute('to'); let to_jid = stanza.getAttribute('to');
const to_resource = Strophe.getResourceFromJid(to_jid); const to_resource = Strophe.getResourceFromJid(to_jid);
...@@ -62742,18 +62745,19 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha ...@@ -62742,18 +62745,19 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha
}); });
}, },
/**
* Returns a chat box or optionally return a newly
* created one if one doesn't exist.
* @private
* @method _converse.ChatBox#getChatBox
* @param { string } jid - The JID of the user whose chat box we want
* @param { boolean } create - Should a new chat box be created if none exists?
* @param { object } attrs - Optional chat box atributes.
*/
getChatBox(jid) { getChatBox(jid) {
let attrs = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; let attrs = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
let create = arguments.length > 2 ? arguments[2] : undefined; let create = arguments.length > 2 ? arguments[2] : undefined;
/* Returns a chat box or optionally return a newly
* created one if one doesn't exist.
*
* Parameters:
* (String) jid - The JID of the user whose chat box we want
* (Boolean) create - Should a new chat box be created if none exists?
* (Object) attrs - Optional chat box atributes.
*/
if (_.isObject(jid)) { if (_.isObject(jid)) {
create = attrs; create = attrs;
attrs = jid; attrs = jid;
...@@ -63220,23 +63224,22 @@ _converse.default_settings = { ...@@ -63220,23 +63224,22 @@ _converse.default_settings = {
websocket_url: undefined, websocket_url: undefined,
whitelisted_plugins: [] whitelisted_plugins: []
}; };
/**
_converse.log = function (message, level) { * Logs messages to the browser's developer console.
let style = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : '';
/* 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.
*
* Available loglevels are 0 for 'debug', 1 for 'info', 2 for 'warn', * Available loglevels are 0 for 'debug', 1 for 'info', 2 for 'warn',
* 3 for 'error' and 4 for 'fatal'. * 3 for 'error' and 4 for 'fatal'.
*
* When using the 'error' or 'warn' loglevels, a full stacktrace will be * When using the 'error' or 'warn' loglevels, a full stacktrace will be
* logged as well. * logged as well.
* @method log
* @private
* @memberOf _converse
* @param { string } message - The message to be logged
* @param { integer } level - The loglevel which allows for filtering of log messages
*/ */
_converse.log = function (message, level) {
let style = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : '';
if (level === strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].LogLevel.ERROR || level === strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].LogLevel.FATAL) { if (level === strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].LogLevel.ERROR || level === strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].LogLevel.FATAL) {
style = style || 'color: maroon'; style = style || 'color: maroon';
} }
...@@ -63280,13 +63283,17 @@ strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].log = function (level, msg) { ...@@ -63280,13 +63283,17 @@ strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].log = function (level, msg) {
strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].error = function (msg) { strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].error = function (msg) {
_converse.log(msg, strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].LogLevel.ERROR); _converse.log(msg, strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].LogLevel.ERROR);
}; };
/**
* Translate the given string based on the current locale.
* Handles all MUC presence stanzas.
* @method __
* @private
* @memberOf _converse
* @param { String } str - The string to translate
*/
_converse.__ = function (str) { _converse.__ = function (str) {
/* Translate the given string based on the current locale.
*
* Parameters:
* (String) str - The string to translate.
*/
if (_lodash_noconflict__WEBPACK_IMPORTED_MODULE_4___default.a.isUndefined(_i18n__WEBPACK_IMPORTED_MODULE_6__["default"])) { if (_lodash_noconflict__WEBPACK_IMPORTED_MODULE_4___default.a.isUndefined(_i18n__WEBPACK_IMPORTED_MODULE_6__["default"])) {
return str; return str;
} }
...@@ -63574,13 +63581,16 @@ _converse.initialize = async function (settings, callback) { ...@@ -63574,13 +63581,16 @@ _converse.initialize = async function (settings, callback) {
// ---------------------- // ----------------------
this.generateResource = () => `/converse.js-${Math.floor(Math.random() * 139749528).toString()}`; this.generateResource = () => `/converse.js-${Math.floor(Math.random() * 139749528).toString()}`;
/**
* Send out a Chat Status Notification (XEP-0352)
* @private
* @method sendCSI
* @memberOf _converse
* @param { String } stat - The user's chat status
*/
this.sendCSI = function (stat) { this.sendCSI = function (stat) {
/* Send out a Chat Status Notification (XEP-0352)
*
* Parameters:
* (String) stat: The user's chat status
*/
_converse.api.send(Object(strophe_js__WEBPACK_IMPORTED_MODULE_0__["$build"])(stat, { _converse.api.send(Object(strophe_js__WEBPACK_IMPORTED_MODULE_0__["$build"])(stat, {
xmlns: strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].NS.CSI xmlns: strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].NS.CSI
})); }));
...@@ -63683,15 +63693,17 @@ _converse.initialize = async function (settings, callback) { ...@@ -63683,15 +63693,17 @@ _converse.initialize = async function (settings, callback) {
'message': message 'message': message
}); });
}; };
/**
* Reject or cancel another user's subscription to our presence updates.
* @method rejectPresenceSubscription
* @private
* @memberOf _converse
* @param { String } jid - The Jabber ID of the user whose subscription is being canceled
* @param { String } message - An optional message to the user
*/
this.rejectPresenceSubscription = function (jid, message) { this.rejectPresenceSubscription = function (jid, message) {
/* Reject or cancel another user's subscription to our presence updates.
*
* Parameters:
* (String) jid - The Jabber ID of the user whose subscription
* is being canceled.
* (String) message - An optional message to the user
*/
const pres = Object(strophe_js__WEBPACK_IMPORTED_MODULE_0__["$pres"])({ const pres = Object(strophe_js__WEBPACK_IMPORTED_MODULE_0__["$pres"])({
to: jid, to: jid,
type: "unsubscribed" type: "unsubscribed"
...@@ -65101,6 +65113,12 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-dis ...@@ -65101,6 +65113,12 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-dis
const _converse = this._converse; // Promises exposed by this plugin const _converse = this._converse; // Promises exposed by this plugin
_converse.api.promises.add('discoInitialized'); _converse.api.promises.add('discoInitialized');
/**
* @class
* @namespace _converse.DiscoEntity
* @memberOf _converse
*/
_converse.DiscoEntity = Backbone.Model.extend({ _converse.DiscoEntity = Backbone.Model.extend({
/* A Disco Entity is a JID addressable entity that can be queried /* A Disco Entity is a JID addressable entity that can be queried
...@@ -65128,14 +65146,15 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-dis ...@@ -65128,14 +65146,15 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-dis
this.items.fetch(); this.items.fetch();
}, },
async getIdentity(category, type) { /**
/* Returns a Promise which resolves with a map indicating * Returns a Promise which resolves with a map indicating
* whether a given identity is provided by this entity. * whether a given identity is provided by this entity.
* * @private
* Parameters: * @method _converse.DiscoEntity#getIdentity
* (String) category - The identity category * @param { String } category - The identity category
* (String) type - The identity type * @param { String } type - The identity type
*/ */
async getIdentity(category, type) {
await this.waitUntilFeaturesDiscovered; await this.waitUntilFeaturesDiscovered;
return this.identities.findWhere({ return this.identities.findWhere({
'category': category, 'category': category,
...@@ -65143,13 +65162,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-dis ...@@ -65143,13 +65162,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-dis
}); });
}, },
async hasFeature(feature) { /**
/* Returns a Promise which resolves with a map indicating * Returns a Promise which resolves with a map indicating
* whether a given feature is supported. * whether a given feature is supported.
* * @private
* Parameters: * @method _converse.DiscoEntity#hasFeature
* (String) feature - The feature that might be supported. * @param { String } feature - The feature that might be supported.
*/ */
async hasFeature(feature) {
await this.waitUntilFeaturesDiscovered; await this.waitUntilFeaturesDiscovered;
if (this.features.findWhere({ if (this.features.findWhere({
...@@ -66655,14 +66675,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -66655,14 +66675,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return this.get('name') || this.get('jid'); return this.get('name') || this.get('jid');
}, },
join(nick, password) { /**
/* Join the groupchat. * Join the groupchat.
* * @private
* Parameters: * @method _converse.ChatRoom#join
* (String) nick: The user's nickname * @param { String } nick - The user's nickname
* (String) password: Optional password, if required by * @param { String } password - Optional password, if required by the groupchat.
* the groupchat.
*/ */
join(nick, password) {
nick = nick ? nick : this.get('nick'); nick = nick ? nick : this.get('nick');
if (!nick) { if (!nick) {
...@@ -66695,13 +66715,12 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -66695,13 +66715,12 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return this; return this;
}, },
leave(exit_msg) {
/* Leave the groupchat. /* Leave the groupchat.
* * @private
* Parameters: * @method _converse.ChatRoom#leave
* (String) exit_msg: Optional message to indicate your * @param { string } exit_msg - Optional message to indicate your reason for leaving
* reason for leaving.
*/ */
leave(exit_msg) {
this.features.destroy(); this.features.destroy();
this.occupants.browserStorage._clear(); this.occupants.browserStorage._clear();
...@@ -66892,12 +66911,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -66892,12 +66911,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
})); }));
}, },
directInvite(recipient, reason) { /**
/* Send a direct invitation as per XEP-0249 * Send a direct invitation as per XEP-0249
* Parameters: * @private
* (String) recipient - JID of the person being invited * @method _converse.ChatRoom#directInvite
* (String) reason - Optional reason for the invitation * @param { String } recipient - JID of the person being invited
* @param { String } reason - Optional reason for the invitation
*/ */
directInvite(recipient, reason) {
if (this.features.get('membersonly')) { if (this.features.get('membersonly')) {
// When inviting to a members-only groupchat, we first add // When inviting to a members-only groupchat, we first add
// the person to the member list by giving them an // the person to the member list by giving them an
...@@ -66998,20 +67019,17 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -66998,20 +67019,17 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
this.features.save(attrs); this.features.save(attrs);
}, },
requestMemberList(affiliation) {
/* Send an IQ stanza to the server, asking it for the /* Send an IQ stanza to the server, asking it for the
* member-list of this groupchat. * member-list of this groupchat.
*
* See: https://xmpp.org/extensions/xep-0045.html#modifymember * See: https://xmpp.org/extensions/xep-0045.html#modifymember
* * @private
* Parameters: * @method _converse.ChatRoom#requestMemberList
* (String) affiliation: The specific member list to * @param { string } affiliation - The specific member list to
* fetch. 'admin', 'owner' or 'member'. * fetch. 'admin', 'owner' or 'member'.
* * @returns:
* Returns: * A promise which resolves once the list has been retrieved.
* A promise which resolves once the list has been
* retrieved.
*/ */
requestMemberList(affiliation) {
affiliation = affiliation || 'member'; affiliation = affiliation || 'member';
const iq = $iq({ const iq = $iq({
to: this.get('jid'), to: this.get('jid'),
...@@ -67024,28 +67042,26 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67024,28 +67042,26 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return _converse.api.sendIQ(iq); return _converse.api.sendIQ(iq);
}, },
setAffiliation(affiliation, members) { /**
/* Send IQ stanzas to the server to set an affiliation for * Send IQ stanzas to the server to set an affiliation for
* the provided JIDs. * the provided JIDs.
*
* See: https://xmpp.org/extensions/xep-0045.html#modifymember * See: https://xmpp.org/extensions/xep-0045.html#modifymember
* *
* XXX: Prosody doesn't accept multiple JIDs' affiliations * Prosody doesn't accept multiple JIDs' affiliations
* being set in one IQ stanza, so as a workaround we send * being set in one IQ stanza, so as a workaround we send
* a separate stanza for each JID. * a separate stanza for each JID.
* Related ticket: https://issues.prosody.im/345 * Related ticket: https://issues.prosody.im/345
* *
* Parameters: * @private
* (String) affiliation: The affiliation * @method _converse.ChatRoom#setAffiliation
* (Object) members: A map of jids, affiliations and * @param { string } affiliation - The affiliation
* @param { object } members - A map of jids, affiliations and
* optionally reasons. Only those entries with the * optionally reasons. Only those entries with the
* same affiliation as being currently set will be * same affiliation as being currently set will be considered.
* considered. * @returns
* * A promise which resolves and fails depending on the XMPP server response.
* Returns:
* A promise which resolves and fails depending on the
* XMPP server response.
*/ */
setAffiliation(affiliation, members) {
members = _.filter(members, member => // We only want those members who have the right members = _.filter(members, member => // We only want those members who have the right
// affiliation (or none, which implies the provided one). // affiliation (or none, which implies the provided one).
_.isUndefined(member.affiliation) || member.affiliation === affiliation); _.isUndefined(member.affiliation) || member.affiliation === affiliation);
...@@ -67055,18 +67071,19 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67055,18 +67071,19 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return Promise.all(promises); return Promise.all(promises);
}, },
saveConfiguration(form) { /**
/* Submit the groupchat configuration form by sending an IQ * Submit the groupchat configuration form by sending an IQ
* stanza to the server. * stanza to the server.
* * @private
* Returns a promise which resolves once the XMPP server * @method _converse.ChatRoom#saveConfiguration
* has return a response IQ. * @param { HTMLElement } form - The configuration form DOM element.
*
* Parameters:
* (HTMLElement) form: The configuration form DOM element.
* If no form is provided, the default configuration * If no form is provided, the default configuration
* values will be used. * values will be used.
* @returns { promise }
* Returns a promise which resolves once the XMPP server
* has return a response IQ.
*/ */
saveConfiguration(form) {
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
const inputs = form ? sizzle(':input:not([type=button]):not([type=submit])', form) : [], const inputs = form ? sizzle(':input:not([type=button]):not([type=submit])', form) : [],
configArray = _.map(inputs, _utils_form__WEBPACK_IMPORTED_MODULE_4__["default"].webForm2xForm); configArray = _.map(inputs, _utils_form__WEBPACK_IMPORTED_MODULE_4__["default"].webForm2xForm);
...@@ -67135,20 +67152,21 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67135,20 +67152,21 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
})); }));
}, },
sendConfiguration(config, callback, errback) { /**
/* Send an IQ stanza with the groupchat configuration. * Send an IQ stanza with the groupchat configuration.
* * @private
* Parameters: * @method _converse.ChatRoom#sendConfiguration
* (Array) config: The groupchat configuration * @param { Array } config - The groupchat configuration
* (Function) callback: Callback upon succesful IQ response * @param { Function } callback - Callback upon succesful IQ response
* The first parameter passed in is IQ containing the * The first parameter passed in is IQ containing the
* groupchat configuration. * groupchat configuration.
* The second is the response IQ from the server. * The second is the response IQ from the server.
* (Function) errback: Callback upon error IQ response * @param { Function } errback - Callback upon error IQ response
* The first parameter passed in is IQ containing the * The first parameter passed in is IQ containing the
* groupchat configuration. * groupchat configuration.
* The second is the response IQ from the server. * The second is the response IQ from the server.
*/ */
sendConfiguration(config, callback, errback) {
const iq = $iq({ const iq = $iq({
to: this.get('jid'), to: this.get('jid'),
type: "set" type: "set"
...@@ -67168,13 +67186,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67168,13 +67186,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return _converse.api.sendIQ(iq).then(callback).catch(errback); return _converse.api.sendIQ(iq).then(callback).catch(errback);
}, },
saveAffiliationAndRole(pres) { /**
/* Parse the presence stanza for the current user's * Parse the presence stanza for the current user's affiliation.
* affiliation. * @private
* * @method _converse.ChatRoom#saveAffiliationAndRole
* Parameters: * @param { XMLElement } pres - A <presence> stanza.
* (XMLElement) pres: A <presence> stanza.
*/ */
saveAffiliationAndRole(pres) {
const item = sizzle(`x[xmlns="${Strophe.NS.MUC_USER}"] item`, pres).pop(); const item = sizzle(`x[xmlns="${Strophe.NS.MUC_USER}"] item`, pres).pop();
const is_self = pres.querySelector("status[code='110']"); const is_self = pres.querySelector("status[code='110']");
...@@ -67196,15 +67214,16 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67196,15 +67214,16 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
} }
}, },
sendAffiliationIQ(affiliation, member) { /**
/* Send an IQ stanza specifying an affiliation change. * Send an IQ stanza specifying an affiliation change.
* * @private
* Paremeters: * @method _converse.ChatRoom#
* (String) affiliation: affiliation (could also be stored * @param { String } affiliation: affiliation
* on the member object). * (could also be stored on the member object).
* (Object) member: Map containing the member's jid and * @param { Object } member: Map containing the member's jid and
* optionally a reason and affiliation. * optionally a reason and affiliation.
*/ */
sendAffiliationIQ(affiliation, member) {
const iq = $iq({ const iq = $iq({
to: this.get('jid'), to: this.get('jid'),
type: "set" type: "set"
...@@ -67223,17 +67242,17 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67223,17 +67242,17 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return _converse.api.sendIQ(iq); return _converse.api.sendIQ(iq);
}, },
setAffiliations(members) { /**
/* Send IQ stanzas to the server to modify the * Send IQ stanzas to the server to modify the
* affiliations in this groupchat. * affiliations in this groupchat.
*
* See: https://xmpp.org/extensions/xep-0045.html#modifymember * See: https://xmpp.org/extensions/xep-0045.html#modifymember
* * @private
* Parameters: * @method _converse.ChatRoom#setAffiliations
* (Object) members: A map of jids, affiliations and optionally reasons * @param { object } members - A map of jids, affiliations and optionally reasons
* (Function) onSuccess: callback for a succesful response * @param { function } onSuccess - callback for a succesful response
* (Function) onError: callback for an error response * @param { function } onError - callback for an error response
*/ */
setAffiliations(members) {
const affiliations = _.uniq(_.map(members, 'affiliation')); const affiliations = _.uniq(_.map(members, 'affiliation'));
return Promise.all(_.map(affiliations, _.partial(this.setAffiliation.bind(this), _, members))); return Promise.all(_.map(affiliations, _.partial(this.setAffiliation.bind(this), _, members)));
...@@ -67253,36 +67272,36 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67253,36 +67272,36 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return [].concat.apply([], result).filter(p => p); return [].concat.apply([], result).filter(p => p);
}, },
updateMemberLists(members, affiliations, deltaFunc) { /**
/* Fetch the lists of users with the given affiliations. * Fetch the lists of users with the given affiliations.
* Then compute the delta between those users and * Then compute the delta between those users and
* the passed in members, and if it exists, send the delta * the passed in members, and if it exists, send the delta
* to the XMPP server to update the member list. * to the XMPP server to update the member list.
* * @private
* Parameters: * @method _converse.ChatRoom#updateMemberLists
* (Object) members: Map of member jids and affiliations. * @param { object } members - Map of member jids and affiliations.
* (String|Array) affiliation: An array of affiliations or * @param { string|array } affiliation - An array of affiliations or
* a string if only one affiliation. * a string if only one affiliation.
* (Function) deltaFunc: The function to compute the delta * @param { function } deltaFunc - The function to compute the delta
* between old and new member lists. * between old and new member lists.
* * @returns { promise }
* Returns:
* A promise which is resolved once the list has been * A promise which is resolved once the list has been
* updated or once it's been established there's no need * updated or once it's been established there's no need
* to update the list. * to update the list.
*/ */
updateMemberLists(members, affiliations, deltaFunc) {
this.getJidsWithAffiliations(affiliations).then(old_members => this.setAffiliations(deltaFunc(members, old_members))).then(() => this.occupants.fetchMembers()).catch(_.partial(_converse.log, _, Strophe.LogLevel.ERROR)); this.getJidsWithAffiliations(affiliations).then(old_members => this.setAffiliations(deltaFunc(members, old_members))).then(() => this.occupants.fetchMembers()).catch(_.partial(_converse.log, _, Strophe.LogLevel.ERROR));
}, },
async checkForReservedNick() { /**
/* Use service-discovery to ask the XMPP server whether * Use service-discovery to ask the XMPP server whether
* this user has a reserved nickname for this groupchat. * this user has a reserved nickname for this groupchat.
* If so, we'll use that, otherwise we render the nickname form. * If so, we'll use that, otherwise we render the nickname form.
* * @private
* Parameters: * @method _converse.ChatRoom#checkForReservedNick
* (Function) callback: Callback upon succesful IQ response * @returns { promise } A promise which resolves with the response IQ
* (Function) errback: Callback upon error IQ response
*/ */
async checkForReservedNick() {
const iq = await _converse.api.sendIQ($iq({ const iq = await _converse.api.sendIQ($iq({
'to': this.get('jid'), 'to': this.get('jid'),
'from': _converse.connection.jid, 'from': _converse.connection.jid,
...@@ -67364,13 +67383,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67364,13 +67383,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
} }
}, },
updateOccupantsOnPresence(pres) { /**
/* Given a presence stanza, update the occupant model * Given a presence stanza, update the occupant model
* based on its contents. * based on its contents.
* * @private
* Parameters: * @method _converse.ChatRoom#updateOccupantsOnPresence
* (XMLElement) pres: The presence stanza * @param { XMLElement } pres - The presence stanza
*/ */
updateOccupantsOnPresence(pres) {
const data = this.parsePresence(pres); const data = this.parsePresence(pres);
if (data.type === 'error' || !data.jid && !data.nick) { if (data.type === 'error' || !data.jid && !data.nick) {
...@@ -67477,12 +67497,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67477,12 +67497,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
acknowledged[xmlns="${Strophe.NS.MARKERS}"]`, stanza).length > 0; acknowledged[xmlns="${Strophe.NS.MARKERS}"]`, stanza).length > 0;
}, },
subjectChangeHandled(attrs) { /**
/* Handle a subject change and return `true` if so. * Handle a subject change and return `true` if so.
* * @private
* Parameters: * @method _converse.ChatRoom#subjectChangeHandled
* (Object) attrs: The message attributes * @param { object } attrs - The message attributes
*/ */
subjectChangeHandled(attrs) {
if (attrs.subject && !attrs.thread && !attrs.message) { if (attrs.subject && !attrs.thread && !attrs.message) {
// https://xmpp.org/extensions/xep-0045.html#subject-mod // https://xmpp.org/extensions/xep-0045.html#subject-mod
// ----------------------------------------------------- // -----------------------------------------------------
...@@ -67501,13 +67522,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67501,13 +67522,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return false; return false;
}, },
ignorableCSN(attrs) { /**
/* Is this a chat state notification that can be ignored, * Is this a chat state notification that can be ignored,
* because it's old or because it's from us. * because it's old or because it's from us.
* * @private
* Parameters: * @method _converse.ChatRoom#ignorableCSN
* (Object) attrs: The message attributes * @param { Object } attrs - The message attributes
*/ */
ignorableCSN(attrs) {
const is_csn = _utils_form__WEBPACK_IMPORTED_MODULE_4__["default"].isOnlyChatStateNotification(attrs), const is_csn = _utils_form__WEBPACK_IMPORTED_MODULE_4__["default"].isOnlyChatStateNotification(attrs),
own_message = Strophe.getResourceFromJid(attrs.from) == this.get('nick'); own_message = Strophe.getResourceFromJid(attrs.from) == this.get('nick');
return is_csn && (attrs.is_delayed || own_message); return is_csn && (attrs.is_delayed || own_message);
...@@ -67537,12 +67559,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67537,12 +67559,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return attrs; return attrs;
}, },
async onMessage(stanza) { /**
/* Handler for all MUC messages sent to this groupchat. * Handler for all MUC messages sent to this groupchat.
* * @private
* Parameters: * @method _converse.ChatRoom#onMessage
* (XMLElement) stanza: The message stanza. * @param { XMLElement } stanza - The message stanza.
*/ */
async onMessage(stanza) {
this.fetchFeaturesIfConfigurationChanged(stanza); this.fetchFeaturesIfConfigurationChanged(stanza);
const original_stanza = stanza, const original_stanza = stanza,
forwarded = sizzle(`forwarded[xmlns="${Strophe.NS.FORWARD}"]`, stanza).pop(); forwarded = sizzle(`forwarded[xmlns="${Strophe.NS.FORWARD}"]`, stanza).pop();
...@@ -67582,12 +67605,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67582,12 +67605,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
}); });
}, },
onPresence(pres) { /**
/* Handles all MUC presence stanzas. * Handles all MUC presence stanzas.
* * @private
* Parameters: * @method _converse.ChatRoom#onPresence
* (XMLElement) pres: The stanza * @param { XMLElement } pres - The stanza
*/ */
onPresence(pres) {
if (pres.getAttribute('type') === 'error') { if (pres.getAttribute('type') === 'error') {
this.save('connection_status', _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].ROOMSTATUS.DISCONNECTED); this.save('connection_status', _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].ROOMSTATUS.DISCONNECTED);
return; return;
...@@ -67606,9 +67630,8 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67606,9 +67630,8 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
} }
}, },
onOwnPresence(pres) { /**
/* Handles a received presence relating to the current * Handles a received presence relating to the current user.
* user.
* *
* For locked groupchats (which are by definition "new"), the * For locked groupchats (which are by definition "new"), the
* groupchat will either be auto-configured or created instantly * groupchat will either be auto-configured or created instantly
...@@ -67618,10 +67641,11 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67618,10 +67641,11 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
* If the groupchat is not locked, then the groupchat will be * If the groupchat is not locked, then the groupchat will be
* auto-configured only if applicable and if the current * auto-configured only if applicable and if the current
* user is the groupchat's owner. * user is the groupchat's owner.
* * @private
* Parameters: * @method _converse.ChatRoom#onOwnPresence
* (XMLElement) pres: The stanza * @param { XMLElement } pres - The stanza
*/ */
onOwnPresence(pres) {
this.saveAffiliationAndRole(pres); this.saveAffiliationAndRole(pres);
const locked_room = pres.querySelector("status[code='201']"); const locked_room = pres.querySelector("status[code='201']");
...@@ -67659,13 +67683,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67659,13 +67683,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
this.save('connection_status', _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].ROOMSTATUS.ENTERED); this.save('connection_status', _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].ROOMSTATUS.ENTERED);
}, },
isUserMentioned(message) { /**
/* Returns a boolean to indicate whether the current user * Returns a boolean to indicate whether the current user
* was mentioned in a message. * was mentioned in a message.
* * @private
* Parameters: * @method _converse.ChatRoom#isUserMentioned
* (String): The text message * @param { String } - The text message
*/ */
isUserMentioned(message) {
const nick = this.get('nick'); const nick = this.get('nick');
if (message.get('references').length) { if (message.get('references').length) {
...@@ -67676,13 +67701,12 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67676,13 +67701,12 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
} }
}, },
incrementUnreadMsgCounter(message) { /* Given a newly received message, update the unread counter if necessary.
/* Given a newly received message, update the unread counter if * @private
* necessary. * @method _converse.ChatRoom#incrementUnreadMsgCounter
* * @param { XMLElement } - The <messsage> stanza
* Parameters:
* (XMLElement): The <messsage> stanza
*/ */
incrementUnreadMsgCounter(message) {
if (!message) { if (!message) {
return; return;
} }
...@@ -67848,15 +67872,15 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -67848,15 +67872,15 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
}; };
} }
}); });
/**
_converse.onDirectMUCInvitation = function (message) { * A direct MUC invitation to join a groupchat has been received
/* A direct MUC invitation to join a groupchat has been received
* See XEP-0249: Direct MUC invitations. * See XEP-0249: Direct MUC invitations.
* * @private
* Parameters: * @method _converse.ChatRoom#onDirectMUCInvitation
* (XMLElement) message: The message stanza containing the * @param { XMLElement } message - The message stanza containing the invitation.
* invitation.
*/ */
_converse.onDirectMUCInvitation = function (message) {
const x_el = sizzle('x[xmlns="jabber:x:conference"]', message).pop(), const x_el = sizzle('x[xmlns="jabber:x:conference"]', message).pop(),
from = Strophe.getBareJidFromJid(message.getAttribute('from')), from = Strophe.getBareJidFromJid(message.getAttribute('from')),
room_jid = x_el.getAttribute('jid'), room_jid = x_el.getAttribute('jid'),
...@@ -68511,11 +68535,15 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -68511,11 +68535,15 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
return true; return true;
}, null, 'presence', null); }, null, 'presence', null);
}; };
/**
_converse.initRoster = function () { * Initialize the Bakcbone collections that represent the contats
/* Initialize the Bakcbone collections that represent the contats
* roster and the roster groups. * roster and the roster groups.
* @private
* @method _converse.initRoster
*/ */
_converse.initRoster = function () {
const storage = _converse.config.get('storage'); const storage = _converse.config.get('storage');
_converse.roster = new _converse.RosterContacts(); _converse.roster = new _converse.RosterContacts();
...@@ -68540,18 +68568,20 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -68540,18 +68568,20 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
_converse.api.trigger('rosterInitialized'); _converse.api.trigger('rosterInitialized');
}; };
/**
_converse.populateRoster = async function () { * Fetch all the roster groups, and then the roster contacts.
let ignore_cache = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : false;
/* Fetch all the roster groups, and then the roster contacts.
* Emit an event after fetching is done in each case. * Emit an event after fetching is done in each case.
* * @private
* Parameters: * @method _converse.populateRoster
* (Bool) ignore_cache - If set to to true, the local cache * @param { Bool } ignore_cache - If set to to true, the local cache
* will be ignored it's guaranteed that the XMPP server * will be ignored it's guaranteed that the XMPP server
* will be queried for the roster. * will be queried for the roster.
*/ */
_converse.populateRoster = async function () {
let ignore_cache = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : false;
if (ignore_cache) { if (ignore_cache) {
_converse.send_initial_presence = true; _converse.send_initial_presence = true;
...@@ -68755,13 +68785,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -68755,13 +68785,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
return this.vcard.get('fullname'); return this.vcard.get('fullname');
}, },
subscribe(message) { /**
/* Send a presence subscription request to this roster contact * Send a presence subscription request to this roster contact
* * @private
* Parameters: * @method _converse.RosterContacts#subscribe
* (String) message - An optional message to explain the * @param { String } message - An optional message to explain the
* reason for the subscription request. * reason for the subscription request.
*/ */
subscribe(message) {
const pres = $pres({ const pres = $pres({
to: this.get('jid'), to: this.get('jid'),
type: "subscribe" type: "subscribe"
...@@ -68786,27 +68817,32 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -68786,27 +68817,32 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
return this; return this;
}, },
ackSubscribe() { /**
/* Upon receiving the presence stanza of type "subscribed", * Upon receiving the presence stanza of type "subscribed",
* the user SHOULD acknowledge receipt of that subscription * the user SHOULD acknowledge receipt of that subscription
* state notification by sending a presence stanza of type * state notification by sending a presence stanza of type
* "subscribe" to the contact * "subscribe" to the contact
* @private
* @method _converse.RosterContacts#ackSubscribe
*/ */
ackSubscribe() {
_converse.api.send($pres({ _converse.api.send($pres({
'type': 'subscribe', 'type': 'subscribe',
'to': this.get('jid') 'to': this.get('jid')
})); }));
}, },
ackUnsubscribe() { /**
/* Upon receiving the presence stanza of type "unsubscribed", * Upon receiving the presence stanza of type "unsubscribed",
* the user SHOULD acknowledge receipt of that subscription state * the user SHOULD acknowledge receipt of that subscription state
* notification by sending a presence stanza of type "unsubscribe" * notification by sending a presence stanza of type "unsubscribe"
* this step lets the user's server know that it MUST no longer * this step lets the user's server know that it MUST no longer
* send notification of the subscription state change to the user. * send notification of the subscription state change to the user.
* Parameters: * @private
* (String) jid - The Jabber ID of the user who is unsubscribing * @method _converse.RosterContacts#ackUnsubscribe
* @param { String } jid - The Jabber ID of the user who is unsubscribing
*/ */
ackUnsubscribe() {
_converse.api.send($pres({ _converse.api.send($pres({
'type': 'unsubscribe', 'type': 'unsubscribe',
'to': this.get('jid') 'to': this.get('jid')
...@@ -68816,21 +68852,25 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -68816,21 +68852,25 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
this.destroy(); this.destroy();
}, },
unauthorize(message) { /**
/* Unauthorize this contact's presence subscription * Unauthorize this contact's presence subscription
* Parameters: * @private
* (String) message - Optional message to send to the person being unauthorized * @method _converse.RosterContacts#unauthorize
* @param { String } message - Optional message to send to the person being unauthorized
*/ */
unauthorize(message) {
_converse.rejectPresenceSubscription(this.get('jid'), message); _converse.rejectPresenceSubscription(this.get('jid'), message);
return this; return this;
}, },
authorize(message) { /**
/* Authorize presence subscription * Authorize presence subscription
* Parameters: * @private
* (String) message - Optional message to send to the person being authorized * @method _converse.RosterContacts#authorize
* @param { String } message - Optional message to send to the person being authorized
*/ */
authorize(message) {
const pres = $pres({ const pres = $pres({
'to': this.get('jid'), 'to': this.get('jid'),
'type': "subscribed" 'type': "subscribed"
...@@ -68845,11 +68885,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -68845,11 +68885,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
return this; return this;
}, },
removeFromRoster() { /**
/* Instruct the XMPP server to remove this contact from our roster * Instruct the XMPP server to remove this contact from our roster
* Parameters: * @private
* (Function) callback * @method _converse.RosterContacts#
* @returns { Promise }
*/ */
removeFromRoster() {
const iq = $iq({ const iq = $iq({
type: 'set' type: 'set'
}).c('query', { }).c('query', {
...@@ -68862,6 +68904,12 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -68862,6 +68904,12 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
} }
}); });
/**
* @class
* @namespace _converse.RosterContacts
* @memberOf _converse
*/
_converse.RosterContacts = Backbone.Collection.extend({ _converse.RosterContacts = Backbone.Collection.extend({
model: _converse.RosterContact, model: _converse.RosterContact,
...@@ -68968,17 +69016,18 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -68968,17 +69016,18 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
return u.isSameBareJID(jid, _converse.connection.jid); return u.isSameBareJID(jid, _converse.connection.jid);
}, },
addAndSubscribe(jid, name, groups, message, attributes) { /**
/* Add a roster contact and then once we have confirmation from * Add a roster contact and then once we have confirmation from
* the XMPP server we subscribe to that contact's presence updates. * the XMPP server we subscribe to that contact's presence updates.
* Parameters: * @private
* (String) jid - The Jabber ID of the user being added and subscribed to. * @method _converse.RosterContacts#addAndSubscribe
* (String) name - The name of that user * @param { String } jid - The Jabber ID of the user being added and subscribed to.
* (Array of Strings) groups - Any roster groups the user might belong to * @param { String } name - The name of that user
* (String) message - An optional message to explain the * @param { Array.String } groups - Any roster groups the user might belong to
* reason for the subscription request. * @param { String } message - An optional message to explain the reason for the subscription request.
* (Object) attributes - Any additional attributes to be stored on the user's model. * @param { Object } attributes - Any additional attributes to be stored on the user's model.
*/ */
addAndSubscribe(jid, name, groups, message, attributes) {
const handler = contact => { const handler = contact => {
if (contact instanceof _converse.RosterContact) { if (contact instanceof _converse.RosterContact) {
contact.subscribe(message); contact.subscribe(message);
...@@ -68988,16 +69037,17 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -68988,16 +69037,17 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
this.addContactToRoster(jid, name, groups, attributes).then(handler, handler); this.addContactToRoster(jid, name, groups, attributes).then(handler, handler);
}, },
sendContactAddIQ(jid, name, groups) { /**
/* Send an IQ stanza to the XMPP server to add a new roster contact. * Send an IQ stanza to the XMPP server to add a new roster contact.
* * @private
* Parameters: * @method _converse.RosterContacts#sendContactAddIQ
* (String) jid - The Jabber ID of the user being added * @param { String } jid - The Jabber ID of the user being added
* (String) name - The name of that user * @param { String } name - The name of that user
* (Array of Strings) groups - Any roster groups the user might belong to * @param { Array.String } groups - Any roster groups the user might belong to
* (Function) callback - A function to call once the IQ is returned * @param { Function } callback - A function to call once the IQ is returned
* (Function) errback - A function to call if an error occurred * @param { Function } errback - A function to call if an error occurred
*/ */
sendContactAddIQ(jid, name, groups) {
name = _.isEmpty(name) ? null : name; name = _.isEmpty(name) ? null : name;
const iq = $iq({ const iq = $iq({
'type': 'set' 'type': 'set'
...@@ -69013,18 +69063,18 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -69013,18 +69063,18 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
return _converse.api.sendIQ(iq); return _converse.api.sendIQ(iq);
}, },
async addContactToRoster(jid, name, groups, attributes) { /**
/* Adds a RosterContact instance to _converse.roster and * Adds a RosterContact instance to _converse.roster and
* registers the contact on the XMPP server. * registers the contact on the XMPP server.
* Returns a promise which is resolved once the XMPP server has * Returns a promise which is resolved once the XMPP server has responded.
* responded. * @private
* * @method _converse.RosterContacts#addContactToRoster
* Parameters: * @param { String } jid - The Jabber ID of the user being added and subscribed to.
* (String) jid - The Jabber ID of the user being added and subscribed to. * @param { String } name - The name of that user
* (String) name - The name of that user * @param { Array.String } groups - Any roster groups the user might belong to
* (Array of Strings) groups - Any roster groups the user might belong to * @param { Object } attributes - Any additional attributes to be stored on the user's model.
* (Object) attributes - Any additional attributes to be stored on the user's model.
*/ */
async addContactToRoster(jid, name, groups, attributes) {
groups = groups || []; groups = groups || [];
try { try {
...@@ -69079,13 +69129,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -69079,13 +69129,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
return _.sum(this.models.filter(model => !_.includes(ignored, model.presence.get('show')))); return _.sum(this.models.filter(model => !_.includes(ignored, model.presence.get('show'))));
}, },
onRosterPush(iq) { /**
/* Handle roster updates from the XMPP server. * Handle roster updates from the XMPP server.
* See: https://xmpp.org/rfcs/rfc6121.html#roster-syntax-actions-push * See: https://xmpp.org/rfcs/rfc6121.html#roster-syntax-actions-push
* * @private
* Parameters: * @method _converse.RosterContacts#onRosterPush
* (XMLElement) IQ - The IQ stanza received from the XMPP server. * @param { XMLElement } IQ - The IQ stanza received from the XMPP server.
*/ */
onRosterPush(iq) {
const id = iq.getAttribute('id'); const id = iq.getAttribute('id');
const from = iq.getAttribute('from'); const from = iq.getAttribute('from');
...@@ -69751,12 +69802,6 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-vca ...@@ -69751,12 +69802,6 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-vca
} }
async function getVCard(_converse, jid) { async function getVCard(_converse, jid) {
/* Request the VCard of another user. Returns a promise.
*
* Parameters:
* (String) jid - The Jabber ID of the user whose VCard
* is being requested.
*/
const to = Strophe.getBareJidFromJid(jid) === _converse.bare_jid ? null : jid; const to = Strophe.getBareJidFromJid(jid) === _converse.bare_jid ? null : jid;
let iq; let iq;
...@@ -70003,9 +70048,7 @@ __webpack_require__.r(__webpack_exports__); ...@@ -70003,9 +70048,7 @@ __webpack_require__.r(__webpack_exports__);
function detectLocale(library_check) { function detectLocale(library_check) {
/* Determine which locale is supported by the user's system as well /* Determine which locale is supported by the user's system as well
* as by the relevant library (e.g. converse.js or moment.js). * as by the relevant library (e.g. converse.js or moment.js).
* * @param { Function } library_check - Returns a boolean indicating whether
* Parameters:
* (Function) library_check - Returns a boolean indicating whether
* the locale is supported. * the locale is supported.
*/ */
var locale, i; var locale, i;
...@@ -70052,14 +70095,13 @@ function getLocale(preferred_locale, isSupportedByLibrary) { ...@@ -70052,14 +70095,13 @@ function getLocale(preferred_locale, isSupportedByLibrary) {
return detectLocale(isSupportedByLibrary) || 'en'; return detectLocale(isSupportedByLibrary) || 'en';
} }
/* Check whether the locale or sub locale (e.g. en-US, en) is supported.
* @param { String } locale - The locale to check for
* @param { Function } available - Returns a boolean indicating whether the locale is supported
*/
function isLocaleAvailable(locale, available) { function isLocaleAvailable(locale, available) {
/* Check whether the locale or sub locale (e.g. en-US, en) is supported.
*
* Parameters:
* (String) locale - The locale to check for
* (Function) available - returns a boolean indicating whether the locale is supported
*/
if (available(locale)) { if (available(locale)) {
return locale; return locale;
} else { } else {
...@@ -70072,6 +70114,10 @@ function isLocaleAvailable(locale, available) { ...@@ -70072,6 +70114,10 @@ function isLocaleAvailable(locale, available) {
} }
let jed_instance; let jed_instance;
/**
* @namespace i18n
*/
/* harmony default export */ __webpack_exports__["default"] = ({ /* harmony default export */ __webpack_exports__["default"] = ({
setLocales(preferred_locale, _converse) { setLocales(preferred_locale, _converse) {
_converse.locale = getLocale(preferred_locale, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_29___default.a.partial(isConverseLocale, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_29___default.a, _converse.locales)); _converse.locale = getLocale(preferred_locale, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_29___default.a.partial(isConverseLocale, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_29___default.a, _converse.locales));
...@@ -70083,7 +70129,7 @@ let jed_instance; ...@@ -70083,7 +70129,7 @@ let jed_instance;
return jed__WEBPACK_IMPORTED_MODULE_27___default.a.sprintf.apply(jed__WEBPACK_IMPORTED_MODULE_27___default.a, arguments); return jed__WEBPACK_IMPORTED_MODULE_27___default.a.sprintf.apply(jed__WEBPACK_IMPORTED_MODULE_27___default.a, arguments);
} }
var t = jed_instance.translate(str); const t = jed_instance.translate(str);
if (arguments.length > 1) { if (arguments.length > 1) {
return t.fetch.apply(t, [].slice.call(arguments, 1)); return t.fetch.apply(t, [].slice.call(arguments, 1));
...@@ -70092,15 +70138,15 @@ let jed_instance; ...@@ -70092,15 +70138,15 @@ let jed_instance;
} }
}, },
fetchTranslations(locale, supported_locales, locale_url) { /**
/* Fetch the translations for the given local at the given URL. * Fetch the translations for the given local at the given URL.
* * @private
* Parameters: * @method i18n#fetchTranslations
* (String) locale: The given i18n locale * @param { String } locale -The given i18n locale
* (Array) supported_locales: List of locales supported * @param { Array } supported_locales - List of locales supported
* (String) locale_url: The URL from which the translations * @param { String } locale_url - The URL from which the translations should be fetched
* should be fetched.
*/ */
fetchTranslations(locale, supported_locales, locale_url) {
return new es6_promise_dist_es6_promise_auto__WEBPACK_IMPORTED_MODULE_28___default.a((resolve, reject) => { return new es6_promise_dist_es6_promise_auto__WEBPACK_IMPORTED_MODULE_28___default.a((resolve, reject) => {
if (!isConverseLocale(locale, supported_locales) || locale === 'en') { if (!isConverseLocale(locale, supported_locales) || locale === 'en') {
return resolve(); return resolve();
...@@ -70339,6 +70385,11 @@ __webpack_require__.r(__webpack_exports__); ...@@ -70339,6 +70385,11 @@ __webpack_require__.r(__webpack_exports__);
/**
* The utils object
* @namespace u
*/
const u = {}; const u = {};
u.toStanza = function (string) { u.toStanza = function (string) {
...@@ -70470,15 +70521,17 @@ u.applyUserSettings = function applyUserSettings(context, settings, user_setting ...@@ -70470,15 +70521,17 @@ u.applyUserSettings = function applyUserSettings(context, settings, user_setting
} }
} }
}; };
/**
u.stringToNode = function (s) { * Converts an HTML string into a DOM Node.
/* Converts an HTML string into a DOM Node.
* Expects that the HTML string has only one top-level element, * Expects that the HTML string has only one top-level element,
* i.e. not multiple ones. * i.e. not multiple ones.
* * @private
* Parameters: * @method u#stringToNode
* (String) s - The HTML string * @param { String } s - The HTML string
*/ */
u.stringToNode = function (s) {
var div = document.createElement('div'); var div = document.createElement('div');
div.innerHTML = s; div.innerHTML = s;
return div.firstElementChild; return div.firstElementChild;
...@@ -70496,40 +70549,44 @@ u.getOuterWidth = function (el) { ...@@ -70496,40 +70549,44 @@ u.getOuterWidth = function (el) {
width += parseInt(style.marginLeft, 10) + parseInt(style.marginRight, 10); width += parseInt(style.marginLeft, 10) + parseInt(style.marginRight, 10);
return width; return width;
}; };
/**
u.stringToElement = function (s) { * Converts an HTML string into a DOM element.
/* Converts an HTML string into a DOM element.
* Expects that the HTML string has only one top-level element, * Expects that the HTML string has only one top-level element,
* i.e. not multiple ones. * i.e. not multiple ones.
* * @private
* Parameters: * @method u#stringToElement
* (String) s - The HTML string * @param { String } s - The HTML string
*/ */
u.stringToElement = function (s) {
var div = document.createElement('div'); var div = document.createElement('div');
div.innerHTML = s; div.innerHTML = s;
return div.firstElementChild; return div.firstElementChild;
}; };
/**
* Checks whether the DOM element matches the given selector.
* @private
* @method u#matchesSelector
* @param { DOMElement } el - The DOM element
* @param { String } selector - The selector
*/
u.matchesSelector = function (el, selector) { u.matchesSelector = function (el, selector) {
/* Checks whether the DOM element matches the given selector.
*
* Parameters:
* (DOMElement) el - The DOM element
* (String) selector - The selector
*/
const match = el.matches || el.matchesSelector || el.msMatchesSelector || el.mozMatchesSelector || el.webkitMatchesSelector || el.oMatchesSelector; const match = el.matches || el.matchesSelector || el.msMatchesSelector || el.mozMatchesSelector || el.webkitMatchesSelector || el.oMatchesSelector;
return match ? match.call(el, selector) : false; return match ? match.call(el, selector) : false;
}; };
/**
* Returns a list of children of the DOM element that match the selector.
* @private
* @method u#queryChildren
* @param { DOMElement } el - the DOM element
* @param { String } selector - the selector they should be matched against
*/
u.queryChildren = function (el, selector) { u.queryChildren = function (el, selector) {
/* Returns a list of children of the DOM element that match the
* selector.
*
* Parameters:
* (DOMElement) el - the DOM element
* (String) selector - the selector they should be matched
* against.
*/
return _lodash_noconflict__WEBPACK_IMPORTED_MODULE_3___default.a.filter(el.childNodes, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_3___default.a.partial(u.matchesSelector, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_3___default.a, selector)); return _lodash_noconflict__WEBPACK_IMPORTED_MODULE_3___default.a.filter(el.childNodes, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_3___default.a.partial(u.matchesSelector, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_3___default.a, selector));
}; };
...@@ -70622,20 +70679,21 @@ u.interpolate = function (string, o) { ...@@ -70622,20 +70679,21 @@ u.interpolate = function (string, o) {
return typeof r === 'string' || typeof r === 'number' ? r : a; return typeof r === 'string' || typeof r === 'number' ? r : a;
}); });
}; };
/**
u.onMultipleEvents = function () { * Call the callback once all the events have been triggered
let events = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : []; * @private
let callback = arguments.length > 1 ? arguments[1] : undefined; * @method u#onMultipleEvents
* @param { Array } events: An array of objects, with keys `object` and
/* Call the callback once all the events have been triggered
*
* Parameters:
* (Array) events: An array of objects, with keys `object` and
* `event`, representing the event name and the object it's * `event`, representing the event name and the object it's
* triggered upon. * triggered upon.
* (Function) callback: The function to call once all events have * @param { Function } callback: The function to call once all events have
* been triggered. * been triggered.
*/ */
u.onMultipleEvents = function () {
let events = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
let callback = arguments.length > 1 ? arguments[1] : undefined;
let triggered = []; let triggered = [];
function handler(result) { function handler(result) {
...@@ -92332,13 +92390,14 @@ __webpack_require__.r(__webpack_exports__); ...@@ -92332,13 +92390,14 @@ __webpack_require__.r(__webpack_exports__);
/**
* Takes an HTML DOM and turns it into an XForm field.
* @private
* @method u#webForm2xForm
* @param { DOMElement } field - the field to convert
*/
_core__WEBPACK_IMPORTED_MODULE_2__["default"].webForm2xForm = function (field) { _core__WEBPACK_IMPORTED_MODULE_2__["default"].webForm2xForm = function (field) {
/* Takes an HTML DOM and turns it into an XForm field.
*
* Parameters:
* (DOMElement) field - the field to convert
*/
let value; let value;
if (field.getAttribute('type') === 'checkbox') { if (field.getAttribute('type') === 'checkbox') {
...@@ -92388,9 +92447,8 @@ const _converse$env = _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_ ...@@ -92388,9 +92447,8 @@ const _converse$env = _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_
Strophe = _converse$env.Strophe, Strophe = _converse$env.Strophe,
sizzle = _converse$env.sizzle, sizzle = _converse$env.sizzle,
_ = _converse$env._; _ = _converse$env._;
/**
_core__WEBPACK_IMPORTED_MODULE_1__["default"].computeAffiliationsDelta = function computeAffiliationsDelta(exclude_existing, remove_absentees, new_list, old_list) { * Given two lists of objects with 'jid', 'affiliation' and
/* Given two lists of objects with 'jid', 'affiliation' and
* 'reason' properties, return a new list containing * 'reason' properties, return a new list containing
* those objects that are new, changed or removed * those objects that are new, changed or removed
* (depending on the 'remove_absentees' boolean). * (depending on the 'remove_absentees' boolean).
...@@ -92400,22 +92458,24 @@ _core__WEBPACK_IMPORTED_MODULE_1__["default"].computeAffiliationsDelta = functio ...@@ -92400,22 +92458,24 @@ _core__WEBPACK_IMPORTED_MODULE_1__["default"].computeAffiliationsDelta = functio
* *
* The 'reason' property is not taken into account when * The 'reason' property is not taken into account when
* comparing whether affiliations have been changed. * comparing whether affiliations have been changed.
* * @private
* Parameters: * @method u#computeAffiliationsDelta
* (Boolean) exclude_existing: Indicates whether JIDs from * @param { boolean } exclude_existing - Indicates whether JIDs from
* the new list which are also in the old list * the new list which are also in the old list
* (regardless of affiliation) should be excluded * (regardless of affiliation) should be excluded
* from the delta. One reason to do this * from the delta. One reason to do this
* would be when you want to add a JID only if it * would be when you want to add a JID only if it
* doesn't have *any* existing affiliation at all. * doesn't have *any* existing affiliation at all.
* (Boolean) remove_absentees: Indicates whether JIDs * @param { boolean } remove_absentees - Indicates whether JIDs
* from the old list which are not in the new list * from the old list which are not in the new list
* should be considered removed and therefore be * should be considered removed and therefore be
* included in the delta with affiliation set * included in the delta with affiliation set
* to 'none'. * to 'none'.
* (Array) new_list: Array containing the new affiliations * @param { array } new_list - Array containing the new affiliations
* (Array) old_list: Array containing the old affiliations * @param { array } old_list - Array containing the old affiliations
*/ */
_core__WEBPACK_IMPORTED_MODULE_1__["default"].computeAffiliationsDelta = function computeAffiliationsDelta(exclude_existing, remove_absentees, new_list, old_list) {
const new_jids = _.map(new_list, 'jid'); const new_jids = _.map(new_list, 'jid');
const old_jids = _.map(old_list, 'jid'); // Get the new affiliations const old_jids = _.map(old_list, 'jid'); // Get the new affiliations
...@@ -96198,14 +96258,16 @@ _headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].nextUntil = functi ...@@ -96198,14 +96258,16 @@ _headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].nextUntil = functi
return matches; return matches;
}; };
/**
_headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].unescapeHTML = function (string) { * Helper method that replace HTML-escaped symbols with equivalent characters
/* Helper method that replace HTML-escaped symbols with equivalent characters
* (e.g. transform occurrences of '&amp;' to '&') * (e.g. transform occurrences of '&amp;' to '&')
* * @private
* Parameters: * @method u#unescapeHTML
* (String) string: a String containing the HTML-escaped symbols. * @param { String } string - a String containing the HTML-escaped symbols.
*/ */
_headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].unescapeHTML = function (string) {
var div = document.createElement('div'); var div = document.createElement('div');
div.innerHTML = string; div.innerHTML = string;
return div.innerText; return div.innerText;
...@@ -96266,16 +96328,17 @@ _headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].slideToggleElement ...@@ -96266,16 +96328,17 @@ _headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].slideToggleElement
return _headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].slideIn(el, duration); return _headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].slideIn(el, duration);
} }
}; };
/**
* Shows/expands an element by sliding it out of itself
* @private
* @method u#slideOut
* @param { HTMLElement } el - The HTML string
* @param { Number } duration - The duration amount in milliseconds
*/
_headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].slideOut = function (el) { _headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].slideOut = function (el) {
let duration = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 200; let duration = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 200;
/* Shows/expands an element by sliding it out of itself
*
* Parameters:
* (HTMLElement) el - The HTML string
* (Number) duration - The duration amount in milliseconds
*/
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
if (_headless_lodash_noconflict__WEBPACK_IMPORTED_MODULE_1___default.a.isNil(el)) { if (_headless_lodash_noconflict__WEBPACK_IMPORTED_MODULE_1___default.a.isNil(el)) {
const err = "Undefined or null element passed into slideOut"; const err = "Undefined or null element passed into slideOut";
...@@ -96422,17 +96485,17 @@ _headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].fadeIn = function ...@@ -96422,17 +96485,17 @@ _headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].fadeIn = function
afterAnimationEnds(el, callback); afterAnimationEnds(el, callback);
} }
}; };
/**
_headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].xForm2webForm = function (field, stanza, domain) { * Takes a field in XMPP XForm (XEP-004: Data Forms) format
/* Takes a field in XMPP XForm (XEP-004: Data Forms) format
* and turns it into an HTML field. * and turns it into an HTML field.
* * Returns either text or a DOM element (which is not ideal, but fine for now).
* Returns either text or a DOM element (which is not ideal, but fine * @private
* for now). * @method u#xForm2webForm
* * @param { XMLElement } field - the field to convert
* Parameters:
* (XMLElement) field - the field to convert
*/ */
_headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].xForm2webForm = function (field, stanza, domain) {
if (field.getAttribute('type') === 'list-single' || field.getAttribute('type') === 'list-multi') { if (field.getAttribute('type') === 'list-single' || field.getAttribute('type') === 'list-multi') {
const values = _headless_lodash_noconflict__WEBPACK_IMPORTED_MODULE_1___default.a.map(_headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].queryChildren(field, 'value'), _headless_lodash_noconflict__WEBPACK_IMPORTED_MODULE_1___default.a.partial(_headless_lodash_noconflict__WEBPACK_IMPORTED_MODULE_1___default.a.get, _headless_lodash_noconflict__WEBPACK_IMPORTED_MODULE_1___default.a, 'textContent')); const values = _headless_lodash_noconflict__WEBPACK_IMPORTED_MODULE_1___default.a.map(_headless_utils_core__WEBPACK_IMPORTED_MODULE_16__["default"].queryChildren(field, 'value'), _headless_lodash_noconflict__WEBPACK_IMPORTED_MODULE_1___default.a.partial(_headless_lodash_noconflict__WEBPACK_IMPORTED_MODULE_1___default.a.get, _headless_lodash_noconflict__WEBPACK_IMPORTED_MODULE_1___default.a, 'textContent'));
...@@ -41,6 +41,5 @@ to fix a bug or to add new functionality. ...@@ -41,6 +41,5 @@ to fix a bug or to add new functionality.
plugin_development plugin_development
api/index api/index
testing testing
events
other_frameworks other_frameworks
builds builds
...@@ -3,29 +3,31 @@ ...@@ -3,29 +3,31 @@
Welcome to the new Converse API documentation, generated with Welcome to the new Converse API documentation, generated with
[JSDoc](http://usejsdoc.org/). [JSDoc](http://usejsdoc.org/).
The old (increasingly out of date and incomplete) API documentation is This documentation replaces the (increasingly out of date and incomplete) [old API documentation](/docs/html/developer_api.html)
currently still [available here](/docs/html/developer_api.html). and [old "Events and Promises" documentation](/docs/html/events.html).
## The public and private API ## The public and private API
Converse has a public API and a private API. Converse has a public API and a private API only available to plugins.
r
The reason we make this distinction between public and private is so that API The reason we make this distinction between public and private is so that API
methods which might can be used to "impersonate" the user, for example by methods which could be used to "impersonate" the user, for example by
sending messages on their behalf, are not available to random scripts running sending messages on their behalf, are not available to random scripts running
in the websites. in your website.
The public API is accessible via the [window.converse](/docs/html/api/converse.html)
global and is therefore available to any JavaScript running in the page.
The public API is accessible via the `window.converse` global and is therefore The private API is only accessible to plugins, which have been whitelisted and
available to all JavaScript running in the page. registered before [converse.initialize](/docs/html/api/converse.html#.initialize)
(which is a public API method) has been called.
Tehe private API is only accessible to plugins, which have been whitelisted and See the [plugin development](/docs/html/plugin_development.html)
registered before `converse.initialize` (which is a public API method) has been
called. See the [plugin development](https://conversejs.org/docs/html/plugin_development.html)
section for more info on writing plugins. section for more info on writing plugins.
Inside a plugin, you can get access to the `_converse.api` object. Note the Inside a plugin, you can get access to the {@link _converse.api}
underscore in front of `_converse`, which indicates that this is a private, object. Note the underscore in front of {@link _converse},
closured object. which indicates that this is a private, closured object.
## API Namespaces ## API Namespaces
...@@ -35,9 +37,26 @@ group relevant methods. ...@@ -35,9 +37,26 @@ group relevant methods.
So, for example, all the XEP-0030 service discovery methods are under the So, for example, all the XEP-0030 service discovery methods are under the
{@link \_converse.api.disco} namespace, in the [private API]{@link \_converse.api}. {@link \_converse.api.disco} namespace, in the [private API]{@link \_converse.api}.
Which means that you access it via `_converse.api.disco`. Which means that you access it via {@link _converse.api.disco}.
### Nested Namespaces
Namespaces can be nested.
Namespaces can be nested. So the {@link \_converse.api.disco} namespace {@link _converse.api} is the top-level namespace, of which {@link \_converse.api.disco}
namespace has {@link \_converse.api.disco.own} as a nested namespace. is a nested, child namespace and {@link \_converse.api.disco.own} is nested another
level deeper.
Not all methods are however within a namespace. For example {@link converse.initialize}. Not all methods are however within a namespace. For example {@link converse.initialize}.
## Stable API versus unstable API
Converse uses [semantic versioning](https://semver.org/) for releases, which means that
we try to maintain a stable API for minor and patch releases and when we do change the
stable API we will make a major release.
In the JSDoc API documentation, all API methods that are **not** marked as *Private*
are considered to be part of the stable API, and you can therefore expect them to
not change between minor and patch releases. If a method is marked as *Private*,
then you could still use it, but we don't provide any guarantee that it won't change
between minor and patch releases.
...@@ -586,18 +586,18 @@ converse.plugins.add('converse-chatview', { ...@@ -586,18 +586,18 @@ converse.plugins.add('converse-chatview', {
); );
}, },
insertDayIndicator (next_msg_el) { /**
/* Inserts an indicator into the chat area, showing the * Inserts an indicator into the chat area, showing the
* day as given by the passed in date. * day as given by the passed in date.
*
* The indicator is only inserted if necessary. * The indicator is only inserted if necessary.
* * @private
* Parameters: * @method _converse.ChatBoxView#insertDayIndicator
* (HTMLElement) next_msg_el - The message element before * @param { HTMLElement } next_msg_el - The message element before
* which the day indicator element must be inserted. * which the day indicator element must be inserted.
* This element must have a "data-isodate" attribute * This element must have a "data-isodate" attribute
* which specifies its creation date. * which specifies its creation date.
*/ */
insertDayIndicator (next_msg_el) {
const prev_msg_el = u.getPreviousElement(next_msg_el, ".message:not(.chat-state-notification)"), const prev_msg_el = u.getPreviousElement(next_msg_el, ".message:not(.chat-state-notification)"),
prev_msg_date = _.isNull(prev_msg_el) ? null : prev_msg_el.getAttribute('data-isodate'), prev_msg_date = _.isNull(prev_msg_el) ? null : prev_msg_el.getAttribute('data-isodate'),
next_msg_date = next_msg_el.getAttribute('data-isodate'); next_msg_date = next_msg_el.getAttribute('data-isodate');
...@@ -616,13 +616,14 @@ converse.plugins.add('converse-chatview', { ...@@ -616,13 +616,14 @@ converse.plugins.add('converse-chatview', {
} }
}, },
getLastMessageDate (cutoff) { /**
/* Return the ISO8601 format date of the latest message. * Return the ISO8601 format date of the latest message.
* * @private
* Parameters: * @method _converse.ChatBoxView#getLastMessageDate
* (Object) cutoff: Moment Date cutoff date. The last * @param { object } cutoff - Moment Date cutoff date. The last
* message received cutoff this date will be returned. * message received cutoff this date will be returned.
*/ */
getLastMessageDate (cutoff) {
const first_msg = u.getFirstChildElement(this.content, '.message:not(.chat-state-notification)'), const first_msg = u.getFirstChildElement(this.content, '.message:not(.chat-state-notification)'),
oldest_date = first_msg ? first_msg.getAttribute('data-isodate') : null; oldest_date = first_msg ? first_msg.getAttribute('data-isodate') : null;
if (!_.isNull(oldest_date) && moment(oldest_date).isAfter(cutoff)) { if (!_.isNull(oldest_date) && moment(oldest_date).isAfter(cutoff)) {
...@@ -713,13 +714,14 @@ converse.plugins.add('converse-chatview', { ...@@ -713,13 +714,14 @@ converse.plugins.add('converse-chatview', {
return !u.isVisible(this.el); return !u.isVisible(this.el);
}, },
insertMessage (view) { /**
/* Given a view representing a message, insert it into the * Given a view representing a message, insert it into the
* content area of the chat box. * content area of the chat box.
* * @private
* Parameters: * @method _converse.ChatBoxView#insertMessage
* (Backbone.View) message: The message Backbone.View * @param { Backbone.View } message - The message Backbone.View
*/ */
insertMessage (view) {
if (view.model.get('type') === 'error') { if (view.model.get('type') === 'error') {
const previous_msg_el = this.content.querySelector(`[data-msgid="${view.model.get('msgid')}"]`); const previous_msg_el = this.content.querySelector(`[data-msgid="${view.model.get('msgid')}"]`);
if (previous_msg_el) { if (previous_msg_el) {
...@@ -746,8 +748,8 @@ converse.plugins.add('converse-chatview', { ...@@ -746,8 +748,8 @@ converse.plugins.add('converse-chatview', {
return this.trigger('messageInserted', view.el); return this.trigger('messageInserted', view.el);
}, },
markFollowups (el) { /**
/* Given a message element, determine wether it should be * Given a message element, determine wether it should be
* marked as a followup message to the previous element. * marked as a followup message to the previous element.
* *
* Also determine whether the element following it is a * Also determine whether the element following it is a
...@@ -757,9 +759,11 @@ converse.plugins.add('converse-chatview', { ...@@ -757,9 +759,11 @@ converse.plugins.add('converse-chatview', {
* author with no other conversation elements inbetween and * author with no other conversation elements inbetween and
* posted within 10 minutes of one another. * posted within 10 minutes of one another.
* *
* Parameters: * @private
* (HTMLElement) el - The message element. * @method _converse.ChatBoxView#markFollowups
* @param { HTMLElement } el - The message element
*/ */
markFollowups (el) {
const from = el.getAttribute('data-from'), const from = el.getAttribute('data-from'),
previous_el = el.previousElementSibling, previous_el = el.previousElementSibling,
date = moment(el.getAttribute('data-isodate')), date = moment(el.getAttribute('data-isodate')),
...@@ -783,15 +787,14 @@ converse.plugins.add('converse-chatview', { ...@@ -783,15 +787,14 @@ converse.plugins.add('converse-chatview', {
} }
}, },
async showMessage (message) { /**
/* Inserts a chat message into the content area of the chat box. * Inserts a chat message into the content area of the chat box.
* * Will also insert a new day indicator if the message is on a different day.
* Will also insert a new day indicator if the message is on a * @private
* different day. * @method _converse.ChatBoxView#showMessage
* * @param { _converse.Message } message - The message object
* Parameters:
* (Backbone.Model) message: The message object
*/ */
async showMessage (message) {
if (!u.isNewMessage(message) && u.isEmptyMessage(message)) { if (!u.isNewMessage(message) && u.isEmptyMessage(message)) {
// Handle archived or delayed messages without any message // Handle archived or delayed messages without any message
// text to show. // text to show.
...@@ -822,12 +825,13 @@ converse.plugins.add('converse-chatview', { ...@@ -822,12 +825,13 @@ converse.plugins.add('converse-chatview', {
} }
}, },
onMessageAdded (message) { /**
/* Handler that gets called when a new message object is created. * Handler that gets called when a new message object is created.
* * @private
* Parameters: * @method _converse.ChatBoxView#onMessageAdded
* (Object) message - The message Backbone object that was added. * @param { object } message - The message Backbone object that was added.
*/ */
onMessageAdded (message) {
this.showMessage(message); this.showMessage(message);
if (message.get('correcting')) { if (message.get('correcting')) {
this.insertIntoTextArea(message.get('message'), true, true); this.insertIntoTextArea(message.get('message'), true, true);
...@@ -865,17 +869,18 @@ converse.plugins.add('converse-chatview', { ...@@ -865,17 +869,18 @@ converse.plugins.add('converse-chatview', {
} }
}, },
setChatState (state, options) { /**
/* Mutator for setting the chat state of this chat session. * Mutator for setting the chat state of this chat session.
* Handles clearing of any chat state notification timeouts and * Handles clearing of any chat state notification timeouts and
* setting new ones if necessary. * setting new ones if necessary.
* Timeouts are set when the state being set is COMPOSING or PAUSED. * Timeouts are set when the state being set is COMPOSING or PAUSED.
* After the timeout, COMPOSING will become PAUSED and PAUSED will become INACTIVE. * After the timeout, COMPOSING will become PAUSED and PAUSED will become INACTIVE.
* See XEP-0085 Chat State Notifications. * See XEP-0085 Chat State Notifications.
* * @private
* Parameters: * @method _converse.ChatBoxView#setChatState
* (string) state - The chat state (consts ACTIVE, COMPOSING, PAUSED, INACTIVE, GONE) * @param { string } state - The chat state (consts ACTIVE, COMPOSING, PAUSED, INACTIVE, GONE)
*/ */
setChatState (state, options) {
if (!_.isUndefined(this.chat_state_timeout)) { if (!_.isUndefined(this.chat_state_timeout)) {
window.clearTimeout(this.chat_state_timeout); window.clearTimeout(this.chat_state_timeout);
delete this.chat_state_timeout; delete this.chat_state_timeout;
......
...@@ -378,11 +378,6 @@ converse.plugins.add('converse-controlbox', { ...@@ -378,11 +378,6 @@ converse.plugins.add('converse-controlbox', {
}, },
showHelpMessages () { showHelpMessages () {
/* Override showHelpMessages in ChatBoxView, for now do nothing.
*
* Parameters:
* (Array) msgs: Array of messages
*/
return; return;
} }
}); });
......
...@@ -203,15 +203,12 @@ converse.plugins.add('converse-muc-views', { ...@@ -203,15 +203,12 @@ converse.plugins.add('converse-muc-views', {
}; };
function insertRoomInfo (el, stanza) {
/* Insert groupchat info (based on returned #disco IQ stanza) /* Insert groupchat info (based on returned #disco IQ stanza)
* * @function insertRoomInfo
* Parameters: * @param { HTMLElement } el - The HTML DOM element that contains the info.
* (HTMLElement) el: The HTML DOM element that should * @param { XMLElement } stanza - The IQ stanza containing the groupchat info.
* contain the info.
* (XMLElement) stanza: The IQ stanza containing the groupchat
* info.
*/ */
function insertRoomInfo (el, stanza) {
// All MUC features found here: https://xmpp.org/registrar/disco-features.html // All MUC features found here: https://xmpp.org/registrar/disco-features.html
el.querySelector('span.spinner').remove(); el.querySelector('span.spinner').remove();
el.querySelector('a.room-info').classList.add('selected'); el.querySelector('a.room-info').classList.add('selected');
...@@ -1110,12 +1107,13 @@ converse.plugins.add('converse-muc-views', { ...@@ -1110,12 +1107,13 @@ converse.plugins.add('converse-muc-views', {
this.model.addHandler('message', 'ChatRoomView.showStatusMessages', this.showStatusMessages.bind(this)); this.model.addHandler('message', 'ChatRoomView.showStatusMessages', this.showStatusMessages.bind(this));
}, },
onPresence (pres) { /**
/* Handles all MUC presence stanzas. * Handles all MUC presence stanzas.
* * @private
* Parameters: * @method _converse.ChatRoomView#onPresence
* (XMLElement) pres: The stanza * @param { XMLElement } pres - The stanza
*/ */
onPresence (pres) {
// XXX: Current thinking is that excessive stanza // XXX: Current thinking is that excessive stanza
// processing inside a view is a "code smell". // processing inside a view is a "code smell".
// Instead stanza processing should happen inside the // Instead stanza processing should happen inside the
...@@ -1138,14 +1136,14 @@ converse.plugins.add('converse-muc-views', { ...@@ -1138,14 +1136,14 @@ converse.plugins.add('converse-muc-views', {
this.fetchMessages(); this.fetchMessages();
}, },
join (nick, password) { /**
/* Join the groupchat. * Join the groupchat.
* * @private
* Parameters: * @method _converse.ChatRoomView#join
* (String) nick: The user's nickname * @param { String } nick - The user's nickname
* (String) password: Optional password, if required by * @param { String } password - Optional password, if required by the groupchat
* the groupchat.
*/ */
join (nick, password) {
if (!nick && !this.model.get('nick')) { if (!nick && !this.model.get('nick')) {
this.checkForReservedNick(); this.checkForReservedNick();
return this; return this;
...@@ -1154,17 +1152,16 @@ converse.plugins.add('converse-muc-views', { ...@@ -1154,17 +1152,16 @@ converse.plugins.add('converse-muc-views', {
return this; return this;
}, },
renderConfigurationForm (stanza) { /**
/* Renders a form given an IQ stanza containing the current * Renders a form given an IQ stanza containing the current
* groupchat configuration. * groupchat configuration.
*
* Returns a promise which resolves once the user has * Returns a promise which resolves once the user has
* either submitted the form, or canceled it. * either submitted the form, or canceled it.
* * @private
* Parameters: * @method _converse.ChatRoomView#renderConfigurationForm
* (XMLElement) stanza: The IQ stanza containing the groupchat * @param { XMLElement } stanza: The IQ stanza containing the groupchat config.
* config.
*/ */
renderConfigurationForm (stanza) {
const container_el = this.el.querySelector('.chatroom-body'); const container_el = this.el.querySelector('.chatroom-body');
_.each(container_el.querySelectorAll('.chatroom-form-container'), u.removeElement); _.each(container_el.querySelectorAll('.chatroom-form-container'), u.removeElement);
_.each(container_el.children, u.hideElement); _.each(container_el.children, u.hideElement);
...@@ -1420,13 +1417,14 @@ converse.plugins.add('converse-muc-views', { ...@@ -1420,13 +1417,14 @@ converse.plugins.add('converse-muc-views', {
u.showElement(container); u.showElement(container);
}, },
getMessageFromStatus (stat, stanza, is_self) { /**
/* Parameters: * @private
* (XMLElement) stat: A <status> element. * @method _converse.ChatRoomView#getMessageFromStatus
* (Boolean) is_self: Whether the element refers to the * @param { XMLElement } stat: A <status> element
* current user. * @param { Boolean } is_self: Whether the element refers to the current user
* (XMLElement) stanza: The original stanza received. * @param { XMLElement } stanza: The original stanza received
*/ */
getMessageFromStatus (stat, stanza, is_self) {
const code = stat.getAttribute('code'); const code = stat.getAttribute('code');
if (code === '110' || (code === '100' && !is_self)) { return; } if (code === '110' || (code === '100' && !is_self)) { return; }
if (code in _converse.muc.info_messages) { if (code in _converse.muc.info_messages) {
...@@ -1697,14 +1695,14 @@ converse.plugins.add('converse-muc-views', { ...@@ -1697,14 +1695,14 @@ converse.plugins.add('converse-muc-views', {
this.scrollDown(); this.scrollDown();
}, },
showStatusMessages (stanza) { /**
/* Check for status codes and communicate their purpose to the user. * Check for status codes and communicate their purpose to the user.
* See: https://xmpp.org/registrar/mucstatus.html * See: https://xmpp.org/registrar/mucstatus.html
* * @private
* Parameters: * @method _converse.ChatRoomView#showStatusMessages
* (XMLElement) stanza: The message or presence stanza * @param { XMLElement } stanza - The message or presence stanza containing the status codes
* containing the status codes.
*/ */
showStatusMessages (stanza) {
const elements = sizzle(`x[xmlns="${Strophe.NS.MUC_USER}"]`, stanza); const elements = sizzle(`x[xmlns="${Strophe.NS.MUC_USER}"]`, stanza);
const is_self = stanza.querySelectorAll("status[code='110']").length; const is_self = stanza.querySelectorAll("status[code='110']").length;
const iteratee = _.partial(this.parseXUserElement.bind(this), _, stanza, is_self); const iteratee = _.partial(this.parseXUserElement.bind(this), _, stanza, is_self);
......
...@@ -159,6 +159,11 @@ converse.plugins.add('converse-register', { ...@@ -159,6 +159,11 @@ converse.plugins.add('converse-register', {
} }
}); });
/**
* @class
* @namespace _converse.RegisterPanel
* @memberOf _converse
*/
_converse.RegisterPanel = Backbone.NativeView.extend({ _converse.RegisterPanel = Backbone.NativeView.extend({
tagName: 'div', tagName: 'div',
id: "converse-register-panel", id: "converse-register-panel",
...@@ -200,20 +205,21 @@ converse.plugins.add('converse-register', { ...@@ -200,20 +205,21 @@ converse.plugins.add('converse-register', {
if (!this._registering) { if (!this._registering) {
connect_cb(req, callback, raw); connect_cb(req, callback, raw);
} else { } else {
if (this.getRegistrationFields(req, callback, raw)) { if (this.getRegistrationFields(req, callback)) {
this._registering = false; this._registering = false;
} }
} }
}; };
}, },
getRegistrationFields (req, _callback, raw) { /**
/* Send an IQ stanza to the XMPP server asking for the * Send an IQ stanza to the XMPP server asking for the registration fields.
* registration fields. * @private
* Parameters: * @method _converse.RegisterPanel#getRegistrationFields
* (Strophe.Request) req - The current request * @param { Strophe.Request } req - The current request
* (Function) callback * @param { Function } callback - The callback function
*/ */
getRegistrationFields (req, _callback) {
const conn = _converse.connection; const conn = _converse.connection;
conn.connected = true; conn.connected = true;
...@@ -248,12 +254,13 @@ converse.plugins.add('converse-register', { ...@@ -248,12 +254,13 @@ converse.plugins.add('converse-register', {
return true; return true;
}, },
onRegistrationFields (stanza) { /**
/* Handler for Registration Fields Request. * Handler for {@link _converse.RegisterPanel#getRegistrationFields}
* * @private
* Parameters: * @method _converse.RegisterPanel#onRegistrationFields
* (XMLElement) elem - The query stanza. * @param { XMLElement } stanza - The query stanza.
*/ */
onRegistrationFields (stanza) {
if (stanza.getAttribute("type") === "error") { if (stanza.getAttribute("type") === "error") {
_converse.connection._changeConnectStatus( _converse.connection._changeConnectStatus(
Strophe.Status.REGIFAIL, Strophe.Status.REGIFAIL,
...@@ -309,13 +316,13 @@ converse.plugins.add('converse-register', { ...@@ -309,13 +316,13 @@ converse.plugins.add('converse-register', {
}, },
onProviderChosen (form) { /**
/* Callback method that gets called when the user has chosen an * Callback method that gets called when the user has chosen an XMPP provider
* XMPP provider. * @private
* * @method _converse.RegisterPanel#onProviderChosen
* Parameters: * @param { HTMLElement } form - The form that was submitted
* (HTMLElement) form - The form that was submitted
*/ */
onProviderChosen (form) {
const domain_input = form.querySelector('input[name=domain]'), const domain_input = form.querySelector('input[name=domain]'),
domain = _.get(domain_input, 'value'); domain = _.get(domain_input, 'value');
if (!domain) { if (!domain) {
...@@ -327,13 +334,13 @@ converse.plugins.add('converse-register', { ...@@ -327,13 +334,13 @@ converse.plugins.add('converse-register', {
this.fetchRegistrationForm(domain.trim()); this.fetchRegistrationForm(domain.trim());
}, },
fetchRegistrationForm (domain_name) { /**
/* This is called with a domain name based on which, it fetches a * Fetch a registration form from the requested domain
* registration form from the requested domain. * @private
* * @method _converse.RegisterPanel#fetchRegistrationForm
* Parameters: * @param { String } domain_name - XMPP server domain
* (String) domain_name - XMPP server domain
*/ */
fetchRegistrationForm (domain_name) {
if (!this.model.get('registration_form_rendered')) { if (!this.model.get('registration_form_rendered')) {
this.renderRegistrationRequest(); this.renderRegistrationRequest();
} }
...@@ -386,16 +393,14 @@ converse.plugins.add('converse-register', { ...@@ -386,16 +393,14 @@ converse.plugins.add('converse-register', {
return this; return this;
}, },
onConnectStatusChanged(status_code) { /**
/* Callback function called by Strophe whenever the * Callback function called by Strophe whenever the connection status changes.
* connection status changes. * Passed to Strophe specifically during a registration attempt.
* * @private
* Passed to Strophe specifically during a registration * @method _converse.RegisterPanel#onConnectStatusChanged
* attempt. * @param { integer } status_code - The Strophe.Status status code
*
* Parameters:
* (Integer) status_code - The Stroph.Status status code
*/ */
onConnectStatusChanged(status_code) {
_converse.log('converse-register: onConnectStatusChanged'); _converse.log('converse-register: onConnectStatusChanged');
if (_.includes([ if (_.includes([
Strophe.Status.DISCONNECTED, Strophe.Status.DISCONNECTED,
...@@ -472,13 +477,14 @@ converse.plugins.add('converse-register', { ...@@ -472,13 +477,14 @@ converse.plugins.add('converse-register', {
}); });
}, },
renderRegistrationForm (stanza) { /**
/* Renders the registration form based on the XForm fields * Renders the registration form based on the XForm fields
* received from the XMPP server. * received from the XMPP server.
* * @private
* Parameters: * @method _converse.RegisterPanel#renderRegistrationForm
* (XMLElement) stanza - The IQ stanza received from the XMPP server. * @param { XMLElement } stanza - The IQ stanza received from the XMPP server.
*/ */
renderRegistrationForm (stanza) {
const form = this.el.querySelector('form'); const form = this.el.querySelector('form');
form.innerHTML = tpl_registration_form({ form.innerHTML = tpl_registration_form({
'__': _converse.__, '__': _converse.__,
...@@ -528,14 +534,14 @@ converse.plugins.add('converse-register', { ...@@ -528,14 +534,14 @@ converse.plugins.add('converse-register', {
flash.classList.remove('hidden'); flash.classList.remove('hidden');
}, },
reportErrors (stanza) { /**
/* Report back to the user any error messages received from the * Report back to the user any error messages received from the
* XMPP server after attempted registration. * XMPP server after attempted registration.
* * @private
* Parameters: * @method _converse.RegisterPanel#reportErrors
* (XMLElement) stanza - The IQ stanza received from the * @param { XMLElement } stanza - The IQ stanza received from the XMPP server
* XMPP server.
*/ */
reportErrors (stanza) {
const errors = stanza.querySelectorAll('error'); const errors = stanza.querySelectorAll('error');
_.each(errors, (error) => { _.each(errors, (error) => {
this.showValidationError(error.textContent); this.showValidationError(error.textContent);
...@@ -568,14 +574,14 @@ converse.plugins.add('converse-register', { ...@@ -568,14 +574,14 @@ converse.plugins.add('converse-register', {
} }
}, },
submitRegistrationForm (form) { /**
/* Handler, when the user submits the registration form. * Handler, when the user submits the registration form.
* Provides form error feedback or starts the registration * Provides form error feedback or starts the registration process.
* process. * @private
* * @method _converse.RegisterPanel#submitRegistrationForm
* Parameters: * @param { HTMLElement } form - The HTML form that was submitted
* (HTMLElement) form - The HTML form that was submitted
*/ */
submitRegistrationForm (form) {
const has_empty_inputs = _.reduce( const has_empty_inputs = _.reduce(
this.el.querySelectorAll('input.required'), this.el.querySelectorAll('input.required'),
function (result, input) { function (result, input) {
...@@ -606,13 +612,12 @@ converse.plugins.add('converse-register', { ...@@ -606,13 +612,12 @@ converse.plugins.add('converse-register', {
this.setFields(iq.tree()); this.setFields(iq.tree());
}, },
setFields (stanza) { /* Stores the values that will be sent to the XMPP server during attempted registration.
/* Stores the values that will be sent to the XMPP server * @private
* during attempted registration. * @method _converse.RegisterPanel#setFields
* * @param { XMLElement } stanza - the IQ stanza that will be sent to the XMPP server.
* Parameters:
* (XMLElement) stanza - the IQ stanza that will be sent to the XMPP server.
*/ */
setFields (stanza) {
const query = stanza.querySelector('query'); const query = stanza.querySelector('query');
const xform = sizzle(`x[xmlns="${Strophe.NS.XFORM}"]`, query); const xform = sizzle(`x[xmlns="${Strophe.NS.XFORM}"]`, query);
if (xform.length > 0) { if (xform.length > 0) {
...@@ -653,14 +658,15 @@ converse.plugins.add('converse-register', { ...@@ -653,14 +658,15 @@ converse.plugins.add('converse-register', {
this.form_type = 'xform'; this.form_type = 'xform';
}, },
_onRegisterIQ (stanza) { /**
/* Callback method that gets called when a return IQ stanza * Callback method that gets called when a return IQ stanza
* is received from the XMPP server, after attempting to * is received from the XMPP server, after attempting to
* register a new user. * register a new user.
* * @private
* Parameters: * @method _converse.RegisterPanel#reportErrors
* (XMLElement) stanza - The IQ stanza. * @param { XMLElement } stanza - The IQ stanza.
*/ */
_onRegisterIQ (stanza) {
if (stanza.getAttribute("type") === "error") { if (stanza.getAttribute("type") === "error") {
_converse.log("Registration failed.", Strophe.LogLevel.ERROR); _converse.log("Registration failed.", Strophe.LogLevel.ERROR);
this.reportErrors(stanza); this.reportErrors(stanza);
......
...@@ -442,13 +442,13 @@ converse.plugins.add('converse-chatboxes', { ...@@ -442,13 +442,13 @@ converse.plugins.add('converse-chatboxes', {
return false; return false;
}, },
createMessageStanza (message) { /**
/* Given a _converse.Message Backbone.Model, return the XML * Given a {@link _converse.Message} return the XML stanza that represents it.
* stanza that represents it. * @private
* * @method _converse.ChatBox#createMessageStanza
* Parameters: * @param { _converse.Message } message - The message object
* (Object) message - The Backbone.Model representing the message
*/ */
createMessageStanza (message) {
const stanza = $msg({ const stanza = $msg({
'from': _converse.connection.jid, 'from': _converse.connection.jid,
'to': this.get('jid'), 'to': this.get('jid'),
...@@ -630,13 +630,14 @@ converse.plugins.add('converse-chatboxes', { ...@@ -630,13 +630,14 @@ converse.plugins.add('converse-chatboxes', {
}); });
}, },
getStanzaIDs (stanza) { /**
/* Extract the XEP-0359 stanza IDs from the passed in stanza * Extract the XEP-0359 stanza IDs from the passed in stanza
* and return a map containing them. * and return a map containing them.
* * @private
* Parameters: * @method _converse.ChatBox#getStanzaIDs
* (XMLElement) stanza - The message stanza * @param { XMLElement } stanza - The message stanza
*/ */
getStanzaIDs (stanza) {
const attrs = {}; const attrs = {};
const stanza_ids = sizzle(`stanza-id[xmlns="${Strophe.NS.SID}"]`, stanza); const stanza_ids = sizzle(`stanza-id[xmlns="${Strophe.NS.SID}"]`, stanza);
if (stanza_ids.length) { if (stanza_ids.length) {
...@@ -659,18 +660,17 @@ converse.plugins.add('converse-chatboxes', { ...@@ -659,18 +660,17 @@ converse.plugins.add('converse-chatboxes', {
return !_.isNil(sizzle(`result[xmlns="${Strophe.NS.MAM}"]`, original_stanza).pop()); return !_.isNil(sizzle(`result[xmlns="${Strophe.NS.MAM}"]`, original_stanza).pop());
}, },
getMessageAttributesFromStanza (stanza, original_stanza) { /**
/* Parses a passed in message stanza and returns an object * Parses a passed in message stanza and returns an object
* of attributes. * of attributes.
* * @private
* Parameters: * @method _converse.ChatBox#getMessageAttributesFromStanza
* (XMLElement) stanza - The message stanza * @param { XMLElement } stanza - The message stanza
* (XMLElement) delay - The <delay> node from the * @param { XMLElement } delay - The <delay> node from the stanza, if there was one.
* stanza, if there was one. * @param { XMLElement } original_stanza - The original stanza, that contains the
* (XMLElement) original_stanza - The original stanza, * message stanza, if it was contained, otherwise it's the message stanza itself.
* that contains the message stanza, if it was
* contained, otherwise it's the message stanza itself.
*/ */
getMessageAttributesFromStanza (stanza, original_stanza) {
const spoiler = sizzle(`spoiler[xmlns="${Strophe.NS.SPOILER}"]`, original_stanza).pop(), const spoiler = sizzle(`spoiler[xmlns="${Strophe.NS.SPOILER}"]`, original_stanza).pop(),
delay = sizzle(`delay[xmlns="${Strophe.NS.DELAY}"]`, original_stanza).pop(), delay = sizzle(`delay[xmlns="${Strophe.NS.DELAY}"]`, original_stanza).pop(),
text = _converse.chatboxes.getMessageBody(stanza) || undefined, text = _converse.chatboxes.getMessageBody(stanza) || undefined,
...@@ -864,13 +864,13 @@ converse.plugins.add('converse-chatboxes', { ...@@ -864,13 +864,13 @@ converse.plugins.add('converse-chatboxes', {
} }
}, },
async onMessage (stanza) { /**
/* Handler method for all incoming single-user chat "message" * Handler method for all incoming single-user chat "message" stanzas.
* stanzas. * @private
* * @method _converse.ChatBox#onMessage
* Parameters: * @param { XMLElement } stanza - The incoming message stanza
* (XMLElement) stanza - The incoming message stanza
*/ */
async onMessage (stanza) {
let to_jid = stanza.getAttribute('to'); let to_jid = stanza.getAttribute('to');
const to_resource = Strophe.getResourceFromJid(to_jid); const to_resource = Strophe.getResourceFromJid(to_jid);
...@@ -970,15 +970,16 @@ converse.plugins.add('converse-chatboxes', { ...@@ -970,15 +970,16 @@ converse.plugins.add('converse-chatboxes', {
_converse.api.trigger('message', {'stanza': original_stanza, 'chatbox': chatbox}); _converse.api.trigger('message', {'stanza': original_stanza, 'chatbox': chatbox});
}, },
getChatBox (jid, attrs={}, create) { /**
/* Returns a chat box or optionally return a newly * Returns a chat box or optionally return a newly
* created one if one doesn't exist. * created one if one doesn't exist.
* * @private
* Parameters: * @method _converse.ChatBox#getChatBox
* (String) jid - The JID of the user whose chat box we want * @param { string } jid - The JID of the user whose chat box we want
* (Boolean) create - Should a new chat box be created if none exists? * @param { boolean } create - Should a new chat box be created if none exists?
* (Object) attrs - Optional chat box atributes. * @param { object } attrs - Optional chat box atributes.
*/ */
getChatBox (jid, attrs={}, create) {
if (_.isObject(jid)) { if (_.isObject(jid)) {
create = attrs; create = attrs;
attrs = jid; attrs = jid;
......
...@@ -225,20 +225,19 @@ _converse.default_settings = { ...@@ -225,20 +225,19 @@ _converse.default_settings = {
}; };
_converse.log = function (message, level, style='') { /**
/* Logs messages to the browser's developer console. * 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.
*
* Available loglevels are 0 for 'debug', 1 for 'info', 2 for 'warn', * Available loglevels are 0 for 'debug', 1 for 'info', 2 for 'warn',
* 3 for 'error' and 4 for 'fatal'. * 3 for 'error' and 4 for 'fatal'.
*
* When using the 'error' or 'warn' loglevels, a full stacktrace will be * When using the 'error' or 'warn' loglevels, a full stacktrace will be
* logged as well. * logged as well.
* @method log
* @private
* @memberOf _converse
* @param { string } message - The message to be logged
* @param { integer } level - The loglevel which allows for filtering of log messages
*/ */
_converse.log = function (message, level, style='') {
if (level === Strophe.LogLevel.ERROR || level === Strophe.LogLevel.FATAL) { if (level === Strophe.LogLevel.ERROR || level === Strophe.LogLevel.FATAL) {
style = style || 'color: maroon'; style = style || 'color: maroon';
} }
...@@ -275,12 +274,15 @@ Strophe.log = function (level, msg) { _converse.log(level+' '+msg, level); }; ...@@ -275,12 +274,15 @@ Strophe.log = function (level, msg) { _converse.log(level+' '+msg, level); };
Strophe.error = function (msg) { _converse.log(msg, Strophe.LogLevel.ERROR); }; Strophe.error = function (msg) { _converse.log(msg, Strophe.LogLevel.ERROR); };
_converse.__ = function (str) { /**
/* Translate the given string based on the current locale. * Translate the given string based on the current locale.
* * Handles all MUC presence stanzas.
* Parameters: * @method __
* (String) str - The string to translate. * @private
* @memberOf _converse
* @param { String } str - The string to translate
*/ */
_converse.__ = function (str) {
if (_.isUndefined(i18n)) { if (_.isUndefined(i18n)) {
return str; return str;
} }
...@@ -565,12 +567,14 @@ _converse.initialize = async function (settings, callback) { ...@@ -565,12 +567,14 @@ _converse.initialize = async function (settings, callback) {
this.generateResource = () => `/converse.js-${Math.floor(Math.random()*139749528).toString()}`; this.generateResource = () => `/converse.js-${Math.floor(Math.random()*139749528).toString()}`;
this.sendCSI = function (stat) { /**
/* Send out a Chat Status Notification (XEP-0352) * Send out a Chat Status Notification (XEP-0352)
* * @private
* Parameters: * @method sendCSI
* (String) stat: The user's chat status * @memberOf _converse
* @param { String } stat - The user's chat status
*/ */
this.sendCSI = function (stat) {
_converse.api.send($build(stat, {xmlns: Strophe.NS.CSI})); _converse.api.send($build(stat, {xmlns: Strophe.NS.CSI}));
_converse.inactive = (stat === _converse.INACTIVE) ? true : false; _converse.inactive = (stat === _converse.INACTIVE) ? true : false;
}; };
...@@ -661,14 +665,15 @@ _converse.initialize = async function (settings, callback) { ...@@ -661,14 +665,15 @@ _converse.initialize = async function (settings, callback) {
}); });
}; };
this.rejectPresenceSubscription = function (jid, message) { /**
/* Reject or cancel another user's subscription to our presence updates. * Reject or cancel another user's subscription to our presence updates.
* * @method rejectPresenceSubscription
* Parameters: * @private
* (String) jid - The Jabber ID of the user whose subscription * @memberOf _converse
* is being canceled. * @param { String } jid - The Jabber ID of the user whose subscription is being canceled
* (String) message - An optional message to the user * @param { String } message - An optional message to the user
*/ */
this.rejectPresenceSubscription = function (jid, message) {
const pres = $pres({to: jid, type: "unsubscribed"}); const pres = $pres({to: jid, type: "unsubscribed"});
if (message && message !== "") { pres.c("status").t(message); } if (message && message !== "") { pres.c("status").t(message); }
_converse.api.send(pres); _converse.api.send(pres);
......
...@@ -23,6 +23,11 @@ converse.plugins.add('converse-disco', { ...@@ -23,6 +23,11 @@ converse.plugins.add('converse-disco', {
_converse.api.promises.add('discoInitialized'); _converse.api.promises.add('discoInitialized');
/**
* @class
* @namespace _converse.DiscoEntity
* @memberOf _converse
*/
_converse.DiscoEntity = Backbone.Model.extend({ _converse.DiscoEntity = Backbone.Model.extend({
/* A Disco Entity is a JID addressable entity that can be queried /* A Disco Entity is a JID addressable entity that can be queried
* for features. * for features.
...@@ -64,14 +69,15 @@ converse.plugins.add('converse-disco', { ...@@ -64,14 +69,15 @@ converse.plugins.add('converse-disco', {
this.items.fetch(); this.items.fetch();
}, },
async getIdentity (category, type) { /**
/* Returns a Promise which resolves with a map indicating * Returns a Promise which resolves with a map indicating
* whether a given identity is provided by this entity. * whether a given identity is provided by this entity.
* * @private
* Parameters: * @method _converse.DiscoEntity#getIdentity
* (String) category - The identity category * @param { String } category - The identity category
* (String) type - The identity type * @param { String } type - The identity type
*/ */
async getIdentity (category, type) {
await this.waitUntilFeaturesDiscovered; await this.waitUntilFeaturesDiscovered;
return this.identities.findWhere({ return this.identities.findWhere({
'category': category, 'category': category,
...@@ -79,13 +85,14 @@ converse.plugins.add('converse-disco', { ...@@ -79,13 +85,14 @@ converse.plugins.add('converse-disco', {
}); });
}, },
async hasFeature (feature) { /**
/* Returns a Promise which resolves with a map indicating * Returns a Promise which resolves with a map indicating
* whether a given feature is supported. * whether a given feature is supported.
* * @private
* Parameters: * @method _converse.DiscoEntity#hasFeature
* (String) feature - The feature that might be supported. * @param { String } feature - The feature that might be supported.
*/ */
async hasFeature (feature) {
await this.waitUntilFeaturesDiscovered await this.waitUntilFeaturesDiscovered
if (this.features.findWhere({'var': feature})) { if (this.features.findWhere({'var': feature})) {
return this; return this;
......
...@@ -291,14 +291,14 @@ converse.plugins.add('converse-muc', { ...@@ -291,14 +291,14 @@ converse.plugins.add('converse-muc', {
return this.get('name') || this.get('jid'); return this.get('name') || this.get('jid');
}, },
join (nick, password) { /**
/* Join the groupchat. * Join the groupchat.
* * @private
* Parameters: * @method _converse.ChatRoom#join
* (String) nick: The user's nickname * @param { String } nick - The user's nickname
* (String) password: Optional password, if required by * @param { String } password - Optional password, if required by the groupchat.
* the groupchat.
*/ */
join (nick, password) {
nick = nick ? nick : this.get('nick'); nick = nick ? nick : this.get('nick');
if (!nick) { if (!nick) {
throw new TypeError('join: You need to provide a valid nickname'); throw new TypeError('join: You need to provide a valid nickname');
...@@ -322,13 +322,12 @@ converse.plugins.add('converse-muc', { ...@@ -322,13 +322,12 @@ converse.plugins.add('converse-muc', {
return this; return this;
}, },
leave (exit_msg) {
/* Leave the groupchat. /* Leave the groupchat.
* * @private
* Parameters: * @method _converse.ChatRoom#leave
* (String) exit_msg: Optional message to indicate your * @param { string } exit_msg - Optional message to indicate your reason for leaving
* reason for leaving.
*/ */
leave (exit_msg) {
this.features.destroy(); this.features.destroy();
this.occupants.browserStorage._clear(); this.occupants.browserStorage._clear();
this.occupants.reset(); this.occupants.reset();
...@@ -479,12 +478,14 @@ converse.plugins.add('converse-muc', { ...@@ -479,12 +478,14 @@ converse.plugins.add('converse-muc', {
); );
}, },
directInvite (recipient, reason) { /**
/* Send a direct invitation as per XEP-0249 * Send a direct invitation as per XEP-0249
* Parameters: * @private
* (String) recipient - JID of the person being invited * @method _converse.ChatRoom#directInvite
* (String) reason - Optional reason for the invitation * @param { String } recipient - JID of the person being invited
* @param { String } reason - Optional reason for the invitation
*/ */
directInvite (recipient, reason) {
if (this.features.get('membersonly')) { if (this.features.get('membersonly')) {
// When inviting to a members-only groupchat, we first add // When inviting to a members-only groupchat, we first add
// the person to the member list by giving them an // the person to the member list by giving them an
...@@ -565,20 +566,17 @@ converse.plugins.add('converse-muc', { ...@@ -565,20 +566,17 @@ converse.plugins.add('converse-muc', {
this.features.save(attrs); this.features.save(attrs);
}, },
requestMemberList (affiliation) {
/* Send an IQ stanza to the server, asking it for the /* Send an IQ stanza to the server, asking it for the
* member-list of this groupchat. * member-list of this groupchat.
*
* See: https://xmpp.org/extensions/xep-0045.html#modifymember * See: https://xmpp.org/extensions/xep-0045.html#modifymember
* * @private
* Parameters: * @method _converse.ChatRoom#requestMemberList
* (String) affiliation: The specific member list to * @param { string } affiliation - The specific member list to
* fetch. 'admin', 'owner' or 'member'. * fetch. 'admin', 'owner' or 'member'.
* * @returns:
* Returns: * A promise which resolves once the list has been retrieved.
* A promise which resolves once the list has been
* retrieved.
*/ */
requestMemberList (affiliation) {
affiliation = affiliation || 'member'; affiliation = affiliation || 'member';
const iq = $iq({to: this.get('jid'), type: "get"}) const iq = $iq({to: this.get('jid'), type: "get"})
.c("query", {xmlns: Strophe.NS.MUC_ADMIN}) .c("query", {xmlns: Strophe.NS.MUC_ADMIN})
...@@ -586,28 +584,26 @@ converse.plugins.add('converse-muc', { ...@@ -586,28 +584,26 @@ converse.plugins.add('converse-muc', {
return _converse.api.sendIQ(iq); return _converse.api.sendIQ(iq);
}, },
setAffiliation (affiliation, members) { /**
/* Send IQ stanzas to the server to set an affiliation for * Send IQ stanzas to the server to set an affiliation for
* the provided JIDs. * the provided JIDs.
*
* See: https://xmpp.org/extensions/xep-0045.html#modifymember * See: https://xmpp.org/extensions/xep-0045.html#modifymember
* *
* XXX: Prosody doesn't accept multiple JIDs' affiliations * Prosody doesn't accept multiple JIDs' affiliations
* being set in one IQ stanza, so as a workaround we send * being set in one IQ stanza, so as a workaround we send
* a separate stanza for each JID. * a separate stanza for each JID.
* Related ticket: https://issues.prosody.im/345 * Related ticket: https://issues.prosody.im/345
* *
* Parameters: * @private
* (String) affiliation: The affiliation * @method _converse.ChatRoom#setAffiliation
* (Object) members: A map of jids, affiliations and * @param { string } affiliation - The affiliation
* @param { object } members - A map of jids, affiliations and
* optionally reasons. Only those entries with the * optionally reasons. Only those entries with the
* same affiliation as being currently set will be * same affiliation as being currently set will be considered.
* considered. * @returns
* * A promise which resolves and fails depending on the XMPP server response.
* Returns:
* A promise which resolves and fails depending on the
* XMPP server response.
*/ */
setAffiliation (affiliation, members) {
members = _.filter(members, (member) => members = _.filter(members, (member) =>
// We only want those members who have the right // We only want those members who have the right
// affiliation (or none, which implies the provided one). // affiliation (or none, which implies the provided one).
...@@ -618,18 +614,19 @@ converse.plugins.add('converse-muc', { ...@@ -618,18 +614,19 @@ converse.plugins.add('converse-muc', {
return Promise.all(promises); return Promise.all(promises);
}, },
saveConfiguration (form) { /**
/* Submit the groupchat configuration form by sending an IQ * Submit the groupchat configuration form by sending an IQ
* stanza to the server. * stanza to the server.
* * @private
* Returns a promise which resolves once the XMPP server * @method _converse.ChatRoom#saveConfiguration
* has return a response IQ. * @param { HTMLElement } form - The configuration form DOM element.
*
* Parameters:
* (HTMLElement) form: The configuration form DOM element.
* If no form is provided, the default configuration * If no form is provided, the default configuration
* values will be used. * values will be used.
* @returns { promise }
* Returns a promise which resolves once the XMPP server
* has return a response IQ.
*/ */
saveConfiguration (form) {
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
const inputs = form ? sizzle(':input:not([type=button]):not([type=submit])', form) : [], const inputs = form ? sizzle(':input:not([type=button]):not([type=submit])', form) : [],
configArray = _.map(inputs, u.webForm2xForm); configArray = _.map(inputs, u.webForm2xForm);
...@@ -689,20 +686,21 @@ converse.plugins.add('converse-muc', { ...@@ -689,20 +686,21 @@ converse.plugins.add('converse-muc', {
); );
}, },
sendConfiguration (config, callback, errback) { /**
/* Send an IQ stanza with the groupchat configuration. * Send an IQ stanza with the groupchat configuration.
* * @private
* Parameters: * @method _converse.ChatRoom#sendConfiguration
* (Array) config: The groupchat configuration * @param { Array } config - The groupchat configuration
* (Function) callback: Callback upon succesful IQ response * @param { Function } callback - Callback upon succesful IQ response
* The first parameter passed in is IQ containing the * The first parameter passed in is IQ containing the
* groupchat configuration. * groupchat configuration.
* The second is the response IQ from the server. * The second is the response IQ from the server.
* (Function) errback: Callback upon error IQ response * @param { Function } errback - Callback upon error IQ response
* The first parameter passed in is IQ containing the * The first parameter passed in is IQ containing the
* groupchat configuration. * groupchat configuration.
* The second is the response IQ from the server. * The second is the response IQ from the server.
*/ */
sendConfiguration (config, callback, errback) {
const iq = $iq({to: this.get('jid'), type: "set"}) const iq = $iq({to: this.get('jid'), type: "set"})
.c("query", {xmlns: Strophe.NS.MUC_OWNER}) .c("query", {xmlns: Strophe.NS.MUC_OWNER})
.c("x", {xmlns: Strophe.NS.XFORM, type: "submit"}); .c("x", {xmlns: Strophe.NS.XFORM, type: "submit"});
...@@ -712,13 +710,13 @@ converse.plugins.add('converse-muc', { ...@@ -712,13 +710,13 @@ converse.plugins.add('converse-muc', {
return _converse.api.sendIQ(iq).then(callback).catch(errback); return _converse.api.sendIQ(iq).then(callback).catch(errback);
}, },
saveAffiliationAndRole (pres) { /**
/* Parse the presence stanza for the current user's * Parse the presence stanza for the current user's affiliation.
* affiliation. * @private
* * @method _converse.ChatRoom#saveAffiliationAndRole
* Parameters: * @param { XMLElement } pres - A <presence> stanza.
* (XMLElement) pres: A <presence> stanza.
*/ */
saveAffiliationAndRole (pres) {
const item = sizzle(`x[xmlns="${Strophe.NS.MUC_USER}"] item`, pres).pop(); const item = sizzle(`x[xmlns="${Strophe.NS.MUC_USER}"] item`, pres).pop();
const is_self = pres.querySelector("status[code='110']"); const is_self = pres.querySelector("status[code='110']");
if (is_self && !_.isNil(item)) { if (is_self && !_.isNil(item)) {
...@@ -733,15 +731,16 @@ converse.plugins.add('converse-muc', { ...@@ -733,15 +731,16 @@ converse.plugins.add('converse-muc', {
} }
}, },
sendAffiliationIQ (affiliation, member) { /**
/* Send an IQ stanza specifying an affiliation change. * Send an IQ stanza specifying an affiliation change.
* * @private
* Paremeters: * @method _converse.ChatRoom#
* (String) affiliation: affiliation (could also be stored * @param { String } affiliation: affiliation
* on the member object). * (could also be stored on the member object).
* (Object) member: Map containing the member's jid and * @param { Object } member: Map containing the member's jid and
* optionally a reason and affiliation. * optionally a reason and affiliation.
*/ */
sendAffiliationIQ (affiliation, member) {
const iq = $iq({to: this.get('jid'), type: "set"}) const iq = $iq({to: this.get('jid'), type: "set"})
.c("query", {xmlns: Strophe.NS.MUC_ADMIN}) .c("query", {xmlns: Strophe.NS.MUC_ADMIN})
.c("item", { .c("item", {
...@@ -755,17 +754,17 @@ converse.plugins.add('converse-muc', { ...@@ -755,17 +754,17 @@ converse.plugins.add('converse-muc', {
return _converse.api.sendIQ(iq); return _converse.api.sendIQ(iq);
}, },
setAffiliations (members) { /**
/* Send IQ stanzas to the server to modify the * Send IQ stanzas to the server to modify the
* affiliations in this groupchat. * affiliations in this groupchat.
*
* See: https://xmpp.org/extensions/xep-0045.html#modifymember * See: https://xmpp.org/extensions/xep-0045.html#modifymember
* * @private
* Parameters: * @method _converse.ChatRoom#setAffiliations
* (Object) members: A map of jids, affiliations and optionally reasons * @param { object } members - A map of jids, affiliations and optionally reasons
* (Function) onSuccess: callback for a succesful response * @param { function } onSuccess - callback for a succesful response
* (Function) onError: callback for an error response * @param { function } onError - callback for an error response
*/ */
setAffiliations (members) {
const affiliations = _.uniq(_.map(members, 'affiliation')); const affiliations = _.uniq(_.map(members, 'affiliation'));
return Promise.all(_.map(affiliations, _.partial(this.setAffiliation.bind(this), _, members))); return Promise.all(_.map(affiliations, _.partial(this.setAffiliation.bind(this), _, members)));
}, },
...@@ -787,39 +786,39 @@ converse.plugins.add('converse-muc', { ...@@ -787,39 +786,39 @@ converse.plugins.add('converse-muc', {
return [].concat.apply([], result).filter(p => p); return [].concat.apply([], result).filter(p => p);
}, },
updateMemberLists (members, affiliations, deltaFunc) { /**
/* Fetch the lists of users with the given affiliations. * Fetch the lists of users with the given affiliations.
* Then compute the delta between those users and * Then compute the delta between those users and
* the passed in members, and if it exists, send the delta * the passed in members, and if it exists, send the delta
* to the XMPP server to update the member list. * to the XMPP server to update the member list.
* * @private
* Parameters: * @method _converse.ChatRoom#updateMemberLists
* (Object) members: Map of member jids and affiliations. * @param { object } members - Map of member jids and affiliations.
* (String|Array) affiliation: An array of affiliations or * @param { string|array } affiliation - An array of affiliations or
* a string if only one affiliation. * a string if only one affiliation.
* (Function) deltaFunc: The function to compute the delta * @param { function } deltaFunc - The function to compute the delta
* between old and new member lists. * between old and new member lists.
* * @returns { promise }
* Returns:
* A promise which is resolved once the list has been * A promise which is resolved once the list has been
* updated or once it's been established there's no need * updated or once it's been established there's no need
* to update the list. * to update the list.
*/ */
updateMemberLists (members, affiliations, deltaFunc) {
this.getJidsWithAffiliations(affiliations) this.getJidsWithAffiliations(affiliations)
.then(old_members => this.setAffiliations(deltaFunc(members, old_members))) .then(old_members => this.setAffiliations(deltaFunc(members, old_members)))
.then(() => this.occupants.fetchMembers()) .then(() => this.occupants.fetchMembers())
.catch(_.partial(_converse.log, _, Strophe.LogLevel.ERROR)); .catch(_.partial(_converse.log, _, Strophe.LogLevel.ERROR));
}, },
async checkForReservedNick () { /**
/* Use service-discovery to ask the XMPP server whether * Use service-discovery to ask the XMPP server whether
* this user has a reserved nickname for this groupchat. * this user has a reserved nickname for this groupchat.
* If so, we'll use that, otherwise we render the nickname form. * If so, we'll use that, otherwise we render the nickname form.
* * @private
* Parameters: * @method _converse.ChatRoom#checkForReservedNick
* (Function) callback: Callback upon succesful IQ response * @returns { promise } A promise which resolves with the response IQ
* (Function) errback: Callback upon error IQ response
*/ */
async checkForReservedNick () {
const iq = await _converse.api.sendIQ( const iq = await _converse.api.sendIQ(
$iq({ $iq({
'to': this.get('jid'), 'to': this.get('jid'),
...@@ -887,13 +886,14 @@ converse.plugins.add('converse-muc', { ...@@ -887,13 +886,14 @@ converse.plugins.add('converse-muc', {
} }
}, },
updateOccupantsOnPresence (pres) { /**
/* Given a presence stanza, update the occupant model * Given a presence stanza, update the occupant model
* based on its contents. * based on its contents.
* * @private
* Parameters: * @method _converse.ChatRoom#updateOccupantsOnPresence
* (XMLElement) pres: The presence stanza * @param { XMLElement } pres - The presence stanza
*/ */
updateOccupantsOnPresence (pres) {
const data = this.parsePresence(pres); const data = this.parsePresence(pres);
if (data.type === 'error' || (!data.jid && !data.nick)) { if (data.type === 'error' || (!data.jid && !data.nick)) {
return true; return true;
...@@ -989,12 +989,13 @@ converse.plugins.add('converse-muc', { ...@@ -989,12 +989,13 @@ converse.plugins.add('converse-muc', {
acknowledged[xmlns="${Strophe.NS.MARKERS}"]`, stanza).length > 0; acknowledged[xmlns="${Strophe.NS.MARKERS}"]`, stanza).length > 0;
}, },
subjectChangeHandled (attrs) { /**
/* Handle a subject change and return `true` if so. * Handle a subject change and return `true` if so.
* * @private
* Parameters: * @method _converse.ChatRoom#subjectChangeHandled
* (Object) attrs: The message attributes * @param { object } attrs - The message attributes
*/ */
subjectChangeHandled (attrs) {
if (attrs.subject && !attrs.thread && !attrs.message) { if (attrs.subject && !attrs.thread && !attrs.message) {
// https://xmpp.org/extensions/xep-0045.html#subject-mod // https://xmpp.org/extensions/xep-0045.html#subject-mod
// ----------------------------------------------------- // -----------------------------------------------------
...@@ -1007,13 +1008,14 @@ converse.plugins.add('converse-muc', { ...@@ -1007,13 +1008,14 @@ converse.plugins.add('converse-muc', {
return false; return false;
}, },
ignorableCSN (attrs) { /**
/* Is this a chat state notification that can be ignored, * Is this a chat state notification that can be ignored,
* because it's old or because it's from us. * because it's old or because it's from us.
* * @private
* Parameters: * @method _converse.ChatRoom#ignorableCSN
* (Object) attrs: The message attributes * @param { Object } attrs - The message attributes
*/ */
ignorableCSN (attrs) {
const is_csn = u.isOnlyChatStateNotification(attrs), const is_csn = u.isOnlyChatStateNotification(attrs),
own_message = Strophe.getResourceFromJid(attrs.from) == this.get('nick'); own_message = Strophe.getResourceFromJid(attrs.from) == this.get('nick');
return is_csn && (attrs.is_delayed || own_message); return is_csn && (attrs.is_delayed || own_message);
...@@ -1038,12 +1040,13 @@ converse.plugins.add('converse-muc', { ...@@ -1038,12 +1040,13 @@ converse.plugins.add('converse-muc', {
return attrs; return attrs;
}, },
async onMessage (stanza) { /**
/* Handler for all MUC messages sent to this groupchat. * Handler for all MUC messages sent to this groupchat.
* * @private
* Parameters: * @method _converse.ChatRoom#onMessage
* (XMLElement) stanza: The message stanza. * @param { XMLElement } stanza - The message stanza.
*/ */
async onMessage (stanza) {
this.fetchFeaturesIfConfigurationChanged(stanza); this.fetchFeaturesIfConfigurationChanged(stanza);
const original_stanza = stanza, const original_stanza = stanza,
...@@ -1076,12 +1079,13 @@ converse.plugins.add('converse-muc', { ...@@ -1076,12 +1079,13 @@ converse.plugins.add('converse-muc', {
_converse.api.trigger('message', {'stanza': original_stanza, 'chatbox': this}); _converse.api.trigger('message', {'stanza': original_stanza, 'chatbox': this});
}, },
onPresence (pres) { /**
/* Handles all MUC presence stanzas. * Handles all MUC presence stanzas.
* * @private
* Parameters: * @method _converse.ChatRoom#onPresence
* (XMLElement) pres: The stanza * @param { XMLElement } pres - The stanza
*/ */
onPresence (pres) {
if (pres.getAttribute('type') === 'error') { if (pres.getAttribute('type') === 'error') {
this.save('connection_status', converse.ROOMSTATUS.DISCONNECTED); this.save('connection_status', converse.ROOMSTATUS.DISCONNECTED);
return; return;
...@@ -1096,9 +1100,8 @@ converse.plugins.add('converse-muc', { ...@@ -1096,9 +1100,8 @@ converse.plugins.add('converse-muc', {
} }
}, },
onOwnPresence (pres) { /**
/* Handles a received presence relating to the current * Handles a received presence relating to the current user.
* user.
* *
* For locked groupchats (which are by definition "new"), the * For locked groupchats (which are by definition "new"), the
* groupchat will either be auto-configured or created instantly * groupchat will either be auto-configured or created instantly
...@@ -1108,10 +1111,11 @@ converse.plugins.add('converse-muc', { ...@@ -1108,10 +1111,11 @@ converse.plugins.add('converse-muc', {
* If the groupchat is not locked, then the groupchat will be * If the groupchat is not locked, then the groupchat will be
* auto-configured only if applicable and if the current * auto-configured only if applicable and if the current
* user is the groupchat's owner. * user is the groupchat's owner.
* * @private
* Parameters: * @method _converse.ChatRoom#onOwnPresence
* (XMLElement) pres: The stanza * @param { XMLElement } pres - The stanza
*/ */
onOwnPresence (pres) {
this.saveAffiliationAndRole(pres); this.saveAffiliationAndRole(pres);
const locked_room = pres.querySelector("status[code='201']"); const locked_room = pres.querySelector("status[code='201']");
...@@ -1148,13 +1152,14 @@ converse.plugins.add('converse-muc', { ...@@ -1148,13 +1152,14 @@ converse.plugins.add('converse-muc', {
this.save('connection_status', converse.ROOMSTATUS.ENTERED); this.save('connection_status', converse.ROOMSTATUS.ENTERED);
}, },
isUserMentioned (message) { /**
/* Returns a boolean to indicate whether the current user * Returns a boolean to indicate whether the current user
* was mentioned in a message. * was mentioned in a message.
* * @private
* Parameters: * @method _converse.ChatRoom#isUserMentioned
* (String): The text message * @param { String } - The text message
*/ */
isUserMentioned (message) {
const nick = this.get('nick'); const nick = this.get('nick');
if (message.get('references').length) { if (message.get('references').length) {
const mentions = message.get('references').filter(ref => (ref.type === 'mention')).map(ref => ref.value); const mentions = message.get('references').filter(ref => (ref.type === 'mention')).map(ref => ref.value);
...@@ -1164,13 +1169,12 @@ converse.plugins.add('converse-muc', { ...@@ -1164,13 +1169,12 @@ converse.plugins.add('converse-muc', {
} }
}, },
incrementUnreadMsgCounter (message) { /* Given a newly received message, update the unread counter if necessary.
/* Given a newly received message, update the unread counter if * @private
* necessary. * @method _converse.ChatRoom#incrementUnreadMsgCounter
* * @param { XMLElement } - The <messsage> stanza
* Parameters:
* (XMLElement): The <messsage> stanza
*/ */
incrementUnreadMsgCounter (message) {
if (!message) { return; } if (!message) { return; }
const body = message.get('message'); const body = message.get('message');
if (_.isNil(body)) { return; } if (_.isNil(body)) { return; }
...@@ -1311,14 +1315,14 @@ converse.plugins.add('converse-muc', { ...@@ -1311,14 +1315,14 @@ converse.plugins.add('converse-muc', {
}); });
_converse.onDirectMUCInvitation = function (message) { /**
/* A direct MUC invitation to join a groupchat has been received * A direct MUC invitation to join a groupchat has been received
* See XEP-0249: Direct MUC invitations. * See XEP-0249: Direct MUC invitations.
* * @private
* Parameters: * @method _converse.ChatRoom#onDirectMUCInvitation
* (XMLElement) message: The message stanza containing the * @param { XMLElement } message - The message stanza containing the invitation.
* invitation.
*/ */
_converse.onDirectMUCInvitation = function (message) {
const x_el = sizzle('x[xmlns="jabber:x:conference"]', message).pop(), const x_el = sizzle('x[xmlns="jabber:x:conference"]', message).pop(),
from = Strophe.getBareJidFromJid(message.getAttribute('from')), from = Strophe.getBareJidFromJid(message.getAttribute('from')),
room_jid = x_el.getAttribute('jid'), room_jid = x_el.getAttribute('jid'),
......
...@@ -45,10 +45,13 @@ converse.plugins.add('converse-roster', { ...@@ -45,10 +45,13 @@ converse.plugins.add('converse-roster', {
}; };
_converse.initRoster = function () { /**
/* Initialize the Bakcbone collections that represent the contats * Initialize the Bakcbone collections that represent the contats
* roster and the roster groups. * roster and the roster groups.
* @private
* @method _converse.initRoster
*/ */
_converse.initRoster = function () {
const storage = _converse.config.get('storage'); const storage = _converse.config.get('storage');
_converse.roster = new _converse.RosterContacts(); _converse.roster = new _converse.RosterContacts();
_converse.roster.browserStorage = new Backbone.BrowserStorage[storage]( _converse.roster.browserStorage = new Backbone.BrowserStorage[storage](
...@@ -75,15 +78,16 @@ converse.plugins.add('converse-roster', { ...@@ -75,15 +78,16 @@ converse.plugins.add('converse-roster', {
}; };
_converse.populateRoster = async function (ignore_cache=false) { /**
/* Fetch all the roster groups, and then the roster contacts. * Fetch all the roster groups, and then the roster contacts.
* Emit an event after fetching is done in each case. * Emit an event after fetching is done in each case.
* * @private
* Parameters: * @method _converse.populateRoster
* (Bool) ignore_cache - If set to to true, the local cache * @param { Bool } ignore_cache - If set to to true, the local cache
* will be ignored it's guaranteed that the XMPP server * will be ignored it's guaranteed that the XMPP server
* will be queried for the roster. * will be queried for the roster.
*/ */
_converse.populateRoster = async function (ignore_cache=false) {
if (ignore_cache) { if (ignore_cache) {
_converse.send_initial_presence = true; _converse.send_initial_presence = true;
try { try {
...@@ -272,13 +276,14 @@ converse.plugins.add('converse-roster', { ...@@ -272,13 +276,14 @@ converse.plugins.add('converse-roster', {
return this.vcard.get('fullname'); return this.vcard.get('fullname');
}, },
subscribe (message) { /**
/* Send a presence subscription request to this roster contact * Send a presence subscription request to this roster contact
* * @private
* Parameters: * @method _converse.RosterContacts#subscribe
* (String) message - An optional message to explain the * @param { String } message - An optional message to explain the
* reason for the subscription request. * reason for the subscription request.
*/ */
subscribe (message) {
const pres = $pres({to: this.get('jid'), type: "subscribe"}); const pres = $pres({to: this.get('jid'), type: "subscribe"});
if (message && message !== "") { if (message && message !== "") {
pres.c("status").t(message).up(); pres.c("status").t(message).up();
...@@ -292,46 +297,55 @@ converse.plugins.add('converse-roster', { ...@@ -292,46 +297,55 @@ converse.plugins.add('converse-roster', {
return this; return this;
}, },
ackSubscribe () { /**
/* Upon receiving the presence stanza of type "subscribed", * Upon receiving the presence stanza of type "subscribed",
* the user SHOULD acknowledge receipt of that subscription * the user SHOULD acknowledge receipt of that subscription
* state notification by sending a presence stanza of type * state notification by sending a presence stanza of type
* "subscribe" to the contact * "subscribe" to the contact
* @private
* @method _converse.RosterContacts#ackSubscribe
*/ */
ackSubscribe () {
_converse.api.send($pres({ _converse.api.send($pres({
'type': 'subscribe', 'type': 'subscribe',
'to': this.get('jid') 'to': this.get('jid')
})); }));
}, },
ackUnsubscribe () { /**
/* Upon receiving the presence stanza of type "unsubscribed", * Upon receiving the presence stanza of type "unsubscribed",
* the user SHOULD acknowledge receipt of that subscription state * the user SHOULD acknowledge receipt of that subscription state
* notification by sending a presence stanza of type "unsubscribe" * notification by sending a presence stanza of type "unsubscribe"
* this step lets the user's server know that it MUST no longer * this step lets the user's server know that it MUST no longer
* send notification of the subscription state change to the user. * send notification of the subscription state change to the user.
* Parameters: * @private
* (String) jid - The Jabber ID of the user who is unsubscribing * @method _converse.RosterContacts#ackUnsubscribe
* @param { String } jid - The Jabber ID of the user who is unsubscribing
*/ */
ackUnsubscribe () {
_converse.api.send($pres({'type': 'unsubscribe', 'to': this.get('jid')})); _converse.api.send($pres({'type': 'unsubscribe', 'to': this.get('jid')}));
this.removeFromRoster(); this.removeFromRoster();
this.destroy(); this.destroy();
}, },
unauthorize (message) { /**
/* Unauthorize this contact's presence subscription * Unauthorize this contact's presence subscription
* Parameters: * @private
* (String) message - Optional message to send to the person being unauthorized * @method _converse.RosterContacts#unauthorize
* @param { String } message - Optional message to send to the person being unauthorized
*/ */
unauthorize (message) {
_converse.rejectPresenceSubscription(this.get('jid'), message); _converse.rejectPresenceSubscription(this.get('jid'), message);
return this; return this;
}, },
authorize (message) { /**
/* Authorize presence subscription * Authorize presence subscription
* Parameters: * @private
* (String) message - Optional message to send to the person being authorized * @method _converse.RosterContacts#authorize
* @param { String } message - Optional message to send to the person being authorized
*/ */
authorize (message) {
const pres = $pres({'to': this.get('jid'), 'type': "subscribed"}); const pres = $pres({'to': this.get('jid'), 'type': "subscribed"});
if (message && message !== "") { if (message && message !== "") {
pres.c("status").t(message); pres.c("status").t(message);
...@@ -340,11 +354,13 @@ converse.plugins.add('converse-roster', { ...@@ -340,11 +354,13 @@ converse.plugins.add('converse-roster', {
return this; return this;
}, },
removeFromRoster () { /**
/* Instruct the XMPP server to remove this contact from our roster * Instruct the XMPP server to remove this contact from our roster
* Parameters: * @private
* (Function) callback * @method _converse.RosterContacts#
* @returns { Promise }
*/ */
removeFromRoster () {
const iq = $iq({type: 'set'}) const iq = $iq({type: 'set'})
.c('query', {xmlns: Strophe.NS.ROSTER}) .c('query', {xmlns: Strophe.NS.ROSTER})
.c('item', {jid: this.get('jid'), subscription: "remove"}); .c('item', {jid: this.get('jid'), subscription: "remove"});
...@@ -352,7 +368,11 @@ converse.plugins.add('converse-roster', { ...@@ -352,7 +368,11 @@ converse.plugins.add('converse-roster', {
} }
}); });
/**
* @class
* @namespace _converse.RosterContacts
* @memberOf _converse
*/
_converse.RosterContacts = Backbone.Collection.extend({ _converse.RosterContacts = Backbone.Collection.extend({
model: _converse.RosterContact, model: _converse.RosterContact,
...@@ -460,17 +480,18 @@ converse.plugins.add('converse-roster', { ...@@ -460,17 +480,18 @@ converse.plugins.add('converse-roster', {
return u.isSameBareJID(jid, _converse.connection.jid); return u.isSameBareJID(jid, _converse.connection.jid);
}, },
addAndSubscribe (jid, name, groups, message, attributes) { /**
/* Add a roster contact and then once we have confirmation from * Add a roster contact and then once we have confirmation from
* the XMPP server we subscribe to that contact's presence updates. * the XMPP server we subscribe to that contact's presence updates.
* Parameters: * @private
* (String) jid - The Jabber ID of the user being added and subscribed to. * @method _converse.RosterContacts#addAndSubscribe
* (String) name - The name of that user * @param { String } jid - The Jabber ID of the user being added and subscribed to.
* (Array of Strings) groups - Any roster groups the user might belong to * @param { String } name - The name of that user
* (String) message - An optional message to explain the * @param { Array.String } groups - Any roster groups the user might belong to
* reason for the subscription request. * @param { String } message - An optional message to explain the reason for the subscription request.
* (Object) attributes - Any additional attributes to be stored on the user's model. * @param { Object } attributes - Any additional attributes to be stored on the user's model.
*/ */
addAndSubscribe (jid, name, groups, message, attributes) {
const handler = (contact) => { const handler = (contact) => {
if (contact instanceof _converse.RosterContact) { if (contact instanceof _converse.RosterContact) {
contact.subscribe(message); contact.subscribe(message);
...@@ -479,16 +500,17 @@ converse.plugins.add('converse-roster', { ...@@ -479,16 +500,17 @@ converse.plugins.add('converse-roster', {
this.addContactToRoster(jid, name, groups, attributes).then(handler, handler); this.addContactToRoster(jid, name, groups, attributes).then(handler, handler);
}, },
sendContactAddIQ (jid, name, groups) { /**
/* Send an IQ stanza to the XMPP server to add a new roster contact. * Send an IQ stanza to the XMPP server to add a new roster contact.
* * @private
* Parameters: * @method _converse.RosterContacts#sendContactAddIQ
* (String) jid - The Jabber ID of the user being added * @param { String } jid - The Jabber ID of the user being added
* (String) name - The name of that user * @param { String } name - The name of that user
* (Array of Strings) groups - Any roster groups the user might belong to * @param { Array.String } groups - Any roster groups the user might belong to
* (Function) callback - A function to call once the IQ is returned * @param { Function } callback - A function to call once the IQ is returned
* (Function) errback - A function to call if an error occurred * @param { Function } errback - A function to call if an error occurred
*/ */
sendContactAddIQ (jid, name, groups) {
name = _.isEmpty(name) ? null : name; name = _.isEmpty(name) ? null : name;
const iq = $iq({'type': 'set'}) const iq = $iq({'type': 'set'})
.c('query', {'xmlns': Strophe.NS.ROSTER}) .c('query', {'xmlns': Strophe.NS.ROSTER})
...@@ -497,18 +519,18 @@ converse.plugins.add('converse-roster', { ...@@ -497,18 +519,18 @@ converse.plugins.add('converse-roster', {
return _converse.api.sendIQ(iq); return _converse.api.sendIQ(iq);
}, },
async addContactToRoster (jid, name, groups, attributes) { /**
/* Adds a RosterContact instance to _converse.roster and * Adds a RosterContact instance to _converse.roster and
* registers the contact on the XMPP server. * registers the contact on the XMPP server.
* Returns a promise which is resolved once the XMPP server has * Returns a promise which is resolved once the XMPP server has responded.
* responded. * @private
* * @method _converse.RosterContacts#addContactToRoster
* Parameters: * @param { String } jid - The Jabber ID of the user being added and subscribed to.
* (String) jid - The Jabber ID of the user being added and subscribed to. * @param { String } name - The name of that user
* (String) name - The name of that user * @param { Array.String } groups - Any roster groups the user might belong to
* (Array of Strings) groups - Any roster groups the user might belong to * @param { Object } attributes - Any additional attributes to be stored on the user's model.
* (Object) attributes - Any additional attributes to be stored on the user's model.
*/ */
async addContactToRoster (jid, name, groups, attributes) {
groups = groups || []; groups = groups || [];
try { try {
await this.sendContactAddIQ(jid, name, groups); await this.sendContactAddIQ(jid, name, groups);
...@@ -551,13 +573,14 @@ converse.plugins.add('converse-roster', { ...@@ -551,13 +573,14 @@ converse.plugins.add('converse-roster', {
return _.sum(this.models.filter((model) => !_.includes(ignored, model.presence.get('show')))); return _.sum(this.models.filter((model) => !_.includes(ignored, model.presence.get('show'))));
}, },
onRosterPush (iq) { /**
/* Handle roster updates from the XMPP server. * Handle roster updates from the XMPP server.
* See: https://xmpp.org/rfcs/rfc6121.html#roster-syntax-actions-push * See: https://xmpp.org/rfcs/rfc6121.html#roster-syntax-actions-push
* * @private
* Parameters: * @method _converse.RosterContacts#onRosterPush
* (XMLElement) IQ - The IQ stanza received from the XMPP server. * @param { XMLElement } IQ - The IQ stanza received from the XMPP server.
*/ */
onRosterPush (iq) {
const id = iq.getAttribute('id'); const id = iq.getAttribute('id');
const from = iq.getAttribute('from'); const from = iq.getAttribute('from');
if (from && from !== _converse.bare_jid) { if (from && from !== _converse.bare_jid) {
......
...@@ -100,12 +100,6 @@ converse.plugins.add('converse-vcard', { ...@@ -100,12 +100,6 @@ converse.plugins.add('converse-vcard', {
} }
async function getVCard (_converse, jid) { async function getVCard (_converse, jid) {
/* Request the VCard of another user. Returns a promise.
*
* Parameters:
* (String) jid - The Jabber ID of the user whose VCard
* is being requested.
*/
const to = Strophe.getBareJidFromJid(jid) === _converse.bare_jid ? null : jid; const to = Strophe.getBareJidFromJid(jid) === _converse.bare_jid ? null : jid;
let iq; let iq;
try { try {
......
...@@ -40680,13 +40680,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha ...@@ -40680,13 +40680,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha
return false; return false;
}, },
createMessageStanza(message) { /**
/* Given a _converse.Message Backbone.Model, return the XML * Given a {@link _converse.Message} return the XML stanza that represents it.
* stanza that represents it. * @private
* * @method _converse.ChatBox#createMessageStanza
* Parameters: * @param { _converse.Message } message - The message object
* (Object) message - The Backbone.Model representing the message
*/ */
createMessageStanza(message) {
const stanza = $msg({ const stanza = $msg({
'from': _converse.connection.jid, 'from': _converse.connection.jid,
'to': this.get('jid'), 'to': this.get('jid'),
...@@ -40896,13 +40896,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha ...@@ -40896,13 +40896,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha
}); });
}, },
getStanzaIDs(stanza) { /**
/* Extract the XEP-0359 stanza IDs from the passed in stanza * Extract the XEP-0359 stanza IDs from the passed in stanza
* and return a map containing them. * and return a map containing them.
* * @private
* Parameters: * @method _converse.ChatBox#getStanzaIDs
* (XMLElement) stanza - The message stanza * @param { XMLElement } stanza - The message stanza
*/ */
getStanzaIDs(stanza) {
const attrs = {}; const attrs = {};
const stanza_ids = sizzle(`stanza-id[xmlns="${Strophe.NS.SID}"]`, stanza); const stanza_ids = sizzle(`stanza-id[xmlns="${Strophe.NS.SID}"]`, stanza);
...@@ -40930,18 +40931,17 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha ...@@ -40930,18 +40931,17 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha
return !_.isNil(sizzle(`result[xmlns="${Strophe.NS.MAM}"]`, original_stanza).pop()); return !_.isNil(sizzle(`result[xmlns="${Strophe.NS.MAM}"]`, original_stanza).pop());
}, },
getMessageAttributesFromStanza(stanza, original_stanza) { /**
/* Parses a passed in message stanza and returns an object * Parses a passed in message stanza and returns an object
* of attributes. * of attributes.
* * @private
* Parameters: * @method _converse.ChatBox#getMessageAttributesFromStanza
* (XMLElement) stanza - The message stanza * @param { XMLElement } stanza - The message stanza
* (XMLElement) delay - The <delay> node from the * @param { XMLElement } delay - The <delay> node from the stanza, if there was one.
* stanza, if there was one. * @param { XMLElement } original_stanza - The original stanza, that contains the
* (XMLElement) original_stanza - The original stanza, * message stanza, if it was contained, otherwise it's the message stanza itself.
* that contains the message stanza, if it was
* contained, otherwise it's the message stanza itself.
*/ */
getMessageAttributesFromStanza(stanza, original_stanza) {
const spoiler = sizzle(`spoiler[xmlns="${Strophe.NS.SPOILER}"]`, original_stanza).pop(), const spoiler = sizzle(`spoiler[xmlns="${Strophe.NS.SPOILER}"]`, original_stanza).pop(),
delay = sizzle(`delay[xmlns="${Strophe.NS.DELAY}"]`, original_stanza).pop(), delay = sizzle(`delay[xmlns="${Strophe.NS.DELAY}"]`, original_stanza).pop(),
text = _converse.chatboxes.getMessageBody(stanza) || undefined, text = _converse.chatboxes.getMessageBody(stanza) || undefined,
...@@ -41152,13 +41152,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha ...@@ -41152,13 +41152,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha
} }
}, },
async onMessage(stanza) { /**
/* Handler method for all incoming single-user chat "message" * Handler method for all incoming single-user chat "message" stanzas.
* stanzas. * @private
* * @method _converse.ChatBox#onMessage
* Parameters: * @param { XMLElement } stanza - The incoming message stanza
* (XMLElement) stanza - The incoming message stanza
*/ */
async onMessage(stanza) {
let to_jid = stanza.getAttribute('to'); let to_jid = stanza.getAttribute('to');
const to_resource = Strophe.getResourceFromJid(to_jid); const to_resource = Strophe.getResourceFromJid(to_jid);
...@@ -41260,18 +41260,19 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha ...@@ -41260,18 +41260,19 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha
}); });
}, },
/**
* Returns a chat box or optionally return a newly
* created one if one doesn't exist.
* @private
* @method _converse.ChatBox#getChatBox
* @param { string } jid - The JID of the user whose chat box we want
* @param { boolean } create - Should a new chat box be created if none exists?
* @param { object } attrs - Optional chat box atributes.
*/
getChatBox(jid) { getChatBox(jid) {
let attrs = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; let attrs = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
let create = arguments.length > 2 ? arguments[2] : undefined; let create = arguments.length > 2 ? arguments[2] : undefined;
/* Returns a chat box or optionally return a newly
* created one if one doesn't exist.
*
* Parameters:
* (String) jid - The JID of the user whose chat box we want
* (Boolean) create - Should a new chat box be created if none exists?
* (Object) attrs - Optional chat box atributes.
*/
if (_.isObject(jid)) { if (_.isObject(jid)) {
create = attrs; create = attrs;
attrs = jid; attrs = jid;
...@@ -41344,8 +41345,6 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha ...@@ -41344,8 +41345,6 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha
_converse.api.listen.on('pluginsInitialized', () => { _converse.api.listen.on('pluginsInitialized', () => {
_converse.chatboxes = new _converse.ChatBoxes(); _converse.chatboxes = new _converse.ChatBoxes();
_converse.emit('chatBoxesInitialized');
/** /**
* Triggered once the _converse.ChatBoxes collection has been initialized. * Triggered once the _converse.ChatBoxes collection has been initialized.
* @event _converse#chatBoxesInitialized * @event _converse#chatBoxesInitialized
...@@ -41353,7 +41352,6 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha ...@@ -41353,7 +41352,6 @@ _converse_core__WEBPACK_IMPORTED_MODULE_2__["default"].plugins.add('converse-cha
* @example _converse.api.waitUntil('chatBoxesInitialized').then(() => { ... }); * @example _converse.api.waitUntil('chatBoxesInitialized').then(() => { ... });
*/ */
_converse.api.trigger('chatBoxesInitialized'); _converse.api.trigger('chatBoxesInitialized');
}); });
...@@ -41741,23 +41739,22 @@ _converse.default_settings = { ...@@ -41741,23 +41739,22 @@ _converse.default_settings = {
websocket_url: undefined, websocket_url: undefined,
whitelisted_plugins: [] whitelisted_plugins: []
}; };
/**
_converse.log = function (message, level) { * Logs messages to the browser's developer console.
let style = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : '';
/* 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.
*
* Available loglevels are 0 for 'debug', 1 for 'info', 2 for 'warn', * Available loglevels are 0 for 'debug', 1 for 'info', 2 for 'warn',
* 3 for 'error' and 4 for 'fatal'. * 3 for 'error' and 4 for 'fatal'.
*
* When using the 'error' or 'warn' loglevels, a full stacktrace will be * When using the 'error' or 'warn' loglevels, a full stacktrace will be
* logged as well. * logged as well.
* @method log
* @private
* @memberOf _converse
* @param { string } message - The message to be logged
* @param { integer } level - The loglevel which allows for filtering of log messages
*/ */
_converse.log = function (message, level) {
let style = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : '';
if (level === strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].LogLevel.ERROR || level === strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].LogLevel.FATAL) { if (level === strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].LogLevel.ERROR || level === strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].LogLevel.FATAL) {
style = style || 'color: maroon'; style = style || 'color: maroon';
} }
...@@ -41801,13 +41798,17 @@ strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].log = function (level, msg) { ...@@ -41801,13 +41798,17 @@ strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].log = function (level, msg) {
strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].error = function (msg) { strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].error = function (msg) {
_converse.log(msg, strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].LogLevel.ERROR); _converse.log(msg, strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].LogLevel.ERROR);
}; };
/**
* Translate the given string based on the current locale.
* Handles all MUC presence stanzas.
* @method __
* @private
* @memberOf _converse
* @param { String } str - The string to translate
*/
_converse.__ = function (str) { _converse.__ = function (str) {
/* Translate the given string based on the current locale.
*
* Parameters:
* (String) str - The string to translate.
*/
if (_lodash_noconflict__WEBPACK_IMPORTED_MODULE_4___default.a.isUndefined(_i18n__WEBPACK_IMPORTED_MODULE_6__["default"])) { if (_lodash_noconflict__WEBPACK_IMPORTED_MODULE_4___default.a.isUndefined(_i18n__WEBPACK_IMPORTED_MODULE_6__["default"])) {
return str; return str;
} }
...@@ -42095,13 +42096,16 @@ _converse.initialize = async function (settings, callback) { ...@@ -42095,13 +42096,16 @@ _converse.initialize = async function (settings, callback) {
// ---------------------- // ----------------------
this.generateResource = () => `/converse.js-${Math.floor(Math.random() * 139749528).toString()}`; this.generateResource = () => `/converse.js-${Math.floor(Math.random() * 139749528).toString()}`;
/**
* Send out a Chat Status Notification (XEP-0352)
* @private
* @method sendCSI
* @memberOf _converse
* @param { String } stat - The user's chat status
*/
this.sendCSI = function (stat) { this.sendCSI = function (stat) {
/* Send out a Chat Status Notification (XEP-0352)
*
* Parameters:
* (String) stat: The user's chat status
*/
_converse.api.send(Object(strophe_js__WEBPACK_IMPORTED_MODULE_0__["$build"])(stat, { _converse.api.send(Object(strophe_js__WEBPACK_IMPORTED_MODULE_0__["$build"])(stat, {
xmlns: strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].NS.CSI xmlns: strophe_js__WEBPACK_IMPORTED_MODULE_0__["Strophe"].NS.CSI
})); }));
...@@ -42204,15 +42208,17 @@ _converse.initialize = async function (settings, callback) { ...@@ -42204,15 +42208,17 @@ _converse.initialize = async function (settings, callback) {
'message': message 'message': message
}); });
}; };
/**
* Reject or cancel another user's subscription to our presence updates.
* @method rejectPresenceSubscription
* @private
* @memberOf _converse
* @param { String } jid - The Jabber ID of the user whose subscription is being canceled
* @param { String } message - An optional message to the user
*/
this.rejectPresenceSubscription = function (jid, message) { this.rejectPresenceSubscription = function (jid, message) {
/* Reject or cancel another user's subscription to our presence updates.
*
* Parameters:
* (String) jid - The Jabber ID of the user whose subscription
* is being canceled.
* (String) message - An optional message to the user
*/
const pres = Object(strophe_js__WEBPACK_IMPORTED_MODULE_0__["$pres"])({ const pres = Object(strophe_js__WEBPACK_IMPORTED_MODULE_0__["$pres"])({
to: jid, to: jid,
type: "unsubscribed" type: "unsubscribed"
...@@ -43038,9 +43044,9 @@ _converse.api = { ...@@ -43038,9 +43044,9 @@ _converse.api = {
* *
* @method _converse.api.trigger * @method _converse.api.trigger
*/ */
'trigger'() { 'trigger'(name) {
/* Event emitter and promise resolver */ /* Event emitter and promise resolver */
_converse.trigger.apply(this, arguments); _converse.trigger.apply(_converse, arguments);
const promise = _converse.promises[name]; const promise = _converse.promises[name];
...@@ -43622,6 +43628,12 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-dis ...@@ -43622,6 +43628,12 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-dis
const _converse = this._converse; // Promises exposed by this plugin const _converse = this._converse; // Promises exposed by this plugin
_converse.api.promises.add('discoInitialized'); _converse.api.promises.add('discoInitialized');
/**
* @class
* @namespace _converse.DiscoEntity
* @memberOf _converse
*/
_converse.DiscoEntity = Backbone.Model.extend({ _converse.DiscoEntity = Backbone.Model.extend({
/* A Disco Entity is a JID addressable entity that can be queried /* A Disco Entity is a JID addressable entity that can be queried
...@@ -43649,14 +43661,15 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-dis ...@@ -43649,14 +43661,15 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-dis
this.items.fetch(); this.items.fetch();
}, },
async getIdentity(category, type) { /**
/* Returns a Promise which resolves with a map indicating * Returns a Promise which resolves with a map indicating
* whether a given identity is provided by this entity. * whether a given identity is provided by this entity.
* * @private
* Parameters: * @method _converse.DiscoEntity#getIdentity
* (String) category - The identity category * @param { String } category - The identity category
* (String) type - The identity type * @param { String } type - The identity type
*/ */
async getIdentity(category, type) {
await this.waitUntilFeaturesDiscovered; await this.waitUntilFeaturesDiscovered;
return this.identities.findWhere({ return this.identities.findWhere({
'category': category, 'category': category,
...@@ -43664,13 +43677,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-dis ...@@ -43664,13 +43677,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-dis
}); });
}, },
async hasFeature(feature) { /**
/* Returns a Promise which resolves with a map indicating * Returns a Promise which resolves with a map indicating
* whether a given feature is supported. * whether a given feature is supported.
* * @private
* Parameters: * @method _converse.DiscoEntity#hasFeature
* (String) feature - The feature that might be supported. * @param { String } feature - The feature that might be supported.
*/ */
async hasFeature(feature) {
await this.waitUntilFeaturesDiscovered; await this.waitUntilFeaturesDiscovered;
if (this.features.findWhere({ if (this.features.findWhere({
...@@ -45176,14 +45190,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -45176,14 +45190,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return this.get('name') || this.get('jid'); return this.get('name') || this.get('jid');
}, },
join(nick, password) { /**
/* Join the groupchat. * Join the groupchat.
* * @private
* Parameters: * @method _converse.ChatRoom#join
* (String) nick: The user's nickname * @param { String } nick - The user's nickname
* (String) password: Optional password, if required by * @param { String } password - Optional password, if required by the groupchat.
* the groupchat.
*/ */
join(nick, password) {
nick = nick ? nick : this.get('nick'); nick = nick ? nick : this.get('nick');
if (!nick) { if (!nick) {
...@@ -45216,13 +45230,12 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -45216,13 +45230,12 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return this; return this;
}, },
leave(exit_msg) {
/* Leave the groupchat. /* Leave the groupchat.
* * @private
* Parameters: * @method _converse.ChatRoom#leave
* (String) exit_msg: Optional message to indicate your * @param { string } exit_msg - Optional message to indicate your reason for leaving
* reason for leaving.
*/ */
leave(exit_msg) {
this.features.destroy(); this.features.destroy();
this.occupants.browserStorage._clear(); this.occupants.browserStorage._clear();
...@@ -45413,12 +45426,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -45413,12 +45426,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
})); }));
}, },
directInvite(recipient, reason) { /**
/* Send a direct invitation as per XEP-0249 * Send a direct invitation as per XEP-0249
* Parameters: * @private
* (String) recipient - JID of the person being invited * @method _converse.ChatRoom#directInvite
* (String) reason - Optional reason for the invitation * @param { String } recipient - JID of the person being invited
* @param { String } reason - Optional reason for the invitation
*/ */
directInvite(recipient, reason) {
if (this.features.get('membersonly')) { if (this.features.get('membersonly')) {
// When inviting to a members-only groupchat, we first add // When inviting to a members-only groupchat, we first add
// the person to the member list by giving them an // the person to the member list by giving them an
...@@ -45519,20 +45534,17 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -45519,20 +45534,17 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
this.features.save(attrs); this.features.save(attrs);
}, },
requestMemberList(affiliation) {
/* Send an IQ stanza to the server, asking it for the /* Send an IQ stanza to the server, asking it for the
* member-list of this groupchat. * member-list of this groupchat.
*
* See: https://xmpp.org/extensions/xep-0045.html#modifymember * See: https://xmpp.org/extensions/xep-0045.html#modifymember
* * @private
* Parameters: * @method _converse.ChatRoom#requestMemberList
* (String) affiliation: The specific member list to * @param { string } affiliation - The specific member list to
* fetch. 'admin', 'owner' or 'member'. * fetch. 'admin', 'owner' or 'member'.
* * @returns:
* Returns: * A promise which resolves once the list has been retrieved.
* A promise which resolves once the list has been
* retrieved.
*/ */
requestMemberList(affiliation) {
affiliation = affiliation || 'member'; affiliation = affiliation || 'member';
const iq = $iq({ const iq = $iq({
to: this.get('jid'), to: this.get('jid'),
...@@ -45545,28 +45557,26 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -45545,28 +45557,26 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return _converse.api.sendIQ(iq); return _converse.api.sendIQ(iq);
}, },
setAffiliation(affiliation, members) { /**
/* Send IQ stanzas to the server to set an affiliation for * Send IQ stanzas to the server to set an affiliation for
* the provided JIDs. * the provided JIDs.
*
* See: https://xmpp.org/extensions/xep-0045.html#modifymember * See: https://xmpp.org/extensions/xep-0045.html#modifymember
* *
* XXX: Prosody doesn't accept multiple JIDs' affiliations * Prosody doesn't accept multiple JIDs' affiliations
* being set in one IQ stanza, so as a workaround we send * being set in one IQ stanza, so as a workaround we send
* a separate stanza for each JID. * a separate stanza for each JID.
* Related ticket: https://issues.prosody.im/345 * Related ticket: https://issues.prosody.im/345
* *
* Parameters: * @private
* (String) affiliation: The affiliation * @method _converse.ChatRoom#setAffiliation
* (Object) members: A map of jids, affiliations and * @param { string } affiliation - The affiliation
* @param { object } members - A map of jids, affiliations and
* optionally reasons. Only those entries with the * optionally reasons. Only those entries with the
* same affiliation as being currently set will be * same affiliation as being currently set will be considered.
* considered. * @returns
* * A promise which resolves and fails depending on the XMPP server response.
* Returns:
* A promise which resolves and fails depending on the
* XMPP server response.
*/ */
setAffiliation(affiliation, members) {
members = _.filter(members, member => // We only want those members who have the right members = _.filter(members, member => // We only want those members who have the right
// affiliation (or none, which implies the provided one). // affiliation (or none, which implies the provided one).
_.isUndefined(member.affiliation) || member.affiliation === affiliation); _.isUndefined(member.affiliation) || member.affiliation === affiliation);
...@@ -45576,18 +45586,19 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -45576,18 +45586,19 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return Promise.all(promises); return Promise.all(promises);
}, },
saveConfiguration(form) { /**
/* Submit the groupchat configuration form by sending an IQ * Submit the groupchat configuration form by sending an IQ
* stanza to the server. * stanza to the server.
* * @private
* Returns a promise which resolves once the XMPP server * @method _converse.ChatRoom#saveConfiguration
* has return a response IQ. * @param { HTMLElement } form - The configuration form DOM element.
*
* Parameters:
* (HTMLElement) form: The configuration form DOM element.
* If no form is provided, the default configuration * If no form is provided, the default configuration
* values will be used. * values will be used.
* @returns { promise }
* Returns a promise which resolves once the XMPP server
* has return a response IQ.
*/ */
saveConfiguration(form) {
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
const inputs = form ? sizzle(':input:not([type=button]):not([type=submit])', form) : [], const inputs = form ? sizzle(':input:not([type=button]):not([type=submit])', form) : [],
configArray = _.map(inputs, _utils_form__WEBPACK_IMPORTED_MODULE_4__["default"].webForm2xForm); configArray = _.map(inputs, _utils_form__WEBPACK_IMPORTED_MODULE_4__["default"].webForm2xForm);
...@@ -45656,20 +45667,21 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -45656,20 +45667,21 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
})); }));
}, },
sendConfiguration(config, callback, errback) { /**
/* Send an IQ stanza with the groupchat configuration. * Send an IQ stanza with the groupchat configuration.
* * @private
* Parameters: * @method _converse.ChatRoom#sendConfiguration
* (Array) config: The groupchat configuration * @param { Array } config - The groupchat configuration
* (Function) callback: Callback upon succesful IQ response * @param { Function } callback - Callback upon succesful IQ response
* The first parameter passed in is IQ containing the * The first parameter passed in is IQ containing the
* groupchat configuration. * groupchat configuration.
* The second is the response IQ from the server. * The second is the response IQ from the server.
* (Function) errback: Callback upon error IQ response * @param { Function } errback - Callback upon error IQ response
* The first parameter passed in is IQ containing the * The first parameter passed in is IQ containing the
* groupchat configuration. * groupchat configuration.
* The second is the response IQ from the server. * The second is the response IQ from the server.
*/ */
sendConfiguration(config, callback, errback) {
const iq = $iq({ const iq = $iq({
to: this.get('jid'), to: this.get('jid'),
type: "set" type: "set"
...@@ -45689,13 +45701,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -45689,13 +45701,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return _converse.api.sendIQ(iq).then(callback).catch(errback); return _converse.api.sendIQ(iq).then(callback).catch(errback);
}, },
saveAffiliationAndRole(pres) { /**
/* Parse the presence stanza for the current user's * Parse the presence stanza for the current user's affiliation.
* affiliation. * @private
* * @method _converse.ChatRoom#saveAffiliationAndRole
* Parameters: * @param { XMLElement } pres - A <presence> stanza.
* (XMLElement) pres: A <presence> stanza.
*/ */
saveAffiliationAndRole(pres) {
const item = sizzle(`x[xmlns="${Strophe.NS.MUC_USER}"] item`, pres).pop(); const item = sizzle(`x[xmlns="${Strophe.NS.MUC_USER}"] item`, pres).pop();
const is_self = pres.querySelector("status[code='110']"); const is_self = pres.querySelector("status[code='110']");
...@@ -45717,15 +45729,16 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -45717,15 +45729,16 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
} }
}, },
sendAffiliationIQ(affiliation, member) { /**
/* Send an IQ stanza specifying an affiliation change. * Send an IQ stanza specifying an affiliation change.
* * @private
* Paremeters: * @method _converse.ChatRoom#
* (String) affiliation: affiliation (could also be stored * @param { String } affiliation: affiliation
* on the member object). * (could also be stored on the member object).
* (Object) member: Map containing the member's jid and * @param { Object } member: Map containing the member's jid and
* optionally a reason and affiliation. * optionally a reason and affiliation.
*/ */
sendAffiliationIQ(affiliation, member) {
const iq = $iq({ const iq = $iq({
to: this.get('jid'), to: this.get('jid'),
type: "set" type: "set"
...@@ -45744,17 +45757,17 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -45744,17 +45757,17 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return _converse.api.sendIQ(iq); return _converse.api.sendIQ(iq);
}, },
setAffiliations(members) { /**
/* Send IQ stanzas to the server to modify the * Send IQ stanzas to the server to modify the
* affiliations in this groupchat. * affiliations in this groupchat.
*
* See: https://xmpp.org/extensions/xep-0045.html#modifymember * See: https://xmpp.org/extensions/xep-0045.html#modifymember
* * @private
* Parameters: * @method _converse.ChatRoom#setAffiliations
* (Object) members: A map of jids, affiliations and optionally reasons * @param { object } members - A map of jids, affiliations and optionally reasons
* (Function) onSuccess: callback for a succesful response * @param { function } onSuccess - callback for a succesful response
* (Function) onError: callback for an error response * @param { function } onError - callback for an error response
*/ */
setAffiliations(members) {
const affiliations = _.uniq(_.map(members, 'affiliation')); const affiliations = _.uniq(_.map(members, 'affiliation'));
return Promise.all(_.map(affiliations, _.partial(this.setAffiliation.bind(this), _, members))); return Promise.all(_.map(affiliations, _.partial(this.setAffiliation.bind(this), _, members)));
...@@ -45774,36 +45787,36 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -45774,36 +45787,36 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return [].concat.apply([], result).filter(p => p); return [].concat.apply([], result).filter(p => p);
}, },
updateMemberLists(members, affiliations, deltaFunc) { /**
/* Fetch the lists of users with the given affiliations. * Fetch the lists of users with the given affiliations.
* Then compute the delta between those users and * Then compute the delta between those users and
* the passed in members, and if it exists, send the delta * the passed in members, and if it exists, send the delta
* to the XMPP server to update the member list. * to the XMPP server to update the member list.
* * @private
* Parameters: * @method _converse.ChatRoom#updateMemberLists
* (Object) members: Map of member jids and affiliations. * @param { object } members - Map of member jids and affiliations.
* (String|Array) affiliation: An array of affiliations or * @param { string|array } affiliation - An array of affiliations or
* a string if only one affiliation. * a string if only one affiliation.
* (Function) deltaFunc: The function to compute the delta * @param { function } deltaFunc - The function to compute the delta
* between old and new member lists. * between old and new member lists.
* * @returns { promise }
* Returns:
* A promise which is resolved once the list has been * A promise which is resolved once the list has been
* updated or once it's been established there's no need * updated or once it's been established there's no need
* to update the list. * to update the list.
*/ */
updateMemberLists(members, affiliations, deltaFunc) {
this.getJidsWithAffiliations(affiliations).then(old_members => this.setAffiliations(deltaFunc(members, old_members))).then(() => this.occupants.fetchMembers()).catch(_.partial(_converse.log, _, Strophe.LogLevel.ERROR)); this.getJidsWithAffiliations(affiliations).then(old_members => this.setAffiliations(deltaFunc(members, old_members))).then(() => this.occupants.fetchMembers()).catch(_.partial(_converse.log, _, Strophe.LogLevel.ERROR));
}, },
async checkForReservedNick() { /**
/* Use service-discovery to ask the XMPP server whether * Use service-discovery to ask the XMPP server whether
* this user has a reserved nickname for this groupchat. * this user has a reserved nickname for this groupchat.
* If so, we'll use that, otherwise we render the nickname form. * If so, we'll use that, otherwise we render the nickname form.
* * @private
* Parameters: * @method _converse.ChatRoom#checkForReservedNick
* (Function) callback: Callback upon succesful IQ response * @returns { promise } A promise which resolves with the response IQ
* (Function) errback: Callback upon error IQ response
*/ */
async checkForReservedNick() {
const iq = await _converse.api.sendIQ($iq({ const iq = await _converse.api.sendIQ($iq({
'to': this.get('jid'), 'to': this.get('jid'),
'from': _converse.connection.jid, 'from': _converse.connection.jid,
...@@ -45885,13 +45898,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -45885,13 +45898,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
} }
}, },
updateOccupantsOnPresence(pres) { /**
/* Given a presence stanza, update the occupant model * Given a presence stanza, update the occupant model
* based on its contents. * based on its contents.
* * @private
* Parameters: * @method _converse.ChatRoom#updateOccupantsOnPresence
* (XMLElement) pres: The presence stanza * @param { XMLElement } pres - The presence stanza
*/ */
updateOccupantsOnPresence(pres) {
const data = this.parsePresence(pres); const data = this.parsePresence(pres);
if (data.type === 'error' || !data.jid && !data.nick) { if (data.type === 'error' || !data.jid && !data.nick) {
...@@ -45998,12 +46012,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -45998,12 +46012,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
acknowledged[xmlns="${Strophe.NS.MARKERS}"]`, stanza).length > 0; acknowledged[xmlns="${Strophe.NS.MARKERS}"]`, stanza).length > 0;
}, },
subjectChangeHandled(attrs) { /**
/* Handle a subject change and return `true` if so. * Handle a subject change and return `true` if so.
* * @private
* Parameters: * @method _converse.ChatRoom#subjectChangeHandled
* (Object) attrs: The message attributes * @param { object } attrs - The message attributes
*/ */
subjectChangeHandled(attrs) {
if (attrs.subject && !attrs.thread && !attrs.message) { if (attrs.subject && !attrs.thread && !attrs.message) {
// https://xmpp.org/extensions/xep-0045.html#subject-mod // https://xmpp.org/extensions/xep-0045.html#subject-mod
// ----------------------------------------------------- // -----------------------------------------------------
...@@ -46022,13 +46037,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -46022,13 +46037,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return false; return false;
}, },
ignorableCSN(attrs) { /**
/* Is this a chat state notification that can be ignored, * Is this a chat state notification that can be ignored,
* because it's old or because it's from us. * because it's old or because it's from us.
* * @private
* Parameters: * @method _converse.ChatRoom#ignorableCSN
* (Object) attrs: The message attributes * @param { Object } attrs - The message attributes
*/ */
ignorableCSN(attrs) {
const is_csn = _utils_form__WEBPACK_IMPORTED_MODULE_4__["default"].isOnlyChatStateNotification(attrs), const is_csn = _utils_form__WEBPACK_IMPORTED_MODULE_4__["default"].isOnlyChatStateNotification(attrs),
own_message = Strophe.getResourceFromJid(attrs.from) == this.get('nick'); own_message = Strophe.getResourceFromJid(attrs.from) == this.get('nick');
return is_csn && (attrs.is_delayed || own_message); return is_csn && (attrs.is_delayed || own_message);
...@@ -46058,12 +46074,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -46058,12 +46074,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
return attrs; return attrs;
}, },
async onMessage(stanza) { /**
/* Handler for all MUC messages sent to this groupchat. * Handler for all MUC messages sent to this groupchat.
* * @private
* Parameters: * @method _converse.ChatRoom#onMessage
* (XMLElement) stanza: The message stanza. * @param { XMLElement } stanza - The message stanza.
*/ */
async onMessage(stanza) {
this.fetchFeaturesIfConfigurationChanged(stanza); this.fetchFeaturesIfConfigurationChanged(stanza);
const original_stanza = stanza, const original_stanza = stanza,
forwarded = sizzle(`forwarded[xmlns="${Strophe.NS.FORWARD}"]`, stanza).pop(); forwarded = sizzle(`forwarded[xmlns="${Strophe.NS.FORWARD}"]`, stanza).pop();
...@@ -46103,12 +46120,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -46103,12 +46120,13 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
}); });
}, },
onPresence(pres) { /**
/* Handles all MUC presence stanzas. * Handles all MUC presence stanzas.
* * @private
* Parameters: * @method _converse.ChatRoom#onPresence
* (XMLElement) pres: The stanza * @param { XMLElement } pres - The stanza
*/ */
onPresence(pres) {
if (pres.getAttribute('type') === 'error') { if (pres.getAttribute('type') === 'error') {
this.save('connection_status', _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].ROOMSTATUS.DISCONNECTED); this.save('connection_status', _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].ROOMSTATUS.DISCONNECTED);
return; return;
...@@ -46127,9 +46145,8 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -46127,9 +46145,8 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
} }
}, },
onOwnPresence(pres) { /**
/* Handles a received presence relating to the current * Handles a received presence relating to the current user.
* user.
* *
* For locked groupchats (which are by definition "new"), the * For locked groupchats (which are by definition "new"), the
* groupchat will either be auto-configured or created instantly * groupchat will either be auto-configured or created instantly
...@@ -46139,10 +46156,11 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -46139,10 +46156,11 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
* If the groupchat is not locked, then the groupchat will be * If the groupchat is not locked, then the groupchat will be
* auto-configured only if applicable and if the current * auto-configured only if applicable and if the current
* user is the groupchat's owner. * user is the groupchat's owner.
* * @private
* Parameters: * @method _converse.ChatRoom#onOwnPresence
* (XMLElement) pres: The stanza * @param { XMLElement } pres - The stanza
*/ */
onOwnPresence(pres) {
this.saveAffiliationAndRole(pres); this.saveAffiliationAndRole(pres);
const locked_room = pres.querySelector("status[code='201']"); const locked_room = pres.querySelector("status[code='201']");
...@@ -46180,13 +46198,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -46180,13 +46198,14 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
this.save('connection_status', _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].ROOMSTATUS.ENTERED); this.save('connection_status', _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].ROOMSTATUS.ENTERED);
}, },
isUserMentioned(message) { /**
/* Returns a boolean to indicate whether the current user * Returns a boolean to indicate whether the current user
* was mentioned in a message. * was mentioned in a message.
* * @private
* Parameters: * @method _converse.ChatRoom#isUserMentioned
* (String): The text message * @param { String } - The text message
*/ */
isUserMentioned(message) {
const nick = this.get('nick'); const nick = this.get('nick');
if (message.get('references').length) { if (message.get('references').length) {
...@@ -46197,13 +46216,12 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -46197,13 +46216,12 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
} }
}, },
incrementUnreadMsgCounter(message) { /* Given a newly received message, update the unread counter if necessary.
/* Given a newly received message, update the unread counter if * @private
* necessary. * @method _converse.ChatRoom#incrementUnreadMsgCounter
* * @param { XMLElement } - The <messsage> stanza
* Parameters:
* (XMLElement): The <messsage> stanza
*/ */
incrementUnreadMsgCounter(message) {
if (!message) { if (!message) {
return; return;
} }
...@@ -46369,15 +46387,15 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc ...@@ -46369,15 +46387,15 @@ _converse_core__WEBPACK_IMPORTED_MODULE_3__["default"].plugins.add('converse-muc
}; };
} }
}); });
/**
_converse.onDirectMUCInvitation = function (message) { * A direct MUC invitation to join a groupchat has been received
/* A direct MUC invitation to join a groupchat has been received
* See XEP-0249: Direct MUC invitations. * See XEP-0249: Direct MUC invitations.
* * @private
* Parameters: * @method _converse.ChatRoom#onDirectMUCInvitation
* (XMLElement) message: The message stanza containing the * @param { XMLElement } message - The message stanza containing the invitation.
* invitation.
*/ */
_converse.onDirectMUCInvitation = function (message) {
const x_el = sizzle('x[xmlns="jabber:x:conference"]', message).pop(), const x_el = sizzle('x[xmlns="jabber:x:conference"]', message).pop(),
from = Strophe.getBareJidFromJid(message.getAttribute('from')), from = Strophe.getBareJidFromJid(message.getAttribute('from')),
room_jid = x_el.getAttribute('jid'), room_jid = x_el.getAttribute('jid'),
...@@ -47032,11 +47050,15 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -47032,11 +47050,15 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
return true; return true;
}, null, 'presence', null); }, null, 'presence', null);
}; };
/**
_converse.initRoster = function () { * Initialize the Bakcbone collections that represent the contats
/* Initialize the Bakcbone collections that represent the contats
* roster and the roster groups. * roster and the roster groups.
* @private
* @method _converse.initRoster
*/ */
_converse.initRoster = function () {
const storage = _converse.config.get('storage'); const storage = _converse.config.get('storage');
_converse.roster = new _converse.RosterContacts(); _converse.roster = new _converse.RosterContacts();
...@@ -47061,18 +47083,20 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -47061,18 +47083,20 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
_converse.api.trigger('rosterInitialized'); _converse.api.trigger('rosterInitialized');
}; };
/**
_converse.populateRoster = async function () { * Fetch all the roster groups, and then the roster contacts.
let ignore_cache = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : false;
/* Fetch all the roster groups, and then the roster contacts.
* Emit an event after fetching is done in each case. * Emit an event after fetching is done in each case.
* * @private
* Parameters: * @method _converse.populateRoster
* (Bool) ignore_cache - If set to to true, the local cache * @param { Bool } ignore_cache - If set to to true, the local cache
* will be ignored it's guaranteed that the XMPP server * will be ignored it's guaranteed that the XMPP server
* will be queried for the roster. * will be queried for the roster.
*/ */
_converse.populateRoster = async function () {
let ignore_cache = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : false;
if (ignore_cache) { if (ignore_cache) {
_converse.send_initial_presence = true; _converse.send_initial_presence = true;
...@@ -47100,6 +47124,7 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -47100,6 +47124,7 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
* position roster groups. * position roster groups.
* @event _converse#rosterGroupsFetched * @event _converse#rosterGroupsFetched
* @example _converse.api.listen.on('rosterGroupsFetched', () => { ... }); * @example _converse.api.listen.on('rosterGroupsFetched', () => { ... });
* @example _converse.api.waitUntil('rosterGroupsFetched').then(() => { ... });
*/ */
_converse.api.trigger('rosterGroupsFetched'); _converse.api.trigger('rosterGroupsFetched');
...@@ -47275,13 +47300,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -47275,13 +47300,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
return this.vcard.get('fullname'); return this.vcard.get('fullname');
}, },
subscribe(message) { /**
/* Send a presence subscription request to this roster contact * Send a presence subscription request to this roster contact
* * @private
* Parameters: * @method _converse.RosterContacts#subscribe
* (String) message - An optional message to explain the * @param { String } message - An optional message to explain the
* reason for the subscription request. * reason for the subscription request.
*/ */
subscribe(message) {
const pres = $pres({ const pres = $pres({
to: this.get('jid'), to: this.get('jid'),
type: "subscribe" type: "subscribe"
...@@ -47306,27 +47332,32 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -47306,27 +47332,32 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
return this; return this;
}, },
ackSubscribe() { /**
/* Upon receiving the presence stanza of type "subscribed", * Upon receiving the presence stanza of type "subscribed",
* the user SHOULD acknowledge receipt of that subscription * the user SHOULD acknowledge receipt of that subscription
* state notification by sending a presence stanza of type * state notification by sending a presence stanza of type
* "subscribe" to the contact * "subscribe" to the contact
* @private
* @method _converse.RosterContacts#ackSubscribe
*/ */
ackSubscribe() {
_converse.api.send($pres({ _converse.api.send($pres({
'type': 'subscribe', 'type': 'subscribe',
'to': this.get('jid') 'to': this.get('jid')
})); }));
}, },
ackUnsubscribe() { /**
/* Upon receiving the presence stanza of type "unsubscribed", * Upon receiving the presence stanza of type "unsubscribed",
* the user SHOULD acknowledge receipt of that subscription state * the user SHOULD acknowledge receipt of that subscription state
* notification by sending a presence stanza of type "unsubscribe" * notification by sending a presence stanza of type "unsubscribe"
* this step lets the user's server know that it MUST no longer * this step lets the user's server know that it MUST no longer
* send notification of the subscription state change to the user. * send notification of the subscription state change to the user.
* Parameters: * @private
* (String) jid - The Jabber ID of the user who is unsubscribing * @method _converse.RosterContacts#ackUnsubscribe
* @param { String } jid - The Jabber ID of the user who is unsubscribing
*/ */
ackUnsubscribe() {
_converse.api.send($pres({ _converse.api.send($pres({
'type': 'unsubscribe', 'type': 'unsubscribe',
'to': this.get('jid') 'to': this.get('jid')
...@@ -47336,21 +47367,25 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -47336,21 +47367,25 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
this.destroy(); this.destroy();
}, },
unauthorize(message) { /**
/* Unauthorize this contact's presence subscription * Unauthorize this contact's presence subscription
* Parameters: * @private
* (String) message - Optional message to send to the person being unauthorized * @method _converse.RosterContacts#unauthorize
* @param { String } message - Optional message to send to the person being unauthorized
*/ */
unauthorize(message) {
_converse.rejectPresenceSubscription(this.get('jid'), message); _converse.rejectPresenceSubscription(this.get('jid'), message);
return this; return this;
}, },
authorize(message) { /**
/* Authorize presence subscription * Authorize presence subscription
* Parameters: * @private
* (String) message - Optional message to send to the person being authorized * @method _converse.RosterContacts#authorize
* @param { String } message - Optional message to send to the person being authorized
*/ */
authorize(message) {
const pres = $pres({ const pres = $pres({
'to': this.get('jid'), 'to': this.get('jid'),
'type': "subscribed" 'type': "subscribed"
...@@ -47365,11 +47400,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -47365,11 +47400,13 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
return this; return this;
}, },
removeFromRoster() { /**
/* Instruct the XMPP server to remove this contact from our roster * Instruct the XMPP server to remove this contact from our roster
* Parameters: * @private
* (Function) callback * @method _converse.RosterContacts#
* @returns { Promise }
*/ */
removeFromRoster() {
const iq = $iq({ const iq = $iq({
type: 'set' type: 'set'
}).c('query', { }).c('query', {
...@@ -47382,6 +47419,12 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -47382,6 +47419,12 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
} }
}); });
/**
* @class
* @namespace _converse.RosterContacts
* @memberOf _converse
*/
_converse.RosterContacts = Backbone.Collection.extend({ _converse.RosterContacts = Backbone.Collection.extend({
model: _converse.RosterContact, model: _converse.RosterContact,
...@@ -47488,17 +47531,18 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -47488,17 +47531,18 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
return u.isSameBareJID(jid, _converse.connection.jid); return u.isSameBareJID(jid, _converse.connection.jid);
}, },
addAndSubscribe(jid, name, groups, message, attributes) { /**
/* Add a roster contact and then once we have confirmation from * Add a roster contact and then once we have confirmation from
* the XMPP server we subscribe to that contact's presence updates. * the XMPP server we subscribe to that contact's presence updates.
* Parameters: * @private
* (String) jid - The Jabber ID of the user being added and subscribed to. * @method _converse.RosterContacts#addAndSubscribe
* (String) name - The name of that user * @param { String } jid - The Jabber ID of the user being added and subscribed to.
* (Array of Strings) groups - Any roster groups the user might belong to * @param { String } name - The name of that user
* (String) message - An optional message to explain the * @param { Array.String } groups - Any roster groups the user might belong to
* reason for the subscription request. * @param { String } message - An optional message to explain the reason for the subscription request.
* (Object) attributes - Any additional attributes to be stored on the user's model. * @param { Object } attributes - Any additional attributes to be stored on the user's model.
*/ */
addAndSubscribe(jid, name, groups, message, attributes) {
const handler = contact => { const handler = contact => {
if (contact instanceof _converse.RosterContact) { if (contact instanceof _converse.RosterContact) {
contact.subscribe(message); contact.subscribe(message);
...@@ -47508,16 +47552,17 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -47508,16 +47552,17 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
this.addContactToRoster(jid, name, groups, attributes).then(handler, handler); this.addContactToRoster(jid, name, groups, attributes).then(handler, handler);
}, },
sendContactAddIQ(jid, name, groups) { /**
/* Send an IQ stanza to the XMPP server to add a new roster contact. * Send an IQ stanza to the XMPP server to add a new roster contact.
* * @private
* Parameters: * @method _converse.RosterContacts#sendContactAddIQ
* (String) jid - The Jabber ID of the user being added * @param { String } jid - The Jabber ID of the user being added
* (String) name - The name of that user * @param { String } name - The name of that user
* (Array of Strings) groups - Any roster groups the user might belong to * @param { Array.String } groups - Any roster groups the user might belong to
* (Function) callback - A function to call once the IQ is returned * @param { Function } callback - A function to call once the IQ is returned
* (Function) errback - A function to call if an error occurred * @param { Function } errback - A function to call if an error occurred
*/ */
sendContactAddIQ(jid, name, groups) {
name = _.isEmpty(name) ? null : name; name = _.isEmpty(name) ? null : name;
const iq = $iq({ const iq = $iq({
'type': 'set' 'type': 'set'
...@@ -47533,18 +47578,18 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -47533,18 +47578,18 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
return _converse.api.sendIQ(iq); return _converse.api.sendIQ(iq);
}, },
async addContactToRoster(jid, name, groups, attributes) { /**
/* Adds a RosterContact instance to _converse.roster and * Adds a RosterContact instance to _converse.roster and
* registers the contact on the XMPP server. * registers the contact on the XMPP server.
* Returns a promise which is resolved once the XMPP server has * Returns a promise which is resolved once the XMPP server has responded.
* responded. * @private
* * @method _converse.RosterContacts#addContactToRoster
* Parameters: * @param { String } jid - The Jabber ID of the user being added and subscribed to.
* (String) jid - The Jabber ID of the user being added and subscribed to. * @param { String } name - The name of that user
* (String) name - The name of that user * @param { Array.String } groups - Any roster groups the user might belong to
* (Array of Strings) groups - Any roster groups the user might belong to * @param { Object } attributes - Any additional attributes to be stored on the user's model.
* (Object) attributes - Any additional attributes to be stored on the user's model.
*/ */
async addContactToRoster(jid, name, groups, attributes) {
groups = groups || []; groups = groups || [];
try { try {
...@@ -47599,13 +47644,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins ...@@ -47599,13 +47644,14 @@ _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins
return _.sum(this.models.filter(model => !_.includes(ignored, model.presence.get('show')))); return _.sum(this.models.filter(model => !_.includes(ignored, model.presence.get('show'))));
}, },
onRosterPush(iq) { /**
/* Handle roster updates from the XMPP server. * Handle roster updates from the XMPP server.
* See: https://xmpp.org/rfcs/rfc6121.html#roster-syntax-actions-push * See: https://xmpp.org/rfcs/rfc6121.html#roster-syntax-actions-push
* * @private
* Parameters: * @method _converse.RosterContacts#onRosterPush
* (XMLElement) IQ - The IQ stanza received from the XMPP server. * @param { XMLElement } IQ - The IQ stanza received from the XMPP server.
*/ */
onRosterPush(iq) {
const id = iq.getAttribute('id'); const id = iq.getAttribute('id');
const from = iq.getAttribute('from'); const from = iq.getAttribute('from');
...@@ -48271,12 +48317,6 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-vca ...@@ -48271,12 +48317,6 @@ _converse_core__WEBPACK_IMPORTED_MODULE_0__["default"].plugins.add('converse-vca
} }
async function getVCard(_converse, jid) { async function getVCard(_converse, jid) {
/* Request the VCard of another user. Returns a promise.
*
* Parameters:
* (String) jid - The Jabber ID of the user whose VCard
* is being requested.
*/
const to = Strophe.getBareJidFromJid(jid) === _converse.bare_jid ? null : jid; const to = Strophe.getBareJidFromJid(jid) === _converse.bare_jid ? null : jid;
let iq; let iq;
...@@ -48568,9 +48608,7 @@ __webpack_require__.r(__webpack_exports__); ...@@ -48568,9 +48608,7 @@ __webpack_require__.r(__webpack_exports__);
function detectLocale(library_check) { function detectLocale(library_check) {
/* Determine which locale is supported by the user's system as well /* Determine which locale is supported by the user's system as well
* as by the relevant library (e.g. converse.js or moment.js). * as by the relevant library (e.g. converse.js or moment.js).
* * @param { Function } library_check - Returns a boolean indicating whether
* Parameters:
* (Function) library_check - Returns a boolean indicating whether
* the locale is supported. * the locale is supported.
*/ */
var locale, i; var locale, i;
...@@ -48617,14 +48655,13 @@ function getLocale(preferred_locale, isSupportedByLibrary) { ...@@ -48617,14 +48655,13 @@ function getLocale(preferred_locale, isSupportedByLibrary) {
return detectLocale(isSupportedByLibrary) || 'en'; return detectLocale(isSupportedByLibrary) || 'en';
} }
/* Check whether the locale or sub locale (e.g. en-US, en) is supported.
* @param { String } locale - The locale to check for
* @param { Function } available - Returns a boolean indicating whether the locale is supported
*/
function isLocaleAvailable(locale, available) { function isLocaleAvailable(locale, available) {
/* Check whether the locale or sub locale (e.g. en-US, en) is supported.
*
* Parameters:
* (String) locale - The locale to check for
* (Function) available - returns a boolean indicating whether the locale is supported
*/
if (available(locale)) { if (available(locale)) {
return locale; return locale;
} else { } else {
...@@ -48637,6 +48674,10 @@ function isLocaleAvailable(locale, available) { ...@@ -48637,6 +48674,10 @@ function isLocaleAvailable(locale, available) {
} }
let jed_instance; let jed_instance;
/**
* @namespace i18n
*/
/* harmony default export */ __webpack_exports__["default"] = ({ /* harmony default export */ __webpack_exports__["default"] = ({
setLocales(preferred_locale, _converse) { setLocales(preferred_locale, _converse) {
_converse.locale = getLocale(preferred_locale, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_29___default.a.partial(isConverseLocale, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_29___default.a, _converse.locales)); _converse.locale = getLocale(preferred_locale, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_29___default.a.partial(isConverseLocale, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_29___default.a, _converse.locales));
...@@ -48648,7 +48689,7 @@ let jed_instance; ...@@ -48648,7 +48689,7 @@ let jed_instance;
return jed__WEBPACK_IMPORTED_MODULE_27___default.a.sprintf.apply(jed__WEBPACK_IMPORTED_MODULE_27___default.a, arguments); return jed__WEBPACK_IMPORTED_MODULE_27___default.a.sprintf.apply(jed__WEBPACK_IMPORTED_MODULE_27___default.a, arguments);
} }
var t = jed_instance.translate(str); const t = jed_instance.translate(str);
if (arguments.length > 1) { if (arguments.length > 1) {
return t.fetch.apply(t, [].slice.call(arguments, 1)); return t.fetch.apply(t, [].slice.call(arguments, 1));
...@@ -48657,15 +48698,15 @@ let jed_instance; ...@@ -48657,15 +48698,15 @@ let jed_instance;
} }
}, },
fetchTranslations(locale, supported_locales, locale_url) { /**
/* Fetch the translations for the given local at the given URL. * Fetch the translations for the given local at the given URL.
* * @private
* Parameters: * @method i18n#fetchTranslations
* (String) locale: The given i18n locale * @param { String } locale -The given i18n locale
* (Array) supported_locales: List of locales supported * @param { Array } supported_locales - List of locales supported
* (String) locale_url: The URL from which the translations * @param { String } locale_url - The URL from which the translations should be fetched
* should be fetched.
*/ */
fetchTranslations(locale, supported_locales, locale_url) {
return new es6_promise_dist_es6_promise_auto__WEBPACK_IMPORTED_MODULE_28___default.a((resolve, reject) => { return new es6_promise_dist_es6_promise_auto__WEBPACK_IMPORTED_MODULE_28___default.a((resolve, reject) => {
if (!isConverseLocale(locale, supported_locales) || locale === 'en') { if (!isConverseLocale(locale, supported_locales) || locale === 'en') {
return resolve(); return resolve();
...@@ -48904,6 +48945,11 @@ __webpack_require__.r(__webpack_exports__); ...@@ -48904,6 +48945,11 @@ __webpack_require__.r(__webpack_exports__);
/**
* The utils object
* @namespace u
*/
const u = {}; const u = {};
u.toStanza = function (string) { u.toStanza = function (string) {
...@@ -49035,15 +49081,17 @@ u.applyUserSettings = function applyUserSettings(context, settings, user_setting ...@@ -49035,15 +49081,17 @@ u.applyUserSettings = function applyUserSettings(context, settings, user_setting
} }
} }
}; };
/**
u.stringToNode = function (s) { * Converts an HTML string into a DOM Node.
/* Converts an HTML string into a DOM Node.
* Expects that the HTML string has only one top-level element, * Expects that the HTML string has only one top-level element,
* i.e. not multiple ones. * i.e. not multiple ones.
* * @private
* Parameters: * @method u#stringToNode
* (String) s - The HTML string * @param { String } s - The HTML string
*/ */
u.stringToNode = function (s) {
var div = document.createElement('div'); var div = document.createElement('div');
div.innerHTML = s; div.innerHTML = s;
return div.firstElementChild; return div.firstElementChild;
...@@ -49061,40 +49109,44 @@ u.getOuterWidth = function (el) { ...@@ -49061,40 +49109,44 @@ u.getOuterWidth = function (el) {
width += parseInt(style.marginLeft, 10) + parseInt(style.marginRight, 10); width += parseInt(style.marginLeft, 10) + parseInt(style.marginRight, 10);
return width; return width;
}; };
/**
u.stringToElement = function (s) { * Converts an HTML string into a DOM element.
/* Converts an HTML string into a DOM element.
* Expects that the HTML string has only one top-level element, * Expects that the HTML string has only one top-level element,
* i.e. not multiple ones. * i.e. not multiple ones.
* * @private
* Parameters: * @method u#stringToElement
* (String) s - The HTML string * @param { String } s - The HTML string
*/ */
u.stringToElement = function (s) {
var div = document.createElement('div'); var div = document.createElement('div');
div.innerHTML = s; div.innerHTML = s;
return div.firstElementChild; return div.firstElementChild;
}; };
/**
* Checks whether the DOM element matches the given selector.
* @private
* @method u#matchesSelector
* @param { DOMElement } el - The DOM element
* @param { String } selector - The selector
*/
u.matchesSelector = function (el, selector) { u.matchesSelector = function (el, selector) {
/* Checks whether the DOM element matches the given selector.
*
* Parameters:
* (DOMElement) el - The DOM element
* (String) selector - The selector
*/
const match = el.matches || el.matchesSelector || el.msMatchesSelector || el.mozMatchesSelector || el.webkitMatchesSelector || el.oMatchesSelector; const match = el.matches || el.matchesSelector || el.msMatchesSelector || el.mozMatchesSelector || el.webkitMatchesSelector || el.oMatchesSelector;
return match ? match.call(el, selector) : false; return match ? match.call(el, selector) : false;
}; };
/**
* Returns a list of children of the DOM element that match the selector.
* @private
* @method u#queryChildren
* @param { DOMElement } el - the DOM element
* @param { String } selector - the selector they should be matched against
*/
u.queryChildren = function (el, selector) { u.queryChildren = function (el, selector) {
/* Returns a list of children of the DOM element that match the
* selector.
*
* Parameters:
* (DOMElement) el - the DOM element
* (String) selector - the selector they should be matched
* against.
*/
return _lodash_noconflict__WEBPACK_IMPORTED_MODULE_3___default.a.filter(el.childNodes, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_3___default.a.partial(u.matchesSelector, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_3___default.a, selector)); return _lodash_noconflict__WEBPACK_IMPORTED_MODULE_3___default.a.filter(el.childNodes, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_3___default.a.partial(u.matchesSelector, _lodash_noconflict__WEBPACK_IMPORTED_MODULE_3___default.a, selector));
}; };
...@@ -49187,20 +49239,21 @@ u.interpolate = function (string, o) { ...@@ -49187,20 +49239,21 @@ u.interpolate = function (string, o) {
return typeof r === 'string' || typeof r === 'number' ? r : a; return typeof r === 'string' || typeof r === 'number' ? r : a;
}); });
}; };
/**
u.onMultipleEvents = function () { * Call the callback once all the events have been triggered
let events = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : []; * @private
let callback = arguments.length > 1 ? arguments[1] : undefined; * @method u#onMultipleEvents
* @param { Array } events: An array of objects, with keys `object` and
/* Call the callback once all the events have been triggered
*
* Parameters:
* (Array) events: An array of objects, with keys `object` and
* `event`, representing the event name and the object it's * `event`, representing the event name and the object it's
* triggered upon. * triggered upon.
* (Function) callback: The function to call once all events have * @param { Function } callback: The function to call once all events have
* been triggered. * been triggered.
*/ */
u.onMultipleEvents = function () {
let events = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
let callback = arguments.length > 1 ? arguments[1] : undefined;
let triggered = []; let triggered = [];
function handler(result) { function handler(result) {
...@@ -70897,13 +70950,14 @@ __webpack_require__.r(__webpack_exports__); ...@@ -70897,13 +70950,14 @@ __webpack_require__.r(__webpack_exports__);
/**
* Takes an HTML DOM and turns it into an XForm field.
* @private
* @method u#webForm2xForm
* @param { DOMElement } field - the field to convert
*/
_core__WEBPACK_IMPORTED_MODULE_2__["default"].webForm2xForm = function (field) { _core__WEBPACK_IMPORTED_MODULE_2__["default"].webForm2xForm = function (field) {
/* Takes an HTML DOM and turns it into an XForm field.
*
* Parameters:
* (DOMElement) field - the field to convert
*/
let value; let value;
if (field.getAttribute('type') === 'checkbox') { if (field.getAttribute('type') === 'checkbox') {
...@@ -70953,9 +71007,8 @@ const _converse$env = _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_ ...@@ -70953,9 +71007,8 @@ const _converse$env = _converse_headless_converse_core__WEBPACK_IMPORTED_MODULE_
Strophe = _converse$env.Strophe, Strophe = _converse$env.Strophe,
sizzle = _converse$env.sizzle, sizzle = _converse$env.sizzle,
_ = _converse$env._; _ = _converse$env._;
/**
_core__WEBPACK_IMPORTED_MODULE_1__["default"].computeAffiliationsDelta = function computeAffiliationsDelta(exclude_existing, remove_absentees, new_list, old_list) { * Given two lists of objects with 'jid', 'affiliation' and
/* Given two lists of objects with 'jid', 'affiliation' and
* 'reason' properties, return a new list containing * 'reason' properties, return a new list containing
* those objects that are new, changed or removed * those objects that are new, changed or removed
* (depending on the 'remove_absentees' boolean). * (depending on the 'remove_absentees' boolean).
...@@ -70965,22 +71018,24 @@ _core__WEBPACK_IMPORTED_MODULE_1__["default"].computeAffiliationsDelta = functio ...@@ -70965,22 +71018,24 @@ _core__WEBPACK_IMPORTED_MODULE_1__["default"].computeAffiliationsDelta = functio
* *
* The 'reason' property is not taken into account when * The 'reason' property is not taken into account when
* comparing whether affiliations have been changed. * comparing whether affiliations have been changed.
* * @private
* Parameters: * @method u#computeAffiliationsDelta
* (Boolean) exclude_existing: Indicates whether JIDs from * @param { boolean } exclude_existing - Indicates whether JIDs from
* the new list which are also in the old list * the new list which are also in the old list
* (regardless of affiliation) should be excluded * (regardless of affiliation) should be excluded
* from the delta. One reason to do this * from the delta. One reason to do this
* would be when you want to add a JID only if it * would be when you want to add a JID only if it
* doesn't have *any* existing affiliation at all. * doesn't have *any* existing affiliation at all.
* (Boolean) remove_absentees: Indicates whether JIDs * @param { boolean } remove_absentees - Indicates whether JIDs
* from the old list which are not in the new list * from the old list which are not in the new list
* should be considered removed and therefore be * should be considered removed and therefore be
* included in the delta with affiliation set * included in the delta with affiliation set
* to 'none'. * to 'none'.
* (Array) new_list: Array containing the new affiliations * @param { array } new_list - Array containing the new affiliations
* (Array) old_list: Array containing the old affiliations * @param { array } old_list - Array containing the old affiliations
*/ */
_core__WEBPACK_IMPORTED_MODULE_1__["default"].computeAffiliationsDelta = function computeAffiliationsDelta(exclude_existing, remove_absentees, new_list, old_list) {
const new_jids = _.map(new_list, 'jid'); const new_jids = _.map(new_list, 'jid');
const old_jids = _.map(old_list, 'jid'); // Get the new affiliations const old_jids = _.map(old_list, 'jid'); // Get the new affiliations
...@@ -44,9 +44,7 @@ import moment from "moment"; ...@@ -44,9 +44,7 @@ import moment from "moment";
function detectLocale (library_check) { function detectLocale (library_check) {
/* Determine which locale is supported by the user's system as well /* Determine which locale is supported by the user's system as well
* as by the relevant library (e.g. converse.js or moment.js). * as by the relevant library (e.g. converse.js or moment.js).
* * @param { Function } library_check - Returns a boolean indicating whether
* Parameters:
* (Function) library_check - Returns a boolean indicating whether
* the locale is supported. * the locale is supported.
*/ */
var locale, i; var locale, i;
...@@ -87,13 +85,11 @@ function getLocale (preferred_locale, isSupportedByLibrary) { ...@@ -87,13 +85,11 @@ function getLocale (preferred_locale, isSupportedByLibrary) {
return detectLocale(isSupportedByLibrary) || 'en'; return detectLocale(isSupportedByLibrary) || 'en';
} }
function isLocaleAvailable (locale, available) { /* Check whether the locale or sub locale (e.g. en-US, en) is supported.
/* Check whether the locale or sub locale (e.g. en-US, en) is supported. * @param { String } locale - The locale to check for
* * @param { Function } available - Returns a boolean indicating whether the locale is supported
* Parameters:
* (String) locale - The locale to check for
* (Function) available - returns a boolean indicating whether the locale is supported
*/ */
function isLocaleAvailable (locale, available) {
if (available(locale)) { if (available(locale)) {
return locale; return locale;
} else { } else {
...@@ -106,6 +102,9 @@ function isLocaleAvailable (locale, available) { ...@@ -106,6 +102,9 @@ function isLocaleAvailable (locale, available) {
let jed_instance; let jed_instance;
/**
* @namespace i18n
*/
export default { export default {
setLocales (preferred_locale, _converse) { setLocales (preferred_locale, _converse) {
...@@ -120,23 +119,23 @@ export default { ...@@ -120,23 +119,23 @@ export default {
if (_.isNil(jed_instance)) { if (_.isNil(jed_instance)) {
return Jed.sprintf.apply(Jed, arguments); return Jed.sprintf.apply(Jed, arguments);
} }
var t = jed_instance.translate(str); const t = jed_instance.translate(str);
if (arguments.length>1) { if (arguments.length > 1) {
return t.fetch.apply(t, [].slice.call(arguments, 1)); return t.fetch.apply(t, [].slice.call(arguments, 1));
} else { } else {
return t.fetch(); return t.fetch();
} }
}, },
fetchTranslations (locale, supported_locales, locale_url) { /**
/* Fetch the translations for the given local at the given URL. * Fetch the translations for the given local at the given URL.
* * @private
* Parameters: * @method i18n#fetchTranslations
* (String) locale: The given i18n locale * @param { String } locale -The given i18n locale
* (Array) supported_locales: List of locales supported * @param { Array } supported_locales - List of locales supported
* (String) locale_url: The URL from which the translations * @param { String } locale_url - The URL from which the translations should be fetched
* should be fetched.
*/ */
fetchTranslations (locale, supported_locales, locale_url) {
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
if (!isConverseLocale(locale, supported_locales) || locale === 'en') { if (!isConverseLocale(locale, supported_locales) || locale === 'en') {
return resolve(); return resolve();
......
...@@ -14,6 +14,11 @@ import { Strophe } from "strophe.js"; ...@@ -14,6 +14,11 @@ import { Strophe } from "strophe.js";
import _ from "../lodash.noconflict"; import _ from "../lodash.noconflict";
import sizzle from "sizzle"; import sizzle from "sizzle";
/**
* The utils object
* @namespace u
*/
const u = {}; const u = {};
u.toStanza = function (string) { u.toStanza = function (string) {
...@@ -147,14 +152,15 @@ u.applyUserSettings = function applyUserSettings (context, settings, user_settin ...@@ -147,14 +152,15 @@ u.applyUserSettings = function applyUserSettings (context, settings, user_settin
} }
}; };
u.stringToNode = function (s) { /**
/* Converts an HTML string into a DOM Node. * Converts an HTML string into a DOM Node.
* Expects that the HTML string has only one top-level element, * Expects that the HTML string has only one top-level element,
* i.e. not multiple ones. * i.e. not multiple ones.
* * @private
* Parameters: * @method u#stringToNode
* (String) s - The HTML string * @param { String } s - The HTML string
*/ */
u.stringToNode = function (s) {
var div = document.createElement('div'); var div = document.createElement('div');
div.innerHTML = s; div.innerHTML = s;
return div.firstElementChild; return div.firstElementChild;
...@@ -170,26 +176,28 @@ u.getOuterWidth = function (el, include_margin=false) { ...@@ -170,26 +176,28 @@ u.getOuterWidth = function (el, include_margin=false) {
return width; return width;
}; };
u.stringToElement = function (s) { /**
/* Converts an HTML string into a DOM element. * Converts an HTML string into a DOM element.
* Expects that the HTML string has only one top-level element, * Expects that the HTML string has only one top-level element,
* i.e. not multiple ones. * i.e. not multiple ones.
* * @private
* Parameters: * @method u#stringToElement
* (String) s - The HTML string * @param { String } s - The HTML string
*/ */
u.stringToElement = function (s) {
var div = document.createElement('div'); var div = document.createElement('div');
div.innerHTML = s; div.innerHTML = s;
return div.firstElementChild; return div.firstElementChild;
}; };
u.matchesSelector = function (el, selector) { /**
/* Checks whether the DOM element matches the given selector. * Checks whether the DOM element matches the given selector.
* * @private
* Parameters: * @method u#matchesSelector
* (DOMElement) el - The DOM element * @param { DOMElement } el - The DOM element
* (String) selector - The selector * @param { String } selector - The selector
*/ */
u.matchesSelector = function (el, selector) {
const match = ( const match = (
el.matches || el.matches ||
el.matchesSelector || el.matchesSelector ||
...@@ -201,15 +209,14 @@ u.matchesSelector = function (el, selector) { ...@@ -201,15 +209,14 @@ u.matchesSelector = function (el, selector) {
return match ? match.call(el, selector) : false; return match ? match.call(el, selector) : false;
}; };
u.queryChildren = function (el, selector) { /**
/* Returns a list of children of the DOM element that match the * Returns a list of children of the DOM element that match the selector.
* selector. * @private
* * @method u#queryChildren
* Parameters: * @param { DOMElement } el - the DOM element
* (DOMElement) el - the DOM element * @param { String } selector - the selector they should be matched against
* (String) selector - the selector they should be matched
* against.
*/ */
u.queryChildren = function (el, selector) {
return _.filter(el.childNodes, _.partial(u.matchesSelector, _, selector)); return _.filter(el.childNodes, _.partial(u.matchesSelector, _, selector));
}; };
...@@ -296,16 +303,17 @@ u.interpolate = function (string, o) { ...@@ -296,16 +303,17 @@ u.interpolate = function (string, o) {
}); });
}; };
u.onMultipleEvents = function (events=[], callback) { /**
/* Call the callback once all the events have been triggered * Call the callback once all the events have been triggered
* * @private
* Parameters: * @method u#onMultipleEvents
* (Array) events: An array of objects, with keys `object` and * @param { Array } events: An array of objects, with keys `object` and
* `event`, representing the event name and the object it's * `event`, representing the event name and the object it's
* triggered upon. * triggered upon.
* (Function) callback: The function to call once all events have * @param { Function } callback: The function to call once all events have
* been triggered. * been triggered.
*/ */
u.onMultipleEvents = function (events=[], callback) {
let triggered = []; let triggered = [];
function handler (result) { function handler (result) {
......
...@@ -10,12 +10,13 @@ import _ from "../lodash.noconflict"; ...@@ -10,12 +10,13 @@ import _ from "../lodash.noconflict";
import tpl_field from "../templates/field.html"; import tpl_field from "../templates/field.html";
import u from "./core"; import u from "./core";
u.webForm2xForm = function (field) { /**
/* Takes an HTML DOM and turns it into an XForm field. * Takes an HTML DOM and turns it into an XForm field.
* * @private
* Parameters: * @method u#webForm2xForm
* (DOMElement) field - the field to convert * @param { DOMElement } field - the field to convert
*/ */
u.webForm2xForm = function (field) {
let value; let value;
if (field.getAttribute('type') === 'checkbox') { if (field.getAttribute('type') === 'checkbox') {
value = field.checked && 1 || 0; value = field.checked && 1 || 0;
......
...@@ -15,8 +15,8 @@ import u from "./core"; ...@@ -15,8 +15,8 @@ import u from "./core";
const { Strophe, sizzle, _ } = converse.env; const { Strophe, sizzle, _ } = converse.env;
u.computeAffiliationsDelta = function computeAffiliationsDelta (exclude_existing, remove_absentees, new_list, old_list) { /**
/* Given two lists of objects with 'jid', 'affiliation' and * Given two lists of objects with 'jid', 'affiliation' and
* 'reason' properties, return a new list containing * 'reason' properties, return a new list containing
* those objects that are new, changed or removed * those objects that are new, changed or removed
* (depending on the 'remove_absentees' boolean). * (depending on the 'remove_absentees' boolean).
...@@ -26,22 +26,23 @@ u.computeAffiliationsDelta = function computeAffiliationsDelta (exclude_existing ...@@ -26,22 +26,23 @@ u.computeAffiliationsDelta = function computeAffiliationsDelta (exclude_existing
* *
* The 'reason' property is not taken into account when * The 'reason' property is not taken into account when
* comparing whether affiliations have been changed. * comparing whether affiliations have been changed.
* * @private
* Parameters: * @method u#computeAffiliationsDelta
* (Boolean) exclude_existing: Indicates whether JIDs from * @param { boolean } exclude_existing - Indicates whether JIDs from
* the new list which are also in the old list * the new list which are also in the old list
* (regardless of affiliation) should be excluded * (regardless of affiliation) should be excluded
* from the delta. One reason to do this * from the delta. One reason to do this
* would be when you want to add a JID only if it * would be when you want to add a JID only if it
* doesn't have *any* existing affiliation at all. * doesn't have *any* existing affiliation at all.
* (Boolean) remove_absentees: Indicates whether JIDs * @param { boolean } remove_absentees - Indicates whether JIDs
* from the old list which are not in the new list * from the old list which are not in the new list
* should be considered removed and therefore be * should be considered removed and therefore be
* included in the delta with affiliation set * included in the delta with affiliation set
* to 'none'. * to 'none'.
* (Array) new_list: Array containing the new affiliations * @param { array } new_list - Array containing the new affiliations
* (Array) old_list: Array containing the old affiliations * @param { array } old_list - Array containing the old affiliations
*/ */
u.computeAffiliationsDelta = function computeAffiliationsDelta (exclude_existing, remove_absentees, new_list, old_list) {
const new_jids = _.map(new_list, 'jid'); const new_jids = _.map(new_list, 'jid');
const old_jids = _.map(old_list, 'jid'); const old_jids = _.map(old_list, 'jid');
......
...@@ -304,13 +304,14 @@ u.nextUntil = function (el, selector, include_self=false) { ...@@ -304,13 +304,14 @@ u.nextUntil = function (el, selector, include_self=false) {
return matches; return matches;
} }
u.unescapeHTML = function (string) { /**
/* Helper method that replace HTML-escaped symbols with equivalent characters * Helper method that replace HTML-escaped symbols with equivalent characters
* (e.g. transform occurrences of '&amp;' to '&') * (e.g. transform occurrences of '&amp;' to '&')
* * @private
* Parameters: * @method u#unescapeHTML
* (String) string: a String containing the HTML-escaped symbols. * @param { String } string - a String containing the HTML-escaped symbols.
*/ */
u.unescapeHTML = function (string) {
var div = document.createElement('div'); var div = document.createElement('div');
div.innerHTML = string; div.innerHTML = string;
return div.innerText; return div.innerText;
...@@ -381,13 +382,14 @@ u.slideToggleElement = function (el, duration) { ...@@ -381,13 +382,14 @@ u.slideToggleElement = function (el, duration) {
}; };
u.slideOut = function (el, duration=200) { /**
/* Shows/expands an element by sliding it out of itself * Shows/expands an element by sliding it out of itself
* * @private
* Parameters: * @method u#slideOut
* (HTMLElement) el - The HTML string * @param { HTMLElement } el - The HTML string
* (Number) duration - The duration amount in milliseconds * @param { Number } duration - The duration amount in milliseconds
*/ */
u.slideOut = function (el, duration=200) {
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
if (_.isNil(el)) { if (_.isNil(el)) {
const err = "Undefined or null element passed into slideOut" const err = "Undefined or null element passed into slideOut"
...@@ -528,16 +530,15 @@ u.fadeIn = function (el, callback) { ...@@ -528,16 +530,15 @@ u.fadeIn = function (el, callback) {
}; };
u.xForm2webForm = function (field, stanza, domain) { /**
/* Takes a field in XMPP XForm (XEP-004: Data Forms) format * Takes a field in XMPP XForm (XEP-004: Data Forms) format
* and turns it into an HTML field. * and turns it into an HTML field.
* * Returns either text or a DOM element (which is not ideal, but fine for now).
* Returns either text or a DOM element (which is not ideal, but fine * @private
* for now). * @method u#xForm2webForm
* * @param { XMLElement } field - the field to convert
* Parameters:
* (XMLElement) field - the field to convert
*/ */
u.xForm2webForm = function (field, stanza, domain) {
if (field.getAttribute('type') === 'list-single' || if (field.getAttribute('type') === 'list-single' ||
field.getAttribute('type') === 'list-multi') { field.getAttribute('type') === 'list-multi') {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment