123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779 |
- // Converse.js (A browser based XMPP chat client)
- // http://conversejs.org
- //
- // This is the utilities module.
- //
- // Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
- // Licensed under the Mozilla Public License (MPLv2)
- //
- /*global define, escape, window */
- (function (root, factory) {
- define([
- "sizzle",
- "es6-promise",
- "lodash.noconflict",
- "strophe",
- "tpl!audio",
- "tpl!file",
- "tpl!image",
- "tpl!video"
- ], factory);
- }(this, function (
- sizzle,
- Promise,
- _,
- Strophe,
- tpl_audio,
- tpl_file,
- tpl_image,
- tpl_video
- ) {
- "use strict";
- const b64_sha1 = Strophe.SHA1.b64_sha1;
- Strophe = Strophe.Strophe;
- const URL_REGEX = /\b(https?:\/\/|www\.|https?:\/\/www\.)[^\s<>]{2,200}\b\/?/g;
- const logger = _.assign({
- 'debug': _.get(console, 'log') ? console.log.bind(console) : _.noop,
- 'error': _.get(console, 'log') ? console.log.bind(console) : _.noop,
- 'info': _.get(console, 'log') ? console.log.bind(console) : _.noop,
- 'warn': _.get(console, 'log') ? console.log.bind(console) : _.noop
- }, console);
- var unescapeHTML = function (htmlEscapedText) {
- /* Helper method that replace HTML-escaped symbols with equivalent characters
- * (e.g. transform occurrences of '&' to '&')
- *
- * Parameters:
- * (String) htmlEscapedText: a String containing the HTML-escaped symbols.
- */
- var div = document.createElement('div');
- div.innerHTML = htmlEscapedText;
- return div.innerText;
- };
- var isImage = function (url) {
- return new Promise((resolve, reject) => {
- var img = new Image();
- var timer = window.setTimeout(function () {
- reject(new Error("Could not determine whether it's an image"));
- img = null;
- }, 3000);
- img.onerror = img.onabort = function () {
- clearTimeout(timer);
- reject(new Error("Could not determine whether it's an image"));
- };
- img.onload = function () {
- clearTimeout(timer);
- resolve(img);
- };
- img.src = url;
- });
- };
- function slideOutWrapup (el) {
- /* Wrapup function for slideOut. */
- el.removeAttribute('data-slider-marker');
- el.classList.remove('collapsed');
- el.style.overflow = "";
- el.style.height = "";
- }
- var u = {};
- u.getNextElement = function (el, selector='*') {
- let next_el = el.nextElementSibling;
- while (!_.isNull(next_el) && !sizzle.matchesSelector(next_el, selector)) {
- next_el = next_el.nextElementSibling;
- }
- return next_el;
- }
- u.getPreviousElement = function (el, selector='*') {
- let prev_el = el.previousSibling;
- while (!_.isNull(prev_el) && !sizzle.matchesSelector(prev_el, selector)) {
- prev_el = prev_el.previousSibling
- }
- return prev_el;
- }
- u.getFirstChildElement = function (el, selector='*') {
- let first_el = el.firstElementChild;
- while (!_.isNull(first_el) && !sizzle.matchesSelector(first_el, selector)) {
- first_el = first_el.nextSibling
- }
- return first_el;
- }
- u.getLastChildElement = function (el, selector='*') {
- let last_el = el.lastElementChild;
- while (!_.isNull(last_el) && !sizzle.matchesSelector(last_el, selector)) {
- last_el = last_el.previousSibling
- }
- return last_el;
- }
- u.calculateElementHeight = function (el) {
- /* Return the height of the passed in DOM element,
- * based on the heights of its children.
- */
- return _.reduce(
- el.children,
- (result, child) => result + child.offsetHeight, 0
- );
- }
- u.addClass = function (className, el) {
- if (el instanceof Element) {
- el.classList.add(className);
- }
- }
- u.removeClass = function (className, el) {
- if (el instanceof Element) {
- el.classList.remove(className);
- }
- return el;
- }
- u.removeElement = function (el) {
- if (!_.isNil(el) && !_.isNil(el.parentNode)) {
- el.parentNode.removeChild(el);
- }
- }
- u.showElement = _.flow(
- _.partial(u.removeClass, 'collapsed'),
- _.partial(u.removeClass, 'hidden')
- )
- u.hideElement = function (el) {
- if (!_.isNil(el)) {
- el.classList.add('hidden');
- }
- return el;
- }
- u.ancestor = function (el, selector) {
- let parent = el;
- while (!_.isNil(parent) && !sizzle.matchesSelector(parent, selector)) {
- parent = parent.parentElement;
- }
- return parent;
- }
- u.nextUntil = function (el, selector, include_self=false) {
- /* Return the element's siblings until one matches the selector. */
- const matches = [];
- let sibling_el = el.nextElementSibling;
- while (!_.isNil(sibling_el) && !sibling_el.matches(selector)) {
- matches.push(sibling_el);
- sibling_el = sibling_el.nextElementSibling;
- }
- return matches;
- }
- u.addHyperlinks = function (text) {
- const list = text.match(URL_REGEX) || [];
- var links = [];
- _.each(list, (match) => {
- const prot = match.indexOf('http://') === 0 || match.indexOf('https://') === 0 ? '' : 'http://';
- const url = prot + encodeURI(decodeURI(match)).replace(/[!'()]/g, escape).replace(/\*/g, "%2A");
- const a = '<a target="_blank" rel="noopener" href="' + url + '">'+ _.escape(match) + '</a>';
- // We first insert a hash of the code that will be inserted, and
- // then later replace that with the code itself. That way we avoid
- // issues when some matches are substrings of others.
- links.push(a);
- text = text.replace(match, b64_sha1(a));
- });
- while (links.length) {
- const a = links.pop();
- text = text.replace(b64_sha1(a), a);
- }
- return text;
- };
- u.renderImageURLs = function (obj) {
- /* Returns a Promise which resolves once all images have been loaded.
- */
- const list = obj.textContent.match(URL_REGEX) || [];
- return Promise.all(
- _.map(list, (url) =>
- new Promise((resolve, reject) =>
- isImage(url).then(function (img) {
- // XXX: need to create a new image, otherwise the event
- // listener doesn't fire
- const i = new Image();
- i.className = 'chat-image';
- i.src = img.src;
- i.addEventListener('load', resolve);
- // We also resolve for non-images, otherwise the
- // Promise.all resolves prematurely.
- i.addEventListener('error', resolve);
- var anchors = sizzle(`a[href="${url}"]`, obj);
- _.each(anchors, (a) => {
- a.replaceChild(i, a.firstChild);
- });
- }).catch(resolve)
- )
- ))
- };
- u.renderFileURL = function (_converse, url) {
- if (url.endsWith('mp3') || url.endsWith('mp4') ||
- url.endsWith('jpg') || url.endsWith('jpeg') ||
- url.endsWith('png') || url.endsWith('gif') ||
- url.endsWith('svg')) {
- return url;
- }
- const name = url.split('/').pop(),
- { __ } = _converse;
- return tpl_file({
- 'url': url,
- 'label_download': __('Download file: "%1$s', name)
- })
- };
- u.renderImageURL = function (_converse, url) {
- const { __ } = _converse;
- if (url.endsWith('jpg') || url.endsWith('jpeg') || url.endsWith('png') ||
- url.endsWith('gif') || url.endsWith('svg')) {
- return tpl_image({
- 'url': url,
- 'label_download': __('Download image file')
- })
- }
- return url;
- };
- u.renderMovieURL = function (_converse, url) {
- const { __ } = _converse;
- if (url.endsWith('mp4')) {
- return tpl_video({
- 'url': url,
- 'label_download': __('Download video file')
- })
- }
- return url;
- };
- u.renderAudioURL = function (_converse, url) {
- const { __ } = _converse;
- if (url.endsWith('mp3')) {
- return tpl_audio({
- 'url': url,
- 'label_download': __('Download audio file')
- })
- }
- return url;
- };
- u.slideInAllElements = function (elements, duration=300) {
- return Promise.all(
- _.map(
- elements,
- _.partial(u.slideIn, _, duration)
- ));
- };
- u.slideToggleElement = function (el, duration) {
- if (_.includes(el.classList, 'collapsed') ||
- _.includes(el.classList, 'hidden')) {
- return u.slideOut(el, duration);
- } else {
- return u.slideIn(el, duration);
- }
- };
- u.hasClass = function (className, el) {
- return _.includes(el.classList, className);
- };
- u.slideOut = function (el, duration=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) => {
- if (_.isNil(el)) {
- const err = "Undefined or null element passed into slideOut"
- logger.warn(err);
- reject(new Error(err));
- return;
- }
- const marker = el.getAttribute('data-slider-marker');
- if (marker) {
- el.removeAttribute('data-slider-marker');
- window.cancelAnimationFrame(marker);
- }
- const end_height = u.calculateElementHeight(el);
- if (window.converse_disable_effects) { // Effects are disabled (for tests)
- el.style.height = end_height + 'px';
- slideOutWrapup(el);
- resolve();
- return;
- }
- if (!u.hasClass('collapsed', el) && !u.hasClass('hidden', el)) {
- resolve();
- return;
- }
- const steps = duration/17; // We assume 17ms per animation which is ~60FPS
- let height = 0;
- function draw () {
- height += end_height/steps;
- if (height < end_height) {
- el.style.height = height + 'px';
- el.setAttribute(
- 'data-slider-marker',
- window.requestAnimationFrame(draw)
- );
- } else {
- // We recalculate the height to work around an apparent
- // browser bug where browsers don't know the correct
- // offsetHeight beforehand.
- el.removeAttribute('data-slider-marker');
- el.style.height = u.calculateElementHeight(el) + 'px';
- el.style.overflow = "";
- el.style.height = "";
- resolve();
- }
- }
- el.style.height = '0';
- el.style.overflow = 'hidden';
- el.classList.remove('hidden');
- el.classList.remove('collapsed');
- el.setAttribute(
- 'data-slider-marker',
- window.requestAnimationFrame(draw)
- );
- });
- };
- u.slideIn = function (el, duration=200) {
- /* Hides/collapses an element by sliding it into itself. */
- return new Promise((resolve, reject) => {
- if (_.isNil(el)) {
- const err = "Undefined or null element passed into slideIn";
- logger.warn(err);
- return reject(new Error(err));
- } else if (_.includes(el.classList, 'collapsed')) {
- return resolve(el);
- } else if (window.converse_disable_effects) { // Effects are disabled (for tests)
- el.classList.add('collapsed');
- el.style.height = "";
- return resolve(el);
- }
- const marker = el.getAttribute('data-slider-marker');
- if (marker) {
- el.removeAttribute('data-slider-marker');
- window.cancelAnimationFrame(marker);
- }
- const original_height = el.offsetHeight,
- steps = duration/17; // We assume 17ms per animation which is ~60FPS
- let height = original_height;
- el.style.overflow = 'hidden';
- function draw () {
- height -= original_height/steps;
- if (height > 0) {
- el.style.height = height + 'px';
- el.setAttribute(
- 'data-slider-marker',
- window.requestAnimationFrame(draw)
- );
- } else {
- el.removeAttribute('data-slider-marker');
- el.classList.add('collapsed');
- el.style.height = "";
- resolve(el);
- }
- }
- el.setAttribute(
- 'data-slider-marker',
- window.requestAnimationFrame(draw)
- );
- });
- };
- function afterAnimationEnds (el, callback) {
- el.classList.remove('visible');
- if (_.isFunction(callback)) {
- callback();
- }
- }
- u.fadeIn = function (el, callback) {
- if (_.isNil(el)) {
- logger.warn("Undefined or null element passed into fadeIn");
- }
- if (window.converse_disable_effects) {
- el.classList.remove('hidden');
- return afterAnimationEnds(el, callback);
- }
- if (_.includes(el.classList, 'hidden')) {
- el.classList.add('visible');
- el.classList.remove('hidden');
- el.addEventListener("webkitAnimationEnd", _.partial(afterAnimationEnds, el, callback));
- el.addEventListener("animationend", _.partial(afterAnimationEnds, el, callback));
- el.addEventListener("oanimationend", _.partial(afterAnimationEnds, el, callback));
- } else {
- afterAnimationEnds(el, callback);
- }
- };
- u.isValidJID = function (jid) {
- return _.compact(jid.split('@')).length === 2 && !jid.startsWith('@') && !jid.endsWith('@');
- };
- u.isValidMUCJID = function (jid) {
- return !jid.startsWith('@') && !jid.endsWith('@');
- };
- u.isSameBareJID = function (jid1, jid2) {
- return Strophe.getBareJidFromJid(jid1).toLowerCase() ===
- Strophe.getBareJidFromJid(jid2).toLowerCase();
- };
- u.getMostRecentMessage = function (model) {
- const messages = model.messages.filter('message');
- return messages[messages.length-1];
- }
- u.isNewMessage = function (message) {
- /* Given a stanza, determine whether it's a new
- * message, i.e. not a MAM archived one.
- */
- if (message instanceof Element) {
- return !sizzle('result[xmlns="'+Strophe.NS.MAM+'"]', message).length &&
- !sizzle('delay[xmlns="'+Strophe.NS.DELAY+'"]', message).length;
- } else {
- return !message.get('delayed');
- }
- };
- u.isOTRMessage = function (message) {
- var body = message.querySelector('body'),
- text = (!_.isNull(body) ? body.textContent: undefined);
- return text && !!text.match(/^\?OTR/);
- };
- u.isHeadlineMessage = function (_converse, message) {
- var from_jid = message.getAttribute('from');
- if (message.getAttribute('type') === 'headline') {
- return true;
- }
- const chatbox = _converse.chatboxes.get(Strophe.getBareJidFromJid(from_jid));
- if (chatbox && chatbox.get('type') === 'chatroom') {
- return false;
- }
- if (message.getAttribute('type') !== 'error' &&
- !_.isNil(from_jid) &&
- !_.includes(from_jid, '@')) {
- // Some servers (I'm looking at you Prosody) don't set the message
- // type to "headline" when sending server messages. For now we
- // check if an @ signal is included, and if not, we assume it's
- // a headline message.
- return true;
- }
- return false;
- };
- u.merge = function merge (first, second) {
- /* Merge the second object into the first one.
- */
- for (var k in second) {
- if (_.isObject(first[k])) {
- merge(first[k], second[k]);
- } else {
- first[k] = second[k];
- }
- }
- };
- u.applyUserSettings = function applyUserSettings (context, settings, user_settings) {
- /* Configuration settings might be nested objects. We only want to
- * add settings which are whitelisted.
- */
- for (var k in settings) {
- if (_.isUndefined(user_settings[k])) {
- continue;
- }
- if (_.isObject(settings[k]) && !_.isArray(settings[k])) {
- applyUserSettings(context[k], settings[k], user_settings[k]);
- } else {
- context[k] = user_settings[k];
- }
- }
- };
- u.stringToNode = function (s) {
- /* Converts an HTML string into a DOM Node.
- * Expects that the HTML string has only one top-level element,
- * i.e. not multiple ones.
- *
- * Parameters:
- * (String) s - The HTML string
- */
- var div = document.createElement('div');
- div.innerHTML = s;
- return div.firstChild;
- };
- u.getOuterWidth = function (el, include_margin=false) {
- var width = el.offsetWidth;
- if (!include_margin) {
- return width;
- }
- var style = window.getComputedStyle(el);
- width += parseInt(style.marginLeft, 10) + parseInt(style.marginRight, 10);
- return width;
- };
- u.stringToElement = function (s) {
- /* Converts an HTML string into a DOM element.
- * Expects that the HTML string has only one top-level element,
- * i.e. not multiple ones.
- *
- * Parameters:
- * (String) s - The HTML string
- */
- var div = document.createElement('div');
- div.innerHTML = s;
- return div.firstElementChild;
- };
- u.matchesSelector = function (el, selector) {
- /* Checks whether the DOM element matches the given selector.
- *
- * Parameters:
- * (DOMElement) el - The DOM element
- * (String) selector - The selector
- */
- return (
- el.matches ||
- el.matchesSelector ||
- el.msMatchesSelector ||
- el.mozMatchesSelector ||
- el.webkitMatchesSelector ||
- el.oMatchesSelector
- ).call(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 _.filter(el.children, _.partial(u.matchesSelector, _, selector));
- };
- u.contains = function (attr, query) {
- return function (item) {
- if (typeof attr === 'object') {
- var value = false;
- _.forEach(attr, function (a) {
- value = value || _.includes(item.get(a).toLowerCase(), query.toLowerCase());
- });
- return value;
- } else if (typeof attr === 'string') {
- return _.includes(item.get(attr).toLowerCase(), query.toLowerCase());
- } else {
- throw new TypeError('contains: wrong attribute type. Must be string or array.');
- }
- };
- };
- u.isOfType = function (type, item) {
- return item.get('type') == type;
- };
- u.isInstance = function (type, item) {
- return item instanceof type;
- };
- u.getAttribute = function (key, item) {
- return item.get(key);
- };
- u.contains.not = function (attr, query) {
- return function (item) {
- return !(u.contains(attr, query)(item));
- };
- };
- u.createFragmentFromText = function (markup) {
- /* Returns a DocumentFragment containing DOM nodes based on the
- * passed-in markup text.
- */
- // http://stackoverflow.com/questions/9334645/create-node-from-markup-string
- var frag = document.createDocumentFragment(),
- tmp = document.createElement('body'), child;
- tmp.innerHTML = markup;
- // Append elements in a loop to a DocumentFragment, so that the
- // browser does not re-render the document for each node.
- while (child = tmp.firstChild) { // eslint-disable-line no-cond-assign
- frag.appendChild(child);
- }
- return frag
- };
- u.addEmoji = function (_converse, emojione, text) {
- if (_converse.use_emojione) {
- return emojione.toImage(text);
- } else {
- return emojione.shortnameToUnicode(text);
- }
- }
- u.getEmojisByCategory = function (_converse, emojione) {
- /* Return a dict of emojis with the categories as keys and
- * lists of emojis in that category as values.
- */
- if (_.isUndefined(_converse.emojis_by_category)) {
- const emojis = _.values(_.mapValues(emojione.emojioneList, function (value, key, o) {
- value._shortname = key;
- return value
- }));
- const tones = [':tone1:', ':tone2:', ':tone3:', ':tone4:', ':tone5:'];
- const excluded = [':kiss_ww:', ':kiss_mm:', ':kiss_woman_man:'];
- const excluded_substrings = [
- ':woman', ':man', ':women_', ':men_', '_man_', '_woman_', '_woman:', '_man:'
- ];
- const excluded_categories = ['modifier', 'regional'];
- const categories = _.difference(
- _.uniq(_.map(emojis, _.partial(_.get, _, 'category'))),
- excluded_categories
- );
- const emojis_by_category = {};
- _.forEach(categories, (cat) => {
- let list = _.sortBy(_.filter(emojis, ['category', cat]), ['uc_base']);
- list = _.filter(
- list,
- (item) => !_.includes(_.concat(tones, excluded), item._shortname) &&
- !_.some(excluded_substrings, _.partial(_.includes, item._shortname))
- );
- if (cat === 'people') {
- const idx = _.findIndex(list, ['uc_base', '1f600']);
- list = _.union(_.slice(list, idx), _.slice(list, 0, idx+1));
- } else if (cat === 'activity') {
- list = _.union(_.slice(list, 27-1), _.slice(list, 0, 27));
- } else if (cat === 'objects') {
- list = _.union(_.slice(list, 24-1), _.slice(list, 0, 24));
- } else if (cat === 'travel') {
- list = _.union(_.slice(list, 17-1), _.slice(list, 0, 17));
- } else if (cat === 'symbols') {
- list = _.union(_.slice(list, 60-1), _.slice(list, 0, 60));
- }
- emojis_by_category[cat] = list;
- });
- _converse.emojis_by_category = emojis_by_category;
- }
- return _converse.emojis_by_category;
- };
- u.getTonedEmojis = function (_converse) {
- _converse.toned_emojis = _.uniq(
- _.map(
- _.filter(
- u.getEmojisByCategory(_converse).people,
- (person) => _.includes(person._shortname, '_tone')
- ),
- (person) => person._shortname.replace(/_tone[1-5]/, '')
- ));
- return _converse.toned_emojis;
- };
- u.isPersistableModel = function (model) {
- return model.collection && model.collection.browserStorage;
- };
- u.getResolveablePromise = function () {
- /* Returns a promise object on which `resolve` or `reject` can be
- * called.
- */
- const wrapper = {};
- const promise = new Promise((resolve, reject) => {
- wrapper.resolve = resolve;
- wrapper.reject = reject;
- })
- _.assign(promise, wrapper);
- return promise;
- };
- u.interpolate = function (string, o) {
- return string.replace(/{{{([^{}]*)}}}/g,
- (a, b) => {
- var r = o[b];
- return typeof r === 'string' || typeof r === 'number' ? r : a;
- });
- };
- u.onMultipleEvents = function (events=[], callback) {
- /* 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
- * triggered upon.
- * (Function) callback: The function to call once all events have
- * been triggered.
- */
- let triggered = [];
- function handler (result) {
- triggered.push(result)
- if (events.length === triggered.length) {
- callback(triggered);
- triggered = [];
- }
- }
- _.each(events, (map) => map.object.on(map.event, handler));
- };
- u.safeSave = function (model, attributes) {
- if (u.isPersistableModel(model)) {
- model.save(attributes);
- } else {
- model.set(attributes);
- }
- }
- u.isVisible = function (el) {
- if (u.hasClass('hidden', el)) {
- return false;
- }
- // XXX: Taken from jQuery's "visible" implementation
- return el.offsetWidth > 0 || el.offsetHeight > 0 || el.getClientRects().length > 0;
- };
- u.triggerEvent = function (el, name, type="Event", bubbles=true, cancelable=true) {
- const evt = document.createEvent(type);
- evt.initEvent(name, bubbles, cancelable);
- el.dispatchEvent(evt);
- };
- u.geoUriToHttp = function(text, geouri_replacement) {
- const regex = /geo:([\-0-9.]+),([\-0-9.]+)(?:,([\-0-9.]+))?(?:\?(.*))?/g;
- return text.replace(regex, geouri_replacement);
- };
- u.httpToGeoUri = function(text, _converse) {
- const replacement = 'geo:$1,$2';
- return text.replace(_converse.geouri_regex, replacement);
- };
- return u;
- }));
|