2018-05-05 20:28:41 +02:00
|
|
|
// Converse.js
|
2016-03-16 12:16:32 +01:00
|
|
|
// http://conversejs.org
|
|
|
|
//
|
2018-05-25 16:25:02 +02:00
|
|
|
// Copyright (c) 2013-2018, the Converse.js developers
|
2016-03-16 12:16:32 +01:00
|
|
|
// Licensed under the Mozilla Public License (MPLv2)
|
|
|
|
|
|
|
|
(function (root, factory) {
|
2018-10-08 01:10:01 +02:00
|
|
|
define(["./converse-core", "./templates/vcard.html"], factory);
|
2018-08-27 21:54:18 +02:00
|
|
|
}(this, function (converse, tpl_vcard) {
|
2016-03-16 12:16:32 +01:00
|
|
|
"use strict";
|
2018-08-27 21:54:18 +02:00
|
|
|
const { Backbone, Promise, Strophe, _, $iq, $build, b64_sha1, moment, sizzle } = converse.env;
|
2018-02-23 22:15:04 +01:00
|
|
|
const u = converse.env.utils;
|
2017-12-02 18:00:18 +01:00
|
|
|
|
|
|
|
|
2016-12-20 11:42:20 +01:00
|
|
|
converse.plugins.add('converse-vcard', {
|
2016-03-16 12:16:32 +01:00
|
|
|
|
2017-07-10 17:46:22 +02:00
|
|
|
initialize () {
|
2016-03-16 12:16:32 +01:00
|
|
|
/* The initialize function gets called as soon as the plugin is
|
|
|
|
* loaded by converse.js's plugin machinery.
|
|
|
|
*/
|
2017-07-10 17:46:22 +02:00
|
|
|
const { _converse } = this;
|
2016-03-16 12:16:32 +01:00
|
|
|
|
2018-05-15 09:47:52 +02:00
|
|
|
_converse.VCard = Backbone.Model.extend({
|
|
|
|
defaults: {
|
|
|
|
'image': _converse.DEFAULT_IMAGE,
|
|
|
|
'image_type': _converse.DEFAULT_IMAGE_TYPE
|
|
|
|
},
|
|
|
|
|
|
|
|
set (key, val, options) {
|
|
|
|
// Override Backbone.Model.prototype.set to make sure that the
|
|
|
|
// default `image` and `image_type` values are maintained.
|
|
|
|
let attrs;
|
|
|
|
if (typeof key === 'object') {
|
|
|
|
attrs = key;
|
|
|
|
options = val;
|
|
|
|
} else {
|
|
|
|
(attrs = {})[key] = val;
|
|
|
|
}
|
|
|
|
if (_.has(attrs, 'image') && !attrs['image']) {
|
|
|
|
attrs['image'] = _converse.DEFAULT_IMAGE;
|
|
|
|
attrs['image_type'] = _converse.DEFAULT_IMAGE_TYPE;
|
|
|
|
return Backbone.Model.prototype.set.call(this, attrs, options);
|
|
|
|
} else {
|
|
|
|
return Backbone.Model.prototype.set.apply(this, arguments);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2018-05-03 18:34:28 +02:00
|
|
|
_converse.VCards = Backbone.Collection.extend({
|
2018-05-15 09:47:52 +02:00
|
|
|
model: _converse.VCard,
|
2018-05-03 18:34:28 +02:00
|
|
|
|
|
|
|
initialize () {
|
2018-05-04 22:50:41 +02:00
|
|
|
this.on('add', (vcard) => _converse.api.vcard.update(vcard));
|
2018-05-03 18:34:28 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2018-06-12 12:13:22 +02:00
|
|
|
function onVCardData (jid, iq, callback) {
|
2018-05-08 17:48:37 +02:00
|
|
|
const vcard = iq.querySelector('vCard');
|
|
|
|
let result = {};
|
|
|
|
if (!_.isNull(vcard)) {
|
|
|
|
result = {
|
|
|
|
'stanza': iq,
|
|
|
|
'fullname': _.get(vcard.querySelector('FN'), 'textContent'),
|
|
|
|
'nickname': _.get(vcard.querySelector('NICKNAME'), 'textContent'),
|
|
|
|
'image': _.get(vcard.querySelector('PHOTO BINVAL'), 'textContent'),
|
|
|
|
'image_type': _.get(vcard.querySelector('PHOTO TYPE'), 'textContent'),
|
|
|
|
'url': _.get(vcard.querySelector('URL'), 'textContent'),
|
|
|
|
'role': _.get(vcard.querySelector('ROLE'), 'textContent'),
|
2018-05-25 16:25:02 +02:00
|
|
|
'email': _.get(vcard.querySelector('EMAIL USERID'), 'textContent'),
|
|
|
|
'vcard_updated': moment().format(),
|
|
|
|
'vcard_error': undefined
|
2018-05-08 17:48:37 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
if (result.image) {
|
2018-08-27 21:54:18 +02:00
|
|
|
const buffer = u.base64ToArrayBuffer(result['image']);
|
|
|
|
crypto.subtle.digest('SHA-1', buffer)
|
|
|
|
.then(ab => {
|
|
|
|
result['image_hash'] = u.arrayBufferToHex(ab);
|
|
|
|
if (callback) callback(result);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
if (callback) callback(result);
|
2018-05-08 17:48:37 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-12 12:13:22 +02:00
|
|
|
function onVCardError (jid, iq, errback) {
|
2018-05-08 17:48:37 +02:00
|
|
|
if (errback) {
|
2018-05-25 16:25:02 +02:00
|
|
|
errback({
|
|
|
|
'stanza': iq,
|
|
|
|
'jid': jid,
|
|
|
|
'vcard_error': moment().format()
|
|
|
|
});
|
2018-05-08 17:48:37 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function createStanza (type, jid, vcard_el) {
|
|
|
|
const iq = $iq(jid ? {'type': type, 'to': jid} : {'type': type});
|
|
|
|
if (!vcard_el) {
|
|
|
|
iq.c("vCard", {'xmlns': Strophe.NS.VCARD});
|
|
|
|
} else {
|
|
|
|
iq.cnode(vcard_el);
|
|
|
|
}
|
|
|
|
return iq;
|
|
|
|
}
|
|
|
|
|
2018-06-12 12:13:22 +02:00
|
|
|
function setVCard (jid, data) {
|
|
|
|
if (!jid) {
|
|
|
|
throw Error("No jid provided for the VCard data");
|
|
|
|
}
|
|
|
|
const vcard_el = Strophe.xmlHtmlNode(tpl_vcard(data)).firstElementChild;
|
|
|
|
return _converse.api.sendIQ(createStanza("set", jid, vcard_el));
|
2018-05-08 17:48:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function getVCard (_converse, jid) {
|
|
|
|
/* Request the VCard of another user. Returns a promise.
|
2018-06-12 12:13:22 +02:00
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* (String) jid - The Jabber ID of the user whose VCard
|
|
|
|
* is being requested.
|
|
|
|
*/
|
|
|
|
const to = Strophe.getBareJidFromJid(jid) === _converse.bare_jid ? null : jid;
|
2018-05-08 17:48:37 +02:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
_converse.connection.sendIQ(
|
2018-06-12 12:13:22 +02:00
|
|
|
createStanza("get", to),
|
|
|
|
_.partial(onVCardData, jid, _, resolve),
|
|
|
|
_.partial(onVCardError, jid, _, resolve),
|
|
|
|
_converse.IQ_TIMEOUT
|
2018-05-08 17:48:37 +02:00
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-07-21 12:41:16 +02:00
|
|
|
/* Event handlers */
|
2018-05-03 18:34:28 +02:00
|
|
|
_converse.initVCardCollection = function () {
|
|
|
|
_converse.vcards = new _converse.VCards();
|
2018-08-22 22:20:15 +02:00
|
|
|
const id = b64_sha1(`converse.vcards`);
|
2018-08-23 09:41:39 +02:00
|
|
|
_converse.vcards.browserStorage = new Backbone.BrowserStorage[_converse.config.get('storage')](id);
|
2018-05-03 18:34:28 +02:00
|
|
|
_converse.vcards.fetch();
|
|
|
|
}
|
2018-08-23 09:41:39 +02:00
|
|
|
_converse.api.listen.on('sessionInitialized', _converse.initVCardCollection);
|
2018-05-03 18:34:28 +02:00
|
|
|
|
|
|
|
|
2017-07-21 12:41:16 +02:00
|
|
|
_converse.on('addClientFeatures', () => {
|
2018-05-11 13:31:48 +02:00
|
|
|
_converse.api.disco.own.features.add(Strophe.NS.VCARD);
|
2017-07-21 12:41:16 +02:00
|
|
|
});
|
|
|
|
|
2017-12-02 18:00:18 +01:00
|
|
|
_.extend(_converse.api, {
|
2018-09-02 15:07:14 +02:00
|
|
|
/**
|
|
|
|
* The XEP-0054 VCard API
|
|
|
|
*
|
|
|
|
* This API lets you access and update user VCards
|
|
|
|
*
|
|
|
|
* @namespace _converse.api.vcard
|
|
|
|
* @memberOf _converse.api
|
|
|
|
*/
|
2017-12-02 18:00:18 +01:00
|
|
|
'vcard': {
|
2018-09-02 15:07:14 +02:00
|
|
|
/**
|
|
|
|
* Enables setting new values for a VCard.
|
|
|
|
*
|
|
|
|
* @method _converse.api.vcard.set
|
|
|
|
* @param {string} jid The JID for which the VCard should be set
|
|
|
|
* @param {object} data A map of VCard keys and values
|
|
|
|
* @example
|
|
|
|
* _converse.api.vcard.set({
|
|
|
|
* 'jid': _converse.bare_jid,
|
|
|
|
* 'fn': 'John Doe',
|
|
|
|
* 'nickname': 'jdoe'
|
|
|
|
* }).then(() => {
|
|
|
|
* // Succes
|
|
|
|
* }).catch(() => {
|
|
|
|
* // Failure
|
|
|
|
* }).
|
|
|
|
*/
|
2018-06-12 12:13:22 +02:00
|
|
|
'set' (jid, data) {
|
|
|
|
return setVCard(jid, data);
|
|
|
|
},
|
2018-05-08 17:48:37 +02:00
|
|
|
|
2018-09-02 15:07:14 +02:00
|
|
|
/**
|
|
|
|
* @method _converse.api.vcard.get
|
|
|
|
* @param {Backbone.Model|string} model Either a `Backbone.Model` instance, or a string JID.
|
|
|
|
* If a `Backbone.Model` instance is passed in, then it must have either a `jid`
|
|
|
|
* attribute or a `muc_jid` attribute.
|
|
|
|
* @param {boolean} [force] A boolean indicating whether the vcard should be
|
|
|
|
* fetched even if it's been fetched before.
|
|
|
|
* @returns {promise} A Promise which resolves with the VCard data for a particular JID or for
|
|
|
|
* a `Backbone.Model` instance which represents an entity with a JID (such as a roster contact,
|
|
|
|
* chat or chatroom occupant).
|
|
|
|
*
|
|
|
|
* @example
|
|
|
|
* _converse.api.waitUntil('rosterContactsFetched').then(() => {
|
|
|
|
* _converse.api.vcard.get('someone@example.org').then(
|
|
|
|
* (vcard) => {
|
|
|
|
* // Do something with the vcard...
|
|
|
|
* }
|
|
|
|
* );
|
|
|
|
* });
|
|
|
|
*/
|
|
|
|
'get' (model, force) {
|
2018-05-01 11:33:32 +02:00
|
|
|
if (_.isString(model)) {
|
|
|
|
return getVCard(_converse, model);
|
2018-05-25 16:25:02 +02:00
|
|
|
} else if (force ||
|
|
|
|
!model.get('vcard_updated') ||
|
|
|
|
!moment(model.get('vcard_error')).isSame(new Date(), "day")) {
|
|
|
|
|
2018-05-25 15:55:34 +02:00
|
|
|
const jid = model.get('jid');
|
2018-05-01 11:33:32 +02:00
|
|
|
if (!jid) {
|
|
|
|
throw new Error("No JID to get vcard for!");
|
|
|
|
}
|
|
|
|
return getVCard(_converse, jid);
|
|
|
|
} else {
|
2018-05-01 14:51:18 +02:00
|
|
|
return Promise.resolve({});
|
2018-05-01 11:33:32 +02:00
|
|
|
}
|
2018-05-01 14:51:18 +02:00
|
|
|
},
|
|
|
|
|
2018-09-02 15:07:14 +02:00
|
|
|
/**
|
|
|
|
* Fetches the VCard associated with a particular `Backbone.Model` instance
|
|
|
|
* (by using its `jid` or `muc_jid` attribute) and then updates the model with the
|
|
|
|
* returned VCard data.
|
|
|
|
*
|
|
|
|
* @method _converse.api.vcard.update
|
|
|
|
* @param {Backbone.Model} model A `Backbone.Model` instance
|
|
|
|
* @param {boolean} [force] A boolean indicating whether the vcard should be
|
|
|
|
* fetched again even if it's been fetched before.
|
|
|
|
* @returns {promise} A promise which resolves once the update has completed.
|
|
|
|
* @example
|
|
|
|
* _converse.api.waitUntil('rosterContactsFetched').then(() => {
|
|
|
|
* const chatbox = _converse.chatboxes.getChatBox('someone@example.org');
|
|
|
|
* _converse.api.vcard.update(chatbox);
|
|
|
|
* });
|
|
|
|
*/
|
2018-05-01 14:51:18 +02:00
|
|
|
'update' (model, force) {
|
2018-09-02 15:07:14 +02:00
|
|
|
return this.get(model, force)
|
|
|
|
.then(vcard => {
|
2018-05-25 16:25:02 +02:00
|
|
|
delete vcard['stanza']
|
|
|
|
model.save(vcard);
|
2018-05-01 18:18:02 +02:00
|
|
|
});
|
2017-12-02 18:00:18 +01:00
|
|
|
}
|
2016-03-16 12:16:32 +01:00
|
|
|
}
|
2017-07-21 12:41:16 +02:00
|
|
|
});
|
2016-03-16 12:16:32 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}));
|