xmpp.chapril.org-conversejs/dist/converse-no-dependencies.js

21796 lines
1.5 MiB
JavaScript
Raw Normal View History

2017-03-05 10:45:18 +01:00
/** Converse.js
*
* An XMPP chat client that runs in the browser.
*
2017-07-22 22:21:05 +02:00
* Version: 3.2.0-rc
2017-03-05 10:45:18 +01:00
*/
/* jshint ignore:start */
(function (root, factory) {
if (typeof define === 'function' && define.amd) {
//Allow using this built library as an AMD module
//in another project. That other project will only
//see this AMD call, not the internal modules in
//the closure below.
define([], factory);
} else {
//Browser globals case.
root.converse = factory();
}
}(this, function () {
//almond, and your modules will be inlined here
/* jshint ignore:end */
/**
* @license almond 0.3.3 Copyright jQuery Foundation and other contributors.
2016-05-03 17:37:10 +02:00
* Released under MIT license, http://github.com/requirejs/almond/LICENSE
*/
//Going sloppy to avoid 'use strict' string cost, but strict practices should
//be followed.
/*global setTimeout: false */
var requirejs, require, define;
(function (undef) {
var main, req, makeMap, handlers,
defined = {},
waiting = {},
config = {},
defining = {},
hasOwn = Object.prototype.hasOwnProperty,
aps = [].slice,
jsSuffixRegExp = /\.js$/;
function hasProp(obj, prop) {
return hasOwn.call(obj, prop);
}
/**
* Given a relative module name, like ./something, normalize it to
* a real name that can be mapped to a path.
* @param {String} name the relative name
* @param {String} baseName a real name that the name arg is relative
* to.
* @returns {String} normalized name
*/
function normalize(name, baseName) {
var nameParts, nameSegment, mapValue, foundMap, lastIndex,
2016-05-03 17:37:10 +02:00
foundI, foundStarMap, starI, i, j, part, normalizedBaseParts,
baseParts = baseName && baseName.split("/"),
map = config.map,
starMap = (map && map['*']) || {};
//Adjust any relative paths.
2016-05-03 17:37:10 +02:00
if (name) {
name = name.split('/');
lastIndex = name.length - 1;
// If wanting node ID compatibility, strip .js from end
// of IDs. Have to do this here, and not in nameToUrl
// because node allows either .js or non .js to map
// to same file.
if (config.nodeIdCompat && jsSuffixRegExp.test(name[lastIndex])) {
name[lastIndex] = name[lastIndex].replace(jsSuffixRegExp, '');
}
// Starts with a '.' so need the baseName
if (name[0].charAt(0) === '.' && baseParts) {
//Convert baseName to array, and lop off the last part,
//so that . matches that 'directory' and not name of the baseName's
//module. For instance, baseName of 'one/two/three', maps to
//'one/two/three.js', but we want the directory, 'one/two' for
//this normalization.
normalizedBaseParts = baseParts.slice(0, baseParts.length - 1);
name = normalizedBaseParts.concat(name);
}
//start trimDots
for (i = 0; i < name.length; i++) {
part = name[i];
if (part === '.') {
name.splice(i, 1);
i -= 1;
} else if (part === '..') {
// If at the start, or previous value is still ..,
// keep them so that when converted to a path it may
// still work when converted to a path, even though
// as an ID it is less than ideal. In larger point
// releases, may be better to just kick out an error.
if (i === 0 || (i === 1 && name[2] === '..') || name[i - 1] === '..') {
continue;
} else if (i > 0) {
name.splice(i - 1, 2);
i -= 2;
}
}
}
2016-05-03 17:37:10 +02:00
//end trimDots
name = name.join('/');
}
//Apply map config if available.
if ((baseParts || starMap) && map) {
nameParts = name.split('/');
for (i = nameParts.length; i > 0; i -= 1) {
nameSegment = nameParts.slice(0, i).join("/");
if (baseParts) {
//Find the longest baseName segment match in the config.
//So, do joins on the biggest to smallest lengths of baseParts.
for (j = baseParts.length; j > 0; j -= 1) {
mapValue = map[baseParts.slice(0, j).join('/')];
//baseName segment has config, find if it has one for
//this name.
if (mapValue) {
mapValue = mapValue[nameSegment];
if (mapValue) {
//Match, update name to the new value.
foundMap = mapValue;
foundI = i;
break;
}
}
}
}
if (foundMap) {
break;
}
//Check for a star map match, but just hold on to it,
//if there is a shorter segment match later in a matching
//config, then favor over this star map.
if (!foundStarMap && starMap && starMap[nameSegment]) {
foundStarMap = starMap[nameSegment];
starI = i;
}
}
if (!foundMap && foundStarMap) {
foundMap = foundStarMap;
foundI = starI;
}
if (foundMap) {
nameParts.splice(0, foundI, foundMap);
name = nameParts.join('/');
}
}
return name;
}
function makeRequire(relName, forceSync) {
return function () {
//A version of a require function that passes a moduleName
//value for items that may need to
//look up paths relative to the moduleName
var args = aps.call(arguments, 0);
//If first arg is not require('string'), and there is only
//one arg, it is the array form without a callback. Insert
//a null so that the following concat is correct.
if (typeof args[0] !== 'string' && args.length === 1) {
args.push(null);
}
return req.apply(undef, args.concat([relName, forceSync]));
};
}
function makeNormalize(relName) {
return function (name) {
return normalize(name, relName);
};
}
function makeLoad(depName) {
return function (value) {
defined[depName] = value;
};
}
function callDep(name) {
if (hasProp(waiting, name)) {
var args = waiting[name];
delete waiting[name];
defining[name] = true;
main.apply(undef, args);
}
if (!hasProp(defined, name) && !hasProp(defining, name)) {
throw new Error('No ' + name);
}
return defined[name];
}
//Turns a plugin!resource to [plugin, resource]
//with the plugin being undefined if the name
//did not have a plugin prefix.
function splitPrefix(name) {
var prefix,
index = name ? name.indexOf('!') : -1;
if (index > -1) {
prefix = name.substring(0, index);
name = name.substring(index + 1, name.length);
}
return [prefix, name];
}
//Creates a parts array for a relName where first part is plugin ID,
//second part is resource ID. Assumes relName has already been normalized.
function makeRelParts(relName) {
return relName ? splitPrefix(relName) : [];
}
/**
* Makes a name map, normalizing the name, and using a plugin
* for normalization if necessary. Grabs a ref to plugin
* too, as an optimization.
*/
makeMap = function (name, relParts) {
var plugin,
parts = splitPrefix(name),
prefix = parts[0],
relResourceName = relParts[1];
name = parts[1];
if (prefix) {
prefix = normalize(prefix, relResourceName);
plugin = callDep(prefix);
}
//Normalize according
if (prefix) {
if (plugin && plugin.normalize) {
name = plugin.normalize(name, makeNormalize(relResourceName));
} else {
name = normalize(name, relResourceName);
}
} else {
name = normalize(name, relResourceName);
parts = splitPrefix(name);
prefix = parts[0];
name = parts[1];
if (prefix) {
plugin = callDep(prefix);
}
}
//Using ridiculous property names for space reasons
return {
f: prefix ? prefix + '!' + name : name, //fullName
n: name,
pr: prefix,
p: plugin
};
};
function makeConfig(name) {
return function () {
return (config && config.config && config.config[name]) || {};
};
}
handlers = {
require: function (name) {
return makeRequire(name);
},
exports: function (name) {
var e = defined[name];
if (typeof e !== 'undefined') {
return e;
} else {
return (defined[name] = {});
}
},
module: function (name) {
return {
id: name,
uri: '',
exports: defined[name],
config: makeConfig(name)
};
}
};
main = function (name, deps, callback, relName) {
var cjsModule, depName, ret, map, i, relParts,
args = [],
callbackType = typeof callback,
usingExports;
//Use name if no relName
relName = relName || name;
relParts = makeRelParts(relName);
//Call the callback to define the module, if necessary.
if (callbackType === 'undefined' || callbackType === 'function') {
//Pull out the defined dependencies and pass the ordered
//values to the callback.
//Default to [require, exports, module] if no deps
deps = !deps.length && callback.length ? ['require', 'exports', 'module'] : deps;
for (i = 0; i < deps.length; i += 1) {
map = makeMap(deps[i], relParts);
depName = map.f;
//Fast path CommonJS standard dependencies.
if (depName === "require") {
args[i] = handlers.require(name);
} else if (depName === "exports") {
//CommonJS module spec 1.1
args[i] = handlers.exports(name);
usingExports = true;
} else if (depName === "module") {
//CommonJS module spec 1.1
cjsModule = args[i] = handlers.module(name);
} else if (hasProp(defined, depName) ||
hasProp(waiting, depName) ||
hasProp(defining, depName)) {
args[i] = callDep(depName);
} else if (map.p) {
map.p.load(map.n, makeRequire(relName, true), makeLoad(depName), {});
args[i] = defined[depName];
} else {
throw new Error(name + ' missing ' + depName);
}
}
ret = callback ? callback.apply(defined[name], args) : undefined;
if (name) {
//If setting exports via "module" is in play,
//favor that over return value and exports. After that,
//favor a non-undefined return value over exports use.
if (cjsModule && cjsModule.exports !== undef &&
cjsModule.exports !== defined[name]) {
defined[name] = cjsModule.exports;
} else if (ret !== undef || !usingExports) {
//Use the return value from the function.
defined[name] = ret;
}
}
} else if (name) {
//May just be an object definition for the module. Only
//worry about defining if have a module name.
defined[name] = callback;
}
};
requirejs = require = req = function (deps, callback, relName, forceSync, alt) {
if (typeof deps === "string") {
if (handlers[deps]) {
//callback in this case is really relName
return handlers[deps](callback);
}
//Just return the module wanted. In this scenario, the
//deps arg is the module name, and second arg (if passed)
//is just the relName.
//Normalize module name, if it contains . or ..
return callDep(makeMap(deps, makeRelParts(callback)).f);
} else if (!deps.splice) {
//deps is a config object, not an array.
config = deps;
if (config.deps) {
req(config.deps, config.callback);
}
if (!callback) {
return;
}
if (callback.splice) {
//callback is an array, which means it is a dependency list.
//Adjust args if there are dependencies
deps = callback;
callback = relName;
relName = null;
} else {
deps = undef;
}
}
//Support require(['a'])
callback = callback || function () {};
//If relName is a function, it is an errback handler,
//so remove it.
if (typeof relName === 'function') {
relName = forceSync;
forceSync = alt;
}
//Simulate async callback;
if (forceSync) {
main(undef, deps, callback, relName);
} else {
//Using a non-zero value because of concern for what old browsers
//do, and latest browsers "upgrade" to 4 if lower value is used:
//http://www.whatwg.org/specs/web-apps/current-work/multipage/timers.html#dom-windowtimers-settimeout:
//If want a value immediately, use require('id') instead -- something
//that works in almond on the global level, but not guaranteed and
//unlikely to work in other AMD implementations.
setTimeout(function () {
main(undef, deps, callback, relName);
}, 4);
}
return req;
};
/**
* Just drops the config on the floor, but returns req in case
* the config return value is used.
*/
req.config = function (cfg) {
return req(cfg);
};
/**
* Expose module registry for debugging and tooling
*/
requirejs._defined = defined;
define = function (name, deps, callback) {
if (typeof name !== 'string') {
throw new Error('See almond README: incorrect module build, no module name');
}
//This module may not have dependencies
if (!deps.splice) {
//deps is not an array, so probably means
//an object literal or factory function for
//the value. Adjust args.
callback = deps;
deps = [];
}
if (!hasProp(defined, name) && !hasProp(waiting, name)) {
waiting[name] = [name, deps, callback];
}
};
define.amd = {
jQuery: true
};
}());
2016-09-16 14:35:02 +02:00
define("almond", function(){});
/*!
* Sizzle CSS Selector Engine v2.2.1
* http://sizzlejs.com/
*
* Copyright jQuery Foundation and other contributors
* Released under the MIT license
* http://jquery.org/license
*
* Date: 2015-10-17
*/
(function( window ) {
var i,
support,
Expr,
getText,
isXML,
tokenize,
compile,
select,
outermostContext,
sortInput,
hasDuplicate,
// Local document vars
setDocument,
document,
docElem,
documentIsHTML,
rbuggyQSA,
rbuggyMatches,
matches,
contains,
// Instance-specific data
expando = "sizzle" + 1 * new Date(),
preferredDoc = window.document,
dirruns = 0,
done = 0,
classCache = createCache(),
tokenCache = createCache(),
compilerCache = createCache(),
sortOrder = function( a, b ) {
if ( a === b ) {
hasDuplicate = true;
}
return 0;
},
// General-purpose constants
MAX_NEGATIVE = 1 << 31,
// Instance methods
hasOwn = ({}).hasOwnProperty,
arr = [],
pop = arr.pop,
push_native = arr.push,
push = arr.push,
slice = arr.slice,
// Use a stripped-down indexOf as it's faster than native
// http://jsperf.com/thor-indexof-vs-for/5
indexOf = function( list, elem ) {
var i = 0,
len = list.length;
for ( ; i < len; i++ ) {
if ( list[i] === elem ) {
return i;
}
}
return -1;
},
booleans = "checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",
// Regular expressions
2016-11-07 15:43:48 +01:00
// http://www.w3.org/TR/css3-selectors/#whitespace
whitespace = "[\\x20\\t\\r\\n\\f]",
// http://www.w3.org/TR/CSS21/syndata.html#value-def-identifier
identifier = "(?:\\\\.|[\\w-]|[^\\x00-\\xa0])+",
2016-11-07 15:43:48 +01:00
// Attribute selectors: http://www.w3.org/TR/selectors/#attribute-selectors
attributes = "\\[" + whitespace + "*(" + identifier + ")(?:" + whitespace +
// Operator (capture 2)
"*([*^$|!~]?=)" + whitespace +
// "Attribute values must be CSS identifiers [capture 5] or strings [capture 3 or capture 4]"
"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|(" + identifier + "))|)" + whitespace +
"*\\]",
pseudos = ":(" + identifier + ")(?:\\((" +
// To reduce the number of selectors needing tokenize in the preFilter, prefer arguments:
// 1. quoted (capture 3; capture 4 or capture 5)
"('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|" +
// 2. simple (capture 6)
"((?:\\\\.|[^\\\\()[\\]]|" + attributes + ")*)|" +
// 3. anything else (capture 2)
".*" +
")\\)|)",
// Leading and non-escaped trailing whitespace, capturing some non-whitespace characters preceding the latter
rwhitespace = new RegExp( whitespace + "+", "g" ),
rtrim = new RegExp( "^" + whitespace + "+|((?:^|[^\\\\])(?:\\\\.)*)" + whitespace + "+$", "g" ),
rcomma = new RegExp( "^" + whitespace + "*," + whitespace + "*" ),
rcombinators = new RegExp( "^" + whitespace + "*([>+~]|" + whitespace + ")" + whitespace + "*" ),
rattributeQuotes = new RegExp( "=" + whitespace + "*([^\\]'\"]*?)" + whitespace + "*\\]", "g" ),
rpseudo = new RegExp( pseudos ),
ridentifier = new RegExp( "^" + identifier + "$" ),
matchExpr = {
"ID": new RegExp( "^#(" + identifier + ")" ),
"CLASS": new RegExp( "^\\.(" + identifier + ")" ),
"TAG": new RegExp( "^(" + identifier + "|[*])" ),
"ATTR": new RegExp( "^" + attributes ),
"PSEUDO": new RegExp( "^" + pseudos ),
"CHILD": new RegExp( "^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\(" + whitespace +
"*(even|odd|(([+-]|)(\\d*)n|)" + whitespace + "*(?:([+-]|)" + whitespace +
"*(\\d+)|))" + whitespace + "*\\)|)", "i" ),
"bool": new RegExp( "^(?:" + booleans + ")$", "i" ),
// For use in libraries implementing .is()
// We use this for POS matching in `select`
"needsContext": new RegExp( "^" + whitespace + "*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\(" +
whitespace + "*((?:-\\d)?\\d*)" + whitespace + "*\\)|)(?=[^-]|$)", "i" )
},
rinputs = /^(?:input|select|textarea|button)$/i,
rheader = /^h\d$/i,
rnative = /^[^{]+\{\s*\[native \w/,
// Easily-parseable/retrievable ID or TAG or CLASS selectors
rquickExpr = /^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,
rsibling = /[+~]/,
rescape = /'|\\/g,
// CSS escapes http://www.w3.org/TR/CSS21/syndata.html#escaped-characters
runescape = new RegExp( "\\\\([\\da-f]{1,6}" + whitespace + "?|(" + whitespace + ")|.)", "ig" ),
funescape = function( _, escaped, escapedWhitespace ) {
var high = "0x" + escaped - 0x10000;
// NaN means non-codepoint
// Support: Firefox<24
// Workaround erroneous numeric interpretation of +"0x"
return high !== high || escapedWhitespace ?
escaped :
high < 0 ?
// BMP codepoint
String.fromCharCode( high + 0x10000 ) :
// Supplemental Plane codepoint (surrogate pair)
String.fromCharCode( high >> 10 | 0xD800, high & 0x3FF | 0xDC00 );
},
// Used for iframes
// See setDocument()
// Removing the function wrapper causes a "Permission Denied"
// error in IE
unloadHandler = function() {
setDocument();
};
// Optimize for push.apply( _, NodeList )
try {
push.apply(
(arr = slice.call( preferredDoc.childNodes )),
preferredDoc.childNodes
);
// Support: Android<4.0
// Detect silently failing push.apply
arr[ preferredDoc.childNodes.length ].nodeType;
} catch ( e ) {
push = { apply: arr.length ?
// Leverage slice if possible
function( target, els ) {
push_native.apply( target, slice.call(els) );
} :
// Support: IE<9
// Otherwise append directly
function( target, els ) {
var j = target.length,
i = 0;
// Can't trust NodeList.length
while ( (target[j++] = els[i++]) ) {}
target.length = j - 1;
}
};
}
function Sizzle( selector, context, results, seed ) {
var m, i, elem, nid, nidselect, match, groups, newSelector,
newContext = context && context.ownerDocument,
// nodeType defaults to 9, since context defaults to document
nodeType = context ? context.nodeType : 9;
results = results || [];
// Return early from calls with invalid selector or context
if ( typeof selector !== "string" || !selector ||
nodeType !== 1 && nodeType !== 9 && nodeType !== 11 ) {
return results;
}
// Try to shortcut find operations (as opposed to filters) in HTML documents
if ( !seed ) {
if ( ( context ? context.ownerDocument || context : preferredDoc ) !== document ) {
setDocument( context );
}
context = context || document;
if ( documentIsHTML ) {
// If the selector is sufficiently simple, try using a "get*By*" DOM method
// (excepting DocumentFragment context, where the methods don't exist)
if ( nodeType !== 11 && (match = rquickExpr.exec( selector )) ) {
// ID selector
if ( (m = match[1]) ) {
// Document context
if ( nodeType === 9 ) {
if ( (elem = context.getElementById( m )) ) {
// Support: IE, Opera, Webkit
// TODO: identify versions
// getElementById can match elements by name instead of ID
if ( elem.id === m ) {
results.push( elem );
return results;
}
} else {
return results;
}
// Element context
} else {
// Support: IE, Opera, Webkit
// TODO: identify versions
// getElementById can match elements by name instead of ID
if ( newContext && (elem = newContext.getElementById( m )) &&
contains( context, elem ) &&
elem.id === m ) {
results.push( elem );
return results;
}
}
// Type selector
} else if ( match[2] ) {
push.apply( results, context.getElementsByTagName( selector ) );
return results;
// Class selector
} else if ( (m = match[3]) && support.getElementsByClassName &&
context.getElementsByClassName ) {
push.apply( results, context.getElementsByClassName( m ) );
return results;
}
}
// Take advantage of querySelectorAll
if ( support.qsa &&
!compilerCache[ selector + " " ] &&
(!rbuggyQSA || !rbuggyQSA.test( selector )) ) {
if ( nodeType !== 1 ) {
newContext = context;
newSelector = selector;
// qSA looks outside Element context, which is not what we want
// Thanks to Andrew Dupont for this workaround technique
// Support: IE <=8
// Exclude object elements
} else if ( context.nodeName.toLowerCase() !== "object" ) {
// Capture the context ID, setting it first if necessary
if ( (nid = context.getAttribute( "id" )) ) {
nid = nid.replace( rescape, "\\$&" );
} else {
context.setAttribute( "id", (nid = expando) );
}
// Prefix every selector in the list
groups = tokenize( selector );
i = groups.length;
nidselect = ridentifier.test( nid ) ? "#" + nid : "[id='" + nid + "']";
while ( i-- ) {
groups[i] = nidselect + " " + toSelector( groups[i] );
}
newSelector = groups.join( "," );
// Expand context for sibling selectors
newContext = rsibling.test( selector ) && testContext( context.parentNode ) ||
context;
}
if ( newSelector ) {
try {
push.apply( results,
newContext.querySelectorAll( newSelector )
);
return results;
} catch ( qsaError ) {
} finally {
if ( nid === expando ) {
context.removeAttribute( "id" );
}
}
}
}
}
}
// All others
return select( selector.replace( rtrim, "$1" ), context, results, seed );
}
/**
* Create key-value caches of limited size
* @returns {function(string, object)} Returns the Object data after storing it on itself with
* property name the (space-suffixed) string and (if the cache is larger than Expr.cacheLength)
* deleting the oldest entry
*/
function createCache() {
var keys = [];
function cache( key, value ) {
// Use (key + " ") to avoid collision with native prototype properties (see Issue #157)
if ( keys.push( key + " " ) > Expr.cacheLength ) {
// Only keep the most recent entries
delete cache[ keys.shift() ];
}
return (cache[ key + " " ] = value);
}
return cache;
}
/**
* Mark a function for special use by Sizzle
* @param {Function} fn The function to mark
*/
function markFunction( fn ) {
fn[ expando ] = true;
return fn;
}
/**
* Support testing using an element
* @param {Function} fn Passed the created div and expects a boolean result
*/
function assert( fn ) {
var div = document.createElement("div");
try {
return !!fn( div );
} catch (e) {
return false;
} finally {
// Remove from its parent by default
if ( div.parentNode ) {
div.parentNode.removeChild( div );
}
// release memory in IE
div = null;
}
}
/**
* Adds the same handler for all of the specified attrs
* @param {String} attrs Pipe-separated list of attributes
* @param {Function} handler The method that will be applied
*/
function addHandle( attrs, handler ) {
var arr = attrs.split("|"),
i = arr.length;
while ( i-- ) {
Expr.attrHandle[ arr[i] ] = handler;
}
}
/**
* Checks document order of two siblings
* @param {Element} a
* @param {Element} b
* @returns {Number} Returns less than 0 if a precedes b, greater than 0 if a follows b
*/
function siblingCheck( a, b ) {
var cur = b && a,
diff = cur && a.nodeType === 1 && b.nodeType === 1 &&
( ~b.sourceIndex || MAX_NEGATIVE ) -
( ~a.sourceIndex || MAX_NEGATIVE );
// Use IE sourceIndex if available on both nodes
if ( diff ) {
return diff;
}
// Check if b follows a
if ( cur ) {
while ( (cur = cur.nextSibling) ) {
if ( cur === b ) {
return -1;
}
}
}
return a ? 1 : -1;
}
/**
* Returns a function to use in pseudos for input types
* @param {String} type
*/
function createInputPseudo( type ) {
return function( elem ) {
var name = elem.nodeName.toLowerCase();
return name === "input" && elem.type === type;
};
}
/**
* Returns a function to use in pseudos for buttons
* @param {String} type
*/
function createButtonPseudo( type ) {
return function( elem ) {
var name = elem.nodeName.toLowerCase();
return (name === "input" || name === "button") && elem.type === type;
};
}
2017-02-03 13:51:07 +01:00
/**
* Returns a function to use in pseudos for positionals
* @param {Function} fn
*/
function createPositionalPseudo( fn ) {
return markFunction(function( argument ) {
argument = +argument;
return markFunction(function( seed, matches ) {
var j,
matchIndexes = fn( [], seed.length, argument ),
i = matchIndexes.length;
// Match elements found at the specified indexes
while ( i-- ) {
if ( seed[ (j = matchIndexes[i]) ] ) {
seed[j] = !(matches[j] = seed[j]);
}
}
});
});
}
/**
* Checks a node for validity as a Sizzle context
* @param {Element|Object=} context
* @returns {Element|Object|Boolean} The input node if acceptable, otherwise a falsy value
*/
function testContext( context ) {
return context && typeof context.getElementsByTagName !== "undefined" && context;
}
// Expose support vars for convenience
support = Sizzle.support = {};
2016-07-28 18:06:31 +02:00
/**
* Detects XML nodes
* @param {Element|Object} elem An element or a document
* @returns {Boolean} True iff elem is a non-HTML XML node
2017-02-03 13:51:07 +01:00
*/
isXML = Sizzle.isXML = function( elem ) {
// documentElement is verified for cases where it doesn't yet exist
// (such as loading iframes in IE - #4833)
var documentElement = elem && (elem.ownerDocument || elem).documentElement;
return documentElement ? documentElement.nodeName !== "HTML" : false;
};
2017-02-03 13:51:07 +01:00
/**
* Sets document-related variables once based on the current document
* @param {Element|Object} [doc] An element or document object to use to set the document
* @returns {Object} Returns the current document
*/
setDocument = Sizzle.setDocument = function( node ) {
var hasCompare, parent,
doc = node ? node.ownerDocument || node : preferredDoc;
2017-02-03 13:51:07 +01:00
// Return early if doc is invalid or already selected
if ( doc === document || doc.nodeType !== 9 || !doc.documentElement ) {
return document;
}
2017-02-03 13:51:07 +01:00
// Update global variables
document = doc;
docElem = document.documentElement;
documentIsHTML = !isXML( document );
// Support: IE 9-11, Edge
// Accessing iframe documents after unload throws "permission denied" errors (jQuery #13936)
if ( (parent = document.defaultView) && parent.top !== parent ) {
// Support: IE 11
if ( parent.addEventListener ) {
parent.addEventListener( "unload", unloadHandler, false );
// Support: IE 9 - 10 only
} else if ( parent.attachEvent ) {
parent.attachEvent( "onunload", unloadHandler );
2017-02-03 13:51:07 +01:00
}
}
2017-02-03 13:51:07 +01:00
/* Attributes
---------------------------------------------------------------------- */
2017-02-03 13:51:07 +01:00
// Support: IE<8
// Verify that getAttribute really returns attributes and not properties
// (excepting IE8 booleans)
support.attributes = assert(function( div ) {
div.className = "i";
return !div.getAttribute("className");
});
2017-02-03 13:51:07 +01:00
/* getElement(s)By*
---------------------------------------------------------------------- */
2017-02-03 13:51:07 +01:00
// Check if getElementsByTagName("*") returns only elements
support.getElementsByTagName = assert(function( div ) {
div.appendChild( document.createComment("") );
return !div.getElementsByTagName("*").length;
});
2017-02-03 13:51:07 +01:00
// Support: IE<9
support.getElementsByClassName = rnative.test( document.getElementsByClassName );
2017-02-03 13:51:07 +01:00
// Support: IE<10
// Check if getElementById returns elements by name
// The broken getElementById methods don't pick up programatically-set names,
// so use a roundabout getElementsByName test
support.getById = assert(function( div ) {
docElem.appendChild( div ).id = expando;
return !document.getElementsByName || !document.getElementsByName( expando ).length;
});
2017-02-03 13:51:07 +01:00
// ID find and filter
if ( support.getById ) {
Expr.find["ID"] = function( id, context ) {
if ( typeof context.getElementById !== "undefined" && documentIsHTML ) {
var m = context.getElementById( id );
return m ? [ m ] : [];
}
};
Expr.filter["ID"] = function( id ) {
var attrId = id.replace( runescape, funescape );
return function( elem ) {
return elem.getAttribute("id") === attrId;
};
};
} else {
// Support: IE6/7
// getElementById is not reliable as a find shortcut
delete Expr.find["ID"];
2017-02-03 13:51:07 +01:00
Expr.filter["ID"] = function( id ) {
var attrId = id.replace( runescape, funescape );
return function( elem ) {
var node = typeof elem.getAttributeNode !== "undefined" &&
elem.getAttributeNode("id");
return node && node.value === attrId;
};
};
}
2017-02-03 13:51:07 +01:00
// Tag
Expr.find["TAG"] = support.getElementsByTagName ?
function( tag, context ) {
if ( typeof context.getElementsByTagName !== "undefined" ) {
return context.getElementsByTagName( tag );
2017-02-03 13:51:07 +01:00
// DocumentFragment nodes don't have gEBTN
} else if ( support.qsa ) {
return context.querySelectorAll( tag );
}
} :
2017-02-03 13:51:07 +01:00
function( tag, context ) {
var elem,
tmp = [],
i = 0,
// By happy coincidence, a (broken) gEBTN appears on DocumentFragment nodes too
results = context.getElementsByTagName( tag );
2017-02-03 13:51:07 +01:00
// Filter out possible comments
if ( tag === "*" ) {
while ( (elem = results[i++]) ) {
if ( elem.nodeType === 1 ) {
tmp.push( elem );
}
}
2017-02-03 13:51:07 +01:00
return tmp;
}
return results;
};
2017-02-03 13:51:07 +01:00
// Class
Expr.find["CLASS"] = support.getElementsByClassName && function( className, context ) {
if ( typeof context.getElementsByClassName !== "undefined" && documentIsHTML ) {
return context.getElementsByClassName( className );
}
};
2017-02-03 13:51:07 +01:00
/* QSA/matchesSelector
---------------------------------------------------------------------- */
2017-02-03 13:51:07 +01:00
// QSA and matchesSelector support
2017-02-03 13:51:07 +01:00
// matchesSelector(:active) reports false when true (IE9/Opera 11.5)
rbuggyMatches = [];
2017-02-03 13:51:07 +01:00
// qSa(:focus) reports false when true (Chrome 21)
// We allow this because of a bug in IE8/9 that throws an error
// whenever `document.activeElement` is accessed on an iframe
// So, we allow :focus to pass through QSA all the time to avoid the IE error
// See http://bugs.jquery.com/ticket/13378
rbuggyQSA = [];
2017-02-03 13:51:07 +01:00
if ( (support.qsa = rnative.test( document.querySelectorAll )) ) {
// Build QSA regex
// Regex strategy adopted from Diego Perini
assert(function( div ) {
// Select is set to empty string on purpose
// This is to test IE's treatment of not explicitly
// setting a boolean content attribute,
// since its presence should be enough
// http://bugs.jquery.com/ticket/12359
docElem.appendChild( div ).innerHTML = "<a id='" + expando + "'></a>" +
"<select id='" + expando + "-\r\\' msallowcapture=''>" +
"<option selected=''></option></select>";
2017-02-03 13:51:07 +01:00
// Support: IE8, Opera 11-12.16
// Nothing should be selected when empty strings follow ^= or $= or *=
// The test attribute must be unknown in Opera but "safe" for WinRT
// http://msdn.microsoft.com/en-us/library/ie/hh465388.aspx#attribute_section
if ( div.querySelectorAll("[msallowcapture^='']").length ) {
rbuggyQSA.push( "[*^$]=" + whitespace + "*(?:''|\"\")" );
}
2017-02-03 13:51:07 +01:00
// Support: IE8
// Boolean attributes and "value" are not treated correctly
if ( !div.querySelectorAll("[selected]").length ) {
rbuggyQSA.push( "\\[" + whitespace + "*(?:value|" + booleans + ")" );
}
// Support: Chrome<29, Android<4.4, Safari<7.0+, iOS<7.0+, PhantomJS<1.9.8+
if ( !div.querySelectorAll( "[id~=" + expando + "-]" ).length ) {
rbuggyQSA.push("~=");
}
// Webkit/Opera - :checked should return selected option elements
// http://www.w3.org/TR/2011/REC-css3-selectors-20110929/#checked
// IE8 throws error here and will not see later tests
if ( !div.querySelectorAll(":checked").length ) {
rbuggyQSA.push(":checked");
}
// Support: Safari 8+, iOS 8+
// https://bugs.webkit.org/show_bug.cgi?id=136851
// In-page `selector#id sibing-combinator selector` fails
if ( !div.querySelectorAll( "a#" + expando + "+*" ).length ) {
rbuggyQSA.push(".#.+[+~]");
}
});
2017-02-03 13:51:07 +01:00
assert(function( div ) {
// Support: Windows 8 Native Apps
// The type and name attributes are restricted during .innerHTML assignment
var input = document.createElement("input");
input.setAttribute( "type", "hidden" );
div.appendChild( input ).setAttribute( "name", "D" );
2017-02-03 13:51:07 +01:00
// Support: IE8
// Enforce case-sensitivity of name attribute
if ( div.querySelectorAll("[name=d]").length ) {
rbuggyQSA.push( "name" + whitespace + "*[*^$|!~]?=" );
}
2017-02-03 13:51:07 +01:00
// FF 3.5 - :enabled/:disabled and hidden elements (hidden elements are still enabled)
// IE8 throws error here and will not see later tests
if ( !div.querySelectorAll(":enabled").length ) {
rbuggyQSA.push( ":enabled", ":disabled" );
}
2017-02-03 13:51:07 +01:00
// Opera 10-11 does not throw on post-comma invalid pseudos
div.querySelectorAll("*,:x");
rbuggyQSA.push(",.*:");
});
2017-02-03 13:51:07 +01:00
}
if ( (support.matchesSelector = rnative.test( (matches = docElem.matches ||
docElem.webkitMatchesSelector ||
docElem.mozMatchesSelector ||
docElem.oMatchesSelector ||
docElem.msMatchesSelector) )) ) {
2017-02-03 13:51:07 +01:00
assert(function( div ) {
// Check to see if it's possible to do matchesSelector
// on a disconnected node (IE 9)
support.disconnectedMatch = matches.call( div, "div" );
2017-02-03 13:51:07 +01:00
// This should fail with an exception
// Gecko does not error, returns false instead
matches.call( div, "[s!='']:x" );
rbuggyMatches.push( "!=", pseudos );
});
}
2017-02-03 13:51:07 +01:00
rbuggyQSA = rbuggyQSA.length && new RegExp( rbuggyQSA.join("|") );
rbuggyMatches = rbuggyMatches.length && new RegExp( rbuggyMatches.join("|") );
2017-02-03 13:51:07 +01:00
/* Contains
---------------------------------------------------------------------- */
hasCompare = rnative.test( docElem.compareDocumentPosition );
2017-02-03 13:51:07 +01:00
// Element contains another
// Purposefully self-exclusive
// As in, an element does not contain itself
contains = hasCompare || rnative.test( docElem.contains ) ?
function( a, b ) {
var adown = a.nodeType === 9 ? a.documentElement : a,
bup = b && b.parentNode;
return a === bup || !!( bup && bup.nodeType === 1 && (
adown.contains ?
adown.contains( bup ) :
a.compareDocumentPosition && a.compareDocumentPosition( bup ) & 16
));
} :
function( a, b ) {
if ( b ) {
while ( (b = b.parentNode) ) {
if ( b === a ) {
return true;
}
}
}
return false;
};
2017-02-03 13:51:07 +01:00
/* Sorting
---------------------------------------------------------------------- */
2017-02-03 13:51:07 +01:00
// Document order sorting
sortOrder = hasCompare ?
function( a, b ) {
2017-02-03 13:51:07 +01:00
// Flag for duplicate removal
if ( a === b ) {
hasDuplicate = true;
return 0;
}
2017-02-03 13:51:07 +01:00
// Sort on method existence if only one input has compareDocumentPosition
var compare = !a.compareDocumentPosition - !b.compareDocumentPosition;
if ( compare ) {
return compare;
}
2017-02-03 13:51:07 +01:00
// Calculate position if both inputs belong to the same document
compare = ( a.ownerDocument || a ) === ( b.ownerDocument || b ) ?
a.compareDocumentPosition( b ) :
2017-02-03 13:51:07 +01:00
// Otherwise we know they are disconnected
1;
2017-02-03 13:51:07 +01:00
// Disconnected nodes
if ( compare & 1 ||
(!support.sortDetached && b.compareDocumentPosition( a ) === compare) ) {
// Choose the first element that is related to our preferred document
if ( a === document || a.ownerDocument === preferredDoc && contains(preferredDoc, a) ) {
return -1;
}
if ( b === document || b.ownerDocument === preferredDoc && contains(preferredDoc, b) ) {
return 1;
2017-02-03 13:51:07 +01:00
}
// Maintain original order
return sortInput ?
( indexOf( sortInput, a ) - indexOf( sortInput, b ) ) :
0;
}
2017-02-03 13:51:07 +01:00
return compare & 4 ? -1 : 1;
} :
function( a, b ) {
// Exit early if the nodes are identical
if ( a === b ) {
hasDuplicate = true;
return 0;
}
2017-02-03 13:51:07 +01:00
var cur,
i = 0,
aup = a.parentNode,
bup = b.parentNode,
ap = [ a ],
bp = [ b ];
2017-02-03 13:51:07 +01:00
// Parentless nodes are either documents or disconnected
if ( !aup || !bup ) {
return a === document ? -1 :
b === document ? 1 :
aup ? -1 :
bup ? 1 :
sortInput ?
( indexOf( sortInput, a ) - indexOf( sortInput, b ) ) :
0;
2017-02-03 13:51:07 +01:00
// If the nodes are siblings, we can do a quick check
} else if ( aup === bup ) {
return siblingCheck( a, b );
}
2017-02-03 13:51:07 +01:00
// Otherwise we need full lists of their ancestors for comparison
cur = a;
while ( (cur = cur.parentNode) ) {
ap.unshift( cur );
}
cur = b;
while ( (cur = cur.parentNode) ) {
bp.unshift( cur );
}
2017-02-03 13:51:07 +01:00
// Walk down the tree looking for a discrepancy
while ( ap[i] === bp[i] ) {
i++;
2017-02-03 13:51:07 +01:00
}
return i ?
// Do a sibling check if the nodes have a common ancestor
siblingCheck( ap[i], bp[i] ) :
// Otherwise nodes in our document sort first
ap[i] === preferredDoc ? -1 :
bp[i] === preferredDoc ? 1 :
0;
};
return document;
};
Sizzle.matches = function( expr, elements ) {
return Sizzle( expr, null, null, elements );
};
Sizzle.matchesSelector = function( elem, expr ) {
// Set document vars if needed
if ( ( elem.ownerDocument || elem ) !== document ) {
setDocument( elem );
2017-02-03 13:51:07 +01:00
}
// Make sure that attribute selectors are quoted
expr = expr.replace( rattributeQuotes, "='$1']" );
2017-02-03 13:51:07 +01:00
if ( support.matchesSelector && documentIsHTML &&
!compilerCache[ expr + " " ] &&
( !rbuggyMatches || !rbuggyMatches.test( expr ) ) &&
( !rbuggyQSA || !rbuggyQSA.test( expr ) ) ) {
2017-02-03 13:51:07 +01:00
try {
var ret = matches.call( elem, expr );
// IE 9's matchesSelector returns false on disconnected nodes
if ( ret || support.disconnectedMatch ||
// As well, disconnected nodes are said to be in a document
// fragment in IE 9
elem.document && elem.document.nodeType !== 11 ) {
return ret;
}
} catch (e) {}
2017-02-03 13:51:07 +01:00
}
return Sizzle( expr, document, null, [ elem ] ).length > 0;
};
2017-02-03 13:51:07 +01:00
Sizzle.contains = function( context, elem ) {
// Set document vars if needed
if ( ( context.ownerDocument || context ) !== document ) {
setDocument( context );
}
return contains( context, elem );
};
2017-02-03 13:51:07 +01:00
Sizzle.attr = function( elem, name ) {
// Set document vars if needed
if ( ( elem.ownerDocument || elem ) !== document ) {
setDocument( elem );
2017-02-03 13:51:07 +01:00
}
var fn = Expr.attrHandle[ name.toLowerCase() ],
// Don't get fooled by Object.prototype properties (jQuery #13807)
val = fn && hasOwn.call( Expr.attrHandle, name.toLowerCase() ) ?
fn( elem, name, !documentIsHTML ) :
undefined;
2017-02-03 13:51:07 +01:00
return val !== undefined ?
val :
support.attributes || !documentIsHTML ?
elem.getAttribute( name ) :
(val = elem.getAttributeNode(name)) && val.specified ?
val.value :
null;
};
Sizzle.error = function( msg ) {
throw new Error( "Syntax error, unrecognized expression: " + msg );
};
2017-02-03 13:51:07 +01:00
/**
* Document sorting and removing duplicates
* @param {ArrayLike} results
2017-02-03 13:51:07 +01:00
*/
Sizzle.uniqueSort = function( results ) {
var elem,
duplicates = [],
j = 0,
i = 0;
2017-02-03 13:51:07 +01:00
// Unless we *know* we can detect duplicates, assume their presence
hasDuplicate = !support.detectDuplicates;
sortInput = !support.sortStable && results.slice( 0 );
results.sort( sortOrder );
2017-02-03 13:51:07 +01:00
if ( hasDuplicate ) {
while ( (elem = results[i++]) ) {
if ( elem === results[ i ] ) {
j = duplicates.push( i );
2017-02-03 13:51:07 +01:00
}
}
while ( j-- ) {
results.splice( duplicates[ j ], 1 );
}
2017-02-03 13:51:07 +01:00
}
// Clear input after sorting to release objects
// See https://github.com/jquery/sizzle/pull/225
sortInput = null;
2017-02-03 13:51:07 +01:00
return results;
};
2017-02-03 13:51:07 +01:00
/**
* Utility function for retrieving the text value of an array of DOM nodes
* @param {Array|Element} elem
2017-02-03 13:51:07 +01:00
*/
getText = Sizzle.getText = function( elem ) {
var node,
ret = "",
i = 0,
nodeType = elem.nodeType;
2017-02-03 13:51:07 +01:00
if ( !nodeType ) {
// If no nodeType, this is expected to be an array
while ( (node = elem[i++]) ) {
// Do not traverse comment nodes
ret += getText( node );
}
} else if ( nodeType === 1 || nodeType === 9 || nodeType === 11 ) {
// Use textContent for elements
// innerText usage removed for consistency of new lines (jQuery #11153)
if ( typeof elem.textContent === "string" ) {
return elem.textContent;
} else {
// Traverse its children
for ( elem = elem.firstChild; elem; elem = elem.nextSibling ) {
ret += getText( elem );
2017-02-03 13:51:07 +01:00
}
}
} else if ( nodeType === 3 || nodeType === 4 ) {
return elem.nodeValue;
}
// Do not include comment or processing instruction nodes
2017-02-03 13:51:07 +01:00
return ret;
2017-02-03 13:51:07 +01:00
};
Expr = Sizzle.selectors = {
2017-02-03 13:51:07 +01:00
// Can be adjusted by the user
cacheLength: 50,
createPseudo: markFunction,
2017-02-03 13:51:07 +01:00
match: matchExpr,
2017-02-03 13:51:07 +01:00
attrHandle: {},
2017-02-03 13:51:07 +01:00
find: {},
2017-02-03 13:51:07 +01:00
relative: {
">": { dir: "parentNode", first: true },
" ": { dir: "parentNode" },
"+": { dir: "previousSibling", first: true },
"~": { dir: "previousSibling" }
},
2017-02-03 13:51:07 +01:00
preFilter: {
"ATTR": function( match ) {
match[1] = match[1].replace( runescape, funescape );
2017-02-03 13:51:07 +01:00
// Move the given value to match[3] whether quoted or unquoted
match[3] = ( match[3] || match[4] || match[5] || "" ).replace( runescape, funescape );
2017-02-03 13:51:07 +01:00
if ( match[2] === "~=" ) {
match[3] = " " + match[3] + " ";
}
2017-02-03 13:51:07 +01:00
return match.slice( 0, 4 );
},
2017-02-03 13:51:07 +01:00
"CHILD": function( match ) {
/* matches from matchExpr["CHILD"]
1 type (only|nth|...)
2 what (child|of-type)
3 argument (even|odd|\d*|\d*n([+-]\d+)?|...)
4 xn-component of xn+y argument ([+-]?\d*n|)
5 sign of xn-component
6 x of xn-component
7 sign of y-component
8 y of y-component
*/
match[1] = match[1].toLowerCase();
2017-02-03 13:51:07 +01:00
if ( match[1].slice( 0, 3 ) === "nth" ) {
// nth-* requires argument
if ( !match[3] ) {
Sizzle.error( match[0] );
}
// numeric x and y parameters for Expr.filter.CHILD
// remember that false/true cast respectively to 0/1
match[4] = +( match[4] ? match[5] + (match[6] || 1) : 2 * ( match[3] === "even" || match[3] === "odd" ) );
match[5] = +( ( match[7] + match[8] ) || match[3] === "odd" );
// other types prohibit arguments
} else if ( match[3] ) {
Sizzle.error( match[0] );
2017-02-03 13:51:07 +01:00
}
return match;
},
2017-02-03 13:51:07 +01:00
"PSEUDO": function( match ) {
var excess,
unquoted = !match[6] && match[2];
2017-02-03 13:51:07 +01:00
if ( matchExpr["CHILD"].test( match[0] ) ) {
return null;
2017-02-03 13:51:07 +01:00
}
// Accept quoted arguments as-is
if ( match[3] ) {
match[2] = match[4] || match[5] || "";
2017-02-03 13:51:07 +01:00
// Strip excess characters from unquoted arguments
} else if ( unquoted && rpseudo.test( unquoted ) &&
// Get excess from tokenize (recursively)
(excess = tokenize( unquoted, true )) &&
// advance to the next closing parenthesis
(excess = unquoted.indexOf( ")", unquoted.length - excess ) - unquoted.length) ) {
2017-02-03 13:51:07 +01:00
// excess is a negative index
match[0] = match[0].slice( 0, excess );
match[2] = unquoted.slice( 0, excess );
2017-02-03 13:51:07 +01:00
}
// Return only captures needed by the pseudo filter method (type and argument)
return match.slice( 0, 3 );
2017-02-03 13:51:07 +01:00
}
},
2017-02-03 13:51:07 +01:00
filter: {
2017-02-03 13:51:07 +01:00
"TAG": function( nodeNameSelector ) {
var nodeName = nodeNameSelector.replace( runescape, funescape ).toLowerCase();
return nodeNameSelector === "*" ?
function() { return true; } :
function( elem ) {
return elem.nodeName && elem.nodeName.toLowerCase() === nodeName;
};
},
2017-02-03 13:51:07 +01:00
"CLASS": function( className ) {
var pattern = classCache[ className + " " ];
2017-02-03 13:51:07 +01:00
return pattern ||
(pattern = new RegExp( "(^|" + whitespace + ")" + className + "(" + whitespace + "|$)" )) &&
classCache( className, function( elem ) {
return pattern.test( typeof elem.className === "string" && elem.className || typeof elem.getAttribute !== "undefined" && elem.getAttribute("class") || "" );
});
},
2017-02-03 13:51:07 +01:00
"ATTR": function( name, operator, check ) {
return function( elem ) {
var result = Sizzle.attr( elem, name );
2017-02-03 13:51:07 +01:00
if ( result == null ) {
return operator === "!=";
}
if ( !operator ) {
return true;
}
2017-02-03 13:51:07 +01:00
result += "";
2017-02-03 13:51:07 +01:00
return operator === "=" ? result === check :
operator === "!=" ? result !== check :
operator === "^=" ? check && result.indexOf( check ) === 0 :
operator === "*=" ? check && result.indexOf( check ) > -1 :
operator === "$=" ? check && result.slice( -check.length ) === check :
operator === "~=" ? ( " " + result.replace( rwhitespace, " " ) + " " ).indexOf( check ) > -1 :
operator === "|=" ? result === check || result.slice( 0, check.length + 1 ) === check + "-" :
false;
};
},
2017-02-03 13:51:07 +01:00
"CHILD": function( type, what, argument, first, last ) {
var simple = type.slice( 0, 3 ) !== "nth",
forward = type.slice( -4 ) !== "last",
ofType = what === "of-type";
2017-02-03 13:51:07 +01:00
return first === 1 && last === 0 ?
2017-02-03 13:51:07 +01:00
// Shortcut for :nth-*(n)
function( elem ) {
return !!elem.parentNode;
} :
2017-02-03 13:51:07 +01:00
function( elem, context, xml ) {
var cache, uniqueCache, outerCache, node, nodeIndex, start,
dir = simple !== forward ? "nextSibling" : "previousSibling",
parent = elem.parentNode,
name = ofType && elem.nodeName.toLowerCase(),
useCache = !xml && !ofType,
diff = false;
2017-02-03 13:51:07 +01:00
if ( parent ) {
2017-02-03 13:51:07 +01:00
// :(first|last|only)-(child|of-type)
if ( simple ) {
while ( dir ) {
node = elem;
while ( (node = node[ dir ]) ) {
if ( ofType ?
node.nodeName.toLowerCase() === name :
node.nodeType === 1 ) {
2017-02-03 13:51:07 +01:00
return false;
}
}
// Reverse direction for :only-* (if we haven't yet done so)
start = dir = type === "only" && !start && "nextSibling";
}
return true;
}
2017-02-03 13:51:07 +01:00
start = [ forward ? parent.firstChild : parent.lastChild ];
2017-02-03 13:51:07 +01:00
// non-xml :nth-child(...) stores cache data on `parent`
if ( forward && useCache ) {
2017-02-03 13:51:07 +01:00
// Seek `elem` from a previously-cached index
2017-02-03 13:51:07 +01:00
// ...in a gzip-friendly way
node = parent;
outerCache = node[ expando ] || (node[ expando ] = {});
2017-02-03 13:51:07 +01:00
// Support: IE <9 only
// Defend against cloned attroperties (jQuery gh-1709)
uniqueCache = outerCache[ node.uniqueID ] ||
(outerCache[ node.uniqueID ] = {});
2017-02-03 13:51:07 +01:00
cache = uniqueCache[ type ] || [];
nodeIndex = cache[ 0 ] === dirruns && cache[ 1 ];
diff = nodeIndex && cache[ 2 ];
node = nodeIndex && parent.childNodes[ nodeIndex ];
2017-02-03 13:51:07 +01:00
while ( (node = ++nodeIndex && node && node[ dir ] ||
2017-02-03 13:51:07 +01:00
// Fallback to seeking `elem` from the start
(diff = nodeIndex = 0) || start.pop()) ) {
2017-02-03 13:51:07 +01:00
// When found, cache indexes on `parent` and break
if ( node.nodeType === 1 && ++diff && node === elem ) {
uniqueCache[ type ] = [ dirruns, nodeIndex, diff ];
break;
}
}
2017-02-03 13:51:07 +01:00
} else {
// Use previously-cached element index if available
if ( useCache ) {
// ...in a gzip-friendly way
node = elem;
outerCache = node[ expando ] || (node[ expando ] = {});
2017-02-03 13:51:07 +01:00
// Support: IE <9 only
// Defend against cloned attroperties (jQuery gh-1709)
uniqueCache = outerCache[ node.uniqueID ] ||
(outerCache[ node.uniqueID ] = {});
2017-02-03 13:51:07 +01:00
cache = uniqueCache[ type ] || [];
nodeIndex = cache[ 0 ] === dirruns && cache[ 1 ];
diff = nodeIndex;
}
2017-02-03 13:51:07 +01:00
// xml :nth-child(...)
// or :nth-last-child(...) or :nth(-last)?-of-type(...)
if ( diff === false ) {
// Use the same loop as above to seek `elem` from the start
while ( (node = ++nodeIndex && node && node[ dir ] ||
(diff = nodeIndex = 0) || start.pop()) ) {
2017-02-03 13:51:07 +01:00
if ( ( ofType ?
node.nodeName.toLowerCase() === name :
node.nodeType === 1 ) &&
++diff ) {
2017-02-03 13:51:07 +01:00
// Cache the index of each encountered element
if ( useCache ) {
outerCache = node[ expando ] || (node[ expando ] = {});
2017-02-03 13:51:07 +01:00
// Support: IE <9 only
// Defend against cloned attroperties (jQuery gh-1709)
uniqueCache = outerCache[ node.uniqueID ] ||
(outerCache[ node.uniqueID ] = {});
2017-02-03 13:51:07 +01:00
uniqueCache[ type ] = [ dirruns, diff ];
}
2017-02-03 13:51:07 +01:00
if ( node === elem ) {
break;
}
}
}
}
}
2017-02-03 13:51:07 +01:00
// Incorporate the offset, then check against cycle size
diff -= last;
return diff === first || ( diff % first === 0 && diff / first >= 0 );
}
};
},
2017-02-03 13:51:07 +01:00
"PSEUDO": function( pseudo, argument ) {
// pseudo-class names are case-insensitive
// http://www.w3.org/TR/selectors/#pseudo-classes
// Prioritize by case sensitivity in case custom pseudos are added with uppercase letters
// Remember that setFilters inherits from pseudos
var args,
fn = Expr.pseudos[ pseudo ] || Expr.setFilters[ pseudo.toLowerCase() ] ||
Sizzle.error( "unsupported pseudo: " + pseudo );
// The user may use createPseudo to indicate that
// arguments are needed to create the filter function
// just as Sizzle does
if ( fn[ expando ] ) {
return fn( argument );
}
2017-02-03 13:51:07 +01:00
// But maintain support for old signatures
if ( fn.length > 1 ) {
args = [ pseudo, pseudo, "", argument ];
return Expr.setFilters.hasOwnProperty( pseudo.toLowerCase() ) ?
markFunction(function( seed, matches ) {
var idx,
matched = fn( seed, argument ),
i = matched.length;
while ( i-- ) {
idx = indexOf( seed, matched[i] );
seed[ idx ] = !( matches[ idx ] = matched[i] );
}
}) :
function( elem ) {
return fn( elem, 0, args );
};
}
2017-02-03 13:51:07 +01:00
return fn;
}
},
2017-02-03 13:51:07 +01:00
pseudos: {
// Potentially complex pseudos
"not": markFunction(function( selector ) {
// Trim the selector passed to compile
// to avoid treating leading and trailing
// spaces as combinators
var input = [],
results = [],
matcher = compile( selector.replace( rtrim, "$1" ) );
2017-02-03 13:51:07 +01:00
return matcher[ expando ] ?
markFunction(function( seed, matches, context, xml ) {
var elem,
unmatched = matcher( seed, null, xml, [] ),
i = seed.length;
2017-02-03 13:51:07 +01:00
// Match elements unmatched by `matcher`
while ( i-- ) {
if ( (elem = unmatched[i]) ) {
seed[i] = !(matches[i] = elem);
}
}
}) :
function( elem, context, xml ) {
input[0] = elem;
matcher( input, null, xml, results );
// Don't keep the element (issue #299)
input[0] = null;
return !results.pop();
};
}),
2017-02-03 13:51:07 +01:00
"has": markFunction(function( selector ) {
return function( elem ) {
return Sizzle( selector, elem ).length > 0;
};
}),
2017-02-03 13:51:07 +01:00
"contains": markFunction(function( text ) {
text = text.replace( runescape, funescape );
return function( elem ) {
return ( elem.textContent || elem.innerText || getText( elem ) ).indexOf( text ) > -1;
};
}),
2017-02-03 13:51:07 +01:00
// "Whether an element is represented by a :lang() selector
// is based solely on the element's language value
// being equal to the identifier C,
// or beginning with the identifier C immediately followed by "-".
// The matching of C against the element's language value is performed case-insensitively.
// The identifier C does not have to be a valid language name."
// http://www.w3.org/TR/selectors/#lang-pseudo
"lang": markFunction( function( lang ) {
// lang value must be a valid identifier
if ( !ridentifier.test(lang || "") ) {
Sizzle.error( "unsupported lang: " + lang );
2017-02-03 13:51:07 +01:00
}
lang = lang.replace( runescape, funescape ).toLowerCase();
return function( elem ) {
var elemLang;
do {
if ( (elemLang = documentIsHTML ?
elem.lang :
elem.getAttribute("xml:lang") || elem.getAttribute("lang")) ) {
2017-02-03 13:51:07 +01:00
elemLang = elemLang.toLowerCase();
return elemLang === lang || elemLang.indexOf( lang + "-" ) === 0;
}
} while ( (elem = elem.parentNode) && elem.nodeType === 1 );
return false;
};
}),
2017-02-03 13:51:07 +01:00
// Miscellaneous
"target": function( elem ) {
var hash = window.location && window.location.hash;
return hash && hash.slice( 1 ) === elem.id;
},
2017-02-03 13:51:07 +01:00
"root": function( elem ) {
return elem === docElem;
},
2017-02-03 13:51:07 +01:00
"focus": function( elem ) {
return elem === document.activeElement && (!document.hasFocus || document.hasFocus()) && !!(elem.type || elem.href || ~elem.tabIndex);
},
2017-02-03 13:51:07 +01:00
// Boolean properties
"enabled": function( elem ) {
return elem.disabled === false;
},
2017-02-03 13:51:07 +01:00
"disabled": function( elem ) {
return elem.disabled === true;
},
2017-02-03 13:51:07 +01:00
"checked": function( elem ) {
// In CSS3, :checked should return both checked and selected elements
// http://www.w3.org/TR/2011/REC-css3-selectors-20110929/#checked
var nodeName = elem.nodeName.toLowerCase();
return (nodeName === "input" && !!elem.checked) || (nodeName === "option" && !!elem.selected);
},
2017-02-03 13:51:07 +01:00
"selected": function( elem ) {
// Accessing this property makes selected-by-default
// options in Safari work properly
if ( elem.parentNode ) {
elem.parentNode.selectedIndex;
2017-02-03 13:51:07 +01:00
}
return elem.selected === true;
},
2017-02-03 13:51:07 +01:00
// Contents
"empty": function( elem ) {
// http://www.w3.org/TR/selectors/#empty-pseudo
// :empty is negated by element (1) or content nodes (text: 3; cdata: 4; entity ref: 5),
// but not by others (comment: 8; processing instruction: 7; etc.)
// nodeType < 6 works because attributes (2) do not appear as children
2017-02-03 13:51:07 +01:00
for ( elem = elem.firstChild; elem; elem = elem.nextSibling ) {
if ( elem.nodeType < 6 ) {
return false;
}
2017-02-03 13:51:07 +01:00
}
return true;
},
"parent": function( elem ) {
return !Expr.pseudos["empty"]( elem );
},
// Element/input types
"header": function( elem ) {
return rheader.test( elem.nodeName );
},
2017-02-03 13:51:07 +01:00
"input": function( elem ) {
return rinputs.test( elem.nodeName );
},
2017-02-03 13:51:07 +01:00
"button": function( elem ) {
var name = elem.nodeName.toLowerCase();
return name === "input" && elem.type === "button" || name === "button";
},
2017-02-03 13:51:07 +01:00
"text": function( elem ) {
var attr;
return elem.nodeName.toLowerCase() === "input" &&
elem.type === "text" &&
2017-02-03 13:51:07 +01:00
// Support: IE<8
// New HTML5 attribute values (e.g., "search") appear with elem.type === "text"
( (attr = elem.getAttribute("type")) == null || attr.toLowerCase() === "text" );
},
2017-02-03 13:51:07 +01:00
// Position-in-collection
"first": createPositionalPseudo(function() {
return [ 0 ];
}),
2017-02-03 13:51:07 +01:00
"last": createPositionalPseudo(function( matchIndexes, length ) {
return [ length - 1 ];
}),
2017-02-03 13:51:07 +01:00
"eq": createPositionalPseudo(function( matchIndexes, length, argument ) {
return [ argument < 0 ? argument + length : argument ];
}),
2017-02-03 13:51:07 +01:00
"even": createPositionalPseudo(function( matchIndexes, length ) {
var i = 0;
for ( ; i < length; i += 2 ) {
matchIndexes.push( i );
}
return matchIndexes;
}),
2017-02-03 13:51:07 +01:00
"odd": createPositionalPseudo(function( matchIndexes, length ) {
var i = 1;
for ( ; i < length; i += 2 ) {
matchIndexes.push( i );
}
return matchIndexes;
}),
2017-02-03 13:51:07 +01:00
"lt": createPositionalPseudo(function( matchIndexes, length, argument ) {
var i = argument < 0 ? argument + length : argument;
for ( ; --i >= 0; ) {
matchIndexes.push( i );
}
return matchIndexes;
}),
2017-02-03 13:51:07 +01:00
"gt": createPositionalPseudo(function( matchIndexes, length, argument ) {
var i = argument < 0 ? argument + length : argument;
for ( ; ++i < length; ) {
matchIndexes.push( i );
2017-02-03 13:51:07 +01:00
}
return matchIndexes;
})
}
};
2017-02-03 13:51:07 +01:00
Expr.pseudos["nth"] = Expr.pseudos["eq"];
2017-02-03 13:51:07 +01:00
// Add button/input type pseudos
for ( i in { radio: true, checkbox: true, file: true, password: true, image: true } ) {
Expr.pseudos[ i ] = createInputPseudo( i );
}
for ( i in { submit: true, reset: true } ) {
Expr.pseudos[ i ] = createButtonPseudo( i );
}
2017-02-03 13:51:07 +01:00
// Easy API for creating new setFilters
function setFilters() {}
setFilters.prototype = Expr.filters = Expr.pseudos;
Expr.setFilters = new setFilters();
2017-02-03 13:51:07 +01:00
tokenize = Sizzle.tokenize = function( selector, parseOnly ) {
var matched, match, tokens, type,
soFar, groups, preFilters,
cached = tokenCache[ selector + " " ];
2017-02-03 13:51:07 +01:00
if ( cached ) {
return parseOnly ? 0 : cached.slice( 0 );
}
2017-02-03 13:51:07 +01:00
soFar = selector;
groups = [];
preFilters = Expr.preFilter;
2017-02-03 13:51:07 +01:00
while ( soFar ) {
2017-02-03 13:51:07 +01:00
// Comma and first run
if ( !matched || (match = rcomma.exec( soFar )) ) {
if ( match ) {
// Don't consume trailing commas as valid
soFar = soFar.slice( match[0].length ) || soFar;
2017-02-03 13:51:07 +01:00
}
groups.push( (tokens = []) );
}
2017-02-03 13:51:07 +01:00
matched = false;
2017-02-03 13:51:07 +01:00
// Combinators
if ( (match = rcombinators.exec( soFar )) ) {
matched = match.shift();
tokens.push({
value: matched,
// Cast descendant combinators to space
type: match[0].replace( rtrim, " " )
});
soFar = soFar.slice( matched.length );
}
2017-02-03 13:51:07 +01:00
// Filters
for ( type in Expr.filter ) {
if ( (match = matchExpr[ type ].exec( soFar )) && (!preFilters[ type ] ||
(match = preFilters[ type ]( match ))) ) {
matched = match.shift();
tokens.push({
value: matched,
type: type,
matches: match
});
soFar = soFar.slice( matched.length );
2017-02-03 13:51:07 +01:00
}
}
2017-02-03 13:51:07 +01:00
if ( !matched ) {
break;
2017-02-03 13:51:07 +01:00
}
}
2017-02-03 13:51:07 +01:00
// Return the length of the invalid excess
// if we're just parsing
// Otherwise, throw an error or return tokens
return parseOnly ?
soFar.length :
soFar ?
Sizzle.error( selector ) :
// Cache the tokens
tokenCache( selector, groups ).slice( 0 );
};
2017-02-03 13:51:07 +01:00
function toSelector( tokens ) {
var i = 0,
len = tokens.length,
selector = "";
for ( ; i < len; i++ ) {
selector += tokens[i].value;
}
return selector;
}
function addCombinator( matcher, combinator, base ) {
var dir = combinator.dir,
checkNonElements = base && dir === "parentNode",
doneName = done++;
return combinator.first ?
// Check against closest ancestor/preceding element
function( elem, context, xml ) {
while ( (elem = elem[ dir ]) ) {
if ( elem.nodeType === 1 || checkNonElements ) {
return matcher( elem, context, xml );
}
}
} :
// Check against all ancestor/preceding elements
function( elem, context, xml ) {
var oldCache, uniqueCache, outerCache,
newCache = [ dirruns, doneName ];
// We can't set arbitrary data on XML nodes, so they don't benefit from combinator caching
if ( xml ) {
while ( (elem = elem[ dir ]) ) {
if ( elem.nodeType === 1 || checkNonElements ) {
if ( matcher( elem, context, xml ) ) {
return true;
}
}
}
} else {
while ( (elem = elem[ dir ]) ) {
if ( elem.nodeType === 1 || checkNonElements ) {
outerCache = elem[ expando ] || (elem[ expando ] = {});
// Support: IE <9 only
// Defend against cloned attroperties (jQuery gh-1709)
uniqueCache = outerCache[ elem.uniqueID ] || (outerCache[ elem.uniqueID ] = {});
if ( (oldCache = uniqueCache[ dir ]) &&
oldCache[ 0 ] === dirruns && oldCache[ 1 ] === doneName ) {
// Assign to newCache so results back-propagate to previous elements
return (newCache[ 2 ] = oldCache[ 2 ]);
} else {
// Reuse newcache so results back-propagate to previous elements
uniqueCache[ dir ] = newCache;
2017-02-03 13:51:07 +01:00
// A match means we're done; a fail means we have to keep checking
if ( (newCache[ 2 ] = matcher( elem, context, xml )) ) {
return true;
}
}
}
}
}
};
}
2017-02-03 13:51:07 +01:00
function elementMatcher( matchers ) {
return matchers.length > 1 ?
function( elem, context, xml ) {
var i = matchers.length;
while ( i-- ) {
if ( !matchers[i]( elem, context, xml ) ) {
return false;
}
}
return true;
} :
matchers[0];
}
2017-02-03 13:51:07 +01:00
function multipleContexts( selector, contexts, results ) {
var i = 0,
len = contexts.length;
for ( ; i < len; i++ ) {
Sizzle( selector, contexts[i], results );
}
return results;
}
2017-02-03 13:51:07 +01:00
function condense( unmatched, map, filter, context, xml ) {
var elem,
newUnmatched = [],
i = 0,
len = unmatched.length,
mapped = map != null;
for ( ; i < len; i++ ) {
if ( (elem = unmatched[i]) ) {
if ( !filter || filter( elem, context, xml ) ) {
newUnmatched.push( elem );
if ( mapped ) {
map.push( i );
2017-02-03 13:51:07 +01:00
}
}
}
}
2017-02-03 13:51:07 +01:00
return newUnmatched;
}
2017-02-03 13:51:07 +01:00
function setMatcher( preFilter, selector, matcher, postFilter, postFinder, postSelector ) {
if ( postFilter && !postFilter[ expando ] ) {
postFilter = setMatcher( postFilter );
}
if ( postFinder && !postFinder[ expando ] ) {
postFinder = setMatcher( postFinder, postSelector );
}
return markFunction(function( seed, results, context, xml ) {
var temp, i, elem,
preMap = [],
postMap = [],
preexisting = results.length,
2017-02-03 13:51:07 +01:00
// Get initial elements from seed or context
elems = seed || multipleContexts( selector || "*", context.nodeType ? [ context ] : context, [] ),
2017-02-03 13:51:07 +01:00
// Prefilter to get matcher input, preserving a map for seed-results synchronization
matcherIn = preFilter && ( seed || !selector ) ?
condense( elems, preMap, preFilter, context, xml ) :
elems,
2017-02-03 13:51:07 +01:00
matcherOut = matcher ?
// If we have a postFinder, or filtered seed, or non-seed postFilter or preexisting results,
postFinder || ( seed ? preFilter : preexisting || postFilter ) ?
2017-02-03 13:51:07 +01:00
// ...intermediate processing is necessary
[] :
2017-02-03 13:51:07 +01:00
// ...otherwise use results directly
results :
matcherIn;
2017-02-03 13:51:07 +01:00
// Find primary matches
if ( matcher ) {
matcher( matcherIn, matcherOut, context, xml );
}
2017-02-03 13:51:07 +01:00
// Apply postFilter
if ( postFilter ) {
temp = condense( matcherOut, postMap );
postFilter( temp, [], context, xml );
2017-02-03 13:51:07 +01:00
// Un-match failing elements by moving them back to matcherIn
i = temp.length;
while ( i-- ) {
if ( (elem = temp[i]) ) {
matcherOut[ postMap[i] ] = !(matcherIn[ postMap[i] ] = elem);
}
}
}
2017-02-03 13:51:07 +01:00
if ( seed ) {
if ( postFinder || preFilter ) {
if ( postFinder ) {
// Get the final matcherOut by condensing this intermediate into postFinder contexts
temp = [];
i = matcherOut.length;
while ( i-- ) {
if ( (elem = matcherOut[i]) ) {
// Restore matcherIn since elem is not yet a final match
temp.push( (matcherIn[i] = elem) );
}
}
postFinder( null, (matcherOut = []), temp, xml );
}
2017-02-03 13:51:07 +01:00
// Move matched elements from seed to results to keep them synchronized
i = matcherOut.length;
while ( i-- ) {
if ( (elem = matcherOut[i]) &&
(temp = postFinder ? indexOf( seed, elem ) : preMap[i]) > -1 ) {
2017-02-03 13:51:07 +01:00
seed[temp] = !(results[temp] = elem);
}
}
}
2017-02-03 13:51:07 +01:00
// Add elements to results, through postFinder if defined
} else {
matcherOut = condense(
matcherOut === results ?
matcherOut.splice( preexisting, matcherOut.length ) :
matcherOut
);
if ( postFinder ) {
postFinder( null, results, matcherOut, xml );
} else {
push.apply( results, matcherOut );
}
}
});
}
2017-02-03 13:51:07 +01:00
function matcherFromTokens( tokens ) {
var checkContext, matcher, j,
len = tokens.length,
leadingRelative = Expr.relative[ tokens[0].type ],
implicitRelative = leadingRelative || Expr.relative[" "],
i = leadingRelative ? 1 : 0,
2017-02-03 13:51:07 +01:00
// The foundational matcher ensures that elements are reachable from top-level context(s)
matchContext = addCombinator( function( elem ) {
return elem === checkContext;
}, implicitRelative, true ),
matchAnyContext = addCombinator( function( elem ) {
return indexOf( checkContext, elem ) > -1;
}, implicitRelative, true ),
matchers = [ function( elem, context, xml ) {
var ret = ( !leadingRelative && ( xml || context !== outermostContext ) ) || (
(checkContext = context).nodeType ?
matchContext( elem, context, xml ) :
matchAnyContext( elem, context, xml ) );
// Avoid hanging onto element (issue #299)
checkContext = null;
return ret;
} ];
2017-02-03 13:51:07 +01:00
for ( ; i < len; i++ ) {
if ( (matcher = Expr.relative[ tokens[i].type ]) ) {
matchers = [ addCombinator(elementMatcher( matchers ), matcher) ];
} else {
matcher = Expr.filter[ tokens[i].type ].apply( null, tokens[i].matches );
2017-02-03 13:51:07 +01:00
// Return special upon seeing a positional matcher
if ( matcher[ expando ] ) {
// Find the next relative operator (if any) for proper handling
j = ++i;
for ( ; j < len; j++ ) {
if ( Expr.relative[ tokens[j].type ] ) {
break;
}
}
return setMatcher(
i > 1 && elementMatcher( matchers ),
i > 1 && toSelector(
// If the preceding token was a descendant combinator, insert an implicit any-element `*`
tokens.slice( 0, i - 1 ).concat({ value: tokens[ i - 2 ].type === " " ? "*" : "" })
).replace( rtrim, "$1" ),
matcher,
i < j && matcherFromTokens( tokens.slice( i, j ) ),
j < len && matcherFromTokens( (tokens = tokens.slice( j )) ),
j < len && toSelector( tokens )
);
}
matchers.push( matcher );
}
}
2017-02-03 13:51:07 +01:00
return elementMatcher( matchers );
}
2017-02-03 13:51:07 +01:00
function matcherFromGroupMatchers( elementMatchers, setMatchers ) {
var bySet = setMatchers.length > 0,
byElement = elementMatchers.length > 0,
superMatcher = function( seed, context, xml, results, outermost ) {
var elem, j, matcher,
matchedCount = 0,
i = "0",
unmatched = seed && [],
setMatched = [],
contextBackup = outermostContext,
// We must always have either seed elements or outermost context
elems = seed || byElement && Expr.find["TAG"]( "*", outermost ),
// Use integer dirruns iff this is the outermost matcher
dirrunsUnique = (dirruns += contextBackup == null ? 1 : Math.random() || 0.1),
len = elems.length;
2017-02-03 13:51:07 +01:00
if ( outermost ) {
outermostContext = context === document || context || outermost;
}
2017-02-03 13:51:07 +01:00
// Add elements passing elementMatchers directly to results
// Support: IE<9, Safari
// Tolerate NodeList properties (IE: "length"; Safari: <number>) matching elements by id
for ( ; i !== len && (elem = elems[i]) != null; i++ ) {
if ( byElement && elem ) {
j = 0;
if ( !context && elem.ownerDocument !== document ) {
setDocument( elem );
xml = !documentIsHTML;
}
while ( (matcher = elementMatchers[j++]) ) {
if ( matcher( elem, context || document, xml) ) {
results.push( elem );
break;
}
}
if ( outermost ) {
dirruns = dirrunsUnique;
}
}
2017-02-03 13:51:07 +01:00
// Track unmatched elements for set filters
if ( bySet ) {
// They will have gone through all possible matchers
if ( (elem = !matcher && elem) ) {
matchedCount--;
}
2017-02-03 13:51:07 +01:00
// Lengthen the array for every element, matched or not
if ( seed ) {
unmatched.push( elem );
}
}
}
2017-02-03 13:51:07 +01:00
// `i` is now the count of elements visited above, and adding it to `matchedCount`
// makes the latter nonnegative.
matchedCount += i;
2017-02-03 13:51:07 +01:00
// Apply set filters to unmatched elements
// NOTE: This can be skipped if there are no unmatched elements (i.e., `matchedCount`
// equals `i`), unless we didn't visit _any_ elements in the above loop because we have
// no element matchers and no seed.
// Incrementing an initially-string "0" `i` allows `i` to remain a string only in that
// case, which will result in a "00" `matchedCount` that differs from `i` but is also
// numerically zero.
if ( bySet && i !== matchedCount ) {
j = 0;
while ( (matcher = setMatchers[j++]) ) {
matcher( unmatched, setMatched, context, xml );
}
2017-02-03 13:51:07 +01:00
if ( seed ) {
// Reintegrate element matches to eliminate the need for sorting
if ( matchedCount > 0 ) {
while ( i-- ) {
if ( !(unmatched[i] || setMatched[i]) ) {
setMatched[i] = pop.call( results );
2017-02-03 13:51:07 +01:00
}
}
}
// Discard index placeholder values to get only actual matches
setMatched = condense( setMatched );
}
2017-02-03 13:51:07 +01:00
// Add matches to results
push.apply( results, setMatched );
2017-02-03 13:51:07 +01:00
// Seedless set matches succeeding multiple successful matchers stipulate sorting
if ( outermost && !seed && setMatched.length > 0 &&
( matchedCount + setMatchers.length ) > 1 ) {
2017-02-03 13:51:07 +01:00
Sizzle.uniqueSort( results );
}
}
2017-02-03 13:51:07 +01:00
// Override manipulation of globals by nested matchers
if ( outermost ) {
dirruns = dirrunsUnique;
outermostContext = contextBackup;
}
2017-02-03 13:51:07 +01:00
return unmatched;
};
2017-02-03 13:51:07 +01:00
return bySet ?
markFunction( superMatcher ) :
superMatcher;
}
2017-02-03 13:51:07 +01:00
compile = Sizzle.compile = function( selector, match /* Internal Use Only */ ) {
var i,
setMatchers = [],
elementMatchers = [],
cached = compilerCache[ selector + " " ];
2017-02-03 13:51:07 +01:00
if ( !cached ) {
// Generate a function of recursive functions that can be used to check each element
if ( !match ) {
match = tokenize( selector );
}
i = match.length;
while ( i-- ) {
cached = matcherFromTokens( match[i] );
if ( cached[ expando ] ) {
setMatchers.push( cached );
} else {
elementMatchers.push( cached );
}
}
2017-02-03 13:51:07 +01:00
// Cache the compiled function
cached = compilerCache( selector, matcherFromGroupMatchers( elementMatchers, setMatchers ) );
2017-02-03 13:51:07 +01:00
// Save selector and tokenization
cached.selector = selector;
}
return cached;
};
2017-02-03 13:51:07 +01:00
/**
* A low-level selection function that works with Sizzle's compiled
* selector functions
* @param {String|Function} selector A selector or a pre-compiled
* selector function built with Sizzle.compile
* @param {Element} context
* @param {Array} [results]
* @param {Array} [seed] A set of elements to match against
*/
select = Sizzle.select = function( selector, context, results, seed ) {
var i, tokens, token, type, find,
compiled = typeof selector === "function" && selector,
match = !seed && tokenize( (selector = compiled.selector || selector) );
2017-02-03 13:51:07 +01:00
results = results || [];
2017-02-03 13:51:07 +01:00
// Try to minimize operations if there is only one selector in the list and no seed
// (the latter of which guarantees us context)
if ( match.length === 1 ) {
2017-02-03 13:51:07 +01:00
// Reduce context if the leading compound selector is an ID
tokens = match[0] = match[0].slice( 0 );
if ( tokens.length > 2 && (token = tokens[0]).type === "ID" &&
support.getById && context.nodeType === 9 && documentIsHTML &&
Expr.relative[ tokens[1].type ] ) {
2017-02-03 13:51:07 +01:00
context = ( Expr.find["ID"]( token.matches[0].replace(runescape, funescape), context ) || [] )[0];
if ( !context ) {
return results;
2017-02-03 13:51:07 +01:00
// Precompiled matchers will still verify ancestry, so step up a level
} else if ( compiled ) {
context = context.parentNode;
2017-02-03 13:51:07 +01:00
}
selector = selector.slice( tokens.shift().value.length );
}
2017-02-03 13:51:07 +01:00
// Fetch a seed set for right-to-left matching
i = matchExpr["needsContext"].test( selector ) ? 0 : tokens.length;
while ( i-- ) {
token = tokens[i];
2017-02-03 13:51:07 +01:00
// Abort if we hit a combinator
if ( Expr.relative[ (type = token.type) ] ) {
break;
}
if ( (find = Expr.find[ type ]) ) {
// Search, expanding context for leading sibling combinators
if ( (seed = find(
token.matches[0].replace( runescape, funescape ),
rsibling.test( tokens[0].type ) && testContext( context.parentNode ) || context
)) ) {
2017-02-03 13:51:07 +01:00
// If seed is empty or no tokens remain, we can return early
tokens.splice( i, 1 );
selector = seed.length && toSelector( tokens );
if ( !selector ) {
push.apply( results, seed );
return results;
}
2017-02-03 13:51:07 +01:00
break;
}
}
}
}
2017-02-03 13:51:07 +01:00
// Compile and execute a filtering function if one is not provided
// Provide `match` to avoid retokenization if we modified the selector above
( compiled || compile( selector, match ) )(
seed,
context,
!documentIsHTML,
results,
!context || rsibling.test( selector ) && testContext( context.parentNode ) || context
);
return results;
};
2017-02-03 13:51:07 +01:00
// One-time assignments
2017-02-03 13:51:07 +01:00
// Sort stability
support.sortStable = expando.split("").sort( sortOrder ).join("") === expando;
2017-02-03 13:51:07 +01:00
// Support: Chrome 14-35+
// Always assume duplicates if they aren't passed to the comparison function
support.detectDuplicates = !!hasDuplicate;
2017-02-03 13:51:07 +01:00
// Initialize against the default document
setDocument();
2017-02-03 13:51:07 +01:00
// Support: Webkit<537.32 - Safari 6.0.3/Chrome 25 (fixed in Chrome 27)
// Detached nodes confoundingly follow *each other*
support.sortDetached = assert(function( div1 ) {
// Should return 1, but returns 4 (following)
return div1.compareDocumentPosition( document.createElement("div") ) & 1;
});
2017-02-03 13:51:07 +01:00
// Support: IE<8
// Prevent attribute/property "interpolation"
// http://msdn.microsoft.com/en-us/library/ms536429%28VS.85%29.aspx
if ( !assert(function( div ) {
div.innerHTML = "<a href='#'></a>";
return div.firstChild.getAttribute("href") === "#" ;
}) ) {
addHandle( "type|href|height|width", function( elem, name, isXML ) {
if ( !isXML ) {
return elem.getAttribute( name, name.toLowerCase() === "type" ? 1 : 2 );
}
});
}
2017-02-03 13:51:07 +01:00
// Support: IE<9
// Use defaultValue in place of getAttribute("value")
if ( !support.attributes || !assert(function( div ) {
div.innerHTML = "<input/>";
div.firstChild.setAttribute( "value", "" );
return div.firstChild.getAttribute( "value" ) === "";
}) ) {
addHandle( "value", function( elem, name, isXML ) {
if ( !isXML && elem.nodeName.toLowerCase() === "input" ) {
return elem.defaultValue;
}
});
}
2017-02-03 13:51:07 +01:00
// Support: IE<9
// Use getAttributeNode to fetch booleans when getAttribute lies
if ( !assert(function( div ) {
return div.getAttribute("disabled") == null;
}) ) {
addHandle( booleans, function( elem, name, isXML ) {
var val;
if ( !isXML ) {
return elem[ name ] === true ? name.toLowerCase() :
(val = elem.getAttributeNode( name )) && val.specified ?
val.value :
null;
}
});
}
2017-02-03 13:51:07 +01:00
// EXPOSE
if ( typeof define === "function" && define.amd ) {
define('sizzle',[],function() { return Sizzle; });
// Sizzle requires that there be a global window in Common-JS like environments
} else if ( typeof module !== "undefined" && module.exports ) {
module.exports = Sizzle;
} else {
window.Sizzle = Sizzle;
}
// EXPOSE
2017-02-03 13:51:07 +01:00
})( window );
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
/*!
* @overview es6-promise - a tiny implementation of Promises/A+.
* @copyright Copyright (c) 2014 Yehuda Katz, Tom Dale, Stefan Penner and contributors (Conversion to ES6 API by Jake Archibald)
* @license Licensed under MIT license
* See https://raw.githubusercontent.com/stefanpenner/es6-promise/master/LICENSE
* @version 4.1.1
*/
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
typeof define === 'function' && define.amd ? define('es6-promise',factory) :
(global.ES6Promise = factory());
}(this, (function () { 'use strict';
function objectOrFunction(x) {
var type = typeof x;
return x !== null && (type === 'object' || type === 'function');
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function isFunction(x) {
return typeof x === 'function';
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var _isArray = undefined;
if (Array.isArray) {
_isArray = Array.isArray;
} else {
_isArray = function (x) {
return Object.prototype.toString.call(x) === '[object Array]';
};
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var isArray = _isArray;
var len = 0;
var vertxNext = undefined;
var customSchedulerFn = undefined;
var asap = function asap(callback, arg) {
queue[len] = callback;
queue[len + 1] = arg;
len += 2;
if (len === 2) {
// If len is 2, that means that we need to schedule an async flush.
// If additional callbacks are queued before the queue is flushed, they
// will be processed by this flush that we are scheduling.
if (customSchedulerFn) {
customSchedulerFn(flush);
} else {
scheduleFlush();
}
}
};
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function setScheduler(scheduleFn) {
customSchedulerFn = scheduleFn;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function setAsap(asapFn) {
asap = asapFn;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var browserWindow = typeof window !== 'undefined' ? window : undefined;
var browserGlobal = browserWindow || {};
var BrowserMutationObserver = browserGlobal.MutationObserver || browserGlobal.WebKitMutationObserver;
var isNode = typeof self === 'undefined' && typeof process !== 'undefined' && ({}).toString.call(process) === '[object process]';
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
// test for web worker but not in IE10
var isWorker = typeof Uint8ClampedArray !== 'undefined' && typeof importScripts !== 'undefined' && typeof MessageChannel !== 'undefined';
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
// node
function useNextTick() {
// node version 0.10.x displays a deprecation warning when nextTick is used recursively
// see https://github.com/cujojs/when/issues/410 for details
return function () {
return process.nextTick(flush);
};
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
// vertx
function useVertxTimer() {
if (typeof vertxNext !== 'undefined') {
return function () {
vertxNext(flush);
};
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
return useSetTimeout();
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function useMutationObserver() {
var iterations = 0;
var observer = new BrowserMutationObserver(flush);
var node = document.createTextNode('');
observer.observe(node, { characterData: true });
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
return function () {
node.data = iterations = ++iterations % 2;
};
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
// web worker
function useMessageChannel() {
var channel = new MessageChannel();
channel.port1.onmessage = flush;
return function () {
return channel.port2.postMessage(0);
};
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function useSetTimeout() {
// Store setTimeout reference so es6-promise will be unaffected by
// other code modifying setTimeout (like sinon.useFakeTimers())
var globalSetTimeout = setTimeout;
return function () {
return globalSetTimeout(flush, 1);
};
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var queue = new Array(1000);
function flush() {
for (var i = 0; i < len; i += 2) {
var callback = queue[i];
var arg = queue[i + 1];
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
callback(arg);
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
queue[i] = undefined;
queue[i + 1] = undefined;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
len = 0;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function attemptVertx() {
try {
var r = require;
var vertx = r('vertx');
vertxNext = vertx.runOnLoop || vertx.runOnContext;
return useVertxTimer();
} catch (e) {
return useSetTimeout();
}
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var scheduleFlush = undefined;
// Decide what async method to use to triggering processing of queued callbacks:
if (isNode) {
scheduleFlush = useNextTick();
} else if (BrowserMutationObserver) {
scheduleFlush = useMutationObserver();
} else if (isWorker) {
scheduleFlush = useMessageChannel();
} else if (browserWindow === undefined && typeof require === 'function') {
scheduleFlush = attemptVertx();
} else {
scheduleFlush = useSetTimeout();
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function then(onFulfillment, onRejection) {
var _arguments = arguments;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var parent = this;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var child = new this.constructor(noop);
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (child[PROMISE_ID] === undefined) {
makePromise(child);
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var _state = parent._state;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (_state) {
(function () {
var callback = _arguments[_state - 1];
asap(function () {
return invokeCallback(_state, child, callback, parent._result);
});
})();
} else {
subscribe(parent, child, onFulfillment, onRejection);
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
return child;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
/**
`Promise.resolve` returns a promise that will become resolved with the
passed `value`. It is shorthand for the following:
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
```javascript
let promise = new Promise(function(resolve, reject){
resolve(1);
});
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
promise.then(function(value){
// value === 1
});
```
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
Instead of writing the above, your code now simply becomes the following:
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
```javascript
let promise = Promise.resolve(1);
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
promise.then(function(value){
// value === 1
});
```
@method resolve
@static
@param {Any} value value that the returned promise will be resolved with
Useful for tooling.
@return {Promise} a promise that will become fulfilled with the given
`value`
*/
function resolve$1(object) {
/*jshint validthis:true */
var Constructor = this;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (object && typeof object === 'object' && object.constructor === Constructor) {
return object;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var promise = new Constructor(noop);
resolve(promise, object);
return promise;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var PROMISE_ID = Math.random().toString(36).substring(16);
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function noop() {}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var PENDING = void 0;
var FULFILLED = 1;
var REJECTED = 2;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var GET_THEN_ERROR = new ErrorObject();
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function selfFulfillment() {
return new TypeError("You cannot resolve a promise with itself");
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function cannotReturnOwn() {
return new TypeError('A promises callback cannot return that same promise.');
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function getThen(promise) {
try {
return promise.then;
} catch (error) {
GET_THEN_ERROR.error = error;
return GET_THEN_ERROR;
}
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function tryThen(then$$1, value, fulfillmentHandler, rejectionHandler) {
try {
then$$1.call(value, fulfillmentHandler, rejectionHandler);
} catch (e) {
return e;
}
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function handleForeignThenable(promise, thenable, then$$1) {
asap(function (promise) {
var sealed = false;
var error = tryThen(then$$1, thenable, function (value) {
if (sealed) {
return;
}
sealed = true;
if (thenable !== value) {
resolve(promise, value);
} else {
fulfill(promise, value);
}
}, function (reason) {
if (sealed) {
return;
}
sealed = true;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
reject(promise, reason);
}, 'Settle: ' + (promise._label || ' unknown promise'));
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (!sealed && error) {
sealed = true;
reject(promise, error);
}
}, promise);
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function handleOwnThenable(promise, thenable) {
if (thenable._state === FULFILLED) {
fulfill(promise, thenable._result);
} else if (thenable._state === REJECTED) {
reject(promise, thenable._result);
} else {
subscribe(thenable, undefined, function (value) {
return resolve(promise, value);
}, function (reason) {
return reject(promise, reason);
});
}
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function handleMaybeThenable(promise, maybeThenable, then$$1) {
if (maybeThenable.constructor === promise.constructor && then$$1 === then && maybeThenable.constructor.resolve === resolve$1) {
handleOwnThenable(promise, maybeThenable);
} else {
if (then$$1 === GET_THEN_ERROR) {
reject(promise, GET_THEN_ERROR.error);
GET_THEN_ERROR.error = null;
} else if (then$$1 === undefined) {
fulfill(promise, maybeThenable);
} else if (isFunction(then$$1)) {
handleForeignThenable(promise, maybeThenable, then$$1);
} else {
fulfill(promise, maybeThenable);
}
}
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function resolve(promise, value) {
if (promise === value) {
reject(promise, selfFulfillment());
} else if (objectOrFunction(value)) {
handleMaybeThenable(promise, value, getThen(value));
} else {
fulfill(promise, value);
}
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function publishRejection(promise) {
if (promise._onerror) {
promise._onerror(promise._result);
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
publish(promise);
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function fulfill(promise, value) {
if (promise._state !== PENDING) {
return;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
promise._result = value;
promise._state = FULFILLED;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (promise._subscribers.length !== 0) {
asap(publish, promise);
}
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function reject(promise, reason) {
if (promise._state !== PENDING) {
return;
}
promise._state = REJECTED;
promise._result = reason;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
asap(publishRejection, promise);
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function subscribe(parent, child, onFulfillment, onRejection) {
var _subscribers = parent._subscribers;
var length = _subscribers.length;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
parent._onerror = null;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
_subscribers[length] = child;
_subscribers[length + FULFILLED] = onFulfillment;
_subscribers[length + REJECTED] = onRejection;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (length === 0 && parent._state) {
asap(publish, parent);
}
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function publish(promise) {
var subscribers = promise._subscribers;
var settled = promise._state;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (subscribers.length === 0) {
return;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var child = undefined,
callback = undefined,
detail = promise._result;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
for (var i = 0; i < subscribers.length; i += 3) {
child = subscribers[i];
callback = subscribers[i + settled];
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (child) {
invokeCallback(settled, child, callback, detail);
} else {
callback(detail);
}
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
promise._subscribers.length = 0;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function ErrorObject() {
this.error = null;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var TRY_CATCH_ERROR = new ErrorObject();
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function tryCatch(callback, detail) {
try {
return callback(detail);
} catch (e) {
TRY_CATCH_ERROR.error = e;
return TRY_CATCH_ERROR;
}
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function invokeCallback(settled, promise, callback, detail) {
var hasCallback = isFunction(callback),
value = undefined,
error = undefined,
succeeded = undefined,
failed = undefined;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (hasCallback) {
value = tryCatch(callback, detail);
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (value === TRY_CATCH_ERROR) {
failed = true;
error = value.error;
value.error = null;
} else {
succeeded = true;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (promise === value) {
reject(promise, cannotReturnOwn());
return;
}
} else {
value = detail;
succeeded = true;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (promise._state !== PENDING) {
// noop
} else if (hasCallback && succeeded) {
resolve(promise, value);
} else if (failed) {
reject(promise, error);
} else if (settled === FULFILLED) {
fulfill(promise, value);
} else if (settled === REJECTED) {
reject(promise, value);
}
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function initializePromise(promise, resolver) {
try {
resolver(function resolvePromise(value) {
resolve(promise, value);
}, function rejectPromise(reason) {
reject(promise, reason);
});
} catch (e) {
reject(promise, e);
}
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var id = 0;
function nextId() {
return id++;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function makePromise(promise) {
promise[PROMISE_ID] = id++;
promise._state = undefined;
promise._result = undefined;
promise._subscribers = [];
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function Enumerator$1(Constructor, input) {
this._instanceConstructor = Constructor;
this.promise = new Constructor(noop);
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (!this.promise[PROMISE_ID]) {
makePromise(this.promise);
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (isArray(input)) {
this.length = input.length;
this._remaining = input.length;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
this._result = new Array(this.length);
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (this.length === 0) {
fulfill(this.promise, this._result);
} else {
this.length = this.length || 0;
this._enumerate(input);
if (this._remaining === 0) {
fulfill(this.promise, this._result);
}
}
} else {
reject(this.promise, validationError());
}
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
function validationError() {
return new Error('Array Methods must be provided an Array');
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
Enumerator$1.prototype._enumerate = function (input) {
for (var i = 0; this._state === PENDING && i < input.length; i++) {
this._eachEntry(input[i], i);
}
};
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
Enumerator$1.prototype._eachEntry = function (entry, i) {
var c = this._instanceConstructor;
var resolve$$1 = c.resolve;
if (resolve$$1 === resolve$1) {
var _then = getThen(entry);
if (_then === then && entry._state !== PENDING) {
this._settledAt(entry._state, i, entry._result);
} else if (typeof _then !== 'function') {
this._remaining--;
this._result[i] = entry;
} else if (c === Promise$3) {
var promise = new c(noop);
handleMaybeThenable(promise, entry, _then);
this._willSettleAt(promise, i);
} else {
this._willSettleAt(new c(function (resolve$$1) {
return resolve$$1(entry);
}), i);
}
} else {
this._willSettleAt(resolve$$1(entry), i);
}
};
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
Enumerator$1.prototype._settledAt = function (state, i, value) {
var promise = this.promise;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (promise._state === PENDING) {
this._remaining--;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (state === REJECTED) {
reject(promise, value);
} else {
this._result[i] = value;
}
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (this._remaining === 0) {
fulfill(promise, this._result);
}
};
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
Enumerator$1.prototype._willSettleAt = function (promise, i) {
var enumerator = this;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
subscribe(promise, undefined, function (value) {
return enumerator._settledAt(FULFILLED, i, value);
}, function (reason) {
return enumerator._settledAt(REJECTED, i, reason);
});
};
2017-07-22 22:21:05 +02:00
/**
`Promise.all` accepts an array of promises, and returns a new promise which
is fulfilled with an array of fulfillment values for the passed promises, or
rejected with the reason of the first passed promise to be rejected. It casts all
elements of the passed iterable to promises as it runs this algorithm.
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
Example:
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
```javascript
let promise1 = resolve(1);
let promise2 = resolve(2);
let promise3 = resolve(3);
let promises = [ promise1, promise2, promise3 ];
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
Promise.all(promises).then(function(array){
// The array here would be [ 1, 2, 3 ];
});
```
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
If any of the `promises` given to `all` are rejected, the first promise
that is rejected will be given as an argument to the returned promises's
rejection handler. For example:
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
Example:
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
```javascript
let promise1 = resolve(1);
let promise2 = reject(new Error("2"));
let promise3 = reject(new Error("3"));
let promises = [ promise1, promise2, promise3 ];
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
Promise.all(promises).then(function(array){
// Code here never runs because there are rejected promises!
}, function(error) {
// error.message === "2"
});
```
@method all
@static
@param {Array} entries array of promises
@param {String} label optional string for labeling the promise.
Useful for tooling.
@return {Promise} promise that is fulfilled when all `promises` have been
fulfilled, or rejected if any of them become rejected.
@static
*/
function all$1(entries) {
return new Enumerator$1(this, entries).promise;
}
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
/**
`Promise.race` returns a new promise which is settled in the same way as the
first passed promise to settle.
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
Example:
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
```javascript
let promise1 = new Promise(function(resolve, reject){
setTimeout(function(){
resolve('promise 1');
}, 200);
});
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
let promise2 = new Promise(function(resolve, reject){
setTimeout(function(){
resolve('promise 2');
}, 100);
});
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
Promise.race([promise1, promise2]).then(function(result){
// result === 'promise 2' because it was resolved before promise1
// was resolved.
});
```
`Promise.race` is deterministic in that only the state of the first
settled promise matters. For example, even if other promises given to the
`promises` array argument are resolved, but the first settled promise has
become rejected before the other promises became fulfilled, the returned
promise will become rejected:
```javascript
let promise1 = new Promise(function(resolve, reject){
setTimeout(function(){
resolve('promise 1');
}, 200);
});
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
let promise2 = new Promise(function(resolve, reject){
setTimeout(function(){
reject(new Error('promise 2'));
}, 100);
});
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
Promise.race([promise1, promise2]).then(function(result){
// Code here never runs
}, function(reason){
// reason.message === 'promise 2' because promise 2 became rejected before
// promise 1 became fulfilled
});
```
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
An example real-world use case is implementing timeouts:
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
```javascript
Promise.race([ajax('foo.json'), timeout(5000)])
```
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
@method race
@static
@param {Array} promises array of promises to observe
Useful for tooling.
@return {Promise} a promise which settles in the same way as the first passed
promise to settle.
*/
function race$1(entries) {
/*jshint validthis:true */
var Constructor = this;
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
if (!isArray(entries)) {
return new Constructor(function (_, reject) {
return reject(new TypeError('You must pass an array to race.'));
});
} else {
return new Constructor(function (resolve, reject) {
var length = entries.length;
for (var i = 0; i < length; i++) {
Constructor.resolve(entries[i]).then(resolve, reject);
}
});
}
}
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
/**
`Promise.reject` returns a promise rejected with the passed `reason`.
It is shorthand for the following:
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
```javascript
let promise = new Promise(function(resolve, reject){
reject(new Error('WHOOPS'));
});
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
promise.then(function(value){
// Code here doesn't run because the promise is rejected!
}, function(reason){
// reason.message === 'WHOOPS'
});
```
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
Instead of writing the above, your code now simply becomes the following:
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
```javascript
let promise = Promise.reject(new Error('WHOOPS'));
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
promise.then(function(value){
// Code here doesn't run because the promise is rejected!
}, function(reason){
// reason.message === 'WHOOPS'
});
```
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
@method reject
@static
@param {Any} reason value that the returned promise will be rejected with.
Useful for tooling.
@return {Promise} a promise rejected with the given `reason`.
*/
function reject$1(reason) {
/*jshint validthis:true */
var Constructor = this;
var promise = new Constructor(noop);
reject(promise, reason);
return promise;
}
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
function needsResolver() {
throw new TypeError('You must pass a resolver function as the first argument to the promise constructor');
}
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
function needsNew() {
throw new TypeError("Failed to construct 'Promise': Please use the 'new' operator, this object constructor cannot be called as a function.");
}
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
/**
Promise objects represent the eventual result of an asynchronous operation. The
primary way of interacting with a promise is through its `then` method, which
registers callbacks to receive either a promise's eventual value or the reason
why the promise cannot be fulfilled.
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
Terminology
-----------
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
- `promise` is an object or function with a `then` method whose behavior conforms to this specification.
- `thenable` is an object or function that defines a `then` method.
- `value` is any legal JavaScript value (including undefined, a thenable, or a promise).
- `exception` is a value that is thrown using the throw statement.
- `reason` is a value that indicates why a promise was rejected.
- `settled` the final resting state of a promise, fulfilled or rejected.
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
A promise can be in one of three states: pending, fulfilled, or rejected.
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
Promises that are fulfilled have a fulfillment value and are in the fulfilled
state. Promises that are rejected have a rejection reason and are in the
rejected state. A fulfillment value is never a thenable.
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
Promises can also be said to *resolve* a value. If this value is also a
promise, then the original promise's settled state will match the value's
settled state. So a promise that *resolves* a promise that rejects will
itself reject, and a promise that *resolves* a promise that fulfills will
itself fulfill.
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
Basic Usage:
------------
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
```js
let promise = new Promise(function(resolve, reject) {
// on success
resolve(value);
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
// on failure
reject(reason);
});
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
promise.then(function(value) {
// on fulfillment
}, function(reason) {
// on rejection
});
```
Advanced Usage:
---------------
Promises shine when abstracting away asynchronous interactions such as
`XMLHttpRequest`s.
```js
function getJSON(url) {
return new Promise(function(resolve, reject){
let xhr = new XMLHttpRequest();
xhr.open('GET', url);
xhr.onreadystatechange = handler;
xhr.responseType = 'json';
xhr.setRequestHeader('Accept', 'application/json');
xhr.send();
function handler() {
if (this.readyState === this.DONE) {
if (this.status === 200) {
resolve(this.response);
} else {
reject(new Error('getJSON: `' + url + '` failed with status: [' + this.status + ']'));
}
}
};
});
}
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
getJSON('/posts.json').then(function(json) {
// on fulfillment
}, function(reason) {
// on rejection
});
```
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
Unlike callbacks, promises are great composable primitives.
```js
Promise.all([
getJSON('/posts'),
getJSON('/comments')
]).then(function(values){
values[0] // => postsJSON
values[1] // => commentsJSON
return values;
});
```
@class Promise
@param {function} resolver
Useful for tooling.
@constructor
*/
2017-07-22 22:21:05 +02:00
function Promise$3(resolver) {
this[PROMISE_ID] = nextId();
this._result = this._state = undefined;
this._subscribers = [];
if (noop !== resolver) {
typeof resolver !== 'function' && needsResolver();
this instanceof Promise$3 ? initializePromise(this, resolver) : needsNew();
}
}
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
Promise$3.all = all$1;
Promise$3.race = race$1;
Promise$3.resolve = resolve$1;
Promise$3.reject = reject$1;
Promise$3._setScheduler = setScheduler;
Promise$3._setAsap = setAsap;
Promise$3._asap = asap;
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
Promise$3.prototype = {
constructor: Promise$3,
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
/**
The primary way of interacting with a promise is through its `then` method,
which registers callbacks to receive either a promise's eventual value or the
reason why the promise cannot be fulfilled.
```js
findUser().then(function(user){
// user is available
}, function(reason){
// user is unavailable, and you are given the reason why
});
```
Chaining
--------
The return value of `then` is itself a promise. This second, 'downstream'
promise is resolved with the return value of the first promise's fulfillment
or rejection handler, or rejected if the handler throws an exception.
```js
findUser().then(function (user) {
return user.name;
}, function (reason) {
return 'default name';
}).then(function (userName) {
// If `findUser` fulfilled, `userName` will be the user's name, otherwise it
// will be `'default name'`
});
findUser().then(function (user) {
throw new Error('Found user, but still unhappy');
}, function (reason) {
throw new Error('`findUser` rejected and we're unhappy');
}).then(function (value) {
// never reached
}, function (reason) {
// if `findUser` fulfilled, `reason` will be 'Found user, but still unhappy'.
// If `findUser` rejected, `reason` will be '`findUser` rejected and we're unhappy'.
});
```
If the downstream promise does not specify a rejection handler, rejection reasons will be propagated further downstream.
```js
findUser().then(function (user) {
throw new PedagogicalException('Upstream error');
}).then(function (value) {
// never reached
}).then(function (value) {
// never reached
}, function (reason) {
// The `PedgagocialException` is propagated all the way down to here
});
```
Assimilation
------------
Sometimes the value you want to propagate to a downstream promise can only be
retrieved asynchronously. This can be achieved by returning a promise in the
fulfillment or rejection handler. The downstream promise will then be pending
until the returned promise is settled. This is called *assimilation*.
```js
findUser().then(function (user) {
return findCommentsByAuthor(user);
}).then(function (comments) {
// The user's comments are now available
});
```
If the assimliated promise rejects, then the downstream promise will also reject.
```js
findUser().then(function (user) {
return findCommentsByAuthor(user);
}).then(function (comments) {
// If `findCommentsByAuthor` fulfills, we'll have the value here
}, function (reason) {
// If `findCommentsByAuthor` rejects, we'll have the reason here
});
```
Simple Example
--------------
Synchronous Example
```javascript
let result;
try {
result = findResult();
// success
} catch(reason) {
// failure
}
```
Errback Example
```js
findResult(function(result, err){
if (err) {
// failure
} else {
// success
}
2017-07-22 22:21:05 +02:00
});
```
Promise Example;
```javascript
findResult().then(function(result){
// success
}, function(reason){
// failure
});
```
Advanced Example
--------------
Synchronous Example
```javascript
let author, books;
try {
author = findAuthor();
books = findBooksByAuthor(author);
// success
} catch(reason) {
// failure
}
```
Errback Example
```js
function foundBooks(books) {
}
function failure(reason) {
}
findAuthor(function(author, err){
if (err) {
failure(err);
// failure
} else {
try {
findBoooksByAuthor(author, function(books, err) {
if (err) {
failure(err);
} else {
try {
foundBooks(books);
} catch(reason) {
failure(reason);
}
}
2017-07-22 22:21:05 +02:00
});
} catch(error) {
failure(err);
}
2017-07-22 22:21:05 +02:00
// success
}
2017-07-22 22:21:05 +02:00
});
```
Promise Example;
```javascript
findAuthor().
then(findBooksByAuthor).
then(function(books){
// found books
}).catch(function(reason){
// something went wrong
});
```
@method then
@param {Function} onFulfilled
@param {Function} onRejected
Useful for tooling.
@return {Promise}
*/
then: then,
/**
`catch` is simply sugar for `then(undefined, onRejection)` which makes it the same
as the catch block of a try/catch statement.
```js
function findAuthor(){
throw new Error('couldn't find that author');
}
2017-07-22 22:21:05 +02:00
// synchronous
try {
findAuthor();
} catch(reason) {
// something went wrong
}
// async with promises
findAuthor().catch(function(reason){
// something went wrong
});
```
@method catch
@param {Function} onRejection
Useful for tooling.
@return {Promise}
*/
'catch': function _catch(onRejection) {
return this.then(null, onRejection);
}
};
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
/*global self*/
function polyfill$1() {
var local = undefined;
if (typeof global !== 'undefined') {
local = global;
} else if (typeof self !== 'undefined') {
local = self;
} else {
try {
local = Function('return this')();
} catch (e) {
throw new Error('polyfill failed because global object is unavailable in this environment');
}
2017-07-22 22:21:05 +02:00
}
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
var P = local.Promise;
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
if (P) {
var promiseToString = null;
try {
promiseToString = Object.prototype.toString.call(P.resolve());
} catch (e) {
// silently ignored
}
if (promiseToString === '[object Promise]' && !P.cast) {
return;
}
}
2017-07-22 22:21:05 +02:00
local.Promise = Promise$3;
}
2017-07-22 22:21:05 +02:00
// Strange compat..
Promise$3.polyfill = polyfill$1;
Promise$3.Promise = Promise$3;
2017-07-22 22:21:05 +02:00
Promise$3.polyfill();
2017-07-22 22:21:05 +02:00
return Promise$3;
2017-07-22 22:21:05 +02:00
})));
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
//# sourceMappingURL=es6-promise.auto.map
;
if (!String.prototype.endsWith) {
String.prototype.endsWith = function (searchString, position) {
var subjectString = this.toString();
if (position === undefined || position > subjectString.length) {
position = subjectString.length;
}
position -= searchString.length;
var lastIndex = subjectString.indexOf(searchString, position);
return lastIndex !== -1 && lastIndex === position;
};
}
if (!String.prototype.startsWith) {
String.prototype.startsWith = function (searchString, position) {
position = position || 0;
return this.substr(position, searchString.length) === searchString;
};
}
if (!String.prototype.splitOnce) {
String.prototype.splitOnce = function (delimiter) {
var components = this.split(delimiter);
return [components.shift(), components.join(delimiter)];
};
}
if (!String.prototype.trim) {
String.prototype.trim = function () {
return this.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, '');
};
}
;
define("polyfill", function(){});
/*!
* jQuery Browser Plugin 0.1.0
* https://github.com/gabceb/jquery-browser-plugin
*
* Original jquery-browser code Copyright 2005, 2015 jQuery Foundation, Inc. and other contributors
* http://jquery.org/license
*
* Modifications Copyright 2015 Gabriel Cebrian
* https://github.com/gabceb
*
* Released under the MIT license
*
* Date: 05-07-2015
*/
/*global window: false */
(function (factory) {
if (typeof define === 'function' && define.amd) {
// AMD. Register as an anonymous module.
define('jquery.browser',['jquery'], function ($) {
return factory($);
});
} else if (typeof module === 'object' && typeof module.exports === 'object') {
// Node-like environment
module.exports = factory(require('jquery'));
} else {
// Browser globals
factory(window.jQuery);
}
}(function(jQuery) {
"use strict";
function uaMatch( ua ) {
// If an UA is not provided, default to the current browser UA.
if ( ua === undefined ) {
ua = window.navigator.userAgent;
}
ua = ua.toLowerCase();
var match = /(edge)\/([\w.]+)/.exec( ua ) ||
/(opr)[\/]([\w.]+)/.exec( ua ) ||
/(chrome)[ \/]([\w.]+)/.exec( ua ) ||
/(iemobile)[\/]([\w.]+)/.exec( ua ) ||
/(version)(applewebkit)[ \/]([\w.]+).*(safari)[ \/]([\w.]+)/.exec( ua ) ||
/(webkit)[ \/]([\w.]+).*(version)[ \/]([\w.]+).*(safari)[ \/]([\w.]+)/.exec( ua ) ||
/(webkit)[ \/]([\w.]+)/.exec( ua ) ||
/(opera)(?:.*version|)[ \/]([\w.]+)/.exec( ua ) ||
/(msie) ([\w.]+)/.exec( ua ) ||
ua.indexOf("trident") >= 0 && /(rv)(?::| )([\w.]+)/.exec( ua ) ||
ua.indexOf("compatible") < 0 && /(mozilla)(?:.*? rv:([\w.]+)|)/.exec( ua ) ||
[];
var platform_match = /(ipad)/.exec( ua ) ||
/(ipod)/.exec( ua ) ||
/(windows phone)/.exec( ua ) ||
/(iphone)/.exec( ua ) ||
/(kindle)/.exec( ua ) ||
/(silk)/.exec( ua ) ||
/(android)/.exec( ua ) ||
/(win)/.exec( ua ) ||
/(mac)/.exec( ua ) ||
/(linux)/.exec( ua ) ||
/(cros)/.exec( ua ) ||
/(playbook)/.exec( ua ) ||
/(bb)/.exec( ua ) ||
/(blackberry)/.exec( ua ) ||
[];
var browser = {},
matched = {
browser: match[ 5 ] || match[ 3 ] || match[ 1 ] || "",
version: match[ 2 ] || match[ 4 ] || "0",
versionNumber: match[ 4 ] || match[ 2 ] || "0",
platform: platform_match[ 0 ] || ""
};
if ( matched.browser ) {
browser[ matched.browser ] = true;
browser.version = matched.version;
browser.versionNumber = parseInt(matched.versionNumber, 10);
}
if ( matched.platform ) {
browser[ matched.platform ] = true;
}
// These are all considered mobile platforms, meaning they run a mobile browser
if ( browser.android || browser.bb || browser.blackberry || browser.ipad || browser.iphone ||
browser.ipod || browser.kindle || browser.playbook || browser.silk || browser[ "windows phone" ]) {
browser.mobile = true;
}
// These are all considered desktop platforms, meaning they run a desktop browser
if ( browser.cros || browser.mac || browser.linux || browser.win ) {
browser.desktop = true;
}
// Chrome, Opera 15+ and Safari are webkit based browsers
if ( browser.chrome || browser.opr || browser.safari ) {
browser.webkit = true;
}
// IE11 has a new token so we will assign it msie to avoid breaking changes
if ( browser.rv || browser.iemobile) {
var ie = "msie";
matched.browser = ie;
browser[ie] = true;
}
// Edge is officially known as Microsoft Edge, so rewrite the key to match
if ( browser.edge ) {
delete browser.edge;
var msedge = "msedge";
matched.browser = msedge;
browser[msedge] = true;
}
// Blackberry browsers are marked as Safari on BlackBerry
if ( browser.safari && browser.blackberry ) {
var blackberry = "blackberry";
matched.browser = blackberry;
browser[blackberry] = true;
}
// Playbook browsers are marked as Safari on Playbook
if ( browser.safari && browser.playbook ) {
var playbook = "playbook";
matched.browser = playbook;
browser[playbook] = true;
}
// BB10 is a newer OS version of BlackBerry
if ( browser.bb ) {
var bb = "blackberry";
matched.browser = bb;
browser[bb] = true;
}
// Opera 15+ are identified as opr
if ( browser.opr ) {
var opera = "opera";
matched.browser = opera;
browser[opera] = true;
}
// Stock Android browsers are marked as Safari on Android.
if ( browser.safari && browser.android ) {
var android = "android";
matched.browser = android;
browser[android] = true;
}
// Kindle browsers are marked as Safari on Kindle
if ( browser.safari && browser.kindle ) {
var kindle = "kindle";
matched.browser = kindle;
browser[kindle] = true;
}
// Kindle Silk browsers are marked as Safari on Kindle
if ( browser.safari && browser.silk ) {
var silk = "silk";
matched.browser = silk;
browser[silk] = true;
}
// Assign the name and platform variable
browser.name = matched.browser;
browser.platform = matched.platform;
return browser;
}
// Run the matching process, also assign the function to the returned object
// for manual, jQuery-free use if desired
window.jQBrowser = uaMatch( window.navigator.userAgent );
window.jQBrowser.uaMatch = uaMatch;
// Only assign to jQuery.browser if jQuery is loaded
if ( jQuery ) {
jQuery.browser = window.jQBrowser;
}
return window.jQBrowser;
}));
/*
jed.js
v0.5.0beta
https://github.com/SlexAxton/Jed
-----------
A gettext compatible i18n library for modern JavaScript Applications
by Alex Sexton - AlexSexton [at] gmail - @SlexAxton
WTFPL license for use
Dojo CLA for contributions
Jed offers the entire applicable GNU gettext spec'd set of
functions, but also offers some nicer wrappers around them.
The api for gettext was written for a language with no function
overloading, so Jed allows a little more of that.
Many thanks to Joshua I. Miller - unrtst@cpan.org - who wrote
gettext.js back in 2008. I was able to vet a lot of my ideas
against his. I also made sure Jed passed against his tests
in order to offer easy upgrades -- jsgettext.berlios.de
*/
(function (root, undef) {
// Set up some underscore-style functions, if you already have
// underscore, feel free to delete this section, and use it
// directly, however, the amount of functions used doesn't
// warrant having underscore as a full dependency.
// Underscore 1.3.0 was used to port and is licensed
// under the MIT License by Jeremy Ashkenas.
var ArrayProto = Array.prototype,
ObjProto = Object.prototype,
slice = ArrayProto.slice,
hasOwnProp = ObjProto.hasOwnProperty,
nativeForEach = ArrayProto.forEach,
breaker = {};
// We're not using the OOP style _ so we don't need the
// extra level of indirection. This still means that you
// sub out for real `_` though.
var _ = {
forEach : function( obj, iterator, context ) {
var i, l, key;
if ( obj === null ) {
return;
}
if ( nativeForEach && obj.forEach === nativeForEach ) {
obj.forEach( iterator, context );
}
else if ( obj.length === +obj.length ) {
for ( i = 0, l = obj.length; i < l; i++ ) {
if ( i in obj && iterator.call( context, obj[i], i, obj ) === breaker ) {
return;
}
}
}
else {
for ( key in obj) {
if ( hasOwnProp.call( obj, key ) ) {
if ( iterator.call (context, obj[key], key, obj ) === breaker ) {
return;
}
}
}
}
},
extend : function( obj ) {
this.forEach( slice.call( arguments, 1 ), function ( source ) {
for ( var prop in source ) {
obj[prop] = source[prop];
}
});
return obj;
}
};
// END Miniature underscore impl
// Jed is a constructor function
var Jed = function ( options ) {
// Some minimal defaults
this.defaults = {
"locale_data" : {
"messages" : {
"" : {
"domain" : "messages",
"lang" : "en",
"plural_forms" : "nplurals=2; plural=(n != 1);"
}
// There are no default keys, though
}
},
// The default domain if one is missing
"domain" : "messages"
};
// Mix in the sent options with the default options
this.options = _.extend( {}, this.defaults, options );
this.textdomain( this.options.domain );
if ( options.domain && ! this.options.locale_data[ this.options.domain ] ) {
throw new Error('Text domain set to non-existent domain: `' + options.domain + '`');
}
};
// The gettext spec sets this character as the default
// delimiter for context lookups.
// e.g.: context\u0004key
// If your translation company uses something different,
// just change this at any time and it will use that instead.
Jed.context_delimiter = String.fromCharCode( 4 );
function getPluralFormFunc ( plural_form_string ) {
return Jed.PF.compile( plural_form_string || "nplurals=2; plural=(n != 1);");
}
function Chain( key, i18n ){
this._key = key;
this._i18n = i18n;
}
// Create a chainable api for adding args prettily
_.extend( Chain.prototype, {
onDomain : function ( domain ) {
this._domain = domain;
return this;
},
withContext : function ( context ) {
this._context = context;
return this;
},
ifPlural : function ( num, pkey ) {
this._val = num;
this._pkey = pkey;
return this;
},
fetch : function ( sArr ) {
if ( {}.toString.call( sArr ) != '[object Array]' ) {
sArr = [].slice.call(arguments);
}
return ( sArr && sArr.length ? Jed.sprintf : function(x){ return x; } )(
this._i18n.dcnpgettext(this._domain, this._context, this._key, this._pkey, this._val),
sArr
);
}
});
// Add functions to the Jed prototype.
// These will be the functions on the object that's returned
// from creating a `new Jed()`
// These seem redundant, but they gzip pretty well.
_.extend( Jed.prototype, {
// The sexier api start point
translate : function ( key ) {
return new Chain( key, this );
},
textdomain : function ( domain ) {
if ( ! domain ) {
return this._textdomain;
}
this._textdomain = domain;
},
gettext : function ( key ) {
return this.dcnpgettext.call( this, undef, undef, key );
},
dgettext : function ( domain, key ) {
return this.dcnpgettext.call( this, domain, undef, key );
},
dcgettext : function ( domain , key /*, category */ ) {
// Ignores the category anyways
return this.dcnpgettext.call( this, domain, undef, key );
},
ngettext : function ( skey, pkey, val ) {
return this.dcnpgettext.call( this, undef, undef, skey, pkey, val );
},
dngettext : function ( domain, skey, pkey, val ) {
return this.dcnpgettext.call( this, domain, undef, skey, pkey, val );
},
dcngettext : function ( domain, skey, pkey, val/*, category */) {
return this.dcnpgettext.call( this, domain, undef, skey, pkey, val );
},
pgettext : function ( context, key ) {
return this.dcnpgettext.call( this, undef, context, key );
},
dpgettext : function ( domain, context, key ) {
return this.dcnpgettext.call( this, domain, context, key );
},
dcpgettext : function ( domain, context, key/*, category */) {
return this.dcnpgettext.call( this, domain, context, key );
},
npgettext : function ( context, skey, pkey, val ) {
return this.dcnpgettext.call( this, undef, context, skey, pkey, val );
},
dnpgettext : function ( domain, context, skey, pkey, val ) {
return this.dcnpgettext.call( this, domain, context, skey, pkey, val );
},
// The most fully qualified gettext function. It has every option.
// Since it has every option, we can use it from every other method.
// This is the bread and butter.
// Technically there should be one more argument in this function for 'Category',
// but since we never use it, we might as well not waste the bytes to define it.
dcnpgettext : function ( domain, context, singular_key, plural_key, val ) {
// Set some defaults
plural_key = plural_key || singular_key;
// Use the global domain default if one
// isn't explicitly passed in
domain = domain || this._textdomain;
// Default the value to the singular case
val = typeof val == 'undefined' ? 1 : val;
var fallback;
// Handle special cases
// No options found
if ( ! this.options ) {
// There's likely something wrong, but we'll return the correct key for english
// We do this by instantiating a brand new Jed instance with the default set
// for everything that could be broken.
fallback = new Jed();
return fallback.dcnpgettext.call( fallback, undefined, undefined, singular_key, plural_key, val );
}
// No translation data provided
if ( ! this.options.locale_data ) {
throw new Error('No locale data provided.');
}
if ( ! this.options.locale_data[ domain ] ) {
throw new Error('Domain `' + domain + '` was not found.');
}
if ( ! this.options.locale_data[ domain ][ "" ] ) {
throw new Error('No locale meta information provided.');
}
// Make sure we have a truthy key. Otherwise we might start looking
// into the empty string key, which is the options for the locale
// data.
if ( ! singular_key ) {
throw new Error('No translation key found.');
}
// Handle invalid numbers, but try casting strings for good measure
if ( typeof val != 'number' ) {
val = parseInt( val, 10 );
if ( isNaN( val ) ) {
throw new Error('The number that was passed in is not a number.');
}
}
var key = context ? context + Jed.context_delimiter + singular_key : singular_key,
locale_data = this.options.locale_data,
dict = locale_data[ domain ],
pluralForms = dict[""].plural_forms || (locale_data.messages || this.defaults.locale_data.messages)[""].plural_forms,
val_idx = getPluralFormFunc(pluralForms)(val) + 1,
val_list,
res;
// Throw an error if a domain isn't found
if ( ! dict ) {
throw new Error('No domain named `' + domain + '` could be found.');
}
val_list = dict[ key ];
// If there is no match, then revert back to
// english style singular/plural with the keys passed in.
if ( ! val_list || val_idx >= val_list.length ) {
if (this.options.missing_key_callback) {
this.options.missing_key_callback(key);
}
res = [ null, singular_key, plural_key ];
return res[ getPluralFormFunc(pluralForms)( val ) + 1 ];
}
res = val_list[ val_idx ];
// This includes empty strings on purpose
if ( ! res ) {
res = [ null, singular_key, plural_key ];
return res[ getPluralFormFunc(pluralForms)( val ) + 1 ];
}
return res;
}
});
// We add in sprintf capabilities for post translation value interolation
// This is not internally used, so you can remove it if you have this
// available somewhere else, or want to use a different system.
// We _slightly_ modify the normal sprintf behavior to more gracefully handle
// undefined values.
/**
sprintf() for JavaScript 0.7-beta1
http://www.diveintojavascript.com/projects/javascript-sprintf
Copyright (c) Alexandru Marasteanu <alexaholic [at) gmail (dot] com>
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
* Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
* Neither the name of sprintf() for JavaScript nor the
names of its contributors may be used to endorse or promote products
derived from this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL Alexandru Marasteanu BE LIABLE FOR ANY
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
var sprintf = (function() {
function get_type(variable) {
return Object.prototype.toString.call(variable).slice(8, -1).toLowerCase();
}
function str_repeat(input, multiplier) {
for (var output = []; multiplier > 0; output[--multiplier] = input) {/* do nothing */}
return output.join('');
}
var str_format = function() {
if (!str_format.cache.hasOwnProperty(arguments[0])) {
str_format.cache[arguments[0]] = str_format.parse(arguments[0]);
}
return str_format.format.call(null, str_format.cache[arguments[0]], arguments);
};
str_format.format = function(parse_tree, argv) {
var cursor = 1, tree_length = parse_tree.length, node_type = '', arg, output = [], i, k, match, pad, pad_character, pad_length;
for (i = 0; i < tree_length; i++) {
node_type = get_type(parse_tree[i]);
if (node_type === 'string') {
output.push(parse_tree[i]);
}
else if (node_type === 'array') {
match = parse_tree[i]; // convenience purposes only
if (match[2]) { // keyword argument
arg = argv[cursor];
for (k = 0; k < match[2].length; k++) {
if (!arg.hasOwnProperty(match[2][k])) {
throw(sprintf('[sprintf] property "%s" does not exist', match[2][k]));
}
arg = arg[match[2][k]];
}
}
else if (match[1]) { // positional argument (explicit)
arg = argv[match[1]];
}
else { // positional argument (implicit)
arg = argv[cursor++];
}
if (/[^s]/.test(match[8]) && (get_type(arg) != 'number')) {
throw(sprintf('[sprintf] expecting number but found %s', get_type(arg)));
}
// Jed EDIT
if ( typeof arg == 'undefined' || arg === null ) {
arg = '';
}
// Jed EDIT
switch (match[8]) {
case 'b': arg = arg.toString(2); break;
case 'c': arg = String.fromCharCode(arg); break;
case 'd': arg = parseInt(arg, 10); break;
case 'e': arg = match[7] ? arg.toExponential(match[7]) : arg.toExponential(); break;
case 'f': arg = match[7] ? parseFloat(arg).toFixed(match[7]) : parseFloat(arg); break;
case 'o': arg = arg.toString(8); break;
case 's': arg = ((arg = String(arg)) && match[7] ? arg.substring(0, match[7]) : arg); break;
case 'u': arg = Math.abs(arg); break;
case 'x': arg = arg.toString(16); break;
case 'X': arg = arg.toString(16).toUpperCase(); break;
}
arg = (/[def]/.test(match[8]) && match[3] && arg >= 0 ? '+'+ arg : arg);
pad_character = match[4] ? match[4] == '0' ? '0' : match[4].charAt(1) : ' ';
pad_length = match[6] - String(arg).length;
pad = match[6] ? str_repeat(pad_character, pad_length) : '';
output.push(match[5] ? arg + pad : pad + arg);
}
}
return output.join('');
};
str_format.cache = {};
str_format.parse = function(fmt) {
var _fmt = fmt, match = [], parse_tree = [], arg_names = 0;
while (_fmt) {
if ((match = /^[^\x25]+/.exec(_fmt)) !== null) {
parse_tree.push(match[0]);
}
else if ((match = /^\x25{2}/.exec(_fmt)) !== null) {
parse_tree.push('%');
}
else if ((match = /^\x25(?:([1-9]\d*)\$|\(([^\)]+)\))?(\+)?(0|'[^$])?(-)?(\d+)?(?:\.(\d+))?([b-fosuxX])/.exec(_fmt)) !== null) {
if (match[2]) {
arg_names |= 1;
var field_list = [], replacement_field = match[2], field_match = [];
if ((field_match = /^([a-z_][a-z_\d]*)/i.exec(replacement_field)) !== null) {
field_list.push(field_match[1]);
while ((replacement_field = replacement_field.substring(field_match[0].length)) !== '') {
if ((field_match = /^\.([a-z_][a-z_\d]*)/i.exec(replacement_field)) !== null) {
field_list.push(field_match[1]);
}
else if ((field_match = /^\[(\d+)\]/.exec(replacement_field)) !== null) {
field_list.push(field_match[1]);
}
else {
throw('[sprintf] huh?');
}
}
}
else {
throw('[sprintf] huh?');
}
match[2] = field_list;
}
else {
arg_names |= 2;
}
if (arg_names === 3) {
throw('[sprintf] mixing positional and named placeholders is not (yet) supported');
}
parse_tree.push(match);
}
else {
throw('[sprintf] huh?');
}
_fmt = _fmt.substring(match[0].length);
}
return parse_tree;
};
return str_format;
})();
var vsprintf = function(fmt, argv) {
argv.unshift(fmt);
return sprintf.apply(null, argv);
};
Jed.parse_plural = function ( plural_forms, n ) {
plural_forms = plural_forms.replace(/n/g, n);
return Jed.parse_expression(plural_forms);
};
Jed.sprintf = function ( fmt, args ) {
if ( {}.toString.call( args ) == '[object Array]' ) {
return vsprintf( fmt, [].slice.call(args) );
}
return sprintf.apply(this, [].slice.call(arguments) );
};
Jed.prototype.sprintf = function () {
return Jed.sprintf.apply(this, arguments);
};
// END sprintf Implementation
// Start the Plural forms section
// This is a full plural form expression parser. It is used to avoid
// running 'eval' or 'new Function' directly against the plural
// forms.
//
// This can be important if you get translations done through a 3rd
// party vendor. I encourage you to use this instead, however, I
// also will provide a 'precompiler' that you can use at build time
// to output valid/safe function representations of the plural form
// expressions. This means you can build this code out for the most
// part.
Jed.PF = {};
Jed.PF.parse = function ( p ) {
var plural_str = Jed.PF.extractPluralExpr( p );
return Jed.PF.parser.parse.call(Jed.PF.parser, plural_str);
};
Jed.PF.compile = function ( p ) {
// Handle trues and falses as 0 and 1
function imply( val ) {
return (val === true ? 1 : val ? val : 0);
}
var ast = Jed.PF.parse( p );
return function ( n ) {
return imply( Jed.PF.interpreter( ast )( n ) );
};
};
Jed.PF.interpreter = function ( ast ) {
return function ( n ) {
var res;
switch ( ast.type ) {
case 'GROUP':
return Jed.PF.interpreter( ast.expr )( n );
case 'TERNARY':
if ( Jed.PF.interpreter( ast.expr )( n ) ) {
return Jed.PF.interpreter( ast.truthy )( n );
}
return Jed.PF.interpreter( ast.falsey )( n );
case 'OR':
return Jed.PF.interpreter( ast.left )( n ) || Jed.PF.interpreter( ast.right )( n );
case 'AND':
return Jed.PF.interpreter( ast.left )( n ) && Jed.PF.interpreter( ast.right )( n );
case 'LT':
return Jed.PF.interpreter( ast.left )( n ) < Jed.PF.interpreter( ast.right )( n );
case 'GT':
return Jed.PF.interpreter( ast.left )( n ) > Jed.PF.interpreter( ast.right )( n );
case 'LTE':
return Jed.PF.interpreter( ast.left )( n ) <= Jed.PF.interpreter( ast.right )( n );
case 'GTE':
return Jed.PF.interpreter( ast.left )( n ) >= Jed.PF.interpreter( ast.right )( n );
case 'EQ':
return Jed.PF.interpreter( ast.left )( n ) == Jed.PF.interpreter( ast.right )( n );
case 'NEQ':
return Jed.PF.interpreter( ast.left )( n ) != Jed.PF.interpreter( ast.right )( n );
case 'MOD':
return Jed.PF.interpreter( ast.left )( n ) % Jed.PF.interpreter( ast.right )( n );
case 'VAR':
return n;
case 'NUM':
return ast.val;
default:
throw new Error("Invalid Token found.");
}
};
};
Jed.PF.extractPluralExpr = function ( p ) {
// trim first
p = p.replace(/^\s\s*/, '').replace(/\s\s*$/, '');
if (! /;\s*$/.test(p)) {
p = p.concat(';');
}
var nplurals_re = /nplurals\=(\d+);/,
plural_re = /plural\=(.*);/,
nplurals_matches = p.match( nplurals_re ),
res = {},
plural_matches;
// Find the nplurals number
if ( nplurals_matches.length > 1 ) {
res.nplurals = nplurals_matches[1];
}
else {
throw new Error('nplurals not found in plural_forms string: ' + p );
}
// remove that data to get to the formula
p = p.replace( nplurals_re, "" );
plural_matches = p.match( plural_re );
if (!( plural_matches && plural_matches.length > 1 ) ) {
throw new Error('`plural` expression not found: ' + p);
}
return plural_matches[ 1 ];
};
/* Jison generated parser */
Jed.PF.parser = (function(){
var parser = {trace: function trace() { },
yy: {},
symbols_: {"error":2,"expressions":3,"e":4,"EOF":5,"?":6,":":7,"||":8,"&&":9,"<":10,"<=":11,">":12,">=":13,"!=":14,"==":15,"%":16,"(":17,")":18,"n":19,"NUMBER":20,"$accept":0,"$end":1},
terminals_: {2:"error",5:"EOF",6:"?",7:":",8:"||",9:"&&",10:"<",11:"<=",12:">",13:">=",14:"!=",15:"==",16:"%",17:"(",18:")",19:"n",20:"NUMBER"},
productions_: [0,[3,2],[4,5],[4,3],[4,3],[4,3],[4,3],[4,3],[4,3],[4,3],[4,3],[4,3],[4,3],[4,1],[4,1]],
performAction: function anonymous(yytext,yyleng,yylineno,yy,yystate,$$,_$) {
var $0 = $$.length - 1;
switch (yystate) {
case 1: return { type : 'GROUP', expr: $$[$0-1] };
break;
case 2:this.$ = { type: 'TERNARY', expr: $$[$0-4], truthy : $$[$0-2], falsey: $$[$0] };
break;
case 3:this.$ = { type: "OR", left: $$[$0-2], right: $$[$0] };
break;
case 4:this.$ = { type: "AND", left: $$[$0-2], right: $$[$0] };
break;
case 5:this.$ = { type: 'LT', left: $$[$0-2], right: $$[$0] };
break;
case 6:this.$ = { type: 'LTE', left: $$[$0-2], right: $$[$0] };
break;
case 7:this.$ = { type: 'GT', left: $$[$0-2], right: $$[$0] };
break;
case 8:this.$ = { type: 'GTE', left: $$[$0-2], right: $$[$0] };
break;
case 9:this.$ = { type: 'NEQ', left: $$[$0-2], right: $$[$0] };
break;
case 10:this.$ = { type: 'EQ', left: $$[$0-2], right: $$[$0] };
break;
case 11:this.$ = { type: 'MOD', left: $$[$0-2], right: $$[$0] };
break;
case 12:this.$ = { type: 'GROUP', expr: $$[$0-1] };
break;
case 13:this.$ = { type: 'VAR' };
break;
case 14:this.$ = { type: 'NUM', val: Number(yytext) };
break;
}
},
table: [{3:1,4:2,17:[1,3],19:[1,4],20:[1,5]},{1:[3]},{5:[1,6],6:[1,7],8:[1,8],9:[1,9],10:[1,10],11:[1,11],12:[1,12],13:[1,13],14:[1,14],15:[1,15],16:[1,16]},{4:17,17:[1,3],19:[1,4],20:[1,5]},{5:[2,13],6:[2,13],7:[2,13],8:[2,13],9:[2,13],10:[2,13],11:[2,13],12:[2,13],13:[2,13],14:[2,13],15:[2,13],16:[2,13],18:[2,13]},{5:[2,14],6:[2,14],7:[2,14],8:[2,14],9:[2,14],10:[2,14],11:[2,14],12:[2,14],13:[2,14],14:[2,14],15:[2,14],16:[2,14],18:[2,14]},{1:[2,1]},{4:18,17:[1,3],19:[1,4],20:[1,5]},{4:19,17:[1,3],19:[1,4],20:[1,5]},{4:20,17:[1,3],19:[1,4],20:[1,5]},{4:21,17:[1,3],19:[1,4],20:[1,5]},{4:22,17:[1,3],19:[1,4],20:[1,5]},{4:23,17:[1,3],19:[1,4],20:[1,5]},{4:24,17:[1,3],19:[1,4],20:[1,5]},{4:25,17:[1,3],19:[1,4],20:[1,5]},{4:26,17:[1,3],19:[1,4],20:[1,5]},{4:27,17:[1,3],19:[1,4],20:[1,5]},{6:[1,7],8:[1,8],9:[1,9],10:[1,10],11:[1,11],12:[1,12],13:[1,13],14:[1,14],15:[1,15],16:[1,16],18:[1,28]},{6:[1,7],7:[1,29],8:[1,8],9:[1,9],10:[1,10],11:[1,11],12:[1,12],13:[1,13],14:[1,14],15:[1,15],16:[1,16]},{5:[2,3],6:[2,3],7:[2,3],8:[2,3],9:[1,9],10:[1,10],11:[1,11],12:[1,12],13:[1,13],14:[1,14],15:[1,15],16:[1,16],18:[2,3]},{5:[2,4],6:[2,4],7:[2,4],8:[2,4],9:[2,4],10:[1,10],11:[1,11],12:[1,12],13:[1,13],14:[1,14],15:[1,15],16:[1,16],18:[2,4]},{5:[2,5],6:[2,5],7:[2,5],8:[2,5],9:[2,5],10:[2,5],11:[2,5],12:[2,5],13:[2,5],14:[2,5],15:[2,5],16:[1,16],18:[2,5]},{5:[2,6],6:[2,6],7:[2,6],8:[2,6],9:[2,6],10:[2,6],11:[2,6],12:[2,6],13:[2,6],14:[2,6],15:[2,6],16:[1,16],18:[2,6]},{5:[2,7],6:[2,7],7:[2,7],8:[2,7],9:[2,7],10:[2,7],11:[2,7],12:[2,7],13:[2,7],14:[2,7],15:[2,7],16:[1,16],18:[2,7]},{5:[2,8],6:[2,8],7:[2,8],8:[2,8],9:[2,8],10:[2,8],11:[2,8],12:[2,8],13:[2,8],14:[2,8],15:[2,8],16:[1,16],18:[2,8]},{5:[2,9],6:[2,9],7:[2,9],8:[2,9],9:[2,9],10:[2,9],11:[2,9],12:[2,9],13:[2,9],14:[2,9],15:[2,9],16:[1,16],18:[2,9]},{5:[2,10],6:[2,10],7:[2,10],8:[2,10],9:[2,10],10:[2,10],11:[2,10],12:[2,10],13:[2,10],14:[2,10],15:[2,10],16:[1,16],18:[2,10]},{5:[2,11],6:[2,11],7:[2,11],8:[2,11],9:[2,11],10:[2,11],11:[2,11],12:[2,11],13:[2,11],14:[2,11],15:[2,11],16:[2,11],18:[2,11]},{5:[2,12],6:[2,12],7:[2,12],8:[2,12],9:[2,12],10:[2,12],11:[2,12],12:[2,12],13:[2,12],14:[2,12],15:[2,12],16:[2,12],18:[2,12]},{4:30,17:[1,3],19:[1,4],20:[1,5]},{5:[2,2],6:[1,7],7:[2,2],8:[1,8],9:[1,9],10:[1,10],11:[1,11],12:[1,12],13:[1,13],14:[1,14],15:[1,15],16:[1,16],18:[2,2]}],
defaultActions: {6:[2,1]},
parseError: function parseError(str, hash) {
throw new Error(str);
},
parse: function parse(input) {
var self = this,
stack = [0],
vstack = [null], // semantic value stack
lstack = [], // location stack
table = this.table,
yytext = '',
yylineno = 0,
yyleng = 0,
recovering = 0,
TERROR = 2,
EOF = 1;
//this.reductionCount = this.shiftCount = 0;
this.lexer.setInput(input);
this.lexer.yy = this.yy;
this.yy.lexer = this.lexer;
if (typeof this.lexer.yylloc == 'undefined')
this.lexer.yylloc = {};
var yyloc = this.lexer.yylloc;
lstack.push(yyloc);
if (typeof this.yy.parseError === 'function')
this.parseError = this.yy.parseError;
function popStack (n) {
stack.length = stack.length - 2*n;
vstack.length = vstack.length - n;
lstack.length = lstack.length - n;
}
function lex() {
var token;
token = self.lexer.lex() || 1; // $end = 1
// if token isn't its numeric value, convert
if (typeof token !== 'number') {
token = self.symbols_[token] || token;
}
return token;
}
var symbol, preErrorSymbol, state, action, a, r, yyval={},p,len,newState, expected;
while (true) {
// retreive state number from top of stack
state = stack[stack.length-1];
// use default actions if available
if (this.defaultActions[state]) {
action = this.defaultActions[state];
} else {
if (symbol == null)
symbol = lex();
// read action for current state and first input
action = table[state] && table[state][symbol];
}
// handle parse error
_handle_error:
if (typeof action === 'undefined' || !action.length || !action[0]) {
if (!recovering) {
// Report error
expected = [];
for (p in table[state]) if (this.terminals_[p] && p > 2) {
expected.push("'"+this.terminals_[p]+"'");
}
var errStr = '';
if (this.lexer.showPosition) {
errStr = 'Parse error on line '+(yylineno+1)+":\n"+this.lexer.showPosition()+"\nExpecting "+expected.join(', ') + ", got '" + this.terminals_[symbol]+ "'";
} else {
errStr = 'Parse error on line '+(yylineno+1)+": Unexpected " +
(symbol == 1 /*EOF*/ ? "end of input" :
("'"+(this.terminals_[symbol] || symbol)+"'"));
}
this.parseError(errStr,
{text: this.lexer.match, token: this.terminals_[symbol] || symbol, line: this.lexer.yylineno, loc: yyloc, expected: expected});
}
// just recovered from another error
if (recovering == 3) {
if (symbol == EOF) {
throw new Error(errStr || 'Parsing halted.');
}
// discard current lookahead and grab another
yyleng = this.lexer.yyleng;
yytext = this.lexer.yytext;
yylineno = this.lexer.yylineno;
yyloc = this.lexer.yylloc;
symbol = lex();
}
// try to recover from error
while (1) {
// check for error recovery rule in this state
if ((TERROR.toString()) in table[state]) {
break;
}
if (state == 0) {
throw new Error(errStr || 'Parsing halted.');
}
popStack(1);
state = stack[stack.length-1];
}
preErrorSymbol = symbol; // save the lookahead token
symbol = TERROR; // insert generic error symbol as new lookahead
state = stack[stack.length-1];
action = table[state] && table[state][TERROR];
recovering = 3; // allow 3 real symbols to be shifted before reporting a new error
}
// this shouldn't happen, unless resolve defaults are off
if (action[0] instanceof Array && action.length > 1) {
throw new Error('Parse Error: multiple actions possible at state: '+state+', token: '+symbol);
}
switch (action[0]) {
case 1: // shift
//this.shiftCount++;
stack.push(symbol);
vstack.push(this.lexer.yytext);
lstack.push(this.lexer.yylloc);
stack.push(action[1]); // push state
symbol = null;
if (!preErrorSymbol) { // normal execution/no error
yyleng = this.lexer.yyleng;
yytext = this.lexer.yytext;
yylineno = this.lexer.yylineno;
yyloc = this.lexer.yylloc;
if (recovering > 0)
recovering--;
} else { // error just occurred, resume old lookahead f/ before error
symbol = preErrorSymbol;
preErrorSymbol = null;
}
break;
case 2: // reduce
//this.reductionCount++;
len = this.productions_[action[1]][1];
// perform semantic action
yyval.$ = vstack[vstack.length-len]; // default to $$ = $1
// default location, uses first token for firsts, last for lasts
yyval._$ = {
first_line: lstack[lstack.length-(len||1)].first_line,
last_line: lstack[lstack.length-1].last_line,
first_column: lstack[lstack.length-(len||1)].first_column,
last_column: lstack[lstack.length-1].last_column
};
r = this.performAction.call(yyval, yytext, yyleng, yylineno, this.yy, action[1], vstack, lstack);
if (typeof r !== 'undefined') {
return r;
}
// pop off stack
if (len) {
stack = stack.slice(0,-1*len*2);
vstack = vstack.slice(0, -1*len);
lstack = lstack.slice(0, -1*len);
}
stack.push(this.productions_[action[1]][0]); // push nonterminal (reduce)
vstack.push(yyval.$);
lstack.push(yyval._$);
// goto new state = table[STATE][NONTERMINAL]
newState = table[stack[stack.length-2]][stack[stack.length-1]];
stack.push(newState);
break;
case 3: // accept
return true;
}
}
return true;
}};/* Jison generated lexer */
var lexer = (function(){
var lexer = ({EOF:1,
parseError:function parseError(str, hash) {
if (this.yy.parseError) {
this.yy.parseError(str, hash);
} else {
throw new Error(str);
}
},
setInput:function (input) {
this._input = input;
this._more = this._less = this.done = false;
this.yylineno = this.yyleng = 0;
this.yytext = this.matched = this.match = '';
this.conditionStack = ['INITIAL'];
this.yylloc = {first_line:1,first_column:0,last_line:1,last_column:0};
return this;
},
input:function () {
var ch = this._input[0];
this.yytext+=ch;
this.yyleng++;
this.match+=ch;
this.matched+=ch;
var lines = ch.match(/\n/);
if (lines) this.yylineno++;
this._input = this._input.slice(1);
return ch;
},
unput:function (ch) {
this._input = ch + this._input;
return this;
},
more:function () {
this._more = true;
return this;
},
pastInput:function () {
var past = this.matched.substr(0, this.matched.length - this.match.length);
return (past.length > 20 ? '...':'') + past.substr(-20).replace(/\n/g, "");
},
upcomingInput:function () {
var next = this.match;
if (next.length < 20) {
next += this._input.substr(0, 20-next.length);
}
return (next.substr(0,20)+(next.length > 20 ? '...':'')).replace(/\n/g, "");
},
showPosition:function () {
var pre = this.pastInput();
var c = new Array(pre.length + 1).join("-");
return pre + this.upcomingInput() + "\n" + c+"^";
},
next:function () {
if (this.done) {
return this.EOF;
}
if (!this._input) this.done = true;
var token,
match,
col,
lines;
if (!this._more) {
this.yytext = '';
this.match = '';
}
var rules = this._currentRules();
for (var i=0;i < rules.length; i++) {
match = this._input.match(this.rules[rules[i]]);
if (match) {
lines = match[0].match(/\n.*/g);
if (lines) this.yylineno += lines.length;
this.yylloc = {first_line: this.yylloc.last_line,
last_line: this.yylineno+1,
first_column: this.yylloc.last_column,
last_column: lines ? lines[lines.length-1].length-1 : this.yylloc.last_column + match[0].length}
this.yytext += match[0];
this.match += match[0];
this.matches = match;
this.yyleng = this.yytext.length;
this._more = false;
this._input = this._input.slice(match[0].length);
this.matched += match[0];
token = this.performAction.call(this, this.yy, this, rules[i],this.conditionStack[this.conditionStack.length-1]);
if (token) return token;
else return;
}
}
if (this._input === "") {
return this.EOF;
} else {
this.parseError('Lexical error on line '+(this.yylineno+1)+'. Unrecognized text.\n'+this.showPosition(),
{text: "", token: null, line: this.yylineno});
}
},
lex:function lex() {
var r = this.next();
if (typeof r !== 'undefined') {
return r;
} else {
return this.lex();
}
},
begin:function begin(condition) {
this.conditionStack.push(condition);
},
popState:function popState() {
return this.conditionStack.pop();
},
_currentRules:function _currentRules() {
return this.conditions[this.conditionStack[this.conditionStack.length-1]].rules;
},
topState:function () {
return this.conditionStack[this.conditionStack.length-2];
},
pushState:function begin(condition) {
this.begin(condition);
}});
lexer.performAction = function anonymous(yy,yy_,$avoiding_name_collisions,YY_START) {
var YYSTATE=YY_START;
switch($avoiding_name_collisions) {
case 0:/* skip whitespace */
break;
case 1:return 20
break;
case 2:return 19
break;
case 3:return 8
break;
case 4:return 9
break;
case 5:return 6
break;
case 6:return 7
break;
case 7:return 11
break;
case 8:return 13
break;
case 9:return 10
break;
case 10:return 12
break;
case 11:return 14
break;
case 12:return 15
break;
case 13:return 16
break;
case 14:return 17
break;
case 15:return 18
break;
case 16:return 5
break;
case 17:return 'INVALID'
break;
}
};
lexer.rules = [/^\s+/,/^[0-9]+(\.[0-9]+)?\b/,/^n\b/,/^\|\|/,/^&&/,/^\?/,/^:/,/^<=/,/^>=/,/^</,/^>/,/^!=/,/^==/,/^%/,/^\(/,/^\)/,/^$/,/^./];
lexer.conditions = {"INITIAL":{"rules":[0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17],"inclusive":true}};return lexer;})()
parser.lexer = lexer;
return parser;
})();
// End parser
// Handle node, amd, and global systems
if (typeof exports !== 'undefined') {
if (typeof module !== 'undefined' && module.exports) {
exports = module.exports = Jed;
}
exports.Jed = Jed;
}
else {
if (typeof define === 'function' && define.amd) {
define('jed', [],function() {
return Jed;
});
}
// Leak a global regardless of module system
root['Jed'] = Jed;
}
})(this);
/**
* @license text 2.0.15 Copyright jQuery Foundation and other contributors.
* Released under MIT license, http://github.com/requirejs/text/LICENSE
*/
/*jslint regexp: true */
/*global require, XMLHttpRequest, ActiveXObject,
define, window, process, Packages,
java, location, Components, FileUtils */
define('text',['module'], function (module) {
'use strict';
var text, fs, Cc, Ci, xpcIsWindows,
progIds = ['Msxml2.XMLHTTP', 'Microsoft.XMLHTTP', 'Msxml2.XMLHTTP.4.0'],
xmlRegExp = /^\s*<\?xml(\s)+version=[\'\"](\d)*.(\d)*[\'\"](\s)*\?>/im,
bodyRegExp = /<body[^>]*>\s*([\s\S]+)\s*<\/body>/im,
hasLocation = typeof location !== 'undefined' && location.href,
defaultProtocol = hasLocation && location.protocol && location.protocol.replace(/\:/, ''),
defaultHostName = hasLocation && location.hostname,
defaultPort = hasLocation && (location.port || undefined),
buildMap = {},
masterConfig = (module.config && module.config()) || {};
function useDefault(value, defaultValue) {
return value === undefined || value === '' ? defaultValue : value;
}
//Allow for default ports for http and https.
function isSamePort(protocol1, port1, protocol2, port2) {
if (port1 === port2) {
return true;
} else if (protocol1 === protocol2) {
if (protocol1 === 'http') {
return useDefault(port1, '80') === useDefault(port2, '80');
} else if (protocol1 === 'https') {
return useDefault(port1, '443') === useDefault(port2, '443');
}
}
return false;
}
text = {
version: '2.0.15',
strip: function (content) {
//Strips <?xml ...?> declarations so that external SVG and XML
//documents can be added to a document without worry. Also, if the string
//is an HTML document, only the part inside the body tag is returned.
if (content) {
content = content.replace(xmlRegExp, "");
var matches = content.match(bodyRegExp);
if (matches) {
content = matches[1];
}
} else {
content = "";
}
return content;
},
jsEscape: function (content) {
return content.replace(/(['\\])/g, '\\$1')
.replace(/[\f]/g, "\\f")
.replace(/[\b]/g, "\\b")
.replace(/[\n]/g, "\\n")
.replace(/[\t]/g, "\\t")
.replace(/[\r]/g, "\\r")
.replace(/[\u2028]/g, "\\u2028")
.replace(/[\u2029]/g, "\\u2029");
},
createXhr: masterConfig.createXhr || function () {
//Would love to dump the ActiveX crap in here. Need IE 6 to die first.
var xhr, i, progId;
if (typeof XMLHttpRequest !== "undefined") {
return new XMLHttpRequest();
} else if (typeof ActiveXObject !== "undefined") {
for (i = 0; i < 3; i += 1) {
progId = progIds[i];
try {
xhr = new ActiveXObject(progId);
} catch (e) {}
if (xhr) {
progIds = [progId]; // so faster next time
break;
}
}
}
return xhr;
},
/**
* Parses a resource name into its component parts. Resource names
* look like: module/name.ext!strip, where the !strip part is
* optional.
* @param {String} name the resource name
* @returns {Object} with properties "moduleName", "ext" and "strip"
* where strip is a boolean.
*/
parseName: function (name) {
var modName, ext, temp,
strip = false,
index = name.lastIndexOf("."),
isRelative = name.indexOf('./') === 0 ||
name.indexOf('../') === 0;
if (index !== -1 && (!isRelative || index > 1)) {
modName = name.substring(0, index);
ext = name.substring(index + 1);
} else {
modName = name;
}
temp = ext || modName;
index = temp.indexOf("!");
if (index !== -1) {
//Pull off the strip arg.
strip = temp.substring(index + 1) === "strip";
temp = temp.substring(0, index);
if (ext) {
ext = temp;
} else {
modName = temp;
}
}
return {
moduleName: modName,
ext: ext,
strip: strip
};
},
xdRegExp: /^((\w+)\:)?\/\/([^\/\\]+)/,
/**
* Is an URL on another domain. Only works for browser use, returns
* false in non-browser environments. Only used to know if an
* optimized .js version of a text resource should be loaded
* instead.
* @param {String} url
* @returns Boolean
*/
useXhr: function (url, protocol, hostname, port) {
var uProtocol, uHostName, uPort,
match = text.xdRegExp.exec(url);
if (!match) {
return true;
}
uProtocol = match[2];
uHostName = match[3];
uHostName = uHostName.split(':');
uPort = uHostName[1];
uHostName = uHostName[0];
return (!uProtocol || uProtocol === protocol) &&
(!uHostName || uHostName.toLowerCase() === hostname.toLowerCase()) &&
((!uPort && !uHostName) || isSamePort(uProtocol, uPort, protocol, port));
},
finishLoad: function (name, strip, content, onLoad) {
content = strip ? text.strip(content) : content;
if (masterConfig.isBuild) {
buildMap[name] = content;
}
onLoad(content);
},
load: function (name, req, onLoad, config) {
//Name has format: some.module.filext!strip
//The strip part is optional.
//if strip is present, then that means only get the string contents
//inside a body tag in an HTML string. For XML/SVG content it means
//removing the <?xml ...?> declarations so the content can be inserted
//into the current doc without problems.
// Do not bother with the work if a build and text will
// not be inlined.
if (config && config.isBuild && !config.inlineText) {
onLoad();
return;
}
masterConfig.isBuild = config && config.isBuild;
var parsed = text.parseName(name),
nonStripName = parsed.moduleName +
(parsed.ext ? '.' + parsed.ext : ''),
url = req.toUrl(nonStripName),
useXhr = (masterConfig.useXhr) ||
text.useXhr;
// Do not load if it is an empty: url
if (url.indexOf('empty:') === 0) {
onLoad();
return;
}
//Load the text. Use XHR if possible and in a browser.
if (!hasLocation || useXhr(url, defaultProtocol, defaultHostName, defaultPort)) {
text.get(url, function (content) {
text.finishLoad(name, parsed.strip, content, onLoad);
}, function (err) {
if (onLoad.error) {
onLoad.error(err);
}
});
} else {
//Need to fetch the resource across domains. Assume
//the resource has been optimized into a JS module. Fetch
//by the module name + extension, but do not include the
//!strip part to avoid file system issues.
req([nonStripName], function (content) {
text.finishLoad(parsed.moduleName + '.' + parsed.ext,
parsed.strip, content, onLoad);
});
}
},
write: function (pluginName, moduleName, write, config) {
if (buildMap.hasOwnProperty(moduleName)) {
var content = text.jsEscape(buildMap[moduleName]);
write.asModule(pluginName + "!" + moduleName,
"define(function () { return '" +
content +
"';});\n");
}
},
writeFile: function (pluginName, moduleName, req, write, config) {
var parsed = text.parseName(moduleName),
extPart = parsed.ext ? '.' + parsed.ext : '',
nonStripName = parsed.moduleName + extPart,
//Use a '.js' file name so that it indicates it is a
//script that can be loaded across domains.
fileName = req.toUrl(parsed.moduleName + extPart) + '.js';
//Leverage own load() method to load plugin value, but only
//write out values that do not have the strip argument,
//to avoid any potential issues with ! in file names.
text.load(nonStripName, req, function (value) {
//Use own write() method to construct full module value.
//But need to create shell that translates writeFile's
//write() to the right interface.
var textWrite = function (contents) {
return write(fileName, contents);
};
textWrite.asModule = function (moduleName, contents) {
return write.asModule(moduleName, fileName, contents);
};
text.write(pluginName, nonStripName, textWrite, config);
}, config);
}
};
if (masterConfig.env === 'node' || (!masterConfig.env &&
typeof process !== "undefined" &&
process.versions &&
!!process.versions.node &&
!process.versions['node-webkit'] &&
!process.versions['atom-shell'])) {
//Using special require.nodeRequire, something added by r.js.
fs = require.nodeRequire('fs');
text.get = function (url, callback, errback) {
try {
var file = fs.readFileSync(url, 'utf8');
//Remove BOM (Byte Mark Order) from utf8 files if it is there.
if (file[0] === '\uFEFF') {
file = file.substring(1);
}
2017-07-22 22:21:05 +02:00
callback(file);
} catch (e) {
if (errback) {
errback(e);
}
}
};
} else if (masterConfig.env === 'xhr' || (!masterConfig.env &&
text.createXhr())) {
text.get = function (url, callback, errback, headers) {
var xhr = text.createXhr(), header;
xhr.open('GET', url, true);
//Allow plugins direct access to xhr headers
if (headers) {
for (header in headers) {
if (headers.hasOwnProperty(header)) {
xhr.setRequestHeader(header.toLowerCase(), headers[header]);
}
}
}
//Allow overrides specified in config
if (masterConfig.onXhr) {
masterConfig.onXhr(xhr, url);
}
xhr.onreadystatechange = function (evt) {
var status, err;
//Do not explicitly handle errors, those should be
//visible via console output in the browser.
if (xhr.readyState === 4) {
status = xhr.status || 0;
if (status > 399 && status < 600) {
//An http 4xx or 5xx error. Signal an error.
err = new Error(url + ' HTTP status: ' + status);
err.xhr = xhr;
if (errback) {
errback(err);
}
} else {
callback(xhr.responseText);
}
if (masterConfig.onXhrComplete) {
masterConfig.onXhrComplete(xhr, url);
}
}
};
xhr.send(null);
};
} else if (masterConfig.env === 'rhino' || (!masterConfig.env &&
typeof Packages !== 'undefined' && typeof java !== 'undefined')) {
//Why Java, why is this so awkward?
text.get = function (url, callback) {
var stringBuffer, line,
encoding = "utf-8",
file = new java.io.File(url),
lineSeparator = java.lang.System.getProperty("line.separator"),
input = new java.io.BufferedReader(new java.io.InputStreamReader(new java.io.FileInputStream(file), encoding)),
content = '';
try {
stringBuffer = new java.lang.StringBuffer();
line = input.readLine();
// Byte Order Mark (BOM) - The Unicode Standard, version 3.0, page 324
// http://www.unicode.org/faq/utf_bom.html
// Note that when we use utf-8, the BOM should appear as "EF BB BF", but it doesn't due to this bug in the JDK:
// http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=4508058
if (line && line.length() && line.charAt(0) === 0xfeff) {
// Eat the BOM, since we've already found the encoding on this file,
// and we plan to concatenating this buffer with others; the BOM should
// only appear at the top of a file.
line = line.substring(1);
}
if (line !== null) {
stringBuffer.append(line);
}
while ((line = input.readLine()) !== null) {
stringBuffer.append(lineSeparator);
stringBuffer.append(line);
}
//Make sure we return a JavaScript string and not a Java string.
content = String(stringBuffer.toString()); //String
} finally {
input.close();
}
callback(content);
};
} else if (masterConfig.env === 'xpconnect' || (!masterConfig.env &&
typeof Components !== 'undefined' && Components.classes &&
Components.interfaces)) {
//Avert your gaze!
Cc = Components.classes;
Ci = Components.interfaces;
Components.utils['import']('resource://gre/modules/FileUtils.jsm');
xpcIsWindows = ('@mozilla.org/windows-registry-key;1' in Cc);
text.get = function (url, callback) {
var inStream, convertStream, fileObj,
readData = {};
if (xpcIsWindows) {
url = url.replace(/\//g, '\\');
}
fileObj = new FileUtils.File(url);
//XPCOM, you so crazy
try {
inStream = Cc['@mozilla.org/network/file-input-stream;1']
.createInstance(Ci.nsIFileInputStream);
inStream.init(fileObj, 1, 0, false);
convertStream = Cc['@mozilla.org/intl/converter-input-stream;1']
.createInstance(Ci.nsIConverterInputStream);
convertStream.init(inStream, "utf-8", inStream.available(),
Ci.nsIConverterInputStream.DEFAULT_REPLACEMENT_CHARACTER);
convertStream.readString(inStream.available(), readData);
convertStream.close();
inStream.close();
callback(readData.value);
} catch (e) {
throw new Error((fileObj && fileObj.path || '') + ': ' + e);
}
2017-07-22 22:21:05 +02:00
};
}
return text;
});
define('text!ca',[],function () { return '{\n "domain": "converse",\n "locale_data": {\n "converse": {\n "": {\n "domain": "converse",\n "plural_forms": "nplurals=2; plural=(n != 1);",\n "lang": "ca"\n },\n "Bookmark this room": [\n null,\n ""\n ],\n "The name for this bookmark:": [\n null,\n ""\n ],\n "Would you like this room to be automatically joined upon startup?": [\n null,\n ""\n ],\n "What should your nickname for this room be?": [\n null,\n ""\n ],\n "Save": [\n null,\n "Desa"\n ],\n "Cancel": [\n null,\n "Cancel·la"\n ],\n "Bookmarks": [\n null,\n ""\n ],\n "Remove this bookmark": [\n null,\n ""\n ],\n "Show more information on this room": [\n null,\n "Mostra més informació d\'aquesta sala"\n ],\n "Click to open this room": [\n null,\n "Feu clic per obrir aquesta sala"\n ],\n "Close this chat box": [\n null,\n "Tanca aquest quadre del xat"\n ],\n "Personal message": [\n null,\n "Missatge personal"\n ],\n "Send": [\n null,\n ""\n ],\n "me": [\n null,\n "jo"\n ],\n "A very large message has been received.This might be due to an attack meant to degrade the chat performance.Output has been shortened.": [\n null,\n ""\n ],\n "Typing from another device": [\n null,\n ""\n ],\n "is typing": [\n null,\n "està escrivint"\n ],\n "Stopped typing on the other device": [\n null,\n ""\n ],\n "has stopped typing": [\n null,\n "ha deixat d\'escriure"\n ],\n "has gone away": [\n null,\n "ha marxat"\n ],\n "Show this menu": [\n null,\n "Mostra aquest menú"\n ],\n "Write in the third person": [\n null,\n "Escriu en tercera persona"\n ],\n "Remove messages": [\n null,\n "Elimina els missatges"\n ],\n "Are you sure you want to clear the messages from this chat box?": [\n null,\n "Segur que voleu esborrar els missatges d\'aquest quadre del xat?"\n ],\n "has gone offline": [\n null,\n "s\'ha desconnectat"\n ],\n "is busy": [\n null,\n "està ocupat"\n ],\n "Clear all messages": [\n null,\n "Esborra tots els missatges"\n ],\n "Insert a smiley": [\n null,\n "Insereix una cara somrient"\n ],\n "Start a call": [\n null,\n "Inicia una trucada"\n ],\n "Contacts": [\n null,\n "Contactes"\n ],\n "XMPP Username:": [\n null,\n "Nom d\'usuari XMPP:"\n ],\n "Password:": [\n null,\n "Contrasenya:"\n ],\n "Click here to log in anonymously": [\n null,\n "Feu clic aquí per iniciar la sessió de manera anònima"\n ],\n "Log In": [\n null,\n "Inicia la sessió"\n ],\n "user@server": [\n null,\n "usuari@servidor"\n ],\n "password": [\n null,\n "contrasenya"\n ],\n "Sign in": [\n null,\n "Inicia la sessió"\n ],\n "I am %1$s": [\n null,\n "Estic %1$s"\n ],\n "Click
/* Lo-Dash Template Loader v1.0.1
* Copyright 2015, Tim Branyen (@tbranyen).
* loader.js may be freely distributed under the MIT license.
*/
(function(global) {
"use strict";
// Cache used to map configuration options between load and write.
var buildMap = {};
// Alias the correct `nodeRequire` method.
var nodeRequire = typeof requirejs === "function" && requirejs.nodeRequire;
// Strips trailing `/` from url fragments.
var stripTrailing = function(prop) {
return prop.replace(/(\/$)/, '');
};
// Define the plugin using the CommonJS syntax.
define('tpl',['require','exports','module','lodash'],function(require, exports) {
var _ = require("lodash");
exports.version = "1.0.1";
// Invoked by the AMD builder, passed the path to resolve, the require
// function, done callback, and the configuration options.
exports.load = function(name, req, load, config) {
var isDojo;
// Dojo provides access to the config object through the req function.
if (!config) {
config = require.rawConfig;
isDojo = true;
}
var contents = "";
var settings = configure(config);
// If the baseUrl and root are the same, just null out the root.
if (stripTrailing(config.baseUrl) === stripTrailing(settings.root)) {
settings.root = '';
}
var url = require.toUrl(settings.root + name + settings.ext);
if (isDojo && url.indexOf(config.baseUrl) !== 0) {
url = stripTrailing(config.baseUrl) + url;
}
// Builds with r.js require Node.js to be installed.
if (config.isBuild) {
// If in Node, get access to the filesystem.
var fs = nodeRequire("fs");
try {
// First try reading the filepath as-is.
contents = String(fs.readFileSync(url));
} catch(ex) {
// If it failed, it's most likely because of a leading `/` and not an
// absolute path. Remove the leading slash and try again.
if (url.slice(0, 1) === "/") {
url = url.slice(1);
}
// Try reading again with the leading `/`.
contents = String(fs.readFileSync(url));
}
// Read in the file synchronously, as RequireJS expects, and return the
// contents. Process as a Lo-Dash template.
buildMap[name] = _.template(contents);
return load();
}
// Create a basic XHR.
var xhr = new XMLHttpRequest();
// Wait for it to load.
xhr.onreadystatechange = function() {
if (xhr.readyState === 4) {
var templateSettings = _.clone(settings.templateSettings);
// Attach the sourceURL.
templateSettings.sourceURL = url;
// Process as a Lo-Dash template and cache.
buildMap[name] = _.template(xhr.responseText, templateSettings);
// Return the compiled template.
load(buildMap[name]);
}
};
// Initiate the fetch.
xhr.open("GET", url, true);
xhr.send(null);
};
// Also invoked by the AMD builder, this writes out a compatible define
// call that will work with loaders such as almond.js that cannot read
// the configuration data.
exports.write = function(pluginName, moduleName, write) {
var template = buildMap[moduleName].source;
// Write out the actual definition
write(strDefine(pluginName, moduleName, template));
};
// This is for curl.js/cram.js build-time support.
exports.compile = function(pluginName, moduleName, req, io, config) {
configure(config);
// Ask cram to fetch the template file (resId) and pass it to `write`.
io.read(moduleName, write, io.error);
function write(template) {
// Write-out define(id,function(){return{/* template */}});
io.write(strDefine(pluginName, moduleName, template));
}
};
// Crafts the written definition form of the module during a build.
function strDefine(pluginName, moduleName, template) {
return [
"define('", pluginName, "!", moduleName, "', ", "['lodash'], ",
[
"function(_) {",
"return ", template, ";",
"}"
].join(""),
");\n"
].join("");
}
function configure(config) {
// Default settings point to the project root and using html files.
var settings = _.extend({
ext: ".html",
root: config.baseUrl,
templateSettings: {}
}, config.lodashLoader);
// Ensure the root has been properly configured with a trailing slash,
// unless it's an empty string or undefined, in which case work off the
// baseUrl.
if (settings.root && settings.root.slice(-1) !== "/") {
settings.root += "/";
}
// Set the custom passed in template settings.
_.extend(_.templateSettings, settings.templateSettings);
return settings;
}
});
})(typeof global === "object" ? global : this);
define('tpl!field', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
__p += '<field var="' +
__e(name) +
'">';
if (_.isArray(value)) { ;
__p += '\n ';
_.each(value,function(arrayValue) { ;
__p += '<value>' +
__e(arrayValue) +
'</value>';
}); ;
__p += '\n';
} else { ;
__p += '\n <value>' +
__e(value) +
'</value>\n';
} ;
__p += '</field>\n';
}
return __p
};});
define('tpl!select_option', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
__p += '<option value="' +
__e(value) +
'" ';
if (selected) { ;
__p += ' selected="selected" ';
} ;
__p += ' >' +
__e(label) +
'</option>\n';
}
return __p
};});
define('tpl!form_select', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
__p += '<label>' +
__e(label) +
'</label>\n<select name="' +
__e(name) +
'" ';
if (multiple) { ;
__p += ' multiple="multiple" ';
} ;
__p += '>' +
((__t = (options)) == null ? '' : __t) +
'</select>\n';
}
return __p
};});
define('tpl!form_textarea', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<label class="label-ta">' +
__e(label) +
'</label>\n<textarea name="' +
__e(name) +
'">' +
__e(value) +
'</textarea>\n';
}
return __p
};});
define('tpl!form_checkbox', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<label>' +
__e(label) +
'</label>\n<input name="' +
__e(name) +
'" type="' +
__e(type) +
'" ' +
__e(checked) +
'>\n';
2017-07-22 22:21:05 +02:00
}
return __p
};});
define('tpl!form_username', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
if (label) { ;
__p += '\n<label>\n ' +
__e(label) +
'\n</label>\n';
} ;
__p += '\n<div class="input-group">\n <input name="' +
__e(name) +
'" type="' +
__e(type) +
'"\n ';
if (value) { ;
__p += ' value="' +
__e(value) +
'" ';
} ;
__p += '\n ';
if (required) { ;
__p += ' class="required" ';
} ;
__p += ' />\n <span title="' +
__e(domain) +
'">' +
__e(domain) +
'</span>\n</div>\n';
}
return __p
};});
define('tpl!form_input', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
if (label) { ;
__p += '\n<label>\n ' +
__e(label) +
'\n</label>\n';
} ;
__p += '\n<input name="' +
__e(name) +
'" type="' +
__e(type) +
'" \n ';
if (value) { ;
__p += ' value="' +
__e(value) +
'" ';
} ;
__p += '\n ';
if (required) { ;
__p += ' class="required" ';
} ;
__p += ' >\n';
}
return __p
};});
define('tpl!form_captcha', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
if (label) { ;
__p += '\n<label>\n ' +
__e(label) +
'\n</label>\n';
} ;
__p += '\n<img src="data:' +
__e(type) +
';base64,' +
__e(data) +
'">\n<input name="' +
__e(name) +
'" type="text" ';
if (required) { ;
__p += ' class="required" ';
} ;
__p += ' >\n\n\n';
}
return __p
};});
var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; };
/*global define, escape, locales, Jed */
(function (root, factory) {
define('utils',["jquery.noconflict", "sizzle", "es6-promise", "jquery.browser", "lodash.noconflict", "locales", "moment_with_locales", "strophe", "tpl!field", "tpl!select_option", "tpl!form_select", "tpl!form_textarea", "tpl!form_checkbox", "tpl!form_username", "tpl!form_input", "tpl!form_captcha"], factory);
})(undefined, function ($, sizzle, Promise, dummy, _, locales, moment, Strophe, tpl_field, tpl_select_option, tpl_form_select, tpl_form_textarea, tpl_form_checkbox, tpl_form_username, tpl_form_input, tpl_form_captcha) {
"use strict";
locales = locales || {};
var b64_sha1 = Strophe.SHA1.b64_sha1;
Strophe = Strophe.Strophe;
var URL_REGEX = /\b(https?:\/\/|www\.|https?:\/\/www\.)[^\s<>]{2,200}\b/g;
var XFORM_TYPE_MAP = {
'text-private': 'password',
'text-single': 'text',
'fixed': 'label',
'boolean': 'checkbox',
'hidden': 'hidden',
'jid-multi': 'textarea',
'list-single': 'dropdown',
'list-multi': 'dropdown'
};
var afterAnimationEnd = function afterAnimationEnd(el, callback) {
el.classList.remove('visible');
if (_.isFunction(callback)) {
callback();
}
2017-07-22 22:21:05 +02:00
};
2017-07-22 22:21:05 +02:00
var unescapeHTML = function unescapeHTML(htmlEscapedText) {
/* Helper method that replace HTML-escaped symbols with equivalent characters
* (e.g. transform occurrences of '&amp;' 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 isImage(url) {
return new Promise(function (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;
});
};
$.fn.hasScrollBar = function () {
if (!$.contains(document, this.get(0))) {
return false;
}
2017-07-22 22:21:05 +02:00
if (this.parent().height() < this.get(0).scrollHeight) {
return true;
}
return false;
};
2017-07-22 22:21:05 +02:00
function calculateSlideStep(height) {
if (height > 100) {
return 10;
} else if (height > 50) {
return 5;
} else {
return 1;
}
}
2017-07-22 22:21:05 +02:00
var utils = {};
2017-07-22 22:21:05 +02:00
// Translation machinery
// ---------------------
utils.__ = function (str) {
if (!utils.isConverseLocale(this.locale) || this.locale === 'en') {
return Jed.sprintf.apply(Jed, arguments);
}
if (typeof this.jed === "undefined") {
this.jed = new Jed(window.JSON.parse(locales[this.locale]));
}
var t = this.jed.translate(str);
if (arguments.length > 1) {
return t.fetch.apply(t, [].slice.call(arguments, 1));
} else {
return t.fetch();
}
};
2017-07-22 22:21:05 +02:00
utils.___ = function (str) {
/* XXX: This is part of a hack to get gettext to scan strings to be
* translated. Strings we cannot send to the function above because
* they require variable interpolation and we don't yet have the
* variables at scan time.
*
* See actionInfoMessages in src/converse-muc.js
*/
return str;
};
2017-07-22 22:21:05 +02:00
utils.isLocaleAvailable = function (locale, available) {
/* Check whether the locale or sub locale (e.g. en-US, en) is supported.
*
* Parameters:
* (Function) available - returns a boolean indicating whether the locale is supported
*/
if (available(locale)) {
return locale;
} else {
var sublocale = locale.split("-")[0];
if (sublocale !== locale && available(sublocale)) {
return sublocale;
}
}
};
2017-07-22 22:21:05 +02:00
utils.addHyperlinks = function (text) {
var list = text.match(URL_REGEX) || [];
var links = [];
_.each(list, function (match) {
var prot = match.indexOf('http://') === 0 || match.indexOf('https://') === 0 ? '' : 'http://';
var url = prot + encodeURI(decodeURI(match)).replace(/[!'()]/g, escape).replace(/\*/g, "%2A");
var 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) {
var a = links.pop();
text = text.replace(b64_sha1(a), a);
}
return text;
};
2017-07-22 22:21:05 +02:00
utils.renderImageURLs = function (obj) {
var list = obj.textContent.match(URL_REGEX) || [];
_.forEach(list, function (url) {
isImage(url).then(function (img) {
img.className = 'chat-image';
var anchors = sizzle("a[href=\"" + url + "\"]", obj);
_.each(anchors, function (a) {
a.innerHTML = img.outerHTML;
});
});
});
return obj;
};
2017-07-22 22:21:05 +02:00
utils.slideInAllElements = function (elements) {
return Promise.all(_.map(elements, _.partial(utils.slideIn, _, 600)));
};
2017-07-22 22:21:05 +02:00
utils.slideToggleElement = function (el) {
if (_.includes(el.classList, 'collapsed')) {
return utils.slideOut(el);
} else {
return utils.slideIn(el);
}
};
2017-07-22 22:21:05 +02:00
utils.slideOut = function (el) {
var duration = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 900;
/* Shows/expands an element by sliding it out of itself. */
function calculateEndHeight(el) {
return _.reduce(el.children, function (result, child) {
return result + child.offsetHeight;
}, 0);
}
function wrapup(el) {
el.removeAttribute('data-slider-marker');
el.classList.remove('collapsed');
el.style.overflow = "";
el.style.height = "";
}
2017-07-22 22:21:05 +02:00
return new Promise(function (resolve, reject) {
if (_.isNil(el)) {
var err = "Undefined or null element passed into slideOut";
console.warn(err);
reject(new Error(err));
return;
}
var interval_marker = el.getAttribute('data-slider-marker');
if (interval_marker) {
el.removeAttribute('data-slider-marker');
window.clearInterval(interval_marker);
}
var end_height = calculateEndHeight(el);
if ($.fx.off) {
// Effects are disabled (for tests)
el.style.height = end_height + 'px';
wrapup(el);
resolve();
return;
}
var step = calculateSlideStep(end_height),
interval = end_height / duration * step;
var h = 0;
interval_marker = window.setInterval(function () {
h += step;
if (h < end_height) {
el.style.height = h + 'px';
} else {
// We recalculate the height to work around an apparent
// browser bug where browsers don't know the correct
// offsetHeight beforehand.
el.style.height = calculateEndHeight(el) + 'px';
window.clearInterval(interval_marker);
wrapup(el);
resolve();
}
}, interval);
el.setAttribute('data-slider-marker', interval_marker);
});
};
utils.slideIn = function (el) {
var duration = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 600;
/* Hides/collapses an element by sliding it into itself. */
return new Promise(function (resolve, reject) {
if (_.isNil(el)) {
var err = "Undefined or null element passed into slideIn";
console.warn(err);
return reject(new Error(err));
} else if (_.includes(el.classList, 'collapsed')) {
return resolve();
} else if ($.fx.off) {
// Effects are disabled (for tests)
el.classList.add('collapsed');
el.style.height = "";
return resolve();
}
var interval_marker = el.getAttribute('data-slider-marker');
if (interval_marker) {
el.removeAttribute('data-slider-marker');
window.clearInterval(interval_marker);
}
var h = el.offsetHeight;
var step = calculateSlideStep(h),
interval = h / duration * step;
2017-07-22 22:21:05 +02:00
el.style.overflow = 'hidden';
2017-07-22 22:21:05 +02:00
interval_marker = window.setInterval(function () {
h -= step;
if (h > 0) {
el.style.height = h + 'px';
} else {
el.removeAttribute('data-slider-marker');
window.clearInterval(interval_marker);
el.classList.add('collapsed');
el.style.height = "";
resolve();
}
}, interval);
el.setAttribute('data-slider-marker', interval_marker);
});
};
2017-07-22 22:21:05 +02:00
utils.fadeIn = function (el, callback) {
if (_.isNil(el)) {
console.warn("Undefined or null element passed into fadeIn");
}
2017-07-22 22:21:05 +02:00
if ($.fx.off) {
el.classList.remove('hidden');
if (_.isFunction(callback)) {
callback();
}
2017-07-22 22:21:05 +02:00
return;
}
2017-07-22 22:21:05 +02:00
if (_.includes(el.classList, 'hidden')) {
/* XXX: This doesn't appear to be working...
el.addEventListener("webkitAnimationEnd", _.partial(afterAnimationEnd, el, callback), false);
el.addEventListener("animationend", _.partial(afterAnimationEnd, el, callback), false);
*/
setTimeout(_.partial(afterAnimationEnd, el, callback), 351);
el.classList.add('visible');
el.classList.remove('hidden');
} else {
2017-07-22 22:21:05 +02:00
afterAnimationEnd(el, callback);
}
2017-07-22 22:21:05 +02:00
};
utils.isSameBareJID = function (jid1, jid2) {
return Strophe.getBareJidFromJid(jid1).toLowerCase() === Strophe.getBareJidFromJid(jid2).toLowerCase();
};
utils.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;
} else {
2017-07-22 22:21:05 +02:00
return !message.get('archive_id');
}
2017-07-22 22:21:05 +02:00
};
2016-12-13 20:46:07 +01:00
2017-07-22 22:21:05 +02:00
utils.isOTRMessage = function (message) {
var body = message.querySelector('body'),
text = !_.isNull(body) ? body.textContent : undefined;
return text && !!text.match(/^\?OTR/);
};
2017-07-22 22:21:05 +02:00
utils.isHeadlineMessage = function (message) {
var from_jid = message.getAttribute('from');
if (message.getAttribute('type') === 'headline') {
return true;
}
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;
};
2017-07-22 22:21:05 +02:00
utils.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];
}
}
};
2017-07-22 22:21:05 +02:00
utils.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];
}
}
};
2017-07-22 22:21:05 +02:00
utils.refreshWebkit = function () {
/* This works around a webkit bug. Refreshes the browser's viewport,
* otherwise chatboxes are not moved along when one is closed.
*/
if ($.browser.webkit && window.requestAnimationFrame) {
window.requestAnimationFrame(function () {
var conversejs = document.getElementById('conversejs');
conversejs.style.display = 'none';
var tmp = conversejs.offsetHeight; // jshint ignore:line
conversejs.style.display = 'block';
});
}
};
2017-07-22 22:21:05 +02:00
utils.webForm2xForm = function (field) {
/* Takes an HTML DOM and turns it into an XForm field.
*
* Parameters:
* (DOMElement) field - the field to convert
*/
var $input = $(field),
value;
if ($input.is('[type=checkbox]')) {
value = $input.is(':checked') && 1 || 0;
} else if ($input.is('textarea')) {
value = [];
var lines = $input.val().split('\n');
for (var vk = 0; vk < lines.length; vk++) {
var val = $.trim(lines[vk]);
if (val === '') continue;
value.push(val);
}
} else {
value = $input.val();
}
return $(tpl_field({
name: $input.attr('name'),
value: value
}))[0];
};
2017-07-22 22:21:05 +02:00
utils.contains = function (attr, query) {
return function (item) {
if ((typeof attr === "undefined" ? "undefined" : _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.');
}
};
};
utils.xForm2webForm = function ($field, $stanza) {
/* Takes a field in XMPP XForm (XEP-004: Data Forms) format
* and turns it into a HTML DOM field.
*
* Parameters:
* (XMLElement) field - the field to convert
*/
// FIXME: take <required> into consideration
var options = [],
j,
$options,
$values,
value,
values;
if ($field.attr('type') === 'list-single' || $field.attr('type') === 'list-multi') {
values = [];
$values = $field.children('value');
for (j = 0; j < $values.length; j++) {
values.push($($values[j]).text());
}
$options = $field.children('option');
for (j = 0; j < $options.length; j++) {
value = $($options[j]).find('value').text();
options.push(tpl_select_option({
value: value,
label: $($options[j]).attr('label'),
selected: _.startsWith(values, value),
required: $field.find('required').length
}));
}
return tpl_form_select({
name: $field.attr('var'),
label: $field.attr('label'),
options: options.join(''),
multiple: $field.attr('type') === 'list-multi',
required: $field.find('required').length
});
} else if ($field.attr('type') === 'fixed') {
return $('<p class="form-help">').text($field.find('value').text());
} else if ($field.attr('type') === 'jid-multi') {
return tpl_form_textarea({
name: $field.attr('var'),
label: $field.attr('label') || '',
value: $field.find('value').text(),
required: $field.find('required').length
});
} else if ($field.attr('type') === 'boolean') {
return tpl_form_checkbox({
name: $field.attr('var'),
type: XFORM_TYPE_MAP[$field.attr('type')],
label: $field.attr('label') || '',
checked: $field.find('value').text() === "1" && 'checked="1"' || '',
required: $field.find('required').length
});
} else if ($field.attr('type') && $field.attr('var') === 'username') {
return tpl_form_username({
domain: ' @' + this.domain,
name: $field.attr('var'),
type: XFORM_TYPE_MAP[$field.attr('type')],
label: $field.attr('label') || '',
value: $field.find('value').text(),
required: $field.find('required').length
});
} else if ($field.attr('type')) {
return tpl_form_input({
name: $field.attr('var'),
type: XFORM_TYPE_MAP[$field.attr('type')],
label: $field.attr('label') || '',
value: $field.find('value').text(),
required: $field.find('required').length
});
} else {
if ($field.attr('var') === 'ocr') {
// Captcha
return _.reduce(_.map($field.find('uri'), $.proxy(function (uri) {
return tpl_form_captcha({
label: this.$field.attr('label'),
name: this.$field.attr('var'),
data: this.$stanza.find('data[cid="' + uri.textContent.replace(/^cid:/, '') + '"]').text(),
type: uri.getAttribute('type'),
required: this.$field.find('required').length
});
}, { '$stanza': $stanza, '$field': $field })), function (memo, num) {
return memo + num;
}, '');
}
}
};
2017-07-22 22:21:05 +02:00
utils.detectLocale = function (library_check) {
/* Determine which locale is supported by the user's system as well
* as by the relevant library (e.g. converse.js or moment.js).
*
* Parameters:
* (Function) library_check - returns a boolean indicating whether
* the locale is supported.
*/
var locale, i;
if (window.navigator.userLanguage) {
locale = utils.isLocaleAvailable(window.navigator.userLanguage, library_check);
}
if (window.navigator.languages && !locale) {
for (i = 0; i < window.navigator.languages.length && !locale; i++) {
locale = utils.isLocaleAvailable(window.navigator.languages[i], library_check);
}
}
2017-07-22 22:21:05 +02:00
if (window.navigator.browserLanguage && !locale) {
locale = utils.isLocaleAvailable(window.navigator.browserLanguage, library_check);
}
if (window.navigator.language && !locale) {
locale = utils.isLocaleAvailable(window.navigator.language, library_check);
}
if (window.navigator.systemLanguage && !locale) {
locale = utils.isLocaleAvailable(window.navigator.systemLanguage, library_check);
}
return locale || 'en';
};
2017-07-22 22:21:05 +02:00
utils.isConverseLocale = function (locale) {
if (!_.isString(locale)) {
return false;
}
return _.includes(_.keys(locales || {}), locale);
};
2017-07-22 22:21:05 +02:00
utils.isMomentLocale = function (locale) {
if (!_.isString(locale)) {
return false;
}
return moment.locale() !== moment.locale(locale);
};
utils.getLocale = function (preferred_locale, isSupportedByLibrary) {
if (_.isString(preferred_locale)) {
if (preferred_locale === 'en' || isSupportedByLibrary(preferred_locale)) {
return preferred_locale;
}
2017-07-22 22:21:05 +02:00
try {
var obj = window.JSON.parse(preferred_locale);
return obj.locale_data.converse[""].lang;
} catch (e) {
console.log(e);
}
}
return utils.detectLocale(isSupportedByLibrary) || 'en';
};
2017-07-22 22:21:05 +02:00
utils.isOfType = function (type, item) {
return item.get('type') == type;
};
2017-07-22 22:21:05 +02:00
utils.isInstance = function (type, item) {
return item instanceof type;
};
2017-07-22 22:21:05 +02:00
utils.getAttribute = function (key, item) {
return item.get(key);
};
2017-07-22 22:21:05 +02:00
utils.contains.not = function (attr, query) {
return function (item) {
return !utils.contains(attr, query)(item);
};
};
2017-07-22 22:21:05 +02:00
utils.createFragmentFromText = function (markup) {
/* Returns a DocumentFragment containing DOM nodes based on the
* passed-in markup text.
*/
2017-07-22 22:21:05 +02:00
// 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;
};
2017-07-22 22:21:05 +02:00
utils.addEmoji = function (_converse, emojione, text) {
if (_converse.use_emojione) {
return emojione.toImage(text);
} else {
return emojione.shortnameToUnicode(text);
}
};
2017-07-22 22:21:05 +02:00
utils.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)) {
var emojis = _.values(_.mapValues(emojione.emojioneList, function (value, key, o) {
value._shortname = key;
return value;
}));
var tones = [':tone1:', ':tone2:', ':tone3:', ':tone4:', ':tone5:'];
var excluded = [':kiss_ww:', ':kiss_mm:', ':kiss_woman_man:'];
var excluded_substrings = [':woman', ':man', ':women_', ':men_', '_man_', '_woman_', '_woman:', '_man:'];
var excluded_categories = ['modifier', 'regional'];
var categories = _.difference(_.uniq(_.map(emojis, _.partial(_.get, _, 'category'))), excluded_categories);
var emojis_by_category = {};
_.forEach(categories, function (cat) {
var list = _.sortBy(_.filter(emojis, ['category', cat]), ['uc_base']);
list = _.filter(list, function (item) {
return !_.includes(_.concat(tones, excluded), item._shortname) && !_.some(excluded_substrings, _.partial(_.includes, item._shortname));
});
if (cat === 'people') {
var 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));
}
2017-07-22 22:21:05 +02:00
emojis_by_category[cat] = list;
});
_converse.emojis_by_category = emojis_by_category;
}
return _converse.emojis_by_category;
};
2017-07-22 22:21:05 +02:00
utils.getTonedEmojis = function (_converse) {
_converse.toned_emojis = _.uniq(_.map(_.filter(utils.getEmojisByCategory(_converse).people, function (person) {
return _.includes(person._shortname, '_tone');
}), function (person) {
return person._shortname.replace(/_tone[1-5]/, '');
}));
return _converse.toned_emojis;
};
2017-07-22 22:21:05 +02:00
utils.isPersistableModel = function (model) {
return model.collection && model.collection.browserStorage;
};
2017-07-22 22:21:05 +02:00
utils.getWrappedPromise = function () {
var wrapper = {};
wrapper.promise = new Promise(function (resolve, reject) {
wrapper.resolve = resolve;
wrapper.reject = reject;
});
return wrapper;
};
2017-07-22 22:21:05 +02:00
utils.safeSave = function (model, attributes) {
if (utils.isPersistableModel(model)) {
model.save(attributes);
} else {
model.set(attributes);
}
};
return utils;
});
//# sourceMappingURL=utils.js.map;
(function (global, factory) {
if (typeof define === "function" && define.amd) {
define('pluggable',['exports', 'lodash'], factory);
} else if (typeof exports !== "undefined") {
factory(exports, require('lodash'));
} else {
var mod = {
exports: {}
};
factory(mod.exports, global._);
global.pluggable = mod.exports;
}
})(this, function (exports, _lodash) {
'use strict';
2016-12-13 20:46:07 +01:00
2017-07-22 22:21:05 +02:00
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.enable = undefined;
2017-07-22 22:21:05 +02:00
var _ = _interopRequireWildcard(_lodash);
2017-07-22 22:21:05 +02:00
function _interopRequireWildcard(obj) {
if (obj && obj.__esModule) {
return obj;
} else {
var newObj = {};
if (obj != null) {
for (var key in obj) {
if (Object.prototype.hasOwnProperty.call(obj, key)) newObj[key] = obj[key];
}
}
2017-07-22 22:21:05 +02:00
newObj.default = obj;
return newObj;
}
}
2017-07-22 22:21:05 +02:00
var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) {
return typeof obj;
} : function (obj) {
return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj;
};
2017-07-22 22:21:05 +02:00
// The `PluginSocket` class contains the plugin architecture, and gets
// created whenever `pluggable.enable(obj);` is called on the object
// that you want to make pluggable.
// You can also see it as the thing into which the plugins are plugged.
// It takes two parameters, first, the object being made pluggable, and
// then the name by which the pluggable object may be referenced on the
// __super__ object (inside overrides).
function PluginSocket(plugged, name) {
this.name = name;
this.plugged = plugged;
if (typeof this.plugged.__super__ === 'undefined') {
this.plugged.__super__ = {};
} else if (typeof this.plugged.__super__ === 'string') {
this.plugged.__super__ = { '__string__': this.plugged.__super__ };
}
this.plugged.__super__[name] = this.plugged;
this.plugins = {};
this.initialized_plugins = [];
}
2017-07-22 22:21:05 +02:00
// Now we add methods to the PluginSocket by adding them to its
// prototype.
_.extend(PluginSocket.prototype, {
2017-07-22 22:21:05 +02:00
// `wrappedOverride` creates a partially applied wrapper function
// that makes sure to set the proper super method when the
// overriding method is called. This is done to enable
// chaining of plugin methods, all the way up to the
// original method.
wrappedOverride: function wrappedOverride(key, value, super_method, default_super) {
if (typeof super_method === "function") {
if (typeof this.__super__ === "undefined") {
/* We're not on the context of the plugged object.
* This can happen when the overridden method is called via
* an event handler or when it's a constructor.
*
* In this case, we simply tack on the __super__ obj.
*/
this.__super__ = default_super;
}
this.__super__[key] = super_method.bind(this);
}
2017-07-22 22:21:05 +02:00
return value.apply(this, _.drop(arguments, 4));
},
2017-07-22 22:21:05 +02:00
// `_overrideAttribute` overrides an attribute on the original object
// (the thing being plugged into).
//
// If the attribute being overridden is a function, then the original
// function will still be available via the `__super__` attribute.
//
// If the same function is being overridden multiple times, then
// the original function will be available at the end of a chain of
// functions, starting from the most recent override, all the way
// back to the original function, each being referenced by the
// previous' __super__ attribute.
//
// For example:
//
// `plugin2.MyFunc.__super__.myFunc => plugin1.MyFunc.__super__.myFunc => original.myFunc`
_overrideAttribute: function _overrideAttribute(key, plugin) {
var value = plugin.overrides[key];
if (typeof value === "function") {
var default_super = {};
default_super[this.name] = this.plugged;
2017-07-22 22:21:05 +02:00
var wrapped_function = _.partial(this.wrappedOverride, key, value, this.plugged[key], default_super);
this.plugged[key] = wrapped_function;
} else {
this.plugged[key] = value;
}
},
2017-07-22 22:21:05 +02:00
_extendObject: function _extendObject(obj, attributes) {
if (!obj.prototype.__super__) {
obj.prototype.__super__ = {};
obj.prototype.__super__[this.name] = this.plugged;
}
var that = this;
_.each(attributes, function (value, key) {
if (key === 'events') {
obj.prototype[key] = _.extend(value, obj.prototype[key]);
} else if (typeof value === 'function') {
// We create a partially applied wrapper function, that
// makes sure to set the proper super method when the
// overriding method is called. This is done to enable
// chaining of plugin methods, all the way up to the
// original method.
var default_super = {};
default_super[that.name] = that.plugged;
2017-07-22 22:21:05 +02:00
var wrapped_function = _.partial(that.wrappedOverride, key, value, obj.prototype[key], default_super);
obj.prototype[key] = wrapped_function;
} else {
obj.prototype[key] = value;
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
});
},
2017-07-22 22:21:05 +02:00
// Plugins can specify optional dependencies (by means of the
// `optional_dependencies` list attribute) which refers to dependencies
// which will be initialized first, before the plugin itself gets initialized.
// They are optional in the sense that if they aren't available, an
// error won't be thrown.
// However, if you want to make these dependencies strict (i.e.
// non-optional), you can set the `strict_plugin_dependencies` attribute to `true`
// on the object being made pluggable (i.e. the object passed to
// `pluggable.enable`).
loadOptionalDependencies: function loadOptionalDependencies(plugin) {
var _this = this;
_.each(plugin.optional_dependencies, function (name) {
var dep = _this.plugins[name];
if (dep) {
if (_.includes(dep.optional_dependencies, plugin.__name__)) {
/* FIXME: circular dependency checking is only one level deep. */
throw "Found a circular dependency between the plugins \"" + plugin.__name__ + "\" and \"" + name + "\"";
}
2017-07-22 22:21:05 +02:00
_this.initializePlugin(dep);
} else {
_this.throwUndefinedDependencyError("Could not find optional dependency \"" + name + "\" " + "for the plugin \"" + plugin.__name__ + "\". " + "If it's needed, make sure it's loaded by require.js");
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
});
},
2017-07-22 22:21:05 +02:00
throwUndefinedDependencyError: function throwUndefinedDependencyError(msg) {
if (this.plugged.strict_plugin_dependencies) {
throw msg;
} else {
console.log(msg);
return;
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
},
2017-07-22 22:21:05 +02:00
// `applyOverrides` is called by initializePlugin. It applies any
// and all overrides of methods or Backbone views and models that
// are defined on any of the plugins.
applyOverrides: function applyOverrides(plugin) {
var _this2 = this;
_.each(Object.keys(plugin.overrides || {}), function (key) {
var override = plugin.overrides[key];
if ((typeof override === 'undefined' ? 'undefined' : _typeof(override)) === "object") {
if (typeof _this2.plugged[key] === 'undefined') {
_this2.throwUndefinedDependencyError("Error: Plugin \"" + plugin.__name__ + "\" tried to override " + key + " but it's not found.");
} else {
2017-07-22 22:21:05 +02:00
_this2._extendObject(_this2.plugged[key], override);
}
2017-07-22 22:21:05 +02:00
} else {
_this2._overrideAttribute(key, plugin);
}
2017-07-22 22:21:05 +02:00
});
},
2017-07-22 22:21:05 +02:00
// `initializePlugin` applies the overrides (if any) defined on all
// the registered plugins and then calls the initialize method for each plugin.
initializePlugin: function initializePlugin(plugin) {
if (!_.includes(_.keys(this.allowed_plugins), plugin.__name__)) {
/* Don't initialize disallowed plugins. */
return;
}
if (_.includes(this.initialized_plugins, plugin.__name__)) {
/* Don't initialize plugins twice, otherwise we get
* infinite recursion in overridden methods.
*/
return;
}
_.extend(plugin, this.properties);
if (plugin.optional_dependencies) {
this.loadOptionalDependencies(plugin);
}
this.applyOverrides(plugin);
if (typeof plugin.initialize === "function") {
plugin.initialize.bind(plugin)(this);
}
this.initialized_plugins.push(plugin.__name__);
},
2017-07-22 22:21:05 +02:00
// `registerPlugin` registers (or inserts, if you'd like) a plugin,
// by adding it to the `plugins` map on the PluginSocket instance.
registerPlugin: function registerPlugin(name, plugin) {
if (name in this.plugins) {
throw new Error('Error: Plugin name ' + name + ' is already taken');
}
plugin.__name__ = name;
this.plugins[name] = plugin;
},
2016-08-12 22:52:21 +02:00
2017-07-22 22:21:05 +02:00
// `initializePlugins` should get called once all plugins have been
// registered. It will then iterate through all the plugins, calling
// `initializePlugin` for each.
// The passed in properties variable is an object with attributes and methods
// which will be attached to the plugins.
initializePlugins: function initializePlugins() {
var properties = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
var whitelist = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : [];
var blacklist = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : [];
2016-08-12 22:52:21 +02:00
2017-07-22 22:21:05 +02:00
if (!_.size(this.plugins)) {
return;
2017-04-23 19:02:44 +02:00
}
2017-07-22 22:21:05 +02:00
this.properties = properties;
this.allowed_plugins = _.pickBy(this.plugins, function (plugin, key) {
return (!whitelist.length || whitelist.length && _.includes(whitelist, key)) && !_.includes(blacklist, key);
});
_.each(_.values(this.allowed_plugins), this.initializePlugin.bind(this));
}
});
2016-06-20 21:11:43 +02:00
2017-07-22 22:21:05 +02:00
function enable(object, name, attrname) {
// Call the `enable` method to make an object pluggable
//
// It takes three parameters:
// - `object`: The object that gets made pluggable.
// - `name`: The string name by which the now pluggable object
// may be referenced on the __super__ obj (in overrides).
// The default value is "plugged".
// - `attrname`: The string name of the attribute on the now
// pluggable object, which refers to the PluginSocket instance
// that gets created.
if (typeof attrname === "undefined") {
attrname = "pluginSocket";
}
if (typeof name === 'undefined') {
name = 'plugged';
}
var ref = {};
ref[attrname] = new PluginSocket(object, name);
return _.extend(object, ref);
}
2016-06-20 21:11:43 +02:00
2017-07-22 22:21:05 +02:00
exports.enable = enable;
exports.default = {
enable: enable
};
});
2016-06-20 21:11:43 +02:00
//# sourceMappingURL=pluggable.js.map
;
2016-06-20 21:11:43 +02:00
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
// Licensed under the Mozilla Public License (MPLv2)
//
/*global Backbone, define, window, document, JSON */
(function (root, factory) {
define('converse-core',["sizzle", "es6-promise", "lodash.noconflict", "lodash.converter", "polyfill", "utils", "moment_with_locales", "strophe", "pluggable", "backbone.noconflict", "backbone.browserStorage", "backbone.overview"], factory);
})(undefined, function (sizzle, Promise, _, lodashConverter, polyfill, utils, moment, Strophe, pluggable, Backbone) {
2016-06-20 21:11:43 +02:00
2017-07-22 22:21:05 +02:00
/* Cannot use this due to Safari bug.
* See https://github.com/jcbrand/converse.js/issues/196
*/
// "use strict";
2017-07-22 22:21:05 +02:00
// Create the FP (functional programming) version of lodash
var fp = lodashConverter(_.runInContext());
2016-06-20 21:11:43 +02:00
2017-07-22 22:21:05 +02:00
// Strophe globals
var _Strophe = Strophe,
$build = _Strophe.$build,
$iq = _Strophe.$iq,
$msg = _Strophe.$msg,
$pres = _Strophe.$pres;
2016-06-20 21:11:43 +02:00
2017-07-22 22:21:05 +02:00
var b64_sha1 = Strophe.SHA1.b64_sha1;
Strophe = Strophe.Strophe;
2016-12-13 20:46:07 +01:00
2017-07-22 22:21:05 +02:00
// Use Mustache style syntax for variable interpolation
/* Configuration of Lodash templates (this config is distinct to the
* config of requirejs-tpl in main.js). This one is for normal inline templates.
*/
_.templateSettings = {
'escape': /\{\{\{([\s\S]+?)\}\}\}/g,
'evaluate': /\{\[([\s\S]+?)\]\}/g,
'interpolate': /\{\{([\s\S]+?)\}\}/g
};
2016-06-20 21:11:43 +02:00
2017-07-22 22:21:05 +02:00
var _converse = {
'templates': {},
'promises': {}
};
2017-07-22 22:21:05 +02:00
_.extend(_converse, Backbone.Events);
2016-06-20 21:11:43 +02:00
2017-07-22 22:21:05 +02:00
_converse.core_plugins = ['converse-bookmarks', 'converse-chatview', 'converse-controlbox', 'converse-core', 'converse-disco', 'converse-dragresize', 'converse-headline', 'converse-mam', 'converse-minimize', 'converse-muc', 'converse-notification', 'converse-otr', 'converse-ping', 'converse-register', 'converse-roomslist', 'converse-rosterview', 'converse-vcard'];
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
// Make converse pluggable
pluggable.enable(_converse, '_converse', 'pluggable');
2017-07-22 22:21:05 +02:00
// Module-level constants
_converse.STATUS_WEIGHTS = {
'offline': 6,
'unavailable': 5,
'xa': 4,
'away': 3,
'dnd': 2,
'chat': 1, // We currently don't differentiate between "chat" and "online"
'online': 1
};
_converse.PRETTY_CHAT_STATUS = {
'offline': 'Offline',
'unavailable': 'Unavailable',
'xa': 'Extended Away',
'away': 'Away',
'dnd': 'Do not disturb',
'chat': 'Chattty',
'online': 'Online'
};
_converse.ANONYMOUS = "anonymous";
_converse.CLOSED = 'closed';
_converse.EXTERNAL = "external";
_converse.LOGIN = "login";
_converse.LOGOUT = "logout";
_converse.OPENED = 'opened';
_converse.PREBIND = "prebind";
var PRETTY_CONNECTION_STATUS = {
0: 'ERROR',
1: 'CONNECTING',
2: 'CONNFAIL',
3: 'AUTHENTICATING',
4: 'AUTHFAIL',
5: 'CONNECTED',
6: 'DISCONNECTED',
7: 'DISCONNECTING',
8: 'ATTACHED',
9: 'REDIRECT'
};
var DEFAULT_IMAGE_TYPE = 'image/png';
var DEFAULT_IMAGE = "iVBORw0KGgoAAAANSUhEUgAAAGAAAABgCAIAAABt+uBvAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH3gwHCy455JBsggAABkJJREFUeNrtnM1PE1sUwHvvTD8otWLHST/Gimi1CEgr6M6FEWuIBo2pujDVsNDEP8GN/4MbN7oxrlipG2OCgZgYlxAbkRYw1KqkIDRCSkM7nXvvW8x7vjyNeQ9m7p1p3z1LQk/v/Dhz7vkEXL161cHl9wI5Ag6IA+KAOCAOiAPigDggLhwQB2S+iNZ+PcYY/SWEEP2HAAAIoSAIoihCCP+ngDDGtVotGAz29/cfOXJEUZSOjg6n06lp2sbGRqlUWlhYyGazS0tLbrdbEASrzgksyeYJId3d3el0uqenRxRFAAAA4KdfIIRgjD9+/Pj8+fOpqSndslofEIQwHA6Pjo4mEon//qmFhYXHjx8vLi4ihBgDEnp7e9l8E0Jo165dQ0NDd+/eDYVC2/qsJElDQ0OEkKWlpa2tLZamxAhQo9EIBoOjo6MXL17csZLe3l5FUT59+lQul5l5JRaAVFWNRqN37tw5ceKEQVWRSOTw4cOFQuHbt2+iKLYCIISQLMu3b99OJpOmKAwEAgcPHszn8+vr6wzsiG6UQQhxuVyXLl0aGBgwUW0sFstkMl6v90fo1KyAMMYDAwPnzp0zXfPg4GAqlWo0Gk0MiBAiy/L58+edTqf5Aa4onj59OhaLYYybFRCEMBaL0fNxBw4cSCQStN0QRUBut3t4eJjq6U+dOiVJElVPRBFQIBDo6+ujCqirqyscDlONGykC2lYyYSR6pBoQQapHZwAoHo/TuARYAOrs7GQASFEUqn6aIiBJkhgA6ujooFpUo6iaTa7koFwnaoWadLNe81tbWwzoaJrWrICWl5cZAFpbW6OabVAEtLi4yABQsVjUNK0pAWWzWQaAcrlcswKanZ1VVZUqHYRQEwOq1Wpv3ryhCmh6erpcLjdrNl+v1ycnJ+l5UELI27dvv3//3qxxEADgy5cvExMT9Mznw4cPtFtAdAPFarU6Pj5eKpVM17yxsfHy5cvV1VXazXu62gVBKBQKT58+rdVqJqrFGL948eLdu3dU8/g/H4FBUaJYLAqC0NPTY9brMD4+PjY25mDSracOCABACJmZmXE6nUePHjWu8NWrV48ePSKEsGlAs7Agfd5nenq6Wq0mk0kjDzY2NvbkyRMIIbP2PLvhBUEQ8vl8NpuNx+M+n29bzhVjvLKycv/+/YmJCcazQuwA6YzW1tYmJyf1SY+2trZ/rRk1Go1SqfT69esHDx4UCgVmNaa/zZ/9ABUhRFXVYDB48uTJeDweiUQkSfL7/T9MA2NcqVTK5fLy8vL8/PzU1FSxWHS5XJaM4wGr9sUwxqqqer3eUCgkSZJuUBBCfTRvc3OzXC6vrKxUKhWn02nhCJ5lM4oQQo/HgxD6+vXr58+fHf8sDOp+HQDg8XgclorFU676dKLlo6yWRdItIBwQB8QBcUCtfosRQjRNQwhhjPUC4w46WXryBSHU1zgEQWBz99EFhDGu1+t+v//48ePxeFxRlD179ng8nh0Efgiher2+vr6ur3HMzMysrq7uTJVdACGEurq6Ll++nEgkPB7Pj9jPoDHqOxyqqubz+WfPnuVyuV9XPeyeagAAAoHArVu3BgcHab8CuVzu4cOHpVKJUnfA5GweY+xyuc6cOXPv3r1IJMLAR8iyPDw8XK/Xi8Wiqqqmm5KZgBBC7e3tN27cuHbtGuPVpf7+/lAoNDs7W61WzfVKpgHSSzw3b95MpVKW3MfRaDQSiczNzVUqFRMZmQOIEOL1eq9fv3727FlL1t50URRFluX5+flqtWpWEGAOIFEUU6nUlStXLKSjy759+xwOx9zcnKZpphzGHMzhcDiTydgk9r1w4YIp7RPTAAmCkMlk2FeLf/tIEKbTab/fbwtAhJBoNGrutpNx6e7uPnTokC1eMU3T0um0DZPMkZER6wERQnw+n/FFSxpy7Nix3bt3WwwIIcRgIWnHkkwmjecfRgGx7DtuV/r6+iwGhDHev3+/bQF1dnYaH6E2CkiWZdsC2rt3r8WAHA5HW1ubbQGZcjajgOwTH/4qNko1Wlg4IA6IA+KAOKBWBUQIsfNojyliKIoRRfH9+/dut9umf3wzpoUNNQ4BAJubmwz+ic+OxefzWWlBhJD29nbug7iT5sIBcUAcEAfEAXFAHBAHxOVn+QMrmWpuPZx12gAAAABJRU5ErkJggg==";
_converse.log = function (message, level) {
if (message instanceof Error) {
message = message.stack;
}
var logger = _.assignIn({
'debug': _.noop,
'error': _.noop,
'info': _.noop,
'warn': _.noop
}, console);
if (level === Strophe.LogLevel.ERROR) {
if (_converse.debug) {
logger.trace("ERROR: " + message);
} else {
logger.error("ERROR: " + message);
}
} else if (level === Strophe.LogLevel.WARN) {
logger.warn("WARNING: " + message);
} else if (level === Strophe.LogLevel.FATAL) {
if (_converse.debug) {
logger.error("FATAL: " + message);
} else {
logger.error("FATAL: " + message);
}
} else if (_converse.debug) {
if (level === Strophe.LogLevel.DEBUG) {
logger.debug("DEBUG: " + message);
} else {
logger.info("INFO: " + message);
}
}
};
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
var PROMISES = ['cachedRoster', 'chatBoxesFetched', 'pluginsInitialized', 'roster', 'rosterContactsFetched', 'rosterGroupsFetched', 'rosterInitialized', 'statusInitialized'];
function addPromise(promise) {
/* Private function, used to add a new promise to the ones already
* available via the `waitUntil` api method.
*/
_converse.promises[promise] = utils.getWrappedPromise();
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
_converse.emit = function (name) {
/* Event emitter and promise resolver */
_converse.trigger.apply(this, arguments);
var promise = _converse.promises[name];
if (!_.isUndefined(promise)) {
promise.resolve();
}
};
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
_converse.initialize = function (settings, callback) {
"use strict";
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
var _this = this;
2017-07-22 22:21:05 +02:00
settings = !_.isUndefined(settings) ? settings : {};
var init_promise = utils.getWrappedPromise();
2017-07-22 22:21:05 +02:00
_.each(PROMISES, addPromise);
2017-07-22 22:21:05 +02:00
if (!_.isUndefined(_converse.chatboxes)) {
// Looks like _converse.initialized was called again without logging
// out or disconnecting in the previous session.
// This happens in tests. We therefore first clean up.
_converse.connection.reset();
_converse.off();
_converse.stopListening();
_converse._tearDown();
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
var unloadevent = void 0;
if ('onpagehide' in window) {
// Pagehide gets thrown in more cases than unload. Specifically it
// gets thrown when the page is cached and not just
// closed/destroyed. It's the only viable event on mobile Safari.
// https://www.webkit.org/blog/516/webkit-page-cache-ii-the-unload-event/
unloadevent = 'pagehide';
} else if ('onbeforeunload' in window) {
unloadevent = 'beforeunload';
} else if ('onunload' in window) {
unloadevent = 'unload';
}
2017-07-22 22:21:05 +02:00
// Logging
Strophe.log = function (level, msg) {
_converse.log(level + ' ' + msg, level);
};
Strophe.error = function (msg) {
_converse.log(msg, Strophe.LogLevel.ERROR);
};
2017-07-22 22:21:05 +02:00
// Add Strophe Namespaces
Strophe.addNamespace('CARBONS', 'urn:xmpp:carbons:2');
Strophe.addNamespace('CHATSTATES', 'http://jabber.org/protocol/chatstates');
Strophe.addNamespace('CSI', 'urn:xmpp:csi:0');
Strophe.addNamespace('DELAY', 'urn:xmpp:delay');
Strophe.addNamespace('HINTS', 'urn:xmpp:hints');
Strophe.addNamespace('MAM', 'urn:xmpp:mam:2');
Strophe.addNamespace('NICK', 'http://jabber.org/protocol/nick');
Strophe.addNamespace('PUBSUB', 'http://jabber.org/protocol/pubsub');
Strophe.addNamespace('ROSTERX', 'http://jabber.org/protocol/rosterx');
Strophe.addNamespace('RSM', 'http://jabber.org/protocol/rsm');
Strophe.addNamespace('XFORM', 'jabber:x:data');
2017-07-22 22:21:05 +02:00
// Instance level constants
this.TIMEOUTS = { // Set as module attr so that we can override in tests.
'PAUSED': 10000,
'INACTIVE': 90000
};
2017-07-22 22:21:05 +02:00
// Internationalization
this.locale = utils.getLocale(settings.i18n, utils.isConverseLocale);
if (!moment.locale) {
//moment.lang is deprecated after 2.8.1, use moment.locale instead
moment.locale = moment.lang;
}
moment.locale(utils.getLocale(settings.i18n, utils.isMomentLocale));
var __ = _converse.__ = utils.__.bind(_converse);
_converse.___ = utils.___;
2017-07-22 22:21:05 +02:00
// XEP-0085 Chat states
// http://xmpp.org/extensions/xep-0085.html
this.INACTIVE = 'inactive';
this.ACTIVE = 'active';
this.COMPOSING = 'composing';
this.PAUSED = 'paused';
this.GONE = 'gone';
2017-07-22 22:21:05 +02:00
// Default configuration values
// ----------------------------
this.default_settings = {
allow_contact_requests: true,
allow_non_roster_messaging: false,
animate: true,
authentication: 'login', // Available values are "login", "prebind", "anonymous" and "external".
auto_away: 0, // Seconds after which user status is set to 'away'
auto_login: false, // Currently only used in connection with anonymous login
auto_reconnect: false,
auto_subscribe: false,
auto_xa: 0, // Seconds after which user status is set to 'xa'
blacklisted_plugins: [],
bosh_service_url: undefined,
connection_options: {},
credentials_url: null, // URL from where login credentials can be fetched
csi_waiting_time: 0, // Support for XEP-0352. Seconds before client is considered idle and CSI is sent out.
debug: false,
default_state: 'online',
expose_rid_and_sid: false,
filter_by_resource: false,
forward_messages: false,
hide_offline_users: false,
include_offline_state: false,
jid: undefined,
keepalive: true,
locked_domain: undefined,
message_carbons: true,
message_storage: 'session',
password: undefined,
prebind_url: null,
priority: 0,
registration_domain: '',
rid: undefined,
roster_groups: true,
show_only_online_users: false,
show_send_button: false,
sid: undefined,
storage: 'session',
strict_plugin_dependencies: false,
synchronize_availability: true,
websocket_url: undefined,
whitelisted_plugins: [],
xhr_custom_status: false,
xhr_custom_status_url: ''
};
_.assignIn(this, this.default_settings);
// Allow only whitelisted configuration attributes to be overwritten
_.assignIn(this, _.pick(settings, _.keys(this.default_settings)));
2017-07-22 22:21:05 +02:00
if (this.authentication === _converse.ANONYMOUS) {
if (this.auto_login && !this.jid) {
throw new Error("Config Error: you need to provide the server's " + "domain via the 'jid' option when using anonymous " + "authentication with auto_login.");
}
}
2017-07-22 22:21:05 +02:00
// Module-level variables
// ----------------------
this.callback = callback || _.noop;
/* When reloading the page:
* For new sessions, we need to send out a presence stanza to notify
* the server/network that we're online.
* When re-attaching to an existing session (e.g. via the keepalive
* option), we don't need to again send out a presence stanza, because
* it's as if "we never left" (see onConnectStatusChanged).
* https://github.com/jcbrand/converse.js/issues/521
*/
this.send_initial_presence = true;
this.msg_counter = 0;
this.user_settings = settings; // Save the user settings so that they can be used by plugins
2017-07-22 22:21:05 +02:00
// Module-level functions
// ----------------------
this.getViewForChatBox = function (chatbox) {
if (!chatbox) {
return;
}
return _converse.chatboxviews.get(chatbox.get('id'));
};
2017-07-22 22:21:05 +02:00
this.generateResource = function () {
return "/converse.js-" + Math.floor(Math.random() * 139749825).toString();
};
2017-07-22 22:21:05 +02:00
this.sendCSI = function (stat) {
/* Send out a Chat Status Notification (XEP-0352)
*
* Parameters:
* (String) stat: The user's chat status
*/
/* Send out a Chat Status Notification (XEP-0352) */
// XXX if (converse.features[Strophe.NS.CSI] || true) {
_converse.connection.send($build(stat, { xmlns: Strophe.NS.CSI }));
_converse.inactive = stat === _converse.INACTIVE ? true : false;
};
2017-07-22 22:21:05 +02:00
this.onUserActivity = function () {
/* Resets counters and flags relating to CSI and auto_away/auto_xa */
if (_converse.idle_seconds > 0) {
_converse.idle_seconds = 0;
}
if (!_converse.connection.authenticated) {
// We can't send out any stanzas when there's no authenticated connection.
// converse can happen when the connection reconnects.
return;
}
if (_converse.inactive) {
_converse.sendCSI(_converse.ACTIVE);
}
if (_converse.auto_changed_status === true) {
_converse.auto_changed_status = false;
// XXX: we should really remember the original state here, and
// then set it back to that...
_converse.xmppstatus.setStatus(_converse.default_state);
}
};
2017-07-22 22:21:05 +02:00
this.onEverySecond = function () {
/* An interval handler running every second.
* Used for CSI and the auto_away and auto_xa features.
*/
if (!_converse.connection.authenticated) {
// We can't send out any stanzas when there's no authenticated connection.
// This can happen when the connection reconnects.
return;
}
var stat = _converse.xmppstatus.getStatus();
_converse.idle_seconds++;
if (_converse.csi_waiting_time > 0 && _converse.idle_seconds > _converse.csi_waiting_time && !_converse.inactive) {
_converse.sendCSI(_converse.INACTIVE);
}
if (_converse.auto_away > 0 && _converse.idle_seconds > _converse.auto_away && stat !== 'away' && stat !== 'xa' && stat !== 'dnd') {
_converse.auto_changed_status = true;
_converse.xmppstatus.setStatus('away');
} else if (_converse.auto_xa > 0 && _converse.idle_seconds > _converse.auto_xa && stat !== 'xa' && stat !== 'dnd') {
_converse.auto_changed_status = true;
_converse.xmppstatus.setStatus('xa');
}
};
2017-07-22 22:21:05 +02:00
this.registerIntervalHandler = function () {
/* Set an interval of one second and register a handler for it.
* Required for the auto_away, auto_xa and csi_waiting_time features.
*/
if (_converse.auto_away < 1 && _converse.auto_xa < 1 && _converse.csi_waiting_time < 1) {
// Waiting time of less then one second means features aren't used.
return;
}
_converse.idle_seconds = 0;
_converse.auto_changed_status = false; // Was the user's status changed by _converse.js?
window.addEventListener('click', _converse.onUserActivity);
window.addEventListener('focus', _converse.onUserActivity);
window.addEventListener('keypress', _converse.onUserActivity);
window.addEventListener('mousemove', _converse.onUserActivity);
window.addEventListener(unloadevent, _converse.onUserActivity);
_converse.everySecondTrigger = window.setInterval(_converse.onEverySecond, 1000);
};
2017-07-22 22:21:05 +02:00
this.giveFeedback = function (subject, klass, message) {
2017-08-08 17:44:01 +02:00
_.forEach(document.querySelectorAll('.conn-feedback'), function (el) {
2017-07-22 22:21:05 +02:00
el.classList.add('conn-feedback');
el.textContent = subject;
if (klass) {
el.classList.add(klass);
} else {
el.classList.remove('error');
}
});
_converse.emit('feedback', {
'klass': klass,
'message': message,
'subject': subject
});
};
2017-07-22 22:21:05 +02:00
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
*/
var pres = $pres({ to: jid, type: "unsubscribed" });
if (message && message !== "") {
pres.c("status").t(message);
}
_converse.connection.send(pres);
};
2017-07-22 22:21:05 +02:00
this.reconnect = _.debounce(function () {
_converse.log('RECONNECTING');
_converse.log('The connection has dropped, attempting to reconnect.');
_converse.giveFeedback(__("Reconnecting"), 'warn', __('The connection has dropped, attempting to reconnect.'));
_converse.connection.reconnecting = true;
_converse._tearDown();
_converse.logIn(null, true);
}, 3000, { 'leading': true });
2017-07-22 22:21:05 +02:00
this.disconnect = function () {
_converse.log('DISCONNECTED');
delete _converse.connection.reconnecting;
_converse.connection.reset();
_converse._tearDown();
_converse.chatboxviews.closeAllChatBoxes();
_converse.emit('disconnected');
};
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
this.onDisconnected = function () {
/* Gets called once strophe's status reaches Strophe.Status.DISCONNECTED.
* Will either start a teardown process for converse.js or attempt
* to reconnect.
*/
if (_converse.disconnection_cause === Strophe.Status.AUTHFAIL) {
if (_converse.credentials_url && _converse.auto_reconnect) {
/* In this case, we reconnect, because we might be receiving
* expirable tokens from the credentials_url.
*/
_converse.emit('will-reconnect');
return _converse.reconnect();
} else {
return _converse.disconnect();
}
} else if (_converse.disconnection_cause === _converse.LOGOUT || _converse.disconnection_reason === "host-unknown" || !_converse.auto_reconnect) {
return _converse.disconnect();
}
_converse.emit('will-reconnect');
_converse.reconnect();
};
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
this.setDisconnectionCause = function (cause, reason, override) {
/* Used to keep track of why we got disconnected, so that we can
* decide on what the next appropriate action is (in onDisconnected)
*/
if (_.isUndefined(cause)) {
delete _converse.disconnection_cause;
delete _converse.disconnection_reason;
} else if (_.isUndefined(_converse.disconnection_cause) || override) {
_converse.disconnection_cause = cause;
_converse.disconnection_reason = reason;
}
};
2017-07-22 22:21:05 +02:00
this.onConnectStatusChanged = function (status, condition) {
/* Callback method called by Strophe as the Strophe.Connection goes
* through various states while establishing or tearing down a
* connection.
*/
_converse.log("Status changed to: " + PRETTY_CONNECTION_STATUS[status]);
if (status === Strophe.Status.CONNECTED || status === Strophe.Status.ATTACHED) {
// By default we always want to send out an initial presence stanza.
_converse.send_initial_presence = true;
_converse.setDisconnectionCause();
if (_converse.connection.reconnecting) {
_converse.log(status === Strophe.Status.CONNECTED ? 'Reconnected' : 'Reattached');
_converse.onConnected(true);
} else {
_converse.log(status === Strophe.Status.CONNECTED ? 'Connected' : 'Attached');
if (_converse.connection.restored) {
// No need to send an initial presence stanza when
// we're restoring an existing session.
_converse.send_initial_presence = false;
}
_converse.onConnected();
}
} else if (status === Strophe.Status.DISCONNECTED) {
_converse.setDisconnectionCause(status, condition);
_converse.onDisconnected();
} else if (status === Strophe.Status.ERROR) {
_converse.giveFeedback(__('Connection error'), 'error', __('An error occurred while connecting to the chat server.'));
} else if (status === Strophe.Status.CONNECTING) {
_converse.giveFeedback(__('Connecting'));
} else if (status === Strophe.Status.AUTHENTICATING) {
_converse.giveFeedback(__('Authenticating'));
} else if (status === Strophe.Status.AUTHFAIL) {
_converse.giveFeedback(__('Authentication Failed'), 'error');
_converse.setDisconnectionCause(status, condition, true);
_converse.onDisconnected();
} else if (status === Strophe.Status.CONNFAIL) {
_converse.giveFeedback(__('Connection failed'), 'error', __("An error occurred while connecting to the chat server: " + condition));
_converse.setDisconnectionCause(status, condition);
} else if (status === Strophe.Status.DISCONNECTING) {
_converse.setDisconnectionCause(status, condition);
}
};
2017-07-22 22:21:05 +02:00
this.incrementMsgCounter = function () {
this.msg_counter += 1;
var unreadMsgCount = this.msg_counter;
if (document.title.search(/^Messages \(\d+\) /) === -1) {
document.title = "Messages (" + unreadMsgCount + ") " + document.title;
} else {
document.title = document.title.replace(/^Messages \(\d+\) /, "Messages (" + unreadMsgCount + ") ");
}
};
2017-07-22 22:21:05 +02:00
this.clearMsgCounter = function () {
this.msg_counter = 0;
if (document.title.search(/^Messages \(\d+\) /) !== -1) {
document.title = document.title.replace(/^Messages \(\d+\) /, "");
}
};
2017-07-22 22:21:05 +02:00
this.initStatus = function () {
return new Promise(function (resolve, reject) {
var promise = new utils.getWrappedPromise();
_this.xmppstatus = new _this.XMPPStatus();
var id = b64_sha1("converse.xmppstatus-" + _converse.bare_jid);
_this.xmppstatus.id = id; // Appears to be necessary for backbone.browserStorage
_this.xmppstatus.browserStorage = new Backbone.BrowserStorage[_converse.storage](id);
_this.xmppstatus.fetch({
success: resolve,
error: resolve
});
_converse.emit('statusInitialized');
});
};
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.initSession = function () {
this.session = new Backbone.Model();
var id = b64_sha1('converse.bosh-session');
this.session.id = id; // Appears to be necessary for backbone.browserStorage
this.session.browserStorage = new Backbone.BrowserStorage[_converse.storage](id);
this.session.fetch();
};
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.clearSession = function () {
if (!_.isUndefined(this.roster)) {
this.roster.browserStorage._clear();
}
if (!_.isUndefined(this.session) && this.session.browserStorage) {
this.session.browserStorage._clear();
}
};
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.logOut = function () {
_converse.chatboxviews.closeAllChatBoxes();
_converse.clearSession();
2017-07-22 22:21:05 +02:00
_converse.setDisconnectionCause(_converse.LOGOUT, undefined, true);
if (!_.isUndefined(_converse.connection)) {
_converse.connection.disconnect();
} else {
_converse._tearDown();
}
_converse.emit('logout');
};
2017-07-22 22:21:05 +02:00
this.saveWindowState = function (ev, hidden) {
// XXX: eventually we should be able to just use
// document.visibilityState (when we drop support for older
// browsers).
var state = void 0;
var event_map = {
'focus': "visible",
'focusin': "visible",
'pageshow': "visible",
'blur': "hidden",
'focusout': "hidden",
'pagehide': "hidden"
};
ev = ev || document.createEvent('Events');
if (ev.type in event_map) {
state = event_map[ev.type];
} else {
state = document[hidden] ? "hidden" : "visible";
}
if (state === 'visible') {
_converse.clearMsgCounter();
}
_converse.windowState = state;
_converse.emit('windowStateChanged', { state: state });
};
2017-07-22 22:21:05 +02:00
this.registerGlobalEventHandlers = function () {
// Taken from:
// http://stackoverflow.com/questions/1060008/is-there-a-way-to-detect-if-a-browser-window-is-not-currently-active
var hidden = "hidden";
// Standards:
if (hidden in document) {
document.addEventListener("visibilitychange", _.partial(_converse.saveWindowState, _, hidden));
} else if ((hidden = "mozHidden") in document) {
document.addEventListener("mozvisibilitychange", _.partial(_converse.saveWindowState, _, hidden));
} else if ((hidden = "webkitHidden") in document) {
document.addEventListener("webkitvisibilitychange", _.partial(_converse.saveWindowState, _, hidden));
} else if ((hidden = "msHidden") in document) {
document.addEventListener("msvisibilitychange", _.partial(_converse.saveWindowState, _, hidden));
} else if ("onfocusin" in document) {
// IE 9 and lower:
document.onfocusin = document.onfocusout = _.partial(_converse.saveWindowState, _, hidden);
} else {
// All others:
window.onpageshow = window.onpagehide = window.onfocus = window.onblur = _.partial(_converse.saveWindowState, _, hidden);
}
// set the initial state (but only if browser supports the Page Visibility API)
if (document[hidden] !== undefined) {
_.partial(_converse.saveWindowState, _, hidden)({ type: document[hidden] ? "blur" : "focus" });
}
};
2017-07-22 22:21:05 +02:00
this.enableCarbons = function () {
var _this2 = this;
2017-07-22 22:21:05 +02:00
/* Ask the XMPP server to enable Message Carbons
* See XEP-0280 https://xmpp.org/extensions/xep-0280.html#enabling
*/
if (!this.message_carbons || this.session.get('carbons_enabled')) {
return;
}
var carbons_iq = new Strophe.Builder('iq', {
from: this.connection.jid,
id: 'enablecarbons',
type: 'set'
}).c('enable', { xmlns: Strophe.NS.CARBONS });
this.connection.addHandler(function (iq) {
if (iq.querySelectorAll('error').length > 0) {
_converse.log('An error occured while trying to enable message carbons.', Strophe.LogLevel.ERROR);
} else {
_this2.session.save({ carbons_enabled: true });
_converse.log('Message carbons have been enabled.');
}
}, null, "iq", null, "enablecarbons");
this.connection.send(carbons_iq);
};
2017-07-22 22:21:05 +02:00
this.initRoster = function () {
/* Initialize the Bakcbone collections that represent the contats
* roster and the roster groups.
*/
_converse.roster = new _converse.RosterContacts();
_converse.roster.browserStorage = new Backbone.BrowserStorage.session(b64_sha1("converse.contacts-" + _converse.bare_jid));
_converse.rostergroups = new _converse.RosterGroups();
_converse.rostergroups.browserStorage = new Backbone.BrowserStorage.session(b64_sha1("converse.roster.groups" + _converse.bare_jid));
_converse.emit('rosterInitialized');
};
2017-07-22 22:21:05 +02:00
this.populateRoster = function () {
/* Fetch all the roster groups, and then the roster contacts.
* Emit an event after fetching is done in each case.
*/
_converse.rostergroups.fetchRosterGroups().then(function () {
_converse.emit('rosterGroupsFetched');
_converse.roster.fetchRosterContacts().then(function () {
_converse.emit('rosterContactsFetched');
_converse.sendInitialPresence();
});
});
};
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.unregisterPresenceHandler = function () {
if (!_.isUndefined(_converse.presence_ref)) {
_converse.connection.deleteHandler(_converse.presence_ref);
delete _converse.presence_ref;
}
};
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.registerPresenceHandler = function () {
_converse.unregisterPresenceHandler();
_converse.presence_ref = _converse.connection.addHandler(function (presence) {
_converse.roster.presenceHandler(presence);
return true;
}, null, 'presence', null);
};
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.sendInitialPresence = function () {
if (_converse.send_initial_presence) {
_converse.xmppstatus.sendPresence();
}
};
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.onStatusInitialized = function (reconnecting) {
/* Continue with session establishment (e.g. fetching chat boxes,
* populating the roster etc.) necessary once the connection has
* been established.
*/
if (reconnecting) {
// No need to recreate the roster, otherwise we lose our
// cached data. However we still emit an event, to give
// event handlers a chance to register views for the
// roster and its groups, before we start populating.
_converse.emit('rosterReadyAfterReconnection');
} else {
_converse.registerIntervalHandler();
_converse.initRoster();
}
// First set up chat boxes, before populating the roster, so that
// the controlbox is properly set up and ready for the rosterview.
_converse.roster.onConnected();
_converse.chatboxes.onConnected();
_converse.populateRoster();
_converse.registerPresenceHandler();
_converse.giveFeedback(__('Contacts'));
if (reconnecting) {
_converse.xmppstatus.sendPresence();
} else {
init_promise.resolve();
_converse.emit('initialized');
}
};
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.setUserJid = function () {
_converse.jid = _converse.connection.jid;
_converse.bare_jid = Strophe.getBareJidFromJid(_converse.connection.jid);
_converse.resource = Strophe.getResourceFromJid(_converse.connection.jid);
_converse.domain = Strophe.getDomainFromJid(_converse.connection.jid);
};
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.onConnected = function (reconnecting) {
/* Called as soon as a new connection has been established, either
* by logging in or by attaching to an existing BOSH session.
*/
// Solves problem of returned PubSub BOSH response not received
// by browser.
_converse.connection.flush();
2017-07-22 22:21:05 +02:00
_converse.setUserJid();
_converse.initSession();
_converse.enableCarbons();
2017-07-22 22:21:05 +02:00
// If there's no xmppstatus obj, then we were never connected to
// begin with, so we set reconnecting to false.
reconnecting = _.isUndefined(_converse.xmppstatus) ? false : reconnecting;
if (reconnecting) {
_converse.onStatusInitialized(true);
_converse.emit('reconnected');
} else {
_converse.chatboxviews.closeAllChatBoxes();
_converse.initStatus().then(_.partial(_converse.onStatusInitialized, false), _.partial(_converse.onStatusInitialized, false)).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
_converse.emit('connected');
}
};
2017-07-22 22:21:05 +02:00
this.RosterContact = Backbone.Model.extend({
2017-07-22 22:21:05 +02:00
defaults: {
'bookmarked': false,
'chat_state': undefined,
'chat_status': 'offline',
'groups': [],
'image': DEFAULT_IMAGE,
'image_type': DEFAULT_IMAGE_TYPE,
'num_unread': 0,
'status': ''
},
2017-07-22 22:21:05 +02:00
initialize: function initialize(attributes) {
var _this3 = this;
2017-07-22 22:21:05 +02:00
var jid = attributes.jid;
2017-07-22 22:21:05 +02:00
var bare_jid = Strophe.getBareJidFromJid(jid).toLowerCase();
var resource = Strophe.getResourceFromJid(jid);
attributes.jid = bare_jid;
this.set(_.assignIn({
'id': bare_jid,
'jid': bare_jid,
'fullname': bare_jid,
'user_id': Strophe.getNodeFromJid(jid),
'resources': resource ? { resource: 0 } : {}
}, attributes));
2017-07-22 22:21:05 +02:00
this.on('destroy', function () {
_this3.removeFromRoster();
});
this.on('change:chat_status', function (item) {
_converse.emit('contactStatusChanged', item.attributes);
});
},
subscribe: function subscribe(message) {
/* Send a presence subscription request to this roster contact
*
* Parameters:
* (String) message - An optional message to explain the
* reason for the subscription request.
*/
this.save('ask', "subscribe"); // ask === 'subscribe' Means we have ask to subscribe to them.
var pres = $pres({ to: this.get('jid'), type: "subscribe" });
if (message && message !== "") {
pres.c("status").t(message).up();
}
var nick = _converse.xmppstatus.get('fullname');
if (nick && nick !== "") {
pres.c('nick', { 'xmlns': Strophe.NS.NICK }).t(nick).up();
}
_converse.connection.send(pres);
return this;
},
ackSubscribe: function ackSubscribe() {
/* Upon receiving the presence stanza of type "subscribed",
* the user SHOULD acknowledge receipt of that subscription
* state notification by sending a presence stanza of type
* "subscribe" to the contact
*/
_converse.connection.send($pres({
'type': 'subscribe',
'to': this.get('jid')
}));
},
ackUnsubscribe: function ackUnsubscribe() {
/* Upon receiving the presence stanza of type "unsubscribed",
* the user SHOULD acknowledge receipt of that subscription state
* notification by sending a presence stanza of type "unsubscribe"
* this step lets the user's server know that it MUST no longer
* send notification of the subscription state change to the user.
* Parameters:
* (String) jid - The Jabber ID of the user who is unsubscribing
*/
_converse.connection.send($pres({ 'type': 'unsubscribe', 'to': this.get('jid') }));
this.destroy(); // Will cause removeFromRoster to be called.
},
unauthorize: function unauthorize(message) {
/* Unauthorize this contact's presence subscription
* Parameters:
* (String) message - Optional message to send to the person being unauthorized
*/
_converse.rejectPresenceSubscription(this.get('jid'), message);
return this;
},
authorize: function authorize(message) {
/* Authorize presence subscription
* Parameters:
* (String) message - Optional message to send to the person being authorized
*/
var pres = $pres({ to: this.get('jid'), type: "subscribed" });
if (message && message !== "") {
pres.c("status").t(message);
}
_converse.connection.send(pres);
return this;
},
addResource: function addResource(presence) {
/* Adds a new resource and it's associated attributes as taken
* from the passed in presence stanza.
*
* Also updates the contact's chat_status if the presence has
* higher priority (and is newer).
*/
var jid = presence.getAttribute('from'),
chat_status = _.propertyOf(presence.querySelector('show'))('textContent') || 'online',
resource = Strophe.getResourceFromJid(jid),
delay = presence.querySelector("delay[xmlns=\"" + Strophe.NS.DELAY + "\"]"),
timestamp = _.isNull(delay) ? moment().format() : moment(delay.getAttribute('stamp')).format();
2017-07-22 22:21:05 +02:00
var priority = _.propertyOf(presence.querySelector('priority'))('textContent') || 0;
priority = _.isNaN(parseInt(priority, 10)) ? 0 : parseInt(priority, 10);
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
var resources = _.isObject(this.get('resources')) ? this.get('resources') : {};
resources[resource] = {
'priority': priority,
'status': chat_status,
'timestamp': timestamp
};
var changed = { 'resources': resources };
var hpr = this.getHighestPriorityResource();
if (priority == hpr.priority && timestamp == hpr.timestamp) {
// Only set the chat status if this is the newest resource
// with the highest priority
changed.chat_status = chat_status;
}
this.save(changed);
return resources;
},
removeResource: function removeResource(resource) {
/* Remove the passed in resource from the contact's resources map.
*
* Also recomputes the chat_status given that there's one less
* resource.
*/
var resources = this.get('resources');
if (!_.isObject(resources)) {
resources = {};
} else {
delete resources[resource];
}
this.save({
'resources': resources,
'chat_status': _.propertyOf(this.getHighestPriorityResource())('status') || 'offline'
});
},
getHighestPriorityResource: function getHighestPriorityResource() {
/* Return the resource with the highest priority.
*
* If multiple resources have the same priority, take the
* newest one.
*/
var resources = this.get('resources');
if (_.isObject(resources) && _.size(resources)) {
var val = _.flow(_.values, _.partial(_.sortBy, _, ['priority', 'timestamp']), _.reverse)(resources)[0];
if (!_.isUndefined(val)) {
return val;
}
}
},
removeFromRoster: function removeFromRoster(callback) {
/* Instruct the XMPP server to remove this contact from our roster
* Parameters:
* (Function) callback
*/
var iq = $iq({ type: 'set' }).c('query', { xmlns: Strophe.NS.ROSTER }).c('item', { jid: this.get('jid'), subscription: "remove" });
_converse.connection.sendIQ(iq, callback, callback);
return this;
}
});
2017-04-04 17:26:06 +02:00
2017-07-22 22:21:05 +02:00
this.RosterContacts = Backbone.Collection.extend({
model: _converse.RosterContact,
2017-07-22 22:21:05 +02:00
comparator: function comparator(contact1, contact2) {
var status1 = contact1.get('chat_status') || 'offline';
var status2 = contact2.get('chat_status') || 'offline';
if (_converse.STATUS_WEIGHTS[status1] === _converse.STATUS_WEIGHTS[status2]) {
var name1 = contact1.get('fullname').toLowerCase();
var name2 = contact2.get('fullname').toLowerCase();
return name1 < name2 ? -1 : name1 > name2 ? 1 : 0;
} else {
return _converse.STATUS_WEIGHTS[status1] < _converse.STATUS_WEIGHTS[status2] ? -1 : 1;
}
},
onConnected: function onConnected() {
/* Called as soon as the connection has been established
* (either after initial login, or after reconnection).
*
* Use the opportunity to register stanza handlers.
*/
this.registerRosterHandler();
this.registerRosterXHandler();
},
registerRosterHandler: function registerRosterHandler() {
/* Register a handler for roster IQ "set" stanzas, which update
* roster contacts.
*/
_converse.connection.addHandler(_converse.roster.onRosterPush.bind(_converse.roster), Strophe.NS.ROSTER, 'iq', "set");
},
registerRosterXHandler: function registerRosterXHandler() {
/* Register a handler for RosterX message stanzas, which are
* used to suggest roster contacts to a user.
*/
var t = 0;
_converse.connection.addHandler(function (msg) {
window.setTimeout(function () {
_converse.connection.flush();
_converse.roster.subscribeToSuggestedItems.bind(_converse.roster)(msg);
}, t);
t += msg.querySelectorAll('item').length * 250;
return true;
}, Strophe.NS.ROSTERX, 'message', null);
},
fetchRosterContacts: function fetchRosterContacts() {
var _this4 = this;
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
/* Fetches the roster contacts, first by trying the
* sessionStorage cache, and if that's empty, then by querying
* the XMPP server.
*
* Returns a promise which resolves once the contacts have been
* fetched.
*/
return new Promise(function (resolve, reject) {
_this4.fetch({
add: true,
success: function success(collection) {
if (collection.length === 0) {
/* We don't have any roster contacts stored in sessionStorage,
* so lets fetch the roster from the XMPP server. We pass in
* 'sendPresence' as callback method, because after initially
* fetching the roster we are ready to receive presence
* updates from our contacts.
*/
_converse.send_initial_presence = true;
_converse.roster.fetchFromServer(resolve);
} else {
_converse.emit('cachedRoster', collection);
resolve();
}
}
});
});
},
subscribeToSuggestedItems: function subscribeToSuggestedItems(msg) {
_.each(msg.querySelectorAll('item'), function (item) {
if (item.getAttribute('action') === 'add') {
_converse.roster.addAndSubscribe(item.getAttribute('jid'), null, _converse.xmppstatus.get('fullname'));
}
});
return true;
},
isSelf: function isSelf(jid) {
return utils.isSameBareJID(jid, _converse.connection.jid);
},
addAndSubscribe: function addAndSubscribe(jid, name, groups, message, attributes) {
/* Add a roster contact and then once we have confirmation from
* the XMPP server we subscribe to that contact's presence updates.
* Parameters:
* (String) jid - The Jabber ID of the user being added and subscribed to.
* (String) name - The name of that user
* (Array of Strings) groups - Any roster groups the user might belong to
* (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.
*/
var handler = function handler(contact) {
if (contact instanceof _converse.RosterContact) {
contact.subscribe(message);
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
};
this.addContact(jid, name, groups, attributes).then(handler, handler);
},
sendContactAddIQ: function sendContactAddIQ(jid, name, groups, callback, errback) {
/* Send an IQ stanza to the XMPP server to add a new roster contact.
*
* Parameters:
* (String) jid - The Jabber ID of the user being added
* (String) name - The name of that user
* (Array of Strings) groups - Any roster groups the user might belong to
* (Function) callback - A function to call once the IQ is returned
* (Function) errback - A function to call if an error occured
*/
name = _.isEmpty(name) ? jid : name;
var iq = $iq({ type: 'set' }).c('query', { xmlns: Strophe.NS.ROSTER }).c('item', { jid: jid, name: name });
_.each(groups, function (group) {
iq.c('group').t(group).up();
2017-02-13 17:16:13 +01:00
});
2017-07-22 22:21:05 +02:00
_converse.connection.sendIQ(iq, callback, errback);
},
addContact: function addContact(jid, name, groups, attributes) {
var _this5 = this;
2017-07-22 22:21:05 +02:00
/* Adds a RosterContact instance to _converse.roster and
* registers the contact on the XMPP server.
* Returns a promise which is resolved once the XMPP server has
* responded.
*
* Parameters:
* (String) jid - The Jabber ID of the user being added and subscribed to.
* (String) name - The name of that user
* (Array of Strings) groups - Any roster groups the user might belong to
* (Object) attributes - Any additional attributes to be stored on the user's model.
*/
return new Promise(function (resolve, reject) {
groups = groups || [];
name = _.isEmpty(name) ? jid : name;
_this5.sendContactAddIQ(jid, name, groups, function () {
var contact = _this5.create(_.assignIn({
ask: undefined,
fullname: name,
groups: groups,
jid: jid,
requesting: false,
subscription: 'none'
}, attributes), { sort: false });
resolve(contact);
}, function (err) {
alert(__("Sorry, there was an error while trying to add " + name + " as a contact."));
_converse.log(err, Strophe.LogLevel.ERROR);
resolve(err);
});
2017-02-13 17:16:13 +01:00
});
2017-07-22 22:21:05 +02:00
},
subscribeBack: function subscribeBack(bare_jid) {
var contact = this.get(bare_jid);
if (contact instanceof _converse.RosterContact) {
contact.authorize().subscribe();
} else {
// Can happen when a subscription is retried or roster was deleted
var handler = function handler(contact) {
if (contact instanceof _converse.RosterContact) {
contact.authorize().subscribe();
}
};
this.addContact(bare_jid, '', [], { 'subscription': 'from' }).then(handler, handler);
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
},
getNumOnlineContacts: function getNumOnlineContacts() {
var ignored = ['offline', 'unavailable'];
if (_converse.show_only_online_users) {
ignored = _.union(ignored, ['dnd', 'xa', 'away']);
}
2017-07-22 22:21:05 +02:00
return _.sum(this.models.filter(function (model) {
return !_.includes(ignored, model.get('chat_status'));
}));
},
onRosterPush: function onRosterPush(iq) {
/* Handle roster updates from the XMPP server.
* See: https://xmpp.org/rfcs/rfc6121.html#roster-syntax-actions-push
*
* Parameters:
* (XMLElement) IQ - The IQ stanza received from the XMPP server.
*/
var id = iq.getAttribute('id');
var from = iq.getAttribute('from');
if (from && from !== "" && Strophe.getBareJidFromJid(from) !== _converse.bare_jid) {
// Receiving client MUST ignore stanza unless it has no from or from = user's bare JID.
// XXX: Some naughty servers apparently send from a full
// JID so we need to explicitly compare bare jids here.
// https://github.com/jcbrand/converse.js/issues/493
_converse.connection.send($iq({ type: 'error', id: id, from: _converse.connection.jid }).c('error', { 'type': 'cancel' }).c('service-unavailable', { 'xmlns': Strophe.NS.ROSTER }));
return true;
}
_converse.connection.send($iq({ type: 'result', id: id, from: _converse.connection.jid }));
var items = sizzle("query[xmlns=\"" + Strophe.NS.ROSTER + "\"] item", iq);
_.each(items, this.updateContact.bind(this));
_converse.emit('rosterPush', iq);
return true;
2017-07-22 22:21:05 +02:00
},
fetchFromServer: function fetchFromServer(callback) {
var _this6 = this,
_arguments = arguments;
2017-07-22 22:21:05 +02:00
/* Get the roster from the XMPP server */
var iq = $iq({ type: 'get', 'id': _converse.connection.getUniqueId('roster') }).c('query', { xmlns: Strophe.NS.ROSTER });
return _converse.connection.sendIQ(iq, function (iq) {
_this6.onReceivedFromServer(iq);
callback.apply(_this6, _arguments);
});
},
onReceivedFromServer: function onReceivedFromServer(iq) {
/* An IQ stanza containing the roster has been received from
* the XMPP server.
*/
var items = sizzle("query[xmlns=\"" + Strophe.NS.ROSTER + "\"] item", iq);
_.each(items, this.updateContact.bind(this));
_converse.emit('roster', iq);
},
updateContact: function updateContact(item) {
/* Update or create RosterContact models based on items
* received in the IQ from the server.
*/
var jid = item.getAttribute('jid');
if (this.isSelf(jid)) {
return;
2016-11-07 15:43:48 +01:00
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
var contact = this.get(jid),
subscription = item.getAttribute("subscription"),
ask = item.getAttribute("ask"),
groups = _.map(item.getElementsByTagName('group'), Strophe.getText);
if (!contact) {
if (subscription === "none" && ask === null || subscription === "remove") {
return; // We're lazy when adding contacts.
}
this.create({
ask: ask,
fullname: item.getAttribute("name") || jid,
groups: groups,
jid: jid,
subscription: subscription
}, { sort: false });
} else {
2017-07-22 22:21:05 +02:00
if (subscription === "remove") {
return contact.destroy(); // will trigger removeFromRoster
}
// We only find out about requesting contacts via the
// presence handler, so if we receive a contact
// here, we know they aren't requesting anymore.
// see docs/DEVELOPER.rst
contact.save({
subscription: subscription,
ask: ask,
requesting: null,
groups: groups
});
}
2017-07-22 22:21:05 +02:00
},
createRequestingContact: function createRequestingContact(presence) {
/* Creates a Requesting Contact.
*
* Note: this method gets completely overridden by converse-vcard.js
*/
var bare_jid = Strophe.getBareJidFromJid(presence.getAttribute('from')),
nick_el = presence.querySelector("nick[xmlns=\"" + Strophe.NS.NICK + "\"]");
var user_data = {
jid: bare_jid,
subscription: 'none',
ask: null,
requesting: true,
fullname: nick_el && nick_el.textContent || bare_jid
};
this.create(user_data);
_converse.emit('contactRequest', user_data);
},
handleIncomingSubscription: function handleIncomingSubscription(presence) {
var jid = presence.getAttribute('from'),
bare_jid = Strophe.getBareJidFromJid(jid),
contact = this.get(bare_jid);
2017-07-22 22:21:05 +02:00
if (!_converse.allow_contact_requests) {
_converse.rejectPresenceSubscription(jid, __("This client does not allow presence subscriptions"));
}
2017-07-22 22:21:05 +02:00
if (_converse.auto_subscribe) {
if (!contact || contact.get('subscription') !== 'to') {
this.subscribeBack(bare_jid);
} else {
contact.authorize();
}
} else {
2017-07-22 22:21:05 +02:00
if (contact) {
if (contact.get('subscription') !== 'none') {
contact.authorize();
} else if (contact.get('ask') === "subscribe") {
contact.authorize();
}
} else {
this.createRequestingContact(presence);
}
}
},
presenceHandler: function presenceHandler(presence) {
var presence_type = presence.getAttribute('type');
if (presence_type === 'error') {
return true;
}
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
var jid = presence.getAttribute('from'),
bare_jid = Strophe.getBareJidFromJid(jid),
resource = Strophe.getResourceFromJid(jid),
chat_status = _.propertyOf(presence.querySelector('show'))('textContent') || 'online',
status_message = _.propertyOf(presence.querySelector('status'))('textContent'),
contact = this.get(bare_jid);
2017-07-22 22:21:05 +02:00
if (this.isSelf(bare_jid)) {
if (_converse.connection.jid !== jid && presence_type !== 'unavailable' && (_converse.synchronize_availability === true || _converse.synchronize_availability === resource)) {
// Another resource has changed its status and
// synchronize_availability option set to update,
// we'll update ours as well.
_converse.xmppstatus.save({ 'status': chat_status });
if (status_message) {
_converse.xmppstatus.save({ 'status_message': status_message });
}
}
return;
} else if (sizzle("query[xmlns=\"" + Strophe.NS.MUC + "\"]", presence).length) {
return; // Ignore MUC
}
2017-07-22 22:21:05 +02:00
if (contact && status_message !== contact.get('status')) {
contact.save({ 'status': status_message });
}
2017-07-22 22:21:05 +02:00
if (presence_type === 'subscribed' && contact) {
contact.ackSubscribe();
} else if (presence_type === 'unsubscribed' && contact) {
contact.ackUnsubscribe();
} else if (presence_type === 'unsubscribe') {
return;
} else if (presence_type === 'subscribe') {
this.handleIncomingSubscription(presence);
} else if (presence_type === 'unavailable' && contact) {
contact.removeResource(resource);
} else if (contact) {
// presence_type is undefined
contact.addResource(presence);
}
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
});
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.RosterGroup = Backbone.Model.extend({
initialize: function initialize(attributes) {
this.set(_.assignIn({
description: __('Click to hide these contacts'),
state: _converse.OPENED
}, attributes));
// Collection of contacts belonging to this group.
this.contacts = new _converse.RosterContacts();
2017-04-23 19:02:44 +02:00
}
2017-07-22 22:21:05 +02:00
});
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
this.RosterGroups = Backbone.Collection.extend({
model: _converse.RosterGroup,
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
fetchRosterGroups: function fetchRosterGroups() {
var _this7 = this;
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
/* Fetches all the roster groups from sessionStorage.
*
* Returns a promise which resolves once the groups have been
* returned.
*/
return new Promise(function (resolve, reject) {
_this7.fetch({
silent: true, // We need to first have all groups before
// we can start positioning them, so we set
// 'silent' to true.
success: resolve
});
});
2017-04-23 19:02:44 +02:00
}
2017-07-22 22:21:05 +02:00
});
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
this.Message = Backbone.Model.extend({
defaults: function defaults() {
return {
msgid: _converse.connection.getUniqueId()
};
}
});
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.Messages = Backbone.Collection.extend({
model: _converse.Message,
comparator: 'time'
});
2017-07-22 22:21:05 +02:00
this.ChatBox = Backbone.Model.extend({
defaults: {
'type': 'chatbox',
'bookmarked': false,
'chat_state': undefined,
'num_unread': 0,
'url': ''
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
this.messages = new _converse.Messages();
this.messages.browserStorage = new Backbone.BrowserStorage[_converse.message_storage](b64_sha1("converse.messages" + this.get('jid') + _converse.bare_jid));
this.save({
// The chat_state will be set to ACTIVE once the chat box is opened
// and we listen for change:chat_state, so shouldn't set it to ACTIVE here.
'box_id': b64_sha1(this.get('jid')),
'time_opened': this.get('time_opened') || moment().valueOf(),
'user_id': Strophe.getNodeFromJid(this.get('jid'))
});
},
getMessageBody: function getMessageBody(message) {
var type = message.getAttribute('type');
return type === 'error' ? _.propertyOf(message.querySelector('error text'))('textContent') : _.propertyOf(message.querySelector('body'))('textContent');
},
getMessageAttributes: function getMessageAttributes(message, delay, original_stanza) {
delay = delay || message.querySelector('delay');
var type = message.getAttribute('type'),
body = this.getMessageBody(message);
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
var delayed = !_.isNull(delay),
is_groupchat = type === 'groupchat',
chat_state = message.getElementsByTagName(_converse.COMPOSING).length && _converse.COMPOSING || message.getElementsByTagName(_converse.PAUSED).length && _converse.PAUSED || message.getElementsByTagName(_converse.INACTIVE).length && _converse.INACTIVE || message.getElementsByTagName(_converse.ACTIVE).length && _converse.ACTIVE || message.getElementsByTagName(_converse.GONE).length && _converse.GONE;
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
var from = void 0;
if (is_groupchat) {
from = Strophe.unescapeNode(Strophe.getResourceFromJid(message.getAttribute('from')));
} else {
from = Strophe.getBareJidFromJid(message.getAttribute('from'));
}
var time = delayed ? delay.getAttribute('stamp') : moment().format();
var sender = void 0,
fullname = void 0;
if (is_groupchat && from === this.get('nick') || !is_groupchat && from === _converse.bare_jid) {
sender = 'me';
fullname = _converse.xmppstatus.get('fullname') || from;
} else {
sender = 'them';
fullname = this.get('fullname') || from;
}
return {
'type': type,
'chat_state': chat_state,
'delayed': delayed,
'fullname': fullname,
'message': body || undefined,
'msgid': message.getAttribute('id'),
'sender': sender,
'time': time
};
},
createMessage: function createMessage(message, delay, original_stanza) {
return this.messages.create(this.getMessageAttributes.apply(this, arguments));
},
newMessageWillBeHidden: function newMessageWillBeHidden() {
/* Returns a boolean to indicate whether a newly received
* message will be visible to the user or not.
*/
return this.get('hidden') || this.get('minimized') || this.isScrolledUp() || _converse.windowState === 'hidden';
},
incrementUnreadMsgCounter: function incrementUnreadMsgCounter(stanza) {
/* Given a newly received message, update the unread counter if
* necessary.
*/
if (_.isNull(stanza.querySelector('body'))) {
return; // The message has no text
}
if (utils.isNewMessage(stanza) && this.newMessageWillBeHidden()) {
this.save({ 'num_unread': this.get('num_unread') + 1 });
_converse.incrementMsgCounter();
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
},
clearUnreadMsgCounter: function clearUnreadMsgCounter() {
this.save({ 'num_unread': 0 });
},
isScrolledUp: function isScrolledUp() {
return this.get('scrolled', true);
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
});
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.ChatBoxes = Backbone.Collection.extend({
comparator: 'time_opened',
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
model: function model(attrs, options) {
return new _converse.ChatBox(attrs, options);
},
registerMessageHandler: function registerMessageHandler() {
_converse.connection.addHandler(this.onMessage.bind(this), null, 'message', 'chat');
_converse.connection.addHandler(this.onErrorMessage.bind(this), null, 'message', 'error');
},
chatBoxMayBeShown: function chatBoxMayBeShown(chatbox) {
return true;
},
onChatBoxesFetched: function onChatBoxesFetched(collection) {
var _this8 = this;
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
/* Show chat boxes upon receiving them from sessionStorage
*
* This method gets overridden entirely in src/converse-controlbox.js
* if the controlbox plugin is active.
*/
collection.each(function (chatbox) {
if (_this8.chatBoxMayBeShown(chatbox)) {
chatbox.trigger('show');
}
});
_converse.emit('chatBoxesFetched');
},
onConnected: function onConnected() {
this.browserStorage = new Backbone.BrowserStorage[_converse.storage](b64_sha1("converse.chatboxes-" + _converse.bare_jid));
this.registerMessageHandler();
this.fetch({
add: true,
success: this.onChatBoxesFetched.bind(this)
});
},
onErrorMessage: function onErrorMessage(message) {
/* Handler method for all incoming error message stanzas
*/
// TODO: we can likely just reuse "onMessage" below
var from_jid = Strophe.getBareJidFromJid(message.getAttribute('from'));
if (utils.isSameBareJID(from_jid, _converse.bare_jid)) {
return true;
}
// Get chat box, but only create a new one when the message has a body.
var chatbox = this.getChatBox(from_jid);
if (!chatbox) {
return true;
}
chatbox.createMessage(message, null, message);
return true;
},
onMessage: function onMessage(message) {
/* Handler method for all incoming single-user chat "message"
* stanzas.
*/
var contact_jid = void 0,
delay = void 0,
resource = void 0,
from_jid = message.getAttribute('from'),
to_jid = message.getAttribute('to');
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
var original_stanza = message,
to_resource = Strophe.getResourceFromJid(to_jid),
is_carbon = !_.isNull(message.querySelector("received[xmlns=\"" + Strophe.NS.CARBONS + "\"]"));
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
if (_converse.filter_by_resource && to_resource && to_resource !== _converse.resource) {
_converse.log("onMessage: Ignoring incoming message intended for a different resource: " + to_jid, Strophe.LogLevel.INFO);
return true;
} else if (utils.isHeadlineMessage(message)) {
// XXX: Ideally we wouldn't have to check for headline
// messages, but Prosody sends headline messages with the
// wrong type ('chat'), so we need to filter them out here.
_converse.log("onMessage: Ignoring incoming headline message sent with type 'chat' from JID: " + from_jid, Strophe.LogLevel.INFO);
return true;
}
var forwarded = message.querySelector('forwarded');
if (!_.isNull(forwarded)) {
var forwarded_message = forwarded.querySelector('message');
var forwarded_from = forwarded_message.getAttribute('from');
if (is_carbon && Strophe.getBareJidFromJid(forwarded_from) !== from_jid) {
// Prevent message forging via carbons
//
// https://xmpp.org/extensions/xep-0280.html#security
return true;
}
message = forwarded_message;
delay = forwarded.querySelector('delay');
from_jid = message.getAttribute('from');
to_jid = message.getAttribute('to');
2016-11-07 15:43:48 +01:00
}
2016-11-30 17:27:20 +01:00
2017-07-22 22:21:05 +02:00
var from_bare_jid = Strophe.getBareJidFromJid(from_jid),
from_resource = Strophe.getResourceFromJid(from_jid),
is_me = from_bare_jid === _converse.bare_jid;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (is_me) {
// I am the sender, so this must be a forwarded message...
contact_jid = Strophe.getBareJidFromJid(to_jid);
resource = Strophe.getResourceFromJid(to_jid);
} else {
contact_jid = from_bare_jid;
resource = from_resource;
}
// Get chat box, but only create a new one when the message has a body.
var chatbox = this.getChatBox(contact_jid, !_.isNull(message.querySelector('body'))),
msgid = message.getAttribute('id');
2017-07-22 22:21:05 +02:00
if (chatbox) {
var messages = msgid && chatbox.messages.findWhere({ msgid: msgid }) || [];
if (_.isEmpty(messages)) {
// Only create the message when we're sure it's not a
// duplicate
chatbox.incrementUnreadMsgCounter(original_stanza);
chatbox.createMessage(message, delay, original_stanza);
}
}
_converse.emit('message', { 'stanza': original_stanza, 'chatbox': chatbox });
return true;
},
createChatBox: function createChatBox(jid, attrs) {
/* Creates a chat box
*
* Parameters:
* (String) jid - The JID of the user for whom a chat box
* gets created.
* (Object) attrs - Optional chat box atributes.
*/
var bare_jid = Strophe.getBareJidFromJid(jid),
roster_item = _converse.roster.get(bare_jid);
var roster_info = {};
if (!_.isUndefined(roster_item)) {
roster_info = {
'fullname': _.isEmpty(roster_item.get('fullname')) ? jid : roster_item.get('fullname'),
'image_type': roster_item.get('image_type'),
'image': roster_item.get('image'),
'url': roster_item.get('url')
};
} else if (!_converse.allow_non_roster_messaging) {
_converse.log("Could not get roster item for JID " + bare_jid + ' and allow_non_roster_messaging is set to false', Strophe.LogLevel.ERROR);
return;
}
return this.create(_.assignIn({
'id': bare_jid,
'jid': bare_jid,
'fullname': jid,
'image_type': DEFAULT_IMAGE_TYPE,
'image': DEFAULT_IMAGE,
'url': ''
}, roster_info, attrs || {}));
},
getChatBox: function getChatBox(jid, create, attrs) {
/* 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.
*/
jid = jid.toLowerCase();
var chatbox = this.get(Strophe.getBareJidFromJid(jid));
if (!chatbox && create) {
chatbox = this.createChatBox(jid, attrs);
}
return chatbox;
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
});
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
this.ChatBoxViews = Backbone.Overview.extend({
initialize: function initialize() {
this.model.on("add", this.onChatBoxAdded, this);
this.model.on("destroy", this.removeChat, this);
},
_ensureElement: function _ensureElement() {
/* Override method from backbone.js
* If the #conversejs element doesn't exist, create it.
*/
if (!this.el) {
var el = document.querySelector('#conversejs');
if (_.isNull(el)) {
el = document.createElement('div');
el.setAttribute('id', 'conversejs');
// Converse.js expects a <body> tag to be present.
document.querySelector('body').appendChild(el);
}
el.innerHTML = '';
this.setElement(el, false);
2017-02-13 17:16:13 +01:00
} else {
2017-07-22 22:21:05 +02:00
this.setElement(_.result(this, 'el'), false);
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
},
onChatBoxAdded: function onChatBoxAdded(item) {
// Views aren't created here, since the core code doesn't
// contain any views. Instead, they're created in overrides in
// plugins, such as in converse-chatview.js and converse-muc.js
return this.get(item.get('id'));
},
removeChat: function removeChat(item) {
this.remove(item.get('id'));
},
closeAllChatBoxes: function closeAllChatBoxes() {
/* This method gets overridden in src/converse-controlbox.js if
* the controlbox plugin is active.
*/
this.each(function (view) {
view.close();
});
return this;
},
chatBoxMayBeShown: function chatBoxMayBeShown(chatbox) {
return this.model.chatBoxMayBeShown(chatbox);
},
getChatBox: function getChatBox(attrs, create) {
var chatbox = this.model.get(attrs.jid);
if (!chatbox && create) {
chatbox = this.model.create(attrs, {
'error': function error(model, response) {
_converse.log(response.responseText);
}
});
}
return chatbox;
},
showChat: function showChat(attrs) {
/* Find the chat box and show it (if it may be shown).
* If it doesn't exist, create it.
*/
var chatbox = this.getChatBox(attrs, true);
if (this.chatBoxMayBeShown(chatbox)) {
chatbox.trigger('show', true);
}
return chatbox;
}
});
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.XMPPStatus = Backbone.Model.extend({
initialize: function initialize() {
var _this9 = this;
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.set({
'status': this.getStatus()
});
this.on('change', function (item) {
if (_.has(item.changed, 'status')) {
_converse.emit('statusChanged', _this9.get('status'));
2016-03-07 18:54:07 +01:00
}
2017-07-22 22:21:05 +02:00
if (_.has(item.changed, 'status_message')) {
_converse.emit('statusMessageChanged', _this9.get('status_message'));
}
});
},
constructPresence: function constructPresence(type, status_message) {
var presence = void 0;
type = _.isString(type) ? type : this.get('status') || _converse.default_state;
status_message = _.isString(status_message) ? status_message : undefined;
// Most of these presence types are actually not explicitly sent,
// but I add all of them here for reference and future proofing.
if (type === 'unavailable' || type === 'probe' || type === 'error' || type === 'unsubscribe' || type === 'unsubscribed' || type === 'subscribe' || type === 'subscribed') {
presence = $pres({ 'type': type });
} else if (type === 'offline') {
presence = $pres({ 'type': 'unavailable' });
} else if (type === 'online') {
presence = $pres();
2016-03-07 18:54:07 +01:00
} else {
2017-07-22 22:21:05 +02:00
presence = $pres().c('show').t(type).up();
}
if (status_message) {
presence.c('status').t(status_message).up();
}
presence.c('priority').t(_.isNaN(Number(_converse.priority)) ? 0 : _converse.priority);
return presence;
},
sendPresence: function sendPresence(type, status_message) {
_converse.connection.send(this.constructPresence(type, status_message));
},
setStatus: function setStatus(value) {
this.sendPresence(value);
this.save({ 'status': value });
},
getStatus: function getStatus() {
return this.get('status') || _converse.default_state;
},
setStatusMessage: function setStatusMessage(status_message) {
this.sendPresence(this.getStatus(), status_message);
this.save({ 'status_message': status_message });
if (this.xhr_custom_status) {
var xhr = new XMLHttpRequest();
xhr.open('POST', this.xhr_custom_status_url, true);
xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded; charset=UTF-8');
xhr.send({ 'msg': status_message });
}
var prev_status = this.get('status_message');
if (prev_status === status_message) {
this.trigger("update-status-ui", this);
}
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
});
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.setUpXMLLogging = function () {
Strophe.log = function (level, msg) {
_converse.log(msg, level);
};
if (this.debug) {
this.connection.xmlInput = function (body) {
_converse.log(body.outerHTML, Strophe.LogLevel.DEBUG);
};
this.connection.xmlOutput = function (body) {
_converse.log(body.outerHTML, Strophe.LogLevel.DEBUG);
};
}
};
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
this.fetchLoginCredentials = function () {
return new Promise(function (resolve, reject) {
var xhr = new XMLHttpRequest();
xhr.open('GET', _converse.credentials_url, true);
xhr.setRequestHeader('Accept', "application/json, text/javascript");
xhr.onload = function () {
if (xhr.status >= 200 && xhr.status < 400) {
var data = JSON.parse(xhr.responseText);
resolve({
'jid': data.jid,
'password': data.password
});
2017-02-13 17:16:13 +01:00
} else {
2017-07-22 22:21:05 +02:00
xhr.onerror();
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
};
xhr.onerror = function () {
delete _converse.connection;
_converse.emit('noResumeableSession');
reject(xhr.responseText);
};
xhr.send();
});
};
this.startNewBOSHSession = function () {
var xhr = new XMLHttpRequest();
xhr.open('GET', _converse.prebind_url, true);
xhr.setRequestHeader('Accept', "application/json, text/javascript");
xhr.onload = function () {
if (xhr.status >= 200 && xhr.status < 400) {
var data = JSON.parse(xhr.responseText);
_converse.connection.attach(data.jid, data.sid, data.rid, _converse.onConnectStatusChanged);
2016-11-07 15:43:48 +01:00
} else {
2017-07-22 22:21:05 +02:00
xhr.onerror();
}
2017-07-22 22:21:05 +02:00
};
xhr.onerror = function () {
delete _converse.connection;
_converse.emit('noResumeableSession');
};
xhr.send();
};
2017-07-22 22:21:05 +02:00
this.restoreBOSHSession = function (jid_is_required) {
/* Tries to restore a cached BOSH session. */
if (!this.jid) {
var msg = "restoreBOSHSession: tried to restore a \"keepalive\" session " + "but we don't have the JID for the user!";
if (jid_is_required) {
throw new Error(msg);
} else {
_converse.log(msg);
}
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
try {
this.connection.restore(this.jid, this.onConnectStatusChanged);
return true;
} catch (e) {
this.log("Could not restore session for jid: " + this.jid + " Error message: " + e.message);
this.clearSession(); // If there's a roster, we want to clear it (see #555)
return false;
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
};
2017-07-22 22:21:05 +02:00
this.attemptPreboundSession = function (reconnecting) {
/* Handle session resumption or initialization when prebind is
* being used.
*/
if (!reconnecting) {
if (this.keepalive && this.restoreBOSHSession(true)) {
return;
}
// No keepalive, or session resumption has failed.
if (this.jid && this.sid && this.rid) {
return this.connection.attach(this.jid, this.sid, this.rid, this.onConnectStatusChanged);
}
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
if (this.prebind_url) {
return this.startNewBOSHSession();
} else {
throw new Error("attemptPreboundSession: If you use prebind and not keepalive, " + "then you MUST supply JID, RID and SID values or a prebind_url.");
2016-03-07 18:54:07 +01:00
}
2017-07-22 22:21:05 +02:00
};
2017-07-22 22:21:05 +02:00
this.attemptNonPreboundSession = function (credentials, reconnecting) {
/* Handle session resumption or initialization when prebind is not being used.
*
* Two potential options exist and are handled in this method:
* 1. keepalive
* 2. auto_login
*/
if (!reconnecting && this.keepalive && this.restoreBOSHSession()) {
return;
}
if (this.auto_login) {
if (credentials) {
// When credentials are passed in, they override prebinding
// or credentials fetching via HTTP
this.autoLogin(credentials);
} else if (this.credentials_url) {
this.fetchLoginCredentials().then(this.autoLogin.bind(this), this.autoLogin.bind(this));
} else if (!this.jid) {
throw new Error("attemptNonPreboundSession: If you use auto_login, " + "you also need to give either a jid value (and if " + "applicable a password) or you need to pass in a URL " + "from where the username and password can be fetched " + "(via credentials_url).");
} else {
this.autoLogin(); // Probably ANONYMOUS login
}
} else if (reconnecting) {
this.autoLogin();
}
};
2017-07-22 22:21:05 +02:00
this.autoLogin = function (credentials) {
if (credentials) {
// If passed in, the credentials come from credentials_url,
// so we set them on the converse object.
this.jid = credentials.jid;
}
if (this.authentication === _converse.ANONYMOUS) {
if (!this.jid) {
throw new Error("Config Error: when using anonymous login " + "you need to provide the server's domain via the 'jid' option. " + "Either when calling converse.initialize, or when calling " + "_converse.api.user.login.");
}
if (!this.connection.reconnecting) {
this.connection.reset();
}
this.connection.connect(this.jid.toLowerCase(), null, this.onConnectStatusChanged);
} else if (this.authentication === _converse.LOGIN) {
var password = _.isNil(credentials) ? _converse.connection.pass || this.password : credentials.password;
if (!password) {
if (this.auto_login) {
throw new Error("initConnection: If you use auto_login and " + "authentication='login' then you also need to provide a password.");
}
_converse.setDisconnectionCause(Strophe.Status.AUTHFAIL, undefined, true);
_converse.disconnect();
return;
}
var resource = Strophe.getResourceFromJid(this.jid);
if (!resource) {
this.jid = this.jid.toLowerCase() + _converse.generateResource();
} else {
this.jid = Strophe.getBareJidFromJid(this.jid).toLowerCase() + '/' + resource;
}
if (!this.connection.reconnecting) {
this.connection.reset();
}
this.connection.connect(this.jid, password, this.onConnectStatusChanged);
}
};
2017-07-22 22:21:05 +02:00
this.logIn = function (credentials, reconnecting) {
// We now try to resume or automatically set up a new session.
// Otherwise the user will be shown a login form.
if (this.authentication === _converse.PREBIND) {
this.attemptPreboundSession(reconnecting);
} else {
this.attemptNonPreboundSession(credentials, reconnecting);
}
};
2017-07-22 22:21:05 +02:00
this.initConnection = function () {
if (this.connection) {
return;
}
if (!this.bosh_service_url && !this.websocket_url) {
throw new Error("initConnection: you must supply a value for either the bosh_service_url or websocket_url or both.");
}
if (('WebSocket' in window || 'MozWebSocket' in window) && this.websocket_url) {
this.connection = new Strophe.Connection(this.websocket_url, this.connection_options);
} else if (this.bosh_service_url) {
this.connection = new Strophe.Connection(this.bosh_service_url, _.assignIn(this.connection_options, { 'keepalive': this.keepalive }));
} else {
throw new Error("initConnection: this browser does not support websockets and bosh_service_url wasn't specified.");
}
};
2017-07-22 22:21:05 +02:00
this._tearDown = function () {
/* Remove those views which are only allowed with a valid
* connection.
*/
_converse.emit('beforeTearDown');
this.unregisterPresenceHandler();
if (this.roster) {
this.roster.off().reset(); // Removes roster contacts
}
this.chatboxes.remove(); // Don't call off(), events won't get re-registered upon reconnect.
delete this.chatboxes.browserStorage;
this.session.destroy();
window.removeEventListener('click', _converse.onUserActivity);
window.removeEventListener('focus', _converse.onUserActivity);
window.removeEventListener('keypress', _converse.onUserActivity);
window.removeEventListener('mousemove', _converse.onUserActivity);
window.removeEventListener(unloadevent, _converse.onUserActivity);
window.clearInterval(_converse.everySecondTrigger);
_converse.emit('afterTearDown');
return this;
};
2017-07-22 22:21:05 +02:00
this.initChatBoxes = function () {
this.chatboxes = new this.ChatBoxes();
this.chatboxviews = new this.ChatBoxViews({ model: this.chatboxes });
2017-02-13 17:16:13 +01:00
};
2017-07-22 22:21:05 +02:00
this.initPlugins = function () {
// If initialize gets called a second time (e.g. during tests), then we
// need to re-apply all plugins (for a new converse instance), and we
// therefore need to clear this array that prevents plugins from being
// initialized twice.
// If initialize is called for the first time, then this array is empty
// in any case.
_converse.pluggable.initialized_plugins = [];
var whitelist = _converse.core_plugins.concat(_converse.whitelisted_plugins);
2017-04-23 19:02:44 +02:00
2017-07-22 22:21:05 +02:00
_converse.pluggable.initializePlugins({
'updateSettings': function updateSettings() {
_converse.log("(DEPRECATION) " + "The `updateSettings` method has been deprecated. " + "Please use `_converse.api.settings.update` instead.", Strophe.LogLevel.WARN);
_converse.api.settings.update.apply(_converse, arguments);
},
2016-05-03 17:37:10 +02:00
2017-07-22 22:21:05 +02:00
'_converse': _converse
}, whitelist, _converse.blacklisted_plugins);
_converse.emit('pluginsInitialized');
2017-02-13 17:16:13 +01:00
};
2017-07-22 22:21:05 +02:00
// Initialization
// --------------
// This is the end of the initialize method.
if (settings.connection) {
this.connection = settings.connection;
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
_converse.initPlugins();
_converse.initChatBoxes();
_converse.initConnection();
_converse.setUpXMLLogging();
_converse.logIn();
_converse.registerGlobalEventHandlers();
2017-07-22 22:21:05 +02:00
if (!_.isUndefined(_converse.connection) && _converse.connection.service === 'jasmine tests') {
return _converse;
} else {
return init_promise.promise;
}
};
2017-07-22 22:21:05 +02:00
// API methods only available to plugins
_converse.api = {
'connection': {
'connected': function connected() {
return _converse.connection && _converse.connection.connected || false;
},
'disconnect': function disconnect() {
_converse.connection.disconnect();
}
},
'emit': function emit() {
_converse.emit.apply(_converse, arguments);
},
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
'user': {
'jid': function jid() {
return _converse.connection.jid;
},
'login': function login(credentials) {
_converse.initConnection();
_converse.logIn(credentials);
},
'logout': function logout() {
_converse.logOut();
},
2017-07-22 22:21:05 +02:00
'status': {
'get': function get() {
return _converse.xmppstatus.get('status');
},
'set': function set(value, message) {
var data = { 'status': value };
if (!_.includes(_.keys(_converse.STATUS_WEIGHTS), value)) {
throw new Error('Invalid availability value. See https://xmpp.org/rfcs/rfc3921.html#rfc.section.2.2.2.1');
}
if (_.isString(message)) {
data.status_message = message;
}
_converse.xmppstatus.sendPresence(value);
_converse.xmppstatus.save(data);
},
2017-07-22 22:21:05 +02:00
'message': {
'get': function get() {
return _converse.xmppstatus.get('status_message');
},
'set': function set(stat) {
_converse.xmppstatus.save({ 'status_message': stat });
}
}
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
},
'settings': {
'update': function update(settings) {
utils.merge(_converse.default_settings, settings);
utils.merge(_converse, settings);
utils.applyUserSettings(_converse, settings, _converse.user_settings);
},
'get': function get(key) {
if (_.includes(_.keys(_converse.default_settings), key)) {
return _converse[key];
}
},
'set': function set(key, val) {
var o = {};
if (_.isObject(key)) {
_.assignIn(_converse, _.pick(key, _.keys(_converse.default_settings)));
} else if (_.isString("string")) {
o[key] = val;
_.assignIn(_converse, _.pick(o, _.keys(_converse.default_settings)));
}
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
},
'promises': {
'add': function add(promises) {
promises = _.isArray(promises) ? promises : [promises];
_.each(promises, addPromise);
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
},
'contacts': {
'get': function get(jids) {
var _transform = function _transform(jid) {
var contact = _converse.roster.get(Strophe.getBareJidFromJid(jid));
if (contact) {
return contact.attributes;
}
return null;
};
if (_.isUndefined(jids)) {
jids = _converse.roster.pluck('jid');
} else if (_.isString(jids)) {
return _transform(jids);
}
return _.map(jids, _transform);
},
'add': function add(jid, name) {
if (!_.isString(jid) || !_.includes(jid, '@')) {
throw new TypeError('contacts.add: invalid jid');
}
_converse.roster.addAndSubscribe(jid, _.isEmpty(name) ? jid : name);
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
},
'chats': {
'open': function open(jids, attrs) {
if (_.isUndefined(jids)) {
_converse.log("chats.open: You need to provide at least one JID", Strophe.LogLevel.ERROR);
return null;
} else if (_.isString(jids)) {
return _converse.getViewForChatBox(_converse.chatboxes.getChatBox(jids, true, attrs).trigger('show'));
}
return _.map(jids, function (jid) {
return _converse.getViewForChatBox(_converse.chatboxes.getChatBox(jid, true, attrs).trigger('show'));
});
},
'get': function get(jids) {
if (_.isUndefined(jids)) {
var result = [];
_converse.chatboxes.each(function (chatbox) {
// FIXME: Leaky abstraction from MUC. We need to add a
// base type for chat boxes, and check for that.
if (chatbox.get('type') !== 'chatroom') {
result.push(_converse.getViewForChatBox(chatbox));
}
});
return result;
} else if (_.isString(jids)) {
return _converse.getViewForChatBox(_converse.chatboxes.getChatBox(jids));
}
return _.map(jids, _.partial(_.flow(_converse.chatboxes.getChatBox.bind(_converse.chatboxes), _converse.getViewForChatBox.bind(_converse)), _, true));
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
},
'tokens': {
'get': function get(id) {
if (!_converse.expose_rid_and_sid || _.isUndefined(_converse.connection)) {
return null;
}
if (id.toLowerCase() === 'rid') {
return _converse.connection.rid || _converse.connection._proto.rid;
} else if (id.toLowerCase() === 'sid') {
return _converse.connection.sid || _converse.connection._proto.sid;
}
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
},
'listen': {
'once': _converse.once.bind(_converse),
'on': _converse.on.bind(_converse),
'not': _converse.off.bind(_converse),
'stanza': function stanza(name, options, handler) {
if (_.isFunction(options)) {
handler = options;
options = {};
} else {
options = options || {};
}
_converse.connection.addHandler(handler, options.ns, name, options.type, options.id, options.from, options);
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
},
'waitUntil': function waitUntil(name) {
var promise = _converse.promises[name];
if (_.isUndefined(promise)) {
return null;
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
return promise.promise;
},
'send': function send(stanza) {
_converse.connection.send(stanza);
}
};
2017-07-22 22:21:05 +02:00
// The public API
return {
'initialize': function initialize(settings, callback) {
return _converse.initialize(settings, callback);
},
'plugins': {
'add': function add(name, plugin) {
plugin.__name__ = name;
if (!_.isUndefined(_converse.pluggable.plugins[name])) {
throw new TypeError("Error: plugin with name \"" + name + "\" has already been " + 'registered!');
2017-02-13 17:16:13 +01:00
} else {
2017-07-22 22:21:05 +02:00
_converse.pluggable.plugins[name] = plugin;
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
}
},
'env': {
'$build': $build,
'$iq': $iq,
'$msg': $msg,
'$pres': $pres,
'Backbone': Backbone,
'Promise': Promise,
'Strophe': Strophe,
'_': _,
'b64_sha1': b64_sha1,
'fp': fp,
'moment': moment,
'sizzle': sizzle,
'utils': utils
}
};
});
//# sourceMappingURL=converse-core.js.map;
/* jshint maxerr: 10000 */
/* jslint unused: true */
/* jshint shadow: true */
/* jshint -W075 */
(function(ns){
// this list must be ordered from largest length of the value array, index 0, to the shortest
ns.emojioneList = {":kiss_mm:":{"uc_base":"1f468-2764-1f48b-1f468","uc_output":"1f468-200d-2764-fe0f-200d-1f48b-200d-1f468","uc_match":"1f468-2764-fe0f-1f48b-1f468","uc_greedy":"1f468-2764-1f48b-1f468","shortnames":[":couplekiss_mm:"],"category":"people"},":kiss_woman_man:":{"uc_base":"1f469-2764-1f48b-1f468","uc_output":"1f469-200d-2764-fe0f-200d-1f48b-200d-1f468","uc_match":"1f469-2764-fe0f-1f48b-1f468","uc_greedy":"1f469-2764-1f48b-1f468","shortnames":[],"category":"people"},":kiss_ww:":{"uc_base":"1f469-2764-1f48b-1f469","uc_output":"1f469-200d-2764-fe0f-200d-1f48b-200d-1f469","uc_match":"1f469-2764-fe0f-1f48b-1f469","uc_greedy":"1f469-2764-1f48b-1f469","shortnames":[":couplekiss_ww:"],"category":"people"},":england:":{"uc_base":"1f3f4-e0067-e0062-e0065-e006e-e0067-e007f","uc_output":"1f3f4-e0067-e0062-e0065-e006e-e0067-e007f","uc_match":"1f3f4-e0067-e0062-e0065-e006e-e0067-e007f","uc_greedy":"1f3f4-e0067-e0062-e0065-e006e-e0067-e007f","shortnames":[],"category":"flags"},":scotland:":{"uc_base":"1f3f4-e0067-e0062-e0073-e0063-e0074-e007f","uc_output":"1f3f4-e0067-e0062-e0073-e0063-e0074-e007f","uc_match":"1f3f4-e0067-e0062-e0073-e0063-e0074-e007f","uc_greedy":"1f3f4-e0067-e0062-e0073-e0063-e0074-e007f","shortnames":[],"category":"flags"},":wales:":{"uc_base":"1f3f4-e0067-e0062-e0077-e006c-e0073-e007f","uc_output":"1f3f4-e0067-e0062-e0077-e006c-e0073-e007f","uc_match":"1f3f4-e0067-e0062-e0077-e006c-e0073-e007f","uc_greedy":"1f3f4-e0067-e0062-e0077-e006c-e0073-e007f","shortnames":[],"category":"flags"},":family_mmbb:":{"uc_base":"1f468-1f468-1f466-1f466","uc_output":"1f468-200d-1f468-200d-1f466-200d-1f466","uc_match":"1f468-1f468-1f466-1f466","uc_greedy":"1f468-1f468-1f466-1f466","shortnames":[],"category":"people"},":family_mmgb:":{"uc_base":"1f468-1f468-1f467-1f466","uc_output":"1f468-200d-1f468-200d-1f467-200d-1f466","uc_match":"1f468-1f468-1f467-1f466","uc_greedy":"1f468-1f468-1f467-1f466","shortnames":[],"category":"people"},":family_mmgg:":{"uc_base":"1f468-1f468-1f467-1f467","uc_output":"1f468-200d-1f468-200d-1f467-200d-1f467","uc_match":"1f468-1f468-1f467-1f467","uc_greedy":"1f468-1f468-1f467-1f467","shortnames":[],"category":"people"},":family_mwbb:":{"uc_base":"1f468-1f469-1f466-1f466","uc_output":"1f468-200d-1f469-200d-1f466-200d-1f466","uc_match":"1f468-1f469-1f466-1f466","uc_greedy":"1f468-1f469-1f466-1f466","shortnames":[],"category":"people"},":family_mwgb:":{"uc_base":"1f468-1f469-1f467-1f466","uc_output":"1f468-200d-1f469-200d-1f467-200d-1f466","uc_match":"1f468-1f469-1f467-1f466","uc_greedy":"1f468-1f469-1f467-1f466","shortnames":[],"category":"people"},":family_mwgg:":{"uc_base":"1f468-1f469-1f467-1f467","uc_output":"1f468-200d-1f469-200d-1f467-200d-1f467","uc_match":"1f468-1f469-1f467-1f467","uc_greedy":"1f468-1f469-1f467-1f467","shortnames":[],"category":"people"},":family_wwbb:":{"uc_base":"1f469-1f469-1f466-1f466","uc_output":"1f469-200d-1f469-200d-1f466-200d-1f466","uc_match":"1f469-1f469-1f466-1f466","uc_greedy":"1f469-1f469-1f466-1f466","shortnames":[],"category":"people"},":family_wwgb:":{"uc_base":"1f469-1f469-1f467-1f466","uc_output":"1f469-200d-1f469-200d-1f467-200d-1f466","uc_match":"1f469-1f469-1f467-1f466","uc_greedy":"1f469-1f469-1f467-1f466","shortnames":[],"category":"people"},":family_wwgg:":{"uc_base":"1f469-1f469-1f467-1f467","uc_output":"1f469-200d-1f469-200d-1f467-200d-1f467","uc_match":"1f469-1f469-1f467-1f467","uc_greedy":"1f469-1f469-1f467-1f467","shortnames":[],"category":"people"},":couple_mm:":{"uc_base":"1f468-2764-1f468","uc_output":"1f468-200d-2764-fe0f-200d-1f468","uc_match":"1f468-2764-fe0f-1f468","uc_greedy":"1f468-2764-1f468","shortnames":[":couple_with_heart_mm:"],"category":"people"},":couple_with_heart_woman_man:":{"uc_base":"1f469-2764-1f468","uc_output":"1f469-200d-2764-fe0f-200d-1f468","uc_match":"1f469-2764-fe0f-1f468","uc_greedy":"1f469-2764-1f468","shortnames":[],"category":"people"},":couple_ww:":{"uc_base":"1f469-2764-1f469","uc_output":"1f469-200d-2764-fe0f-200d-1f469","uc_match":"1f469-2764-fe0f-1f469","uc_greedy":"1f469-2764-1f469","shortnames":[":
var tmpShortNames = [], emoji;
for (emoji in ns.emojioneList) {
if (!ns.emojioneList.hasOwnProperty(emoji) || (emoji === '')) continue;
tmpShortNames.push(emoji.replace(/[+]/g, "\\$&"));
for (var i = 0; i < ns.emojioneList[emoji].shortnames.length; i++) {
tmpShortNames.push(ns.emojioneList[emoji].shortnames[i].replace(/[+]/g, "\\$&"));
}
}
ns.shortnames = tmpShortNames.join('|');
// javascript escapes here must be ordered from largest length to shortest
ns.jsEscapeMap = {"\uD83D\uDC69\u200D\u2764\uFE0F\u200D\uD83D\uDC8B\u200D\uD83D\uDC69":"1f469-2764-1f48b-1f469","\uD83D\uDC68\u200D\u2764\uFE0F\u200D\uD83D\uDC8B\u200D\uD83D\uDC68":"1f468-2764-1f48b-1f468","\uD83D\uDC69\u200D\u2764\uFE0F\u200D\uD83D\uDC8B\u200D\uD83D\uDC68":"1f469-2764-1f48b-1f468","\uD83C\uDFF4\uDB40\uDC67\uDB40\uDC62\uDB40\uDC65\uDB40\uDC6E\uDB40\uDC67\uDB40\uDC7F":"1f3f4-e0067-e0062-e0065-e006e-e0067-e007f","\uD83C\uDFF4\uDB40\uDC67\uDB40\uDC62\uDB40\uDC73\uDB40\uDC63\uDB40\uDC74\uDB40\uDC7F":"1f3f4-e0067-e0062-e0073-e0063-e0074-e007f","\uD83C\uDFF4\uDB40\uDC67\uDB40\uDC62\uDB40\uDC77\uDB40\uDC6C\uDB40\uDC73\uDB40\uDC7F":"1f3f4-e0067-e0062-e0077-e006c-e0073-e007f","\uD83D\uDC68\u200D\uD83D\uDC68\u200D\uD83D\uDC66\u200D\uD83D\uDC66":"1f468-1f468-1f466-1f466","\uD83D\uDC68\u200D\uD83D\uDC68\u200D\uD83D\uDC67\u200D\uD83D\uDC66":"1f468-1f468-1f467-1f466","\uD83D\uDC68\u200D\uD83D\uDC68\u200D\uD83D\uDC67\u200D\uD83D\uDC67":"1f468-1f468-1f467-1f467","\uD83D\uDC68\u200D\uD83D\uDC69\u200D\uD83D\uDC66\u200D\uD83D\uDC66":"1f468-1f469-1f466-1f466","\uD83D\uDC68\u200D\uD83D\uDC69\u200D\uD83D\uDC67\u200D\uD83D\uDC66":"1f468-1f469-1f467-1f466","\uD83D\uDC68\u200D\uD83D\uDC69\u200D\uD83D\uDC67\u200D\uD83D\uDC67":"1f468-1f469-1f467-1f467","\uD83D\uDC69\u200D\uD83D\uDC69\u200D\uD83D\uDC66\u200D\uD83D\uDC66":"1f469-1f469-1f466-1f466","\uD83D\uDC69\u200D\uD83D\uDC69\u200D\uD83D\uDC67\u200D\uD83D\uDC66":"1f469-1f469-1f467-1f466","\uD83D\uDC69\u200D\uD83D\uDC69\u200D\uD83D\uDC67\u200D\uD83D\uDC67":"1f469-1f469-1f467-1f467","\uD83D\uDC68\u200D\u2764\u200D\uD83D\uDC8B\u200D\uD83D\uDC68":"1f468-2764-1f48b-1f468","\uD83D\uDC69\u200D\u2764\u200D\uD83D\uDC8B\u200D\uD83D\uDC68":"1f469-2764-1f48b-1f468","\uD83D\uDC69\u200D\u2764\u200D\uD83D\uDC8B\u200D\uD83D\uDC69":"1f469-2764-1f48b-1f469","\uD83D\uDC69\u200D\u2764\uFE0F\u200D\uD83D\uDC69":"1f469-2764-1f469","\uD83D\uDC68\u200D\u2764\uFE0F\u200D\uD83D\uDC68":"1f468-2764-1f468","\uD83D\uDC69\u200D\u2764\uFE0F\u200D\uD83D\uDC68":"1f469-2764-1f468","\uD83D\uDD75\uFE0F\uD83C\uDFFB\u200D\u2640\uFE0F":"1f575-1f3fb-2640","\uD83D\uDD75\uFE0F\uD83C\uDFFB\u200D\u2642\uFE0F":"1f575-1f3fb-2642","\uD83D\uDD75\uFE0F\uD83C\uDFFC\u200D\u2640\uFE0F":"1f575-1f3fc-2640","\uD83D\uDD75\uFE0F\uD83C\uDFFC\u200D\u2642\uFE0F":"1f575-1f3fc-2642","\uD83D\uDD75\uFE0F\uD83C\uDFFD\u200D\u2640\uFE0F":"1f575-1f3fd-2640","\uD83D\uDD75\uFE0F\uD83C\uDFFD\u200D\u2642\uFE0F":"1f575-1f3fd-2642","\uD83D\uDD75\uFE0F\uD83C\uDFFE\u200D\u2640\uFE0F":"1f575-1f3fe-2640","\uD83D\uDD75\uFE0F\uD83C\uDFFE\u200D\u2642\uFE0F":"1f575-1f3fe-2642","\uD83D\uDD75\uFE0F\uD83C\uDFFF\u200D\u2640\uFE0F":"1f575-1f3ff-2640","\uD83D\uDD75\uFE0F\uD83C\uDFFF\u200D\u2642\uFE0F":"1f575-1f3ff-2642","\uD83C\uDFCB\uFE0F\uD83C\uDFFB\u200D\u2640\uFE0F":"1f3cb-1f3fb-2640","\uD83C\uDFCB\uFE0F\uD83C\uDFFB\u200D\u2642\uFE0F":"1f3cb-1f3fb-2642","\uD83C\uDFCB\uFE0F\uD83C\uDFFC\u200D\u2640\uFE0F":"1f3cb-1f3fc-2640","\uD83C\uDFCB\uFE0F\uD83C\uDFFC\u200D\u2642\uFE0F":"1f3cb-1f3fc-2642","\uD83C\uDFCB\uFE0F\uD83C\uDFFD\u200D\u2640\uFE0F":"1f3cb-1f3fd-2640","\uD83C\uDFCB\uFE0F\uD83C\uDFFD\u200D\u2642\uFE0F":"1f3cb-1f3fd-2642","\uD83C\uDFCB\uFE0F\uD83C\uDFFE\u200D\u2640\uFE0F":"1f3cb-1f3fe-2640","\uD83C\uDFCB\uFE0F\uD83C\uDFFE\u200D\u2642\uFE0F":"1f3cb-1f3fe-2642","\uD83C\uDFCB\uFE0F\uD83C\uDFFF\u200D\u2640\uFE0F":"1f3cb-1f3ff-2640","\uD83C\uDFCB\uFE0F\uD83C\uDFFF\u200D\u2642\uFE0F":"1f3cb-1f3ff-2642","\uD83C\uDFCC\uFE0F\uD83C\uDFFB\u200D\u2640\uFE0F":"1f3cc-1f3fb-2640","\uD83C\uDFCC\uFE0F\uD83C\uDFFB\u200D\u2642\uFE0F":"1f3cc-1f3fb-2642","\uD83C\uDFCC\uFE0F\uD83C\uDFFC\u200D\u2640\uFE0F":"1f3cc-1f3fc-2640","\uD83C\uDFCC\uFE0F\uD83C\uDFFC\u200D\u2642\uFE0F":"1f3cc-1f3fc-2642","\uD83C\uDFCC\uFE0F\uD83C\uDFFD\u200D\u2640\uFE0F":"1f3cc-1f3fd-2640","\uD83C\uDFCC\uFE0F\uD83C\uDFFD\u200D\u2642\uFE0F":"1f3cc-1f3fd-2642","\uD83C\uDFCC\uFE0F\uD83C\uDFFE\u200D\u2640\uFE0F":"1f3cc-1f3fe-2640","\uD83C\uDFCC\uFE0F\uD83C\uDFFE\u200D\u2642\uFE0F":"1f3cc-1f3fe-2642","\uD83C\uDFCC\uFE0F\uD83C\uDFFF\u200D\u2640\uFE0F":"1f3cc-1f3ff-2640","\uD83C\uDFCC\uFE0F\uD83C\uDFFF\u200D\u
ns.jsEscapeMapGreedy = {"\uD83D\uDC69\u2764\uD83D\uDC8B\uD83D\uDC69":"1f469-2764-1f48b-1f469","\uD83D\uDC68\u2764\uD83D\uDC8B\uD83D\uDC68":"1f468-2764-1f48b-1f468","\uD83D\uDC69\u2764\uD83D\uDC8B\uD83D\uDC68":"1f469-2764-1f48b-1f468","\uD83D\uDC69\u2764\uD83D\uDC69":"1f469-2764-1f469","\uD83D\uDC68\u2764\uD83D\uDC68":"1f468-2764-1f468","\uD83C\uDFCC\uD83C\uDFFB\u2642":"1f3cc-1f3fb-2642","\uD83C\uDFCC\uD83C\uDFFC\u2642":"1f3cc-1f3fc-2642","\uD83C\uDFCC\uD83C\uDFFD\u2642":"1f3cc-1f3fd-2642","\uD83C\uDFCC\uD83C\uDFFE\u2642":"1f3cc-1f3fe-2642","\uD83C\uDFCC\uD83C\uDFFF\u2642":"1f3cc-1f3ff-2642","\uD83C\uDFCC\uD83C\uDFFB\u2640":"1f3cc-1f3fb-2640","\uD83C\uDFCC\uD83C\uDFFC\u2640":"1f3cc-1f3fc-2640","\uD83C\uDFCC\uD83C\uDFFD\u2640":"1f3cc-1f3fd-2640","\uD83C\uDFCC\uD83C\uDFFE\u2640":"1f3cc-1f3fe-2640","\uD83C\uDFCC\uD83C\uDFFF\u2640":"1f3cc-1f3ff-2640","\uD83D\uDC68\uD83C\uDFFB\u2696":"1f468-1f3fb-2696","\uD83D\uDC68\uD83C\uDFFC\u2696":"1f468-1f3fc-2696","\uD83D\uDC68\uD83C\uDFFD\u2696":"1f468-1f3fd-2696","\uD83D\uDC68\uD83C\uDFFE\u2696":"1f468-1f3fe-2696","\uD83D\uDC68\uD83C\uDFFF\u2696":"1f468-1f3ff-2696","\uD83D\uDC69\uD83C\uDFFB\u2696":"1f469-1f3fb-2696","\uD83D\uDC69\uD83C\uDFFC\u2696":"1f469-1f3fc-2696","\uD83D\uDC69\uD83C\uDFFD\u2696":"1f469-1f3fd-2696","\uD83D\uDC69\uD83C\uDFFE\u2696":"1f469-1f3fe-2696","\uD83D\uDC69\uD83C\uDFFF\u2696":"1f469-1f3ff-2696","\uD83D\uDC68\uD83C\uDFFB\u2708":"1f468-1f3fb-2708","\uD83D\uDC68\uD83C\uDFFC\u2708":"1f468-1f3fc-2708","\uD83D\uDC68\uD83C\uDFFD\u2708":"1f468-1f3fd-2708","\uD83D\uDC68\uD83C\uDFFE\u2708":"1f468-1f3fe-2708","\uD83D\uDC68\uD83C\uDFFF\u2708":"1f468-1f3ff-2708","\uD83D\uDC69\uD83C\uDFFB\u2708":"1f469-1f3fb-2708","\uD83D\uDC69\uD83C\uDFFC\u2708":"1f469-1f3fc-2708","\uD83D\uDC69\uD83C\uDFFD\u2708":"1f469-1f3fd-2708","\uD83D\uDC69\uD83C\uDFFE\u2708":"1f469-1f3fe-2708","\uD83D\uDC69\uD83C\uDFFF\u2708":"1f469-1f3ff-2708","\uD83D\uDC69\u2764\uD83D\uDC68":"1f469-2764-1f468","\uD83D\uDC68\uD83C\uDFFB\u2695":"1f468-1f3fb-2695","\uD83D\uDC68\uD83C\uDFFC\u2695":"1f468-1f3fc-2695","\uD83D\uDC68\uD83C\uDFFD\u2695":"1f468-1f3fd-2695","\uD83D\uDC68\uD83C\uDFFE\u2695":"1f468-1f3fe-2695","\uD83D\uDC68\uD83C\uDFFF\u2695":"1f468-1f3ff-2695","\uD83D\uDC69\uD83C\uDFFB\u2695":"1f469-1f3fb-2695","\uD83D\uDC69\uD83C\uDFFC\u2695":"1f469-1f3fc-2695","\uD83D\uDC69\uD83C\uDFFD\u2695":"1f469-1f3fd-2695","\uD83D\uDC69\uD83C\uDFFE\u2695":"1f469-1f3fe-2695","\uD83D\uDC69\uD83C\uDFFF\u2695":"1f469-1f3ff-2695","\uD83D\uDC6E\uD83C\uDFFB\u2640":"1f46e-1f3fb-2640","\uD83D\uDC6E\uD83C\uDFFB\u2642":"1f46e-1f3fb-2642","\uD83D\uDC6E\uD83C\uDFFC\u2640":"1f46e-1f3fc-2640","\uD83D\uDC6E\uD83C\uDFFC\u2642":"1f46e-1f3fc-2642","\uD83D\uDC6E\uD83C\uDFFD\u2640":"1f46e-1f3fd-2640","\uD83D\uDC6E\uD83C\uDFFD\u2642":"1f46e-1f3fd-2642","\uD83D\uDC6E\uD83C\uDFFE\u2640":"1f46e-1f3fe-2640","\uD83D\uDC6E\uD83C\uDFFE\u2642":"1f46e-1f3fe-2642","\uD83D\uDC6E\uD83C\uDFFF\u2640":"1f46e-1f3ff-2640","\uD83D\uDC6E\uD83C\uDFFF\u2642":"1f46e-1f3ff-2642","\uD83D\uDC71\uD83C\uDFFB\u2640":"1f471-1f3fb-2640","\uD83D\uDC71\uD83C\uDFFB\u2642":"1f471-1f3fb-2642","\uD83D\uDC71\uD83C\uDFFC\u2640":"1f471-1f3fc-2640","\uD83D\uDC71\uD83C\uDFFC\u2642":"1f471-1f3fc-2642","\uD83D\uDC71\uD83C\uDFFD\u2640":"1f471-1f3fd-2640","\uD83D\uDC71\uD83C\uDFFD\u2642":"1f471-1f3fd-2642","\uD83D\uDC71\uD83C\uDFFE\u2640":"1f471-1f3fe-2640","\uD83D\uDC71\uD83C\uDFFE\u2642":"1f471-1f3fe-2642","\uD83D\uDC71\uD83C\uDFFF\u2640":"1f471-1f3ff-2640","\uD83D\uDC71\uD83C\uDFFF\u2642":"1f471-1f3ff-2642","\uD83D\uDC73\uD83C\uDFFB\u2640":"1f473-1f3fb-2640","\uD83D\uDC73\uD83C\uDFFB\u2642":"1f473-1f3fb-2642","\uD83D\uDC73\uD83C\uDFFC\u2640":"1f473-1f3fc-2640","\uD83D\uDC73\uD83C\uDFFC\u2642":"1f473-1f3fc-2642","\uD83D\uDC73\uD83C\uDFFD\u2640":"1f473-1f3fd-2640","\uD83D\uDC73\uD83C\uDFFD\u2642":"1f473-1f3fd-2642","\uD83D\uDC73\uD83C\uDFFE\u2640":"1f473-1f3fe-2640","\uD83D\uDC73\uD83C\uDFFE\u2642":"1f473-1f3fe-2642","\uD83D\uDC73\uD83C\uDFFF\u2640":"1f473-1f3ff-2640","\uD83D\uDC73\uD83C\uDFFF\u2642":"1f473-1f3ff-2642","\uD83D\uDC77\uD83C\uDFFB\u2640":"1f477-1f3fb-2640","\uD83D\u
ns.asciiList = {
'*\\0/*':'1f646',
'*\\O/*':'1f646',
'-___-':'1f611',
':\'-)':'1f602',
'\':-)':'1f605',
'\':-D':'1f605',
'>:-)':'1f606',
'\':-(':'1f613',
'>:-(':'1f620',
':\'-(':'1f622',
'O:-)':'1f607',
'0:-3':'1f607',
'0:-)':'1f607',
'0;^)':'1f607',
'O;-)':'1f607',
'0;-)':'1f607',
'O:-3':'1f607',
'-__-':'1f611',
':-Þ':'1f61b',
'</3':'1f494',
':\')':'1f602',
':-D':'1f603',
'\':)':'1f605',
'\'=)':'1f605',
'\':D':'1f605',
'\'=D':'1f605',
'>:)':'1f606',
'>;)':'1f606',
'>=)':'1f606',
';-)':'1f609',
'*-)':'1f609',
';-]':'1f609',
';^)':'1f609',
'\':(':'1f613',
'\'=(':'1f613',
':-*':'1f618',
':^*':'1f618',
'>:P':'1f61c',
'X-P':'1f61c',
'>:[':'1f61e',
':-(':'1f61e',
':-[':'1f61e',
'>:(':'1f620',
':\'(':'1f622',
';-(':'1f622',
'>.<':'1f623',
'#-)':'1f635',
'%-)':'1f635',
'X-)':'1f635',
'\\0/':'1f646',
'\\O/':'1f646',
'0:3':'1f607',
'0:)':'1f607',
'O:)':'1f607',
'O=)':'1f607',
'O:3':'1f607',
'B-)':'1f60e',
'8-)':'1f60e',
'B-D':'1f60e',
'8-D':'1f60e',
'-_-':'1f611',
'>:\\':'1f615',
'>:/':'1f615',
':-/':'1f615',
':-.':'1f615',
':-P':'1f61b',
':Þ':'1f61b',
':-b':'1f61b',
':-O':'1f62e',
'O_O':'1f62e',
'>:O':'1f62e',
':-X':'1f636',
':-#':'1f636',
':-)':'1f642',
'(y)':'1f44d',
'<3':'2764',
':D':'1f603',
'=D':'1f603',
';)':'1f609',
'*)':'1f609',
';]':'1f609',
';D':'1f609',
':*':'1f618',
'=*':'1f618',
':(':'1f61e',
':[':'1f61e',
'=(':'1f61e',
':@':'1f620',
';(':'1f622',
'D:':'1f628',
':$':'1f633',
'=$':'1f633',
'#)':'1f635',
'%)':'1f635',
'X)':'1f635',
'B)':'1f60e',
'8)':'1f60e',
':/':'1f615',
':\\':'1f615',
'=/':'1f615',
'=\\':'1f615',
':L':'1f615',
'=L':'1f615',
':P':'1f61b',
'=P':'1f61b',
':b':'1f61b',
':O':'1f62e',
':X':'1f636',
':#':'1f636',
'=X':'1f636',
'=#':'1f636',
':)':'1f642',
'=]':'1f642',
'=)':'1f642',
':]':'1f642'
};
2017-07-22 22:21:05 +02:00
ns.asciiRegexp = '(\\*\\\\0\\/\\*|\\*\\\\O\\/\\*|\\-___\\-|\\:\'\\-\\)|\'\\:\\-\\)|\'\\:\\-D|\\>\\:\\-\\)|>\\:\\-\\)|\'\\:\\-\\(|\\>\\:\\-\\(|>\\:\\-\\(|\\:\'\\-\\(|O\\:\\-\\)|0\\:\\-3|0\\:\\-\\)|0;\\^\\)|O;\\-\\)|0;\\-\\)|O\\:\\-3|\\-__\\-|\\:\\-Þ|\\:\\-Þ|\\<\\/3|<\\/3|\\:\'\\)|\\:\\-D|\'\\:\\)|\'\\=\\)|\'\\:D|\'\\=D|\\>\\:\\)|>\\:\\)|\\>;\\)|>;\\)|\\>\\=\\)|>\\=\\)|;\\-\\)|\\*\\-\\)|;\\-\\]|;\\^\\)|\'\\:\\(|\'\\=\\(|\\:\\-\\*|\\:\\^\\*|\\>\\:P|>\\:P|X\\-P|\\>\\:\\[|>\\:\\[|\\:\\-\\(|\\:\\-\\[|\\>\\:\\(|>\\:\\(|\\:\'\\(|;\\-\\(|\\>\\.\\<|>\\.<|#\\-\\)|%\\-\\)|X\\-\\)|\\\\0\\/|\\\\O\\/|0\\:3|0\\:\\)|O\\:\\)|O\\=\\)|O\\:3|B\\-\\)|8\\-\\)|B\\-D|8\\-D|\\-_\\-|\\>\\:\\\\|>\\:\\\\|\\>\\:\\/|>\\:\\/|\\:\\-\\/|\\:\\-\\.|\\:\\-P|\\:Þ|\\:Þ|\\:\\-b|\\:\\-O|O_O|\\>\\:O|>\\:O|\\:\\-X|\\:\\-#|\\:\\-\\)|\\(y\\)|\\<3|<3|\\:D|\\=D|;\\)|\\*\\)|;\\]|;D|\\:\\*|\\=\\*|\\:\\(|\\:\\[|\\=\\(|\\:@|;\\(|D\\:|\\:\\$|\\=\\$|#\\)|%\\)|X\\)|B\\)|8\\)|\\:\\/|\\:\\\\|\\=\\/|\\=\\\\|\\:L|\\=L|\\:P|\\=P|\\:b|\\:O|\\:X|\\:#|\\=X|\\=#|\\:\\)|\\=\\]|\\=\\)|\\:\\])';
ns.emojiVersion = '3.1'; // you can [optionally] modify this to load alternate emoji versions. see readme for backwards compatibility and version options
ns.emojiSize = '32';
ns.greedyMatch = false; // set to true for greedy unicode matching
ns.imagePathPNG = 'https://cdn.jsdelivr.net/emojione/assets/' + ns.emojiVersion + '/png/';
ns.defaultPathPNG = ns.imagePathPNG;
ns.imageTitleTag = true; // set to false to remove title attribute from img tag
ns.sprites = false; // if this is true then sprite markup will be used
ns.spriteSize = '32';
ns.unicodeAlt = true; // use the unicode char as the alt attribute (makes copy and pasting the resulting text better)
ns.ascii = false; // change to true to convert ascii smileys
ns.riskyMatchAscii = false; // set true to match ascii without leading/trailing space char
ns.regShortNames = new RegExp("<object[^>]*>.*?<\/object>|<span[^>]*>.*?<\/span>|<(?:object|embed|svg|img|div|span|p|a)[^>]*>|("+ns.shortnames+")", "gi");
ns.regAscii = new RegExp("<object[^>]*>.*?<\/object>|<span[^>]*>.*?<\/span>|<(?:object|embed|svg|img|div|span|p|a)[^>]*>|((\\s|^)"+ns.asciiRegexp+"(?=\\s|$|[!,.?]))", "gi");
ns.regAsciiRisky = new RegExp("<object[^>]*>.*?<\/object>|<span[^>]*>.*?<\/span>|<(?:object|embed|svg|img|div|span|p|a)[^>]*>|(()"+ns.asciiRegexp+"())", "gi");
ns.regUnicode = new RegExp("<object[^>]*>.*?<\/object>|<span[^>]*>.*?<\/span>|<(?:object|embed|svg|img|div|span|p|a)[^>]*>|(?:\uD83C\uDFF3)\uFE0F?\u200D?(?:\uD83C\uDF08)|(?:\uD83D\uDC41)\uFE0F?\u200D?(?:\uD83D\uDDE8)\uFE0F?|[#-9]\uFE0F?\u20E3|(?:(?:\uD83C\uDFF4)(?:\uDB40[\uDC60-\uDCFF]){1,6})|(?:\uD83C[\uDDE0-\uDDFF]){2}|(?:(?:\uD83D[\uDC68\uDC69]))\uFE0F?(?:\uD83C[\uDFFA-\uDFFF])?\u200D?(?:[\u2695\u2696\u2708]|\uD83C[\uDF3E-\uDFED]|\uD83D[\uDCBB\uDCBC\uDD27\uDD2C\uDE80\uDE92])|(?:\uD83D[\uDC68\uDC69]|\uD83E[\uDDD0-\uDDDF])(?:\uD83C[\uDFFA-\uDFFF])?\u200D?[\u2640\u2642\u2695\u2696\u2708]?\uFE0F?|(?:(?:\u2764|\uD83D[\uDC66-\uDC69\uDC8B])[\u200D\uFE0F]{0,2}){1,3}(?:\u2764|\uD83D[\uDC66-\uDC69\uDC8B])|(?:(?:\u2764|\uD83D[\uDC66-\uDC69\uDC8B])\uFE0F?){2,4}|(?:\uD83D[\uDC68\uDC69\uDC6E\uDC71-\uDC87\uDD75\uDE45-\uDE4E]|\uD83E[\uDD26\uDD37]|\uD83C[\uDFC3-\uDFCC]|\uD83E[\uDD38-\uDD3E]|\uD83D[\uDEA3-\uDEB6]|\u26f9|\uD83D\uDC6F)\uFE0F?(?:\uD83C[\uDFFB-\uDFFF])?\u200D?[\u2640\u2642]?\uFE0F?|(?:[\u261D\u26F9\u270A-\u270D]|\uD83C[\uDF85-\uDFCC]|\uD83D[\uDC42-\uDCAA\uDD74-\uDD96\uDE45-\uDE4F\uDEA3-\uDECC]|\uD83E[\uDD18-\uDD3E])\uFE0F?(?:\uD83C[\uDFFB-\uDFFF])|(?:[\u2194-\u2199\u21a9-\u21aa]\uFE0F?|[\u0023\u002a]|[\u3030\u303d]\uFE0F?|(?:\ud83c[\udd70-\udd71]|\ud83c\udd8e|\ud83c[\udd91-\udd9a])\uFE0F?|\u24c2\uFE0F?|[\u3297\u3299]\uFE0F?|(?:\ud83c[\ude01-\ude02]|\ud83c\ude1a|\ud83c\ude2f|\ud83c[\ude32-\ude3a]|\ud83c[\ude50-\ude51])\uFE0F?|[\u203c\u2049]\uFE0F?|[\u25aa-\u25ab\u25b6\u25c0\u25fb-\u25fe]\uFE0F?|[\u00a9\u00ae]\uFE0F?|[\u2122\u2139]\uFE0F?|\ud83c\udc04\uFE0F?|[\u2b05-\u2b07\u2b1b-\u2b1c\u2b50\u2b55]\uFE0F?|[\u231a-\u231b\u2328\u23cf\u23e9-\u23f3\u23f8-\u23fa]\uFE0F?|\ud83c\udccf|[\u2934\u2935]\uFE0F?)|[\u2700-\u27bf]\uFE0F?|[\ud800-\udbff][\udc00-\udfff]\uFE0F?|[\u2600-\u26FF]\uFE0F?|[\u0030-\u0039]\uFE0F", "g");
ns.toImage = function(str) {
str = ns.unicodeToImage(str);
str = ns.shortnameToImage(str);
return str;
};
2017-07-22 22:21:05 +02:00
// Uses toShort to transform all unicode into a standard shortname
// then transforms the shortname into unicode
// This is done for standardization when converting several unicode types
ns.unifyUnicode = function(str) {
str = ns.toShort(str);
str = ns.shortnameToUnicode(str);
return str;
};
// Replace shortnames (:wink:) with Ascii equivalents ( ;^) )
// Useful for systems that dont support unicode nor images
ns.shortnameToAscii = function(str) {
var unicode,
// something to keep in mind here is that array flip will destroy
// half of the ascii text "emojis" because the unicode numbers are duplicated
// this is ok for what it's being used for
unicodeToAscii = ns.objectFlip(ns.asciiList);
str = str.replace(ns.regShortNames, function(shortname) {
if( (typeof shortname === 'undefined') || (shortname === '') || (!(shortname in ns.emojioneList)) ) {
// if the shortname doesnt exist just return the entire match
return shortname;
}
else {
unicode = ns.emojioneList[shortname].uc_output;
if(typeof unicodeToAscii[unicode] !== 'undefined') {
return unicodeToAscii[unicode];
2017-02-13 17:16:13 +01:00
} else {
2017-07-22 22:21:05 +02:00
return shortname;
2017-02-13 17:16:13 +01:00
}
}
2017-07-22 22:21:05 +02:00
});
return str;
};
2017-07-22 22:21:05 +02:00
// will output unicode from shortname
// useful for sending emojis back to mobile devices
ns.shortnameToUnicode = function(str) {
// replace regular shortnames first
var unicode,fname;
str = str.replace(ns.regShortNames, function(shortname) {
if( (typeof shortname === 'undefined') || (shortname === '') || (!(shortname in ns.emojioneList)) ) {
// if the shortname doesnt exist just return the entire matchhju
return shortname;
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
unicode = ns.emojioneList[shortname].uc_output.toUpperCase();
fname = ns.emojioneList[shortname].uc_base;
return ns.convert(unicode);
});
2017-07-22 22:21:05 +02:00
// if ascii smileys are turned on, then we'll replace them!
if (ns.ascii) {
var asciiRX = ns.riskyMatchAscii ? ns.regAsciiRisky : ns.regAscii;
str = str.replace(asciiRX, function(entire, m1, m2, m3) {
if( (typeof m3 === 'undefined') || (m3 === '') || (!(ns.unescapeHTML(m3) in ns.asciiList)) ) {
// if the ascii doesnt exist just return the entire match
return entire;
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
m3 = ns.unescapeHTML(m3);
unicode = ns.asciiList[m3].toUpperCase();
return m2+ns.convert(unicode);
});
}
return str;
};
ns.shortnameToImage = function(str) {
// replace regular shortnames first
var replaceWith,shortname,unicode,fname,alt,category,title,size,ePath;
var mappedUnicode = ns.mapUnicodeToShort();
str = str.replace(ns.regShortNames, function(shortname) {
if( (typeof shortname === 'undefined') || (shortname === '') || (ns.shortnames.indexOf(shortname) === -1) ) {
// if the shortname doesnt exist just return the entire match
return shortname;
2017-06-23 20:25:33 +02:00
}
2017-07-22 22:21:05 +02:00
else {
// map shortname to parent
if (!ns.emojioneList[shortname]) {
for ( var emoji in ns.emojioneList ) {
if (!ns.emojioneList.hasOwnProperty(emoji) || (emoji === '')) continue;
if (ns.emojioneList[emoji].shortnames.indexOf(shortname) === -1) continue;
shortname = emoji;
break;
}
}
unicode = ns.emojioneList[shortname].uc_output;
fname = ns.emojioneList[shortname].uc_base;
category = (fname.includes("-1f3f")) ? 'diversity' : ns.emojioneList[shortname].category;
title = ns.imageTitleTag ? 'title="' + shortname + '"' : '';
size = (ns.spriteSize == '32' || ns.spriteSize == '64') ? ns.spriteSize : '32';
//if the image path has not changed, we'll assume the default cdn path, otherwise we'll assume the provided path
ePath = (ns.imagePathPNG != ns.defaultPathPNG) ? ns.imagePathPNG : ns.defaultPathPNG + ns.emojiSize + '/';
// depending on the settings, we'll either add the native unicode as the alt tag, otherwise the shortname
alt = (ns.unicodeAlt) ? ns.convert(unicode.toUpperCase()) : shortname;
if(ns.sprites) {
replaceWith = '<span class="emojione emojione-' + size + '-' + category + ' _' + fname + '" ' + title + '>' + alt + '</span>';
}
else {
replaceWith = '<img class="emojione" alt="' + alt + '" ' + title + ' src="' + ePath + fname + '.png"/>';
}
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
return replaceWith;
2017-06-23 20:25:33 +02:00
}
2017-07-22 22:21:05 +02:00
});
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
// if ascii smileys are turned on, then we'll replace them!
if (ns.ascii) {
var asciiRX = ns.riskyMatchAscii ? ns.regAsciiRisky : ns.regAscii;
str = str.replace(asciiRX, function(entire, m1, m2, m3) {
if( (typeof m3 === 'undefined') || (m3 === '') || (!(ns.unescapeHTML(m3) in ns.asciiList)) ) {
// if the ascii doesnt exist just return the entire match
return entire;
}
m3 = ns.unescapeHTML(m3);
unicode = ns.asciiList[m3];
shortname = mappedUnicode[unicode];
category = (unicode.includes("-1f3f")) ? 'diversity' : ns.emojioneList[shortname].category;
title = ns.imageTitleTag ? 'title="' + ns.escapeHTML(m3) + '"' : '';
size = (ns.spriteSize == '32' || ns.spriteSize == '64') ? ns.spriteSize : '32';
//if the image path has not changed, we'll assume the default cdn path, otherwise we'll assume the provided path
ePath = (ns.imagePathPNG != ns.defaultPathPNG) ? ns.imagePathPNG : ns.defaultPathPNG + ns.emojiSize + '/';
// depending on the settings, we'll either add the native unicode as the alt tag, otherwise the shortname
alt = (ns.unicodeAlt) ? ns.convert(unicode.toUpperCase()) : ns.escapeHTML(m3);
if(ns.sprites) {
replaceWith = m2+'<span class="emojione emojione-' + size + '-' + category + ' _' + unicode +'" ' + title + '>' + alt + '</span>';
}
else {
replaceWith = m2+'<img class="emojione" alt="'+alt+'" ' + title + ' src="' + ePath + unicode + '.png"/>';
}
return replaceWith;
2017-02-13 17:16:13 +01:00
});
2017-07-22 22:21:05 +02:00
}
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
return str;
};
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
ns.unicodeToImage = function(str) {
var replaceWith,unicode,short,fname,alt,category,title,size,ePath;
var mappedUnicode = ns.mapUnicodeToShort();
var eList = ns.emojioneList;
str = str.replace(ns.regUnicode, function(unicodeChar) {
if( (typeof unicodeChar === 'undefined') || (unicodeChar === '') )
{
return unicodeChar;
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
else if ( unicodeChar in ns.jsEscapeMap )
{
fname = ns.jsEscapeMap[unicodeChar];
}
else if ( ns.greedyMatch && unicodeChar in ns.jsEscapeMapGreedy )
{
fname = ns.jsEscapeMapGreedy[unicodeChar];
}
else
{
return unicodeChar;
2017-06-23 20:25:33 +02:00
}
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
// then map to shortname and locate the filename
short = mappedUnicode[fname];
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
// then pull the unicode output from emojioneList
fname = eList[short].uc_base;
unicode = eList[short].uc_output;
category = (fname.includes("-1f3f")) ? 'diversity' : eList[short].category;
size = (ns.spriteSize == '32' || ns.spriteSize == '64') ? ns.spriteSize : '32';
//if the image path has not changed, we'll assume the default cdn path, otherwise we'll assume the provided path
ePath = (ns.imagePathPNG != ns.defaultPathPNG) ? ns.imagePathPNG : ns.defaultPathPNG + ns.emojiSize + '/';
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
// depending on the settings, we'll either add the native unicode as the alt tag, otherwise the shortname
alt = (ns.unicodeAlt) ? ns.convert(unicode.toUpperCase()) : short;
title = ns.imageTitleTag ? 'title="' + short + '"' : '';
if(ns.sprites) {
replaceWith = '<span class="emojione emojione-' + size + '-' + category + ' _' + fname + '" ' + title + '>' + alt + '</span>';
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
else {
replaceWith = '<img class="emojione" alt="' + alt + '" ' + title + ' src="' + ePath + fname + '.png"/>';
2017-02-13 17:16:13 +01:00
}
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
return replaceWith;
});
// if ascii smileys are turned on, then we'll replace them!
if (ns.ascii) {
var asciiRX = ns.riskyMatchAscii ? ns.regAsciiRisky : ns.regAscii;
str = str.replace(asciiRX, function(entire, m1, m2, m3) {
if( (typeof m3 === 'undefined') || (m3 === '') || (!(ns.unescapeHTML(m3) in ns.asciiList)) ) {
// if the ascii doesnt exist just return the entire match
return entire;
}
m3 = ns.unescapeHTML(m3);
unicode = ns.asciiList[m3];
shortname = mappedUnicode[unicode];
category = (unicode.includes("-1f3f")) ? 'diversity' : ns.emojioneList[shortname].category;
title = ns.imageTitleTag ? 'title="' + ns.escapeHTML(m3) + '"' : '';
size = (ns.spriteSize == '32' || ns.spriteSize == '64') ? ns.spriteSize : '32';
//if the image path has not changed, we'll assume the default cdn path, otherwise we'll assume the provided path
ePath = (ns.imagePathPNG != ns.defaultPathPNG) ? ns.imagePathPNG : ns.defaultPathPNG + ns.emojiSize + '/';
// depending on the settings, we'll either add the native unicode as the alt tag, otherwise the shortname
alt = (ns.unicodeAlt) ? ns.convert(unicode.toUpperCase()) : ns.escapeHTML(m3);
if(ns.sprites) {
replaceWith = m2+'<span class="emojione emojione-' + size + '-' + category + ' _' + unicode +'" ' + title + '>' + alt + '</span>';
}
else {
replaceWith = m2+'<img class="emojione" alt="'+alt+'" ' + title + ' src="' + ePath + unicode + '.png"/>';
}
return replaceWith;
});
}
return str;
};
// this is really just unicodeToShortname() but I opted for the shorthand name to match toImage()
ns.toShort = function(str) {
var find = ns.unicodeCharRegex();
return ns.replaceAll(str, find);
};
// for converting unicode code points and code pairs to their respective characters
ns.convert = function(unicode) {
if(unicode.indexOf("-") > -1) {
var parts = [];
var s = unicode.split('-');
for(var i = 0; i < s.length; i++) {
var part = parseInt(s[i], 16);
if (part >= 0x10000 && part <= 0x10FFFF) {
var hi = Math.floor((part - 0x10000) / 0x400) + 0xD800;
var lo = ((part - 0x10000) % 0x400) + 0xDC00;
part = (String.fromCharCode(hi) + String.fromCharCode(lo));
}
else {
part = String.fromCharCode(part);
}
parts.push(part);
2016-03-16 12:49:35 +01:00
}
2017-07-22 22:21:05 +02:00
return parts.join('');
}
else {
var s = parseInt(unicode, 16);
if (s >= 0x10000 && s <= 0x10FFFF) {
var hi = Math.floor((s - 0x10000) / 0x400) + 0xD800;
var lo = ((s - 0x10000) % 0x400) + 0xDC00;
return (String.fromCharCode(hi) + String.fromCharCode(lo));
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
else {
return String.fromCharCode(s);
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
}
};
2017-07-22 22:21:05 +02:00
ns.escapeHTML = function (string) {
var escaped = {
'&' : '&amp;',
'<' : '&lt;',
'>' : '&gt;',
'"' : '&quot;',
'\'': '&#039;'
2017-02-13 17:16:13 +01:00
};
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
return string.replace(/[&<>"']/g, function (match) {
return escaped[match];
});
};
ns.unescapeHTML = function (string) {
var unescaped = {
'&amp;' : '&',
'&#38;' : '&',
'&#x26;' : '&',
'&lt;' : '<',
'&#60;' : '<',
'&#x3C;' : '<',
'&gt;' : '>',
'&#62;' : '>',
'&#x3E;' : '>',
'&quot;' : '"',
'&#34;' : '"',
'&#x22;' : '"',
'&apos;' : '\'',
'&#39;' : '\'',
'&#x27;' : '\''
2017-02-13 17:16:13 +01:00
};
2017-07-22 22:21:05 +02:00
return string.replace(/&(?:amp|#38|#x26|lt|#60|#x3C|gt|#62|#x3E|apos|#39|#x27|quot|#34|#x22);/ig, function (match) {
return unescaped[match];
});
};
ns.shortnameConversionMap = function() {
var map = [], emoji;
for (emoji in ns.emojioneList) {
if (!ns.emojioneList.hasOwnProperty(emoji) || (emoji === '')) continue;
map[ns.convert(ns.emojioneList[emoji].uc_output)] = emoji;
}
return map;
};
ns.unicodeCharRegex = function() {
var map = [];
for (emoji in ns.emojioneList) {
if (!ns.emojioneList.hasOwnProperty(emoji) || (emoji === '')) continue;
map.push(ns.convert(ns.emojioneList[emoji].uc_output));
}
return map.join('|');
};
ns.mapEmojioneList = function (addToMapStorage) {
for (var shortname in ns.emojioneList) {
if (!ns.emojioneList.hasOwnProperty(shortname)) { continue; }
var unicode = ns.emojioneList[shortname].uc_base;
addToMapStorage(unicode, shortname);
}
};
ns.mapUnicodeToShort = function() {
if (!ns.memMapShortToUnicode) {
ns.memMapShortToUnicode = {};
ns.mapEmojioneList(function (unicode, shortname) {
ns.memMapShortToUnicode[unicode] = shortname;
});
}
return ns.memMapShortToUnicode;
};
2017-07-22 22:21:05 +02:00
ns.memorizeReplacement = function() {
if (!ns.unicodeReplacementRegEx || !ns.memMapShortToUnicodeCharacters) {
var unicodeList = [];
ns.memMapShortToUnicodeCharacters = {};
ns.mapEmojioneList(function (unicode, shortname) {
var emojiCharacter = ns.convert(unicode);
ns.memMapShortToUnicodeCharacters[emojiCharacter] = shortname;
unicodeList.push(emojiCharacter);
});
ns.unicodeReplacementRegEx = unicodeList.join('|');
}
};
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
ns.mapUnicodeCharactersToShort = function() {
ns.memorizeReplacement();
return ns.memMapShortToUnicodeCharacters;
};
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
//reverse an object
ns.objectFlip = function (obj) {
var key, tmp_obj = {};
2016-05-03 17:37:10 +02:00
2017-07-22 22:21:05 +02:00
for (key in obj) {
if (obj.hasOwnProperty(key)) {
tmp_obj[obj[key]] = key;
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
}
2017-07-22 22:21:05 +02:00
return tmp_obj;
};
2017-07-22 22:21:05 +02:00
ns.escapeRegExp = function(string) {
return string.replace(/[-[\]{}()*+?.,;:&\\^$#\s]/g, "\\$&");
};
2017-07-22 22:21:05 +02:00
ns.replaceAll = function(string, find) {
var escapedFind = ns.escapeRegExp(find); //sorted largest output to smallest output
var search = new RegExp("<object[^>]*>.*?<\/object>|<span[^>]*>.*?<\/span>|<(?:object|embed|svg|img|div|span|p|a)[^>]*>|("+escapedFind+")", "gi");
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
// callback prevents replacing anything inside of these common html tags as well as between an <object></object> tag
2017-07-22 22:21:05 +02:00
var replace = function(entire, m1) {
return ((typeof m1 === 'undefined') || (m1 === '')) ? entire : ns.shortnameConversionMap()[m1];
2017-02-13 17:16:13 +01:00
};
2017-07-22 22:21:05 +02:00
return string.replace(search,replace);
};
2017-07-22 22:21:05 +02:00
}(this.emojione = this.emojione || {}));
if(typeof module === "object") module.exports = this.emojione;
define("emojione", (function (global) {
return function () {
var ret, fn;
return ret || global.emojione;
};
}(this)));
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
/**
* 默认配置
*
* @author 老雷<leizongmin@gmail.com>
*/
2017-07-22 22:21:05 +02:00
var FilterCSS = require('cssfilter').FilterCSS;
var getDefaultCSSWhiteList = require('cssfilter').getDefaultWhiteList;
var _ = require('./util');
// 默认白名单
function getDefaultWhiteList () {
return {
a: ['target', 'href', 'title'],
abbr: ['title'],
address: [],
area: ['shape', 'coords', 'href', 'alt'],
article: [],
aside: [],
audio: ['autoplay', 'controls', 'loop', 'preload', 'src'],
b: [],
bdi: ['dir'],
bdo: ['dir'],
big: [],
blockquote: ['cite'],
br: [],
caption: [],
center: [],
cite: [],
code: [],
col: ['align', 'valign', 'span', 'width'],
colgroup: ['align', 'valign', 'span', 'width'],
dd: [],
del: ['datetime'],
details: ['open'],
div: [],
dl: [],
dt: [],
em: [],
font: ['color', 'size', 'face'],
footer: [],
h1: [],
h2: [],
h3: [],
h4: [],
h5: [],
h6: [],
header: [],
hr: [],
i: [],
img: ['src', 'alt', 'title', 'width', 'height'],
ins: ['datetime'],
li: [],
mark: [],
nav: [],
ol: [],
p: [],
pre: [],
s: [],
section:[],
small: [],
span: [],
sub: [],
sup: [],
strong: [],
table: ['width', 'border', 'align', 'valign'],
tbody: ['align', 'valign'],
td: ['width', 'rowspan', 'colspan', 'align', 'valign'],
tfoot: ['align', 'valign'],
th: ['width', 'rowspan', 'colspan', 'align', 'valign'],
thead: ['align', 'valign'],
tr: ['rowspan', 'align', 'valign'],
tt: [],
u: [],
ul: [],
video: ['autoplay', 'controls', 'loop', 'preload', 'src', 'height', 'width']
};
}
2017-07-22 22:21:05 +02:00
// 默认CSS Filter
var defaultCSSFilter = new FilterCSS();
2017-07-22 22:21:05 +02:00
/**
* 匹配到标签时的处理方法
*
* @param {String} tag
* @param {String} html
* @param {Object} options
* @return {String}
*/
function onTag (tag, html, options) {
// do nothing
}
2017-07-22 22:21:05 +02:00
/**
* 匹配到不在白名单上的标签时的处理方法
*
* @param {String} tag
* @param {String} html
* @param {Object} options
* @return {String}
*/
function onIgnoreTag (tag, html, options) {
// do nothing
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
/**
* 匹配到标签属性时的处理方法
*
* @param {String} tag
* @param {String} name
* @param {String} value
* @return {String}
*/
function onTagAttr (tag, name, value) {
// do nothing
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
/**
* 匹配到不在白名单上的标签属性时的处理方法
*
* @param {String} tag
* @param {String} name
* @param {String} value
* @return {String}
*/
function onIgnoreTagAttr (tag, name, value) {
// do nothing
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
/**
* HTML转义
*
* @param {String} html
*/
function escapeHtml (html) {
return html.replace(REGEXP_LT, '&lt;').replace(REGEXP_GT, '&gt;');
}
2017-07-22 22:21:05 +02:00
/**
* 安全的标签属性值
*
* @param {String} tag
* @param {String} name
* @param {String} value
* @param {Object} cssFilter
* @return {String}
*/
function safeAttrValue (tag, name, value, cssFilter) {
// 转换为友好的属性值,再做判断
value = friendlyAttrValue(value);
if (name === 'href' || name === 'src') {
// 过滤 href 和 src 属性
// 仅允许 http:// | https:// | mailto: | / | # 开头的地址
value = _.trim(value);
if (value === '#') return '#';
if (!(value.substr(0, 7) === 'http://' ||
value.substr(0, 8) === 'https://' ||
value.substr(0, 7) === 'mailto:' ||
value[0] === '#' ||
value[0] === '/')) {
return '';
}
} else if (name === 'background') {
// 过滤 background 属性 这个xss漏洞较老了可能已经不适用
// javascript:
REGEXP_DEFAULT_ON_TAG_ATTR_4.lastIndex = 0;
if (REGEXP_DEFAULT_ON_TAG_ATTR_4.test(value)) {
return '';
}
} else if (name === 'style') {
// /*注释*/
/*REGEXP_DEFAULT_ON_TAG_ATTR_3.lastIndex = 0;
if (REGEXP_DEFAULT_ON_TAG_ATTR_3.test(value)) {
return '';
}*/
// expression()
REGEXP_DEFAULT_ON_TAG_ATTR_7.lastIndex = 0;
if (REGEXP_DEFAULT_ON_TAG_ATTR_7.test(value)) {
return '';
}
// url()
REGEXP_DEFAULT_ON_TAG_ATTR_8.lastIndex = 0;
if (REGEXP_DEFAULT_ON_TAG_ATTR_8.test(value)) {
REGEXP_DEFAULT_ON_TAG_ATTR_4.lastIndex = 0;
if (REGEXP_DEFAULT_ON_TAG_ATTR_4.test(value)) {
return '';
}
}
if (cssFilter !== false) {
cssFilter = cssFilter || defaultCSSFilter;
value = cssFilter.process(value);
}
}
2017-07-22 22:21:05 +02:00
// 输出时需要转义<>"
value = escapeAttrValue(value);
return value;
}
2017-07-22 22:21:05 +02:00
// 正则表达式
var REGEXP_LT = /</g;
var REGEXP_GT = />/g;
var REGEXP_QUOTE = /"/g;
var REGEXP_QUOTE_2 = /&quot;/g;
var REGEXP_ATTR_VALUE_1 = /&#([a-zA-Z0-9]*);?/img;
var REGEXP_ATTR_VALUE_COLON = /&colon;?/img;
var REGEXP_ATTR_VALUE_NEWLINE = /&newline;?/img;
var REGEXP_DEFAULT_ON_TAG_ATTR_3 = /\/\*|\*\//mg;
var REGEXP_DEFAULT_ON_TAG_ATTR_4 = /((j\s*a\s*v\s*a|v\s*b|l\s*i\s*v\s*e)\s*s\s*c\s*r\s*i\s*p\s*t\s*|m\s*o\s*c\s*h\s*a)\:/ig;
var REGEXP_DEFAULT_ON_TAG_ATTR_5 = /^[\s"'`]*(d\s*a\s*t\s*a\s*)\:/ig;
var REGEXP_DEFAULT_ON_TAG_ATTR_6 = /^[\s"'`]*(d\s*a\s*t\s*a\s*)\:\s*image\//ig;
var REGEXP_DEFAULT_ON_TAG_ATTR_7 = /e\s*x\s*p\s*r\s*e\s*s\s*s\s*i\s*o\s*n\s*\(.*/ig;
var REGEXP_DEFAULT_ON_TAG_ATTR_8 = /u\s*r\s*l\s*\(.*/ig;
2017-07-22 22:21:05 +02:00
/**
* 对双引号进行转义
*
* @param {String} str
* @return {String} str
*/
function escapeQuote (str) {
return str.replace(REGEXP_QUOTE, '&quot;');
}
/**
* 对双引号进行转义
*
* @param {String} str
* @return {String} str
*/
function unescapeQuote (str) {
return str.replace(REGEXP_QUOTE_2, '"');
}
/**
* 对html实体编码进行转义
*
* @param {String} str
* @return {String}
*/
function escapeHtmlEntities (str) {
return str.replace(REGEXP_ATTR_VALUE_1, function replaceUnicode (str, code) {
return (code[0] === 'x' || code[0] === 'X')
? String.fromCharCode(parseInt(code.substr(1), 16))
: String.fromCharCode(parseInt(code, 10));
});
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
/**
* 对html5新增的危险实体编码进行转义
*
* @param {String} str
* @return {String}
*/
function escapeDangerHtml5Entities (str) {
return str.replace(REGEXP_ATTR_VALUE_COLON, ':')
.replace(REGEXP_ATTR_VALUE_NEWLINE, ' ');
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
/**
* 清除不可见字符
*
* @param {String} str
* @return {String}
*/
function clearNonPrintableCharacter (str) {
var str2 = '';
for (var i = 0, len = str.length; i < len; i++) {
str2 += str.charCodeAt(i) < 32 ? ' ' : str.charAt(i);
}
return _.trim(str2);
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
/**
* 将标签的属性值转换成一般字符便于分析
*
* @param {String} str
* @return {String}
*/
function friendlyAttrValue (str) {
str = unescapeQuote(str); // 双引号
str = escapeHtmlEntities(str); // 转换HTML实体编码
str = escapeDangerHtml5Entities(str); // 转换危险的HTML5新增实体编码
str = clearNonPrintableCharacter(str); // 清除不可见字符
return str;
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
/**
* 转义用于输出的标签属性值
*
* @param {String} str
* @return {String}
*/
function escapeAttrValue (str) {
str = escapeQuote(str);
str = escapeHtml(str);
return str;
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
/**
* 去掉不在白名单中的标签onIgnoreTag处理方法
*/
function onIgnoreTagStripAll () {
return '';
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
/**
* 删除标签体
*
* @param {array} tags 要删除的标签列表
* @param {function} next 对不在列表中的标签的处理函数可选
*/
function StripTagBody (tags, next) {
if (typeof(next) !== 'function') {
next = function () {};
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var isRemoveAllTag = !Array.isArray(tags);
function isRemoveTag (tag) {
if (isRemoveAllTag) return true;
return (_.indexOf(tags, tag) !== -1);
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var removeList = []; // 要删除的位置范围列表
var posStart = false; // 当前标签开始位置
return {
onIgnoreTag: function (tag, html, options) {
if (isRemoveTag(tag)) {
if (options.isClosing) {
var ret = '[/removed]';
var end = options.position + ret.length;
removeList.push([posStart !== false ? posStart : options.position, end]);
posStart = false;
return ret;
} else {
if (!posStart) {
posStart = options.position;
}
return '[removed]';
}
} else {
return next(tag, html, options);
}
},
remove: function (html) {
var rethtml = '';
var lastPos = 0;
_.forEach(removeList, function (pos) {
rethtml += html.slice(lastPos, pos[0]);
lastPos = pos[1];
});
rethtml += html.slice(lastPos);
return rethtml;
}
};
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
/**
* 去除备注标签
*
* @param {String} html
* @return {String}
*/
function stripCommentTag (html) {
return html.replace(STRIP_COMMENT_TAG_REGEXP, '');
}
var STRIP_COMMENT_TAG_REGEXP = /<!--[\s\S]*?-->/g;
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
/**
* 去除不可见字符
*
* @param {String} html
* @return {String}
*/
function stripBlankChar (html) {
var chars = html.split('');
chars = chars.filter(function (char) {
var c = char.charCodeAt(0);
if (c === 127) return false;
if (c <= 31) {
if (c === 10 || c === 13) return true;
return false;
}
return true;
});
return chars.join('');
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
exports.whiteList = getDefaultWhiteList();
exports.getDefaultWhiteList = getDefaultWhiteList;
exports.onTag = onTag;
exports.onIgnoreTag = onIgnoreTag;
exports.onTagAttr = onTagAttr;
exports.onIgnoreTagAttr = onIgnoreTagAttr;
exports.safeAttrValue = safeAttrValue;
exports.escapeHtml = escapeHtml;
exports.escapeQuote = escapeQuote;
exports.unescapeQuote = unescapeQuote;
exports.escapeHtmlEntities = escapeHtmlEntities;
exports.escapeDangerHtml5Entities = escapeDangerHtml5Entities;
exports.clearNonPrintableCharacter = clearNonPrintableCharacter;
exports.friendlyAttrValue = friendlyAttrValue;
exports.escapeAttrValue = escapeAttrValue;
exports.onIgnoreTagStripAll = onIgnoreTagStripAll;
exports.StripTagBody = StripTagBody;
exports.stripCommentTag = stripCommentTag;
exports.stripBlankChar = stripBlankChar;
exports.cssFilter = defaultCSSFilter;
exports.getDefaultCSSWhiteList = getDefaultCSSWhiteList;
},{"./util":4,"cssfilter":8}],2:[function(require,module,exports){
/**
* 模块入口
*
* @author 老雷<leizongmin@gmail.com>
*/
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
var DEFAULT = require('./default');
var parser = require('./parser');
var FilterXSS = require('./xss');
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
/**
* XSS过滤
*
* @param {String} html 要过滤的HTML代码
* @param {Object} options 选项whiteList, onTag, onTagAttr, onIgnoreTag, onIgnoreTagAttr, safeAttrValue, escapeHtml
* @return {String}
*/
function filterXSS (html, options) {
var xss = new FilterXSS(options);
return xss.process(html);
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
// 输出
exports = module.exports = filterXSS;
exports.FilterXSS = FilterXSS;
for (var i in DEFAULT) exports[i] = DEFAULT[i];
for (var i in parser) exports[i] = parser[i];
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
// 在浏览器端使用
if (typeof window !== 'undefined') {
window.filterXSS = module.exports;
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
},{"./default":1,"./parser":3,"./xss":5}],3:[function(require,module,exports){
/**
* 简单 HTML Parser
*
* @author 老雷<leizongmin@gmail.com>
*/
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
var _ = require('./util');
2017-07-22 22:21:05 +02:00
/**
* 获取标签的名称
*
* @param {String} html '<a hef="#">'
* @return {String}
*/
function getTagName (html) {
var i = html.indexOf(' ');
if (i === -1) {
var tagName = html.slice(1, -1);
} else {
var tagName = html.slice(1, i + 1);
}
tagName = _.trim(tagName).toLowerCase();
if (tagName.slice(0, 1) === '/') tagName = tagName.slice(1);
if (tagName.slice(-1) === '/') tagName = tagName.slice(0, -1);
return tagName;
}
2017-07-22 22:21:05 +02:00
/**
* 是否为闭合标签
*
* @param {String} html '<a hef="#">'
* @return {Boolean}
*/
function isClosing (html) {
return (html.slice(0, 2) === '</');
}
2017-07-22 22:21:05 +02:00
/**
* 分析HTML代码调用相应的函数处理返回处理后的HTML
*
* @param {String} html
* @param {Function} onTag 处理标签的函数
* 参数格式 function (sourcePosition, position, tag, html, isClosing)
* @param {Function} escapeHtml 对HTML进行转义的函数
* @return {String}
*/
function parseTag (html, onTag, escapeHtml) {
'user strict';
var rethtml = ''; // 待返回的HTML
var lastPos = 0; // 上一个标签结束位置
var tagStart = false; // 当前标签开始位置
var quoteStart = false; // 引号开始位置
var currentPos = 0; // 当前位置
var len = html.length; // HTML长度
var currentHtml = ''; // 当前标签的HTML代码
var currentTagName = ''; // 当前标签的名称
// 逐个分析字符
for (currentPos = 0; currentPos < len; currentPos++) {
var c = html.charAt(currentPos);
if (tagStart === false) {
if (c === '<') {
tagStart = currentPos;
continue;
}
} else {
if (quoteStart === false) {
if (c === '<') {
rethtml += escapeHtml(html.slice(lastPos, currentPos));
tagStart = currentPos;
lastPos = currentPos;
continue;
}
if (c === '>') {
rethtml += escapeHtml(html.slice(lastPos, tagStart));
currentHtml = html.slice(tagStart, currentPos + 1);
currentTagName = getTagName(currentHtml);
rethtml += onTag(tagStart,
rethtml.length,
currentTagName,
currentHtml,
isClosing(currentHtml));
lastPos = currentPos + 1;
tagStart = false;
continue;
}
// HTML标签内的引号仅当前一个字符是等于号时才有效
if ((c === '"' || c === "'") && html.charAt(currentPos - 1) === '=') {
quoteStart = c;
continue;
}
} else {
if (c === quoteStart) {
quoteStart = false;
continue;
}
}
}
}
if (lastPos < html.length) {
rethtml += escapeHtml(html.substr(lastPos));
}
2017-07-22 22:21:05 +02:00
return rethtml;
}
2017-07-22 22:21:05 +02:00
// 不符合属性名称规则的正则表达式
var REGEXP_ATTR_NAME = /[^a-zA-Z0-9_:\.\-]/img;
2017-07-22 22:21:05 +02:00
/**
* 分析标签HTML代码调用相应的函数处理返回HTML
*
* @param {String} html 如标签'<a href="#" target="_blank">' 则为 'href="#" target="_blank"'
* @param {Function} onAttr 处理属性值的函数
* 函数格式 function (name, value)
* @return {String}
*/
function parseAttr (html, onAttr) {
'user strict';
var lastPos = 0; // 当前位置
var retAttrs = []; // 待返回的属性列表
var tmpName = false; // 临时属性名称
var len = html.length; // HTML代码长度
function addAttr (name, value) {
name = _.trim(name);
name = name.replace(REGEXP_ATTR_NAME, '').toLowerCase();
if (name.length < 1) return;
var ret = onAttr(name, value || '');
if (ret) retAttrs.push(ret);
};
2017-07-22 22:21:05 +02:00
// 逐个分析字符
for (var i = 0; i < len; i++) {
var c = html.charAt(i);
var v, j;
if (tmpName === false && c === '=') {
tmpName = html.slice(lastPos, i);
lastPos = i + 1;
continue;
}
if (tmpName !== false) {
// HTML标签内的引号仅当前一个字符是等于号时才有效
if (i === lastPos && (c === '"' || c === "'") && html.charAt(i - 1) === '=') {
j = html.indexOf(c, i + 1);
if (j === -1) {
break;
} else {
v = _.trim(html.slice(lastPos + 1, j));
addAttr(tmpName, v);
tmpName = false;
i = j;
lastPos = i + 1;
continue;
}
}
}
if (c === ' ') {
if (tmpName === false) {
j = findNextEqual(html, i);
if (j === -1) {
v = _.trim(html.slice(lastPos, i));
addAttr(v);
tmpName = false;
lastPos = i + 1;
continue;
} else {
i = j - 1;
continue;
}
} else {
j = findBeforeEqual(html, i - 1);
if (j === -1) {
v = _.trim(html.slice(lastPos, i));
v = stripQuoteWrap(v);
addAttr(tmpName, v);
tmpName = false;
lastPos = i + 1;
continue;
} else {
continue;
}
}
}
}
2017-07-22 22:21:05 +02:00
if (lastPos < html.length) {
if (tmpName === false) {
addAttr(html.slice(lastPos));
} else {
addAttr(tmpName, stripQuoteWrap(_.trim(html.slice(lastPos))));
}
}
2017-07-22 22:21:05 +02:00
return _.trim(retAttrs.join(' '));
}
function findNextEqual (str, i) {
for (; i < str.length; i++) {
var c = str[i];
if (c === ' ') continue;
if (c === '=') return i;
return -1;
}
}
2017-07-22 22:21:05 +02:00
function findBeforeEqual (str, i) {
for (; i > 0; i--) {
var c = str[i];
if (c === ' ') continue;
if (c === '=') return i;
return -1;
}
}
2017-07-22 22:21:05 +02:00
function isQuoteWrapString (text) {
if ((text[0] === '"' && text[text.length - 1] === '"') ||
(text[0] === '\'' && text[text.length - 1] === '\'')) {
return true;
} else {
return false;
}
};
2017-07-22 22:21:05 +02:00
function stripQuoteWrap (text) {
if (isQuoteWrapString(text)) {
return text.substr(1, text.length - 2);
} else {
return text;
}
};
2016-05-03 17:37:10 +02:00
2017-07-22 22:21:05 +02:00
exports.parseTag = parseTag;
exports.parseAttr = parseAttr;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
},{"./util":4}],4:[function(require,module,exports){
module.exports = {
indexOf: function (arr, item) {
var i, j;
if (Array.prototype.indexOf) {
return arr.indexOf(item);
}
for (i = 0, j = arr.length; i < j; i++) {
if (arr[i] === item) {
return i;
}
}
return -1;
},
forEach: function (arr, fn, scope) {
var i, j;
if (Array.prototype.forEach) {
return arr.forEach(fn, scope);
}
for (i = 0, j = arr.length; i < j; i++) {
fn.call(scope, arr[i], i, arr);
}
},
trim: function (str) {
if (String.prototype.trim) {
return str.trim();
}
return str.replace(/(^\s*)|(\s*$)/g, '');
}
};
2017-07-22 22:21:05 +02:00
},{}],5:[function(require,module,exports){
/**
* 过滤XSS
*
* @author 老雷<leizongmin@gmail.com>
*/
2017-07-22 22:21:05 +02:00
var FilterCSS = require('cssfilter').FilterCSS;
var DEFAULT = require('./default');
var parser = require('./parser');
var parseTag = parser.parseTag;
var parseAttr = parser.parseAttr;
var _ = require('./util');
2017-02-13 17:16:13 +01:00
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
/**
* 返回值是否为空
*
* @param {Object} obj
* @return {Boolean}
*/
function isNull (obj) {
return (obj === undefined || obj === null);
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
/**
* 取标签内的属性列表字符串
*
* @param {String} html
* @return {Object}
* - {String} html
* - {Boolean} closing
*/
function getAttrs (html) {
var i = html.indexOf(' ');
if (i === -1) {
return {
html: '',
closing: (html[html.length - 2] === '/')
};
}
html = _.trim(html.slice(i + 1, -1));
var isClosing = (html[html.length - 1] === '/');
if (isClosing) html = _.trim(html.slice(0, -1));
return {
html: html,
closing: isClosing
};
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
/**
* 浅拷贝对象
*
* @param {Object} obj
* @return {Object}
*/
function shallowCopyObject (obj) {
var ret = {};
for (var i in obj) {
ret[i] = obj[i];
}
return ret;
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
/**
* XSS过滤对象
*
* @param {Object} options
* 选项whiteList, onTag, onTagAttr, onIgnoreTag,
* onIgnoreTagAttr, safeAttrValue, escapeHtml
* stripIgnoreTagBody, allowCommentTag, stripBlankChar
* css{whiteList, onAttr, onIgnoreAttr} css=false表示禁用cssfilter
*/
function FilterXSS (options) {
options = shallowCopyObject(options || {});
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (options.stripIgnoreTag) {
if (options.onIgnoreTag) {
console.error('Notes: cannot use these two options "stripIgnoreTag" and "onIgnoreTag" at the same time');
}
options.onIgnoreTag = DEFAULT.onIgnoreTagStripAll;
}
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
options.whiteList = options.whiteList || DEFAULT.whiteList;
options.onTag = options.onTag || DEFAULT.onTag;
options.onTagAttr = options.onTagAttr || DEFAULT.onTagAttr;
options.onIgnoreTag = options.onIgnoreTag || DEFAULT.onIgnoreTag;
options.onIgnoreTagAttr = options.onIgnoreTagAttr || DEFAULT.onIgnoreTagAttr;
options.safeAttrValue = options.safeAttrValue || DEFAULT.safeAttrValue;
options.escapeHtml = options.escapeHtml || DEFAULT.escapeHtml;
this.options = options;
if (options.css === false) {
this.cssFilter = false;
} else {
options.css = options.css || {};
this.cssFilter = new FilterCSS(options.css);
}
}
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
/**
* 开始处理
*
* @param {String} html
* @return {String}
*/
FilterXSS.prototype.process = function (html) {
// 兼容各种奇葩输入
html = html || '';
html = html.toString();
if (!html) return '';
var me = this;
var options = me.options;
var whiteList = options.whiteList;
var onTag = options.onTag;
var onIgnoreTag = options.onIgnoreTag;
var onTagAttr = options.onTagAttr;
var onIgnoreTagAttr = options.onIgnoreTagAttr;
var safeAttrValue = options.safeAttrValue;
var escapeHtml = options.escapeHtml;
var cssFilter = me.cssFilter;
// 是否清除不可见字符
if (options.stripBlankChar) {
html = DEFAULT.stripBlankChar(html);
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
// 是否禁止备注标签
if (!options.allowCommentTag) {
html = DEFAULT.stripCommentTag(html);
}
2016-05-03 17:37:10 +02:00
2017-07-22 22:21:05 +02:00
// 如果开启了stripIgnoreTagBody
var stripIgnoreTagBody = false;
if (options.stripIgnoreTagBody) {
var stripIgnoreTagBody = DEFAULT.StripTagBody(options.stripIgnoreTagBody, onIgnoreTag);
onIgnoreTag = stripIgnoreTagBody.onIgnoreTag;
}
2016-05-03 17:37:10 +02:00
2017-07-22 22:21:05 +02:00
var retHtml = parseTag(html, function (sourcePosition, position, tag, html, isClosing) {
var info = {
sourcePosition: sourcePosition,
position: position,
isClosing: isClosing,
isWhite: (tag in whiteList)
};
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
// 调用onTag处理
var ret = onTag(tag, html, info);
if (!isNull(ret)) return ret;
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
// 默认标签处理方法
if (info.isWhite) {
// 白名单标签,解析标签属性
// 如果是闭合标签,则不需要解析属性
if (info.isClosing) {
return '</' + tag + '>';
}
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
var attrs = getAttrs(html);
var whiteAttrList = whiteList[tag];
var attrsHtml = parseAttr(attrs.html, function (name, value) {
// 调用onTagAttr处理
var isWhiteAttr = (_.indexOf(whiteAttrList, name) !== -1);
var ret = onTagAttr(tag, name, value, isWhiteAttr);
if (!isNull(ret)) return ret;
// 默认的属性处理方法
if (isWhiteAttr) {
// 白名单属性调用safeAttrValue过滤属性值
value = safeAttrValue(tag, name, value, cssFilter);
if (value) {
return name + '="' + value + '"';
} else {
return name;
}
} else {
// 非白名单属性调用onIgnoreTagAttr处理
var ret = onIgnoreTagAttr(tag, name, value, isWhiteAttr);
if (!isNull(ret)) return ret;
return;
}
});
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
// 构造新的标签代码
var html = '<' + tag;
if (attrsHtml) html += ' ' + attrsHtml;
if (attrs.closing) html += ' /';
html += '>';
return html;
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
} else {
// 非白名单标签调用onIgnoreTag处理
var ret = onIgnoreTag(tag, html, info);
if (!isNull(ret)) return ret;
return escapeHtml(html);
}
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
}, escapeHtml);
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
// 如果开启了stripIgnoreTagBody需要对结果再进行处理
if (stripIgnoreTagBody) {
retHtml = stripIgnoreTagBody.remove(retHtml);
}
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
return retHtml;
};
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
module.exports = FilterXSS;
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
},{"./default":1,"./parser":3,"./util":4,"cssfilter":8}],6:[function(require,module,exports){
/**
* cssfilter
*
* @author 老雷<leizongmin@gmail.com>
*/
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
var DEFAULT = require('./default');
var parseStyle = require('./parser');
var _ = require('./util');
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
/**
* 返回值是否为空
*
* @param {Object} obj
* @return {Boolean}
*/
function isNull (obj) {
return (obj === undefined || obj === null);
}
2017-07-22 22:21:05 +02:00
/**
* 浅拷贝对象
*
* @param {Object} obj
* @return {Object}
*/
function shallowCopyObject (obj) {
var ret = {};
for (var i in obj) {
ret[i] = obj[i];
}
return ret;
}
2017-07-22 22:21:05 +02:00
/**
* 创建CSS过滤器
*
* @param {Object} options
* - {Object} whiteList
* - {Object} onAttr
* - {Object} onIgnoreAttr
*/
function FilterCSS (options) {
options = shallowCopyObject(options || {});
options.whiteList = options.whiteList || DEFAULT.whiteList;
options.onAttr = options.onAttr || DEFAULT.onAttr;
options.onIgnoreAttr = options.onIgnoreAttr || DEFAULT.onIgnoreAttr;
this.options = options;
}
2017-07-22 22:21:05 +02:00
FilterCSS.prototype.process = function (css) {
// 兼容各种奇葩输入
css = css || '';
css = css.toString();
if (!css) return '';
var me = this;
var options = me.options;
var whiteList = options.whiteList;
var onAttr = options.onAttr;
var onIgnoreAttr = options.onIgnoreAttr;
var retCSS = parseStyle(css, function (sourcePosition, position, name, value, source) {
var check = whiteList[name];
var isWhite = false;
if (check === true) isWhite = check;
else if (typeof check === 'function') isWhite = check(value);
else if (check instanceof RegExp) isWhite = check.test(value);
if (isWhite !== true) isWhite = false;
var opts = {
position: position,
sourcePosition: sourcePosition,
source: source,
isWhite: isWhite
};
2017-07-22 22:21:05 +02:00
if (isWhite) {
2017-07-22 22:21:05 +02:00
var ret = onAttr(name, value, opts);
if (isNull(ret)) {
return name + ':' + value;
} else {
return ret;
}
2017-07-22 22:21:05 +02:00
} else {
2017-07-22 22:21:05 +02:00
var ret = onIgnoreAttr(name, value, opts);
if (!isNull(ret)) {
return ret;
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
}
});
2017-07-22 22:21:05 +02:00
return retCSS;
};
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
module.exports = FilterCSS;
},{"./default":7,"./parser":9,"./util":10}],7:[function(require,module,exports){
/**
* cssfilter
*
* @author 老雷<leizongmin@gmail.com>
*/
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
function getDefaultWhiteList () {
// 白名单值说明:
// true: 允许该属性
// Function: function (val) { } 返回true表示允许该属性其他值均表示不允许
// RegExp: regexp.test(val) 返回true表示允许该属性其他值均表示不允许
// 除上面列出的值外均表示不允许
var whiteList = {};
whiteList['align-content'] = false; // default: auto
whiteList['align-items'] = false; // default: auto
whiteList['align-self'] = false; // default: auto
whiteList['alignment-adjust'] = false; // default: auto
whiteList['alignment-baseline'] = false; // default: baseline
whiteList['all'] = false; // default: depending on individual properties
whiteList['anchor-point'] = false; // default: none
whiteList['animation'] = false; // default: depending on individual properties
whiteList['animation-delay'] = false; // default: 0
whiteList['animation-direction'] = false; // default: normal
whiteList['animation-duration'] = false; // default: 0
whiteList['animation-fill-mode'] = false; // default: none
whiteList['animation-iteration-count'] = false; // default: 1
whiteList['animation-name'] = false; // default: none
whiteList['animation-play-state'] = false; // default: running
whiteList['animation-timing-function'] = false; // default: ease
whiteList['azimuth'] = false; // default: center
whiteList['backface-visibility'] = false; // default: visible
whiteList['background'] = true; // default: depending on individual properties
whiteList['background-attachment'] = true; // default: scroll
whiteList['background-clip'] = true; // default: border-box
whiteList['background-color'] = true; // default: transparent
whiteList['background-image'] = true; // default: none
whiteList['background-origin'] = true; // default: padding-box
whiteList['background-position'] = true; // default: 0% 0%
whiteList['background-repeat'] = true; // default: repeat
whiteList['background-size'] = true; // default: auto
whiteList['baseline-shift'] = false; // default: baseline
whiteList['binding'] = false; // default: none
whiteList['bleed'] = false; // default: 6pt
whiteList['bookmark-label'] = false; // default: content()
whiteList['bookmark-level'] = false; // default: none
whiteList['bookmark-state'] = false; // default: open
whiteList['border'] = true; // default: depending on individual properties
whiteList['border-bottom'] = true; // default: depending on individual properties
whiteList['border-bottom-color'] = true; // default: current color
whiteList['border-bottom-left-radius'] = true; // default: 0
whiteList['border-bottom-right-radius'] = true; // default: 0
whiteList['border-bottom-style'] = true; // default: none
whiteList['border-bottom-width'] = true; // default: medium
whiteList['border-collapse'] = true; // default: separate
whiteList['border-color'] = true; // default: depending on individual properties
whiteList['border-image'] = true; // default: none
whiteList['border-image-outset'] = true; // default: 0
whiteList['border-image-repeat'] = true; // default: stretch
whiteList['border-image-slice'] = true; // default: 100%
whiteList['border-image-source'] = true; // default: none
whiteList['border-image-width'] = true; // default: 1
whiteList['border-left'] = true; // default: depending on individual properties
whiteList['border-left-color'] = true; // default: current color
whiteList['border-left-style'] = true; // default: none
whiteList['border-left-width'] = true; // default: medium
whiteList['border-radius'] = true; // default: 0
whiteList['border-right'] = true; // default: depending on individual properties
whiteList['border-right-color'] = true; // default: current color
whiteList['border-right-style'] = true; // default: none
whiteList['border-right-width'] = true; // default: medium
whiteList['border-spacing'] = true; // default: 0
whiteList['border-style'] = true; // default: depending on individual properties
whiteList['border-top'] = true; // default: depending on individual properties
whiteList['border-top-color'] = true; // default: current color
whiteList['border-top-left-radius'] = true; // default: 0
whiteList['border-top-right-radius'] = true; // default: 0
whiteList['border-top-style'] = true; // default: none
whiteList['border-top-width'] = true; // default: medium
whiteList['border-width'] = true; // default: depending on individual properties
whiteList['bottom'] = false; // default: auto
whiteList['box-decoration-break'] = true; // default: slice
whiteList['box-shadow'] = true; // default: none
whiteList['box-sizing'] = true; // default: content-box
whiteList['box-snap'] = true; // default: none
whiteList['box-suppress'] = true; // default: show
whiteList['break-after'] = true; // default: auto
whiteList['break-before'] = true; // default: auto
whiteList['break-inside'] = true; // default: auto
whiteList['caption-side'] = false; // default: top
whiteList['chains'] = false; // default: none
whiteList['clear'] = true; // default: none
whiteList['clip'] = false; // default: auto
whiteList['clip-path'] = false; // default: none
whiteList['clip-rule'] = false; // default: nonzero
whiteList['color'] = true; // default: implementation dependent
whiteList['color-interpolation-filters'] = true; // default: auto
whiteList['column-count'] = false; // default: auto
whiteList['column-fill'] = false; // default: balance
whiteList['column-gap'] = false; // default: normal
whiteList['column-rule'] = false; // default: depending on individual properties
whiteList['column-rule-color'] = false; // default: current color
whiteList['column-rule-style'] = false; // default: medium
whiteList['column-rule-width'] = false; // default: medium
whiteList['column-span'] = false; // default: none
whiteList['column-width'] = false; // default: auto
whiteList['columns'] = false; // default: depending on individual properties
whiteList['contain'] = false; // default: none
whiteList['content'] = false; // default: normal
whiteList['counter-increment'] = false; // default: none
whiteList['counter-reset'] = false; // default: none
whiteList['counter-set'] = false; // default: none
whiteList['crop'] = false; // default: auto
whiteList['cue'] = false; // default: depending on individual properties
whiteList['cue-after'] = false; // default: none
whiteList['cue-before'] = false; // default: none
whiteList['cursor'] = false; // default: auto
whiteList['direction'] = false; // default: ltr
whiteList['display'] = true; // default: depending on individual properties
whiteList['display-inside'] = true; // default: auto
whiteList['display-list'] = true; // default: none
whiteList['display-outside'] = true; // default: inline-level
whiteList['dominant-baseline'] = false; // default: auto
whiteList['elevation'] = false; // default: level
whiteList['empty-cells'] = false; // default: show
whiteList['filter'] = false; // default: none
whiteList['flex'] = false; // default: depending on individual properties
whiteList['flex-basis'] = false; // default: auto
whiteList['flex-direction'] = false; // default: row
whiteList['flex-flow'] = false; // default: depending on individual properties
whiteList['flex-grow'] = false; // default: 0
whiteList['flex-shrink'] = false; // default: 1
whiteList['flex-wrap'] = false; // default: nowrap
whiteList['float'] = false; // default: none
whiteList['float-offset'] = false; // default: 0 0
whiteList['flood-color'] = false; // default: black
whiteList['flood-opacity'] = false; // default: 1
whiteList['flow-from'] = false; // default: none
whiteList['flow-into'] = false; // default: none
whiteList['font'] = true; // default: depending on individual properties
whiteList['font-family'] = true; // default: implementation dependent
whiteList['font-feature-settings'] = true; // default: normal
whiteList['font-kerning'] = true; // default: auto
whiteList['font-language-override'] = true; // default: normal
whiteList['font-size'] = true; // default: medium
whiteList['font-size-adjust'] = true; // default: none
whiteList['font-stretch'] = true; // default: normal
whiteList['font-style'] = true; // default: normal
whiteList['font-synthesis'] = true; // default: weight style
whiteList['font-variant'] = true; // default: normal
whiteList['font-variant-alternates'] = true; // default: normal
whiteList['font-variant-caps'] = true; // default: normal
whiteList['font-variant-east-asian'] = true; // default: normal
whiteList['font-variant-ligatures'] = true; // default: normal
whiteList['font-variant-numeric'] = true; // default: normal
whiteList['font-variant-position'] = true; // default: normal
whiteList['font-weight'] = true; // default: normal
whiteList['grid'] = false; // default: depending on individual properties
whiteList['grid-area'] = false; // default: depending on individual properties
whiteList['grid-auto-columns'] = false; // default: auto
whiteList['grid-auto-flow'] = false; // default: none
whiteList['grid-auto-rows'] = false; // default: auto
whiteList['grid-column'] = false; // default: depending on individual properties
whiteList['grid-column-end'] = false; // default: auto
whiteList['grid-column-start'] = false; // default: auto
whiteList['grid-row'] = false; // default: depending on individual properties
whiteList['grid-row-end'] = false; // default: auto
whiteList['grid-row-start'] = false; // default: auto
whiteList['grid-template'] = false; // default: depending on individual properties
whiteList['grid-template-areas'] = false; // default: none
whiteList['grid-template-columns'] = false; // default: none
whiteList['grid-template-rows'] = false; // default: none
whiteList['hanging-punctuation'] = false; // default: none
whiteList['height'] = true; // default: auto
whiteList['hyphens'] = false; // default: manual
whiteList['icon'] = false; // default: auto
whiteList['image-orientation'] = false; // default: auto
whiteList['image-resolution'] = false; // default: normal
whiteList['ime-mode'] = false; // default: auto
whiteList['initial-letters'] = false; // default: normal
whiteList['inline-box-align'] = false; // default: last
whiteList['justify-content'] = false; // default: auto
whiteList['justify-items'] = false; // default: auto
whiteList['justify-self'] = false; // default: auto
whiteList['left'] = false; // default: auto
whiteList['letter-spacing'] = true; // default: normal
whiteList['lighting-color'] = true; // default: white
whiteList['line-box-contain'] = false; // default: block inline replaced
whiteList['line-break'] = false; // default: auto
whiteList['line-grid'] = false; // default: match-parent
whiteList['line-height'] = false; // default: normal
whiteList['line-snap'] = false; // default: none
whiteList['line-stacking'] = false; // default: depending on individual properties
whiteList['line-stacking-ruby'] = false; // default: exclude-ruby
whiteList['line-stacking-shift'] = false; // default: consider-shifts
whiteList['line-stacking-strategy'] = false; // default: inline-line-height
whiteList['list-style'] = true; // default: depending on individual properties
whiteList['list-style-image'] = true; // default: none
whiteList['list-style-position'] = true; // default: outside
whiteList['list-style-type'] = true; // default: disc
whiteList['margin'] = true; // default: depending on individual properties
whiteList['margin-bottom'] = true; // default: 0
whiteList['margin-left'] = true; // default: 0
whiteList['margin-right'] = true; // default: 0
whiteList['margin-top'] = true; // default: 0
whiteList['marker-offset'] = false; // default: auto
whiteList['marker-side'] = false; // default: list-item
whiteList['marks'] = false; // default: none
whiteList['mask'] = false; // default: border-box
whiteList['mask-box'] = false; // default: see individual properties
whiteList['mask-box-outset'] = false; // default: 0
whiteList['mask-box-repeat'] = false; // default: stretch
whiteList['mask-box-slice'] = false; // default: 0 fill
whiteList['mask-box-source'] = false; // default: none
whiteList['mask-box-width'] = false; // default: auto
whiteList['mask-clip'] = false; // default: border-box
whiteList['mask-image'] = false; // default: none
whiteList['mask-origin'] = false; // default: border-box
whiteList['mask-position'] = false; // default: center
whiteList['mask-repeat'] = false; // default: no-repeat
whiteList['mask-size'] = false; // default: border-box
whiteList['mask-source-type'] = false; // default: auto
whiteList['mask-type'] = false; // default: luminance
whiteList['max-height'] = true; // default: none
whiteList['max-lines'] = false; // default: none
whiteList['max-width'] = true; // default: none
whiteList['min-height'] = true; // default: 0
whiteList['min-width'] = true; // default: 0
whiteList['move-to'] = false; // default: normal
whiteList['nav-down'] = false; // default: auto
whiteList['nav-index'] = false; // default: auto
whiteList['nav-left'] = false; // default: auto
whiteList['nav-right'] = false; // default: auto
whiteList['nav-up'] = false; // default: auto
whiteList['object-fit'] = false; // default: fill
whiteList['object-position'] = false; // default: 50% 50%
whiteList['opacity'] = false; // default: 1
whiteList['order'] = false; // default: 0
whiteList['orphans'] = false; // default: 2
whiteList['outline'] = false; // default: depending on individual properties
whiteList['outline-color'] = false; // default: invert
whiteList['outline-offset'] = false; // default: 0
whiteList['outline-style'] = false; // default: none
whiteList['outline-width'] = false; // default: medium
whiteList['overflow'] = false; // default: depending on individual properties
whiteList['overflow-wrap'] = false; // default: normal
whiteList['overflow-x'] = false; // default: visible
whiteList['overflow-y'] = false; // default: visible
whiteList['padding'] = true; // default: depending on individual properties
whiteList['padding-bottom'] = true; // default: 0
whiteList['padding-left'] = true; // default: 0
whiteList['padding-right'] = true; // default: 0
whiteList['padding-top'] = true; // default: 0
whiteList['page'] = false; // default: auto
whiteList['page-break-after'] = false; // default: auto
whiteList['page-break-before'] = false; // default: auto
whiteList['page-break-inside'] = false; // default: auto
whiteList['page-policy'] = false; // default: start
whiteList['pause'] = false; // default: implementation dependent
whiteList['pause-after'] = false; // default: implementation dependent
whiteList['pause-before'] = false; // default: implementation dependent
whiteList['perspective'] = false; // default: none
whiteList['perspective-origin'] = false; // default: 50% 50%
whiteList['pitch'] = false; // default: medium
whiteList['pitch-range'] = false; // default: 50
whiteList['play-during'] = false; // default: auto
whiteList['position'] = false; // default: static
whiteList['presentation-level'] = false; // default: 0
whiteList['quotes'] = false; // default: text
whiteList['region-fragment'] = false; // default: auto
whiteList['resize'] = false; // default: none
whiteList['rest'] = false; // default: depending on individual properties
whiteList['rest-after'] = false; // default: none
whiteList['rest-before'] = false; // default: none
whiteList['richness'] = false; // default: 50
whiteList['right'] = false; // default: auto
whiteList['rotation'] = false; // default: 0
whiteList['rotation-point'] = false; // default: 50% 50%
whiteList['ruby-align'] = false; // default: auto
whiteList['ruby-merge'] = false; // default: separate
whiteList['ruby-position'] = false; // default: before
whiteList['shape-image-threshold'] = false; // default: 0.0
whiteList['shape-outside'] = false; // default: none
whiteList['shape-margin'] = false; // default: 0
whiteList['size'] = false; // default: auto
whiteList['speak'] = false; // default: auto
whiteList['speak-as'] = false; // default: normal
whiteList['speak-header'] = false; // default: once
whiteList['speak-numeral'] = false; // default: continuous
whiteList['speak-punctuation'] = false; // default: none
whiteList['speech-rate'] = false; // default: medium
whiteList['stress'] = false; // default: 50
whiteList['string-set'] = false; // default: none
whiteList['tab-size'] = false; // default: 8
whiteList['table-layout'] = false; // default: auto
whiteList['text-align'] = true; // default: start
whiteList['text-align-last'] = true; // default: auto
whiteList['text-combine-upright'] = true; // default: none
whiteList['text-decoration'] = true; // default: none
whiteList['text-decoration-color'] = true; // default: currentColor
whiteList['text-decoration-line'] = true; // default: none
whiteList['text-decoration-skip'] = true; // default: objects
whiteList['text-decoration-style'] = true; // default: solid
whiteList['text-emphasis'] = true; // default: depending on individual properties
whiteList['text-emphasis-color'] = true; // default: currentColor
whiteList['text-emphasis-position'] = true; // default: over right
whiteList['text-emphasis-style'] = true; // default: none
whiteList['text-height'] = true; // default: auto
whiteList['text-indent'] = true; // default: 0
whiteList['text-justify'] = true; // default: auto
whiteList['text-orientation'] = true; // default: mixed
whiteList['text-overflow'] = true; // default: clip
whiteList['text-shadow'] = true; // default: none
whiteList['text-space-collapse'] = true; // default: collapse
whiteList['text-transform'] = true; // default: none
whiteList['text-underline-position'] = true; // default: auto
whiteList['text-wrap'] = true; // default: normal
whiteList['top'] = false; // default: auto
whiteList['transform'] = false; // default: none
whiteList['transform-origin'] = false; // default: 50% 50% 0
whiteList['transform-style'] = false; // default: flat
whiteList['transition'] = false; // default: depending on individual properties
whiteList['transition-delay'] = false; // default: 0s
whiteList['transition-duration'] = false; // default: 0s
whiteList['transition-property'] = false; // default: all
whiteList['transition-timing-function'] = false; // default: ease
whiteList['unicode-bidi'] = false; // default: normal
whiteList['vertical-align'] = false; // default: baseline
whiteList['visibility'] = false; // default: visible
whiteList['voice-balance'] = false; // default: center
whiteList['voice-duration'] = false; // default: auto
whiteList['voice-family'] = false; // default: implementation dependent
whiteList['voice-pitch'] = false; // default: medium
whiteList['voice-range'] = false; // default: medium
whiteList['voice-rate'] = false; // default: normal
whiteList['voice-stress'] = false; // default: normal
whiteList['voice-volume'] = false; // default: medium
whiteList['volume'] = false; // default: medium
whiteList['white-space'] = false; // default: normal
whiteList['widows'] = false; // default: 2
whiteList['width'] = true; // default: auto
whiteList['will-change'] = false; // default: auto
whiteList['word-break'] = true; // default: normal
whiteList['word-spacing'] = true; // default: normal
whiteList['word-wrap'] = true; // default: normal
whiteList['wrap-flow'] = false; // default: auto
whiteList['wrap-through'] = false; // default: wrap
whiteList['writing-mode'] = false; // default: horizontal-tb
whiteList['z-index'] = false; // default: auto
return whiteList;
}
2017-07-22 22:21:05 +02:00
/**
* 匹配到白名单上的一个属性时
*
* @param {String} name
* @param {String} value
* @param {Object} options
* @return {String}
*/
function onAttr (name, value, options) {
// do nothing
}
2017-07-22 22:21:05 +02:00
/**
* 匹配到不在白名单上的一个属性时
*
* @param {String} name
* @param {String} value
* @param {Object} options
* @return {String}
*/
function onIgnoreAttr (name, value, options) {
// do nothing
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
exports.whiteList = getDefaultWhiteList();
exports.getDefaultWhiteList = getDefaultWhiteList;
exports.onAttr = onAttr;
exports.onIgnoreAttr = onIgnoreAttr;
2017-07-22 22:21:05 +02:00
},{}],8:[function(require,module,exports){
/**
* cssfilter
*
* @author 老雷<leizongmin@gmail.com>
*/
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var DEFAULT = require('./default');
var FilterCSS = require('./css');
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
/**
* XSS过滤
*
* @param {String} css 要过滤的CSS代码
* @param {Object} options 选项whiteList, onAttr, onIgnoreAttr
* @return {String}
*/
function filterCSS (html, options) {
var xss = new FilterCSS(options);
return xss.process(html);
}
2017-07-22 22:21:05 +02:00
// 输出
exports = module.exports = filterCSS;
exports.FilterCSS = FilterCSS;
for (var i in DEFAULT) exports[i] = DEFAULT[i];
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
// 在浏览器端使用
if (typeof window !== 'undefined') {
window.filterCSS = module.exports;
}
2017-07-22 22:21:05 +02:00
},{"./css":6,"./default":7}],9:[function(require,module,exports){
/**
* cssfilter
*
* @author 老雷<leizongmin@gmail.com>
*/
2017-07-22 22:21:05 +02:00
var _ = require('./util');
2017-04-04 17:26:06 +02:00
2017-07-22 22:21:05 +02:00
/**
* 解析style
*
* @param {String} css
* @param {Function} onAttr 处理属性的函数
* 参数格式 function (sourcePosition, position, name, value, source)
* @return {String}
*/
function parseStyle (css, onAttr) {
css = _.trimRight(css);
if (css[css.length - 1] !== ';') css += ';';
var cssLength = css.length;
var isParenthesisOpen = false;
var lastPos = 0;
var i = 0;
var retCSS = '';
function addNewAttr () {
// 如果没有正常的闭合圆括号,则直接忽略当前属性
if (!isParenthesisOpen) {
var source = _.trim(css.slice(lastPos, i));
var j = source.indexOf(':');
if (j !== -1) {
var name = _.trim(source.slice(0, j));
var value = _.trim(source.slice(j + 1));
// 必须有属性名称
if (name) {
var ret = onAttr(lastPos, retCSS.length, name, value, source);
if (ret) retCSS += ret + '; ';
2017-04-04 17:26:06 +02:00
}
2017-07-22 22:21:05 +02:00
}
}
lastPos = i + 1;
}
2017-07-22 22:21:05 +02:00
for (; i < cssLength; i++) {
var c = css[i];
if (c === '/' && css[i + 1] === '*') {
// 备注开始
var j = css.indexOf('*/', i + 2);
// 如果没有正常的备注结束,则后面的部分全部跳过
if (j === -1) break;
// 直接将当前位置调到备注结尾,并且初始化状态
i = j + 1;
lastPos = i + 1;
isParenthesisOpen = false;
} else if (c === '(') {
isParenthesisOpen = true;
} else if (c === ')') {
isParenthesisOpen = false;
} else if (c === ';') {
if (isParenthesisOpen) {
// 在圆括号里面,忽略
} else {
addNewAttr();
}
} else if (c === '\n') {
addNewAttr();
}
}
2017-02-13 17:16:13 +01:00
2017-07-22 22:21:05 +02:00
return _.trim(retCSS);
}
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
module.exports = parseStyle;
},{"./util":10}],10:[function(require,module,exports){
module.exports = {
indexOf: function (arr, item) {
var i, j;
if (Array.prototype.indexOf) {
return arr.indexOf(item);
}
for (i = 0, j = arr.length; i < j; i++) {
if (arr[i] === item) {
return i;
}
}
return -1;
},
forEach: function (arr, fn, scope) {
var i, j;
if (Array.prototype.forEach) {
return arr.forEach(fn, scope);
}
for (i = 0, j = arr.length; i < j; i++) {
fn.call(scope, arr[i], i, arr);
}
},
trim: function (str) {
if (String.prototype.trim) {
return str.trim();
}
return str.replace(/(^\s*)|(\s*$)/g, '');
},
trimRight: function (str) {
if (String.prototype.trimRight) {
return str.trimRight();
}
return str.replace(/(\s*$)/g, '');
}
};
},{}]},{},[2]);
define("xss", (function (global) {
return function () {
var ret, fn;
fn = function (xss_noconflict) {
return {
filterXSS: window.filterXSS,
filterCSS: window.filterCSS
2017-02-13 17:16:13 +01:00
}
2017-07-22 22:21:05 +02:00
};
ret = fn.apply(global, arguments);
return ret;
2017-02-13 17:16:13 +01:00
};
2017-07-22 22:21:05 +02:00
}(this)));
2016-03-07 18:54:07 +01:00
define('tpl!chatbox', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-06-23 20:25:33 +02:00
__p += '<div class="flyout box-flyout">\n <div class="chat-head chat-head-chatbox">\n <a class="chatbox-btn close-chatbox-button icon-close" title="' +
2017-02-13 17:16:13 +01:00
__e(info_close) +
2016-11-07 15:43:48 +01:00
'"></a>\n <div class="chat-title">\n ';
if (url) { ;
__p += '\n <a href="' +
2017-02-13 17:16:13 +01:00
__e(url) +
2016-11-07 15:43:48 +01:00
'" target="_blank" rel="noopener" class="user">\n ';
} ;
__p += '\n ' +
2017-02-13 17:16:13 +01:00
__e( title ) +
2016-11-07 15:43:48 +01:00
'\n ';
if (url) { ;
__p += '\n </a>\n ';
} ;
2017-04-04 17:26:06 +02:00
__p += '\n <p class="user-custom-message"><p/>\n </div>\n </div>\n <div class="chat-body">\n <div class="chat-content ';
if (show_send_button) { ;
__p += 'chat-content-sendbutton';
} ;
__p += '"></div>\n <div class="new-msgs-indicator hidden">▼ ' +
2017-02-13 17:16:13 +01:00
__e( unread_msgs ) +
2016-11-07 15:43:48 +01:00
' ▼</div>\n ';
if (show_textarea) { ;
2017-07-22 22:21:05 +02:00
__p += '\n <form class="sendXMPPMessage">\n ';
if (show_toolbar) { ;
__p += '\n <ul class="chat-toolbar no-text-select"></ul>\n ';
} ;
2017-04-04 17:26:06 +02:00
__p += '\n <textarea\n type="text"\n class="chat-textarea ';
if (show_send_button) { ;
__p += 'chat-textarea-send-button';
} ;
__p += '"\n placeholder="' +
2017-02-13 17:16:13 +01:00
__e(label_personal_message) +
2017-04-04 17:26:06 +02:00
'"/>\n\n ';
if (show_send_button) { ;
2017-04-23 19:02:44 +02:00
__p += '\n <button type="submit" class="pure-button send-button">' +
__e( label_send ) +
'</button>\n ';
2017-04-04 17:26:06 +02:00
} ;
__p += '\n </form>\n ';
} ;
__p += '\n </div>\n</div>\n';
2016-03-07 18:54:07 +01:00
}
return __p
};});
define('tpl!new_day', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<time class="chat-info chat-date" data-isodate="' +
2017-02-13 17:16:13 +01:00
__e(isodate) +
'">' +
2017-02-13 17:16:13 +01:00
__e(datestring) +
2016-11-07 15:43:48 +01:00
'</time>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!action', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<div class="chat-message ' +
2017-02-13 17:16:13 +01:00
__e(extra_classes) +
'" data-isodate="' +
2017-02-13 17:16:13 +01:00
__e(isodate) +
'">\n <span class="chat-msg-author chat-msg-' +
2017-02-13 17:16:13 +01:00
__e(sender) +
'">' +
2017-02-13 17:16:13 +01:00
__e(time) +
' **' +
2017-02-13 17:16:13 +01:00
__e(username) +
2017-03-05 10:45:18 +01:00
'&nbsp;</span>\n <span class="chat-msg-content chat-action"><!-- message gets added here via renderMessage --></span>\n</div>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2017-07-22 22:21:05 +02:00
define('tpl!emojis', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
_.forEach(emojis_by_category, function (obj, category) { ;
__p += '\n <ul class="emoji-picker emoji-picker-' +
__e(category) +
' ';
if (current_category !== category) { ;
__p += ' hidden ';
} ;
__p += '">\n ';
_.forEach(emojis_by_category[category], function (emoji) { ;
__p += '\n <li class="emoji insert-emoji ';
if (shouldBeHidden(emoji._shortname, current_skintone, toned_emojis)) { ;
__p += ' hidden ';
}; ;
__p += '"\n data-emoji="' +
__e(emoji._shortname) +
'">\n <a href="#" data-emoji="' +
__e(emoji._shortname) +
'"> ' +
((__t = ( transform(emoji._shortname) )) == null ? '' : __t) +
' </a>\n </li>\n ';
}); ;
__p += '\n </ul>\n';
}); ;
__p += '\n<ul class="emoji-toolbar">\n <li class="emoji-category-picker">\n <ul>\n ';
_.forEach(emojis_by_category, function (obj, category) { ;
__p += '\n <li data-category="' +
__e(category) +
'" class="emoji-category ';
if (current_category === category) { ;
__p += ' picked ';
} ;
__p += '">\n <a class="pick-category" href="#" data-category="' +
__e(category) +
'"> ' +
((__t = ( transform(emojis_by_category[category][0]._shortname) )) == null ? '' : __t) +
' </a>\n </li>\n ';
}); ;
__p += '\n </ul>\n </li>\n <li class="emoji-skintone-picker">\n <ul>\n ';
_.forEach(skintones, function (skintone) { ;
__p += '\n <li data-skintone="' +
__e(skintone) +
'" class="emoji-skintone ';
if (current_skintone === skintone) { ;
__p += ' picked ';
} ;
__p += '">\n <a class="pick-skintone" href="#" data-skintone="' +
__e(skintone) +
'"> ' +
((__t = ( transform(':'+skintone+':') )) == null ? '' : __t) +
' </a>\n </li>\n ';
}); ;
__p += '\n </ul>\n </li>\n</ul>\n';
}
return __p
};});
define('tpl!message', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<div class="chat-message ' +
2017-02-13 17:16:13 +01:00
__e(extra_classes) +
'" data-isodate="' +
2017-02-13 17:16:13 +01:00
__e(isodate) +
'" data-msgid="' +
2017-02-13 17:16:13 +01:00
__e(msgid) +
'">\n <span class="chat-msg-author chat-msg-' +
2017-02-13 17:16:13 +01:00
__e(sender) +
'">' +
2017-02-13 17:16:13 +01:00
__e(time) +
' ' +
2017-02-13 17:16:13 +01:00
__e(username) +
2016-11-07 15:43:48 +01:00
':&nbsp;</span>\n <span class="chat-msg-content"><!-- message gets added here via renderMessage --></span>\n</div>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-03-07 18:54:07 +01:00
2017-04-23 19:02:44 +02:00
define('tpl!help_message', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<div class="chat-' +
__e(type) +
'">' +
__e(message) +
'</div>\n';
}
return __p
};});
define('tpl!toolbar', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-07-22 22:21:05 +02:00
if (use_emoji) { ;
__p += '\n <li class="toggle-toolbar-menu toggle-smiley icon-happy" title="' +
2017-02-13 17:16:13 +01:00
__e(label_insert_smiley) +
2017-07-22 22:21:05 +02:00
'">\n <ul class="emoji-picker"></ul>\n </li>\n';
} ;
__p += '\n';
if (show_call_button) { ;
__p += '\n<li class="toggle-call"><a class="icon-phone" title="' +
2017-02-13 17:16:13 +01:00
__e(label_start_call) +
2016-11-07 15:43:48 +01:00
'"></a></li>\n';
} ;
__p += '\n';
if (show_clear_button) { ;
2017-06-23 20:25:33 +02:00
__p += '\n<li class="toggle-clear"><a class="icon-trash" title="' +
2017-02-13 17:16:13 +01:00
__e(label_clear) +
2016-11-07 15:43:48 +01:00
'"></a></li>\n';
} ;
__p += '\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-03-07 18:54:07 +01:00
define('tpl!avatar', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '';
with (obj) {
2017-06-23 20:25:33 +02:00
__p += '<canvas height="' +
((__t = (height)) == null ? '' : __t) +
'px" width="' +
((__t = (width)) == null ? '' : __t) +
'px" class="avatar"></canvas>\n';
}
return __p
};});
define('tpl!spinner', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '';
with (obj) {
__p += '<span class="spinner centered"/>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
2017-02-03 13:51:07 +01:00
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
2016-11-07 15:43:48 +01:00
// Licensed under the Mozilla Public License (MPLv2)
//
2017-04-23 19:02:44 +02:00
/*global define */
2016-03-07 18:54:07 +01:00
2016-11-07 15:43:48 +01:00
(function (root, factory) {
2017-07-22 22:21:05 +02:00
define('converse-chatview',["jquery.noconflict", "converse-core", "emojione", "xss", "tpl!chatbox", "tpl!new_day", "tpl!action", "tpl!emojis", "tpl!message", "tpl!help_message", "tpl!toolbar", "tpl!avatar", "tpl!spinner"], factory);
})(undefined, function ($, converse, emojione, xss, tpl_chatbox, tpl_new_day, tpl_action, tpl_emojis, tpl_message, tpl_help_message, tpl_toolbar, tpl_avatar, tpl_spinner) {
2016-11-07 15:43:48 +01:00
"use strict";
2017-07-22 22:21:05 +02:00
var _converse$env = converse.env,
$msg = _converse$env.$msg,
Backbone = _converse$env.Backbone,
Strophe = _converse$env.Strophe,
_ = _converse$env._,
b64_sha1 = _converse$env.b64_sha1,
moment = _converse$env.moment,
utils = _converse$env.utils;
2016-03-07 18:54:07 +01:00
2016-11-07 15:43:48 +01:00
var KEY = {
ENTER: 13,
FORWARD_SLASH: 47
};
2016-03-07 18:54:07 +01:00
2017-02-03 13:51:07 +01:00
converse.plugins.add('converse-chatview', {
2016-03-07 18:54:07 +01:00
2016-11-07 15:43:48 +01:00
overrides: {
// Overrides mentioned here will be picked up by converse.js's
// plugin architecture they will replace existing methods on the
// relevant objects or classes.
//
// New functions which don't exist yet can also be added.
2017-07-22 22:21:05 +02:00
//
registerGlobalEventHandlers: function registerGlobalEventHandlers() {
this.__super__.registerGlobalEventHandlers();
document.addEventListener('click', function (ev) {
if (_.includes(ev.target.classList, 'toggle-toolbar-menu') || _.includes(ev.target.classList, 'insert-emoji')) {
return;
}
utils.slideInAllElements(document.querySelectorAll('.toolbar-menu'));
});
},
2016-03-07 18:54:07 +01:00
2016-11-07 15:43:48 +01:00
ChatBoxViews: {
2017-07-22 22:21:05 +02:00
onChatBoxAdded: function onChatBoxAdded(item) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
var view = this.get(item.get('id'));
if (!view) {
2017-07-22 22:21:05 +02:00
view = new _converse.ChatBoxView({ model: item });
2016-11-07 15:43:48 +01:00
this.add(item.get('id'), view);
return view;
} else {
return this.__super__.onChatBoxAdded.apply(this, arguments);
}
}
}
},
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-11-07 15:43:48 +01:00
/* The initialize function gets called as soon as the plugin is
* loaded by converse.js's plugin machinery.
*/
2017-02-03 13:51:07 +01:00
var _converse = this._converse,
__ = _converse.__;
2017-07-22 22:21:05 +02:00
2017-07-05 11:59:55 +02:00
_converse.api.settings.update({
2017-07-22 22:21:05 +02:00
'use_emojione': true,
'emojione_image_path': emojione.imagePathPNG,
'chatview_avatar_height': 32,
'chatview_avatar_width': 32,
'show_toolbar': true,
'time_format': 'HH:mm',
'visible_toolbar_buttons': {
'emoji': true,
2016-11-07 15:43:48 +01:00
'call': false,
'clear': true
2017-07-22 22:21:05 +02:00
}
});
emojione.imagePathPNG = _converse.emojione_image_path;
emojione.ascii = true;
function onWindowStateChanged(data) {
_converse.chatboxviews.each(function (chatboxview) {
chatboxview.onWindowStateChanged(data.state);
});
}
_converse.api.listen.on('windowStateChanged', onWindowStateChanged);
_converse.EmojiPicker = Backbone.Model.extend({
defaults: {
'current_category': 'people',
'current_skintone': '',
'scroll_position': 0
},
initialize: function initialize() {
var id = "converse.emoji-" + _converse.bare_jid;
this.id = id;
this.browserStorage = new Backbone.BrowserStorage[_converse.storage](id);
}
});
_converse.EmojiPickerView = Backbone.View.extend({
className: 'emoji-picker-container toolbar-menu collapsed',
events: {
'click .emoji-category-picker li.emoji-category': 'chooseCategory',
'click .emoji-skintone-picker li.emoji-skintone': 'chooseSkinTone'
},
initialize: function initialize() {
this.model.on('change:current_skintone', this.render, this);
this.model.on('change:current_category', this.render, this);
this.setScrollPosition = _.debounce(this.setScrollPosition, 50);
2017-07-22 22:21:05 +02:00
},
render: function render() {
var _this = this;
var emojis_html = tpl_emojis(_.extend(this.model.toJSON(), {
'transform': _converse.use_emojione ? emojione.shortnameToImage : emojione.shortnameToUnicode,
'emojis_by_category': utils.getEmojisByCategory(_converse, emojione),
'toned_emojis': utils.getTonedEmojis(_converse),
'skintones': ['tone1', 'tone2', 'tone3', 'tone4', 'tone5'],
'shouldBeHidden': this.shouldBeHidden
}));
this.el.innerHTML = emojis_html;
2017-08-08 17:44:01 +02:00
_.forEach(this.el.querySelectorAll('.emoji-picker'), function (el) {
el.addEventListener('scroll', _this.setScrollPosition.bind(_this));
2017-07-22 22:21:05 +02:00
});
this.restoreScrollPosition();
return this;
},
shouldBeHidden: function shouldBeHidden(shortname, current_skintone, toned_emojis) {
/* Helper method for the template which decides whether an
* emoji should be hidden, based on which skin tone is
* currently being applied.
*/
if (_.includes(shortname, '_tone')) {
if (!current_skintone || !_.includes(shortname, current_skintone)) {
return true;
}
} else {
if (current_skintone && _.includes(toned_emojis, shortname)) {
return true;
}
}
return false;
},
restoreScrollPosition: function restoreScrollPosition() {
var current_picker = _.difference(this.el.querySelectorAll('.emoji-picker'), this.el.querySelectorAll('.emoji-picker.hidden'));
if (current_picker.length === 1 && this.model.get('scroll_position')) {
current_picker[0].scrollTop = this.model.get('scroll_position');
}
},
2017-08-08 17:44:01 +02:00
setScrollPosition: function setScrollPosition(ev) {
2017-07-22 22:21:05 +02:00
this.model.save('scroll_position', ev.target.scrollTop);
},
chooseSkinTone: function chooseSkinTone(ev) {
ev.preventDefault();
ev.stopPropagation();
var target = ev.target.nodeName === 'IMG' ? ev.target.parentElement : ev.target;
var skintone = target.getAttribute("data-skintone").trim();
if (this.model.get('current_skintone') === skintone) {
this.model.save({ 'current_skintone': '' });
} else {
this.model.save({ 'current_skintone': skintone });
}
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
chooseCategory: function chooseCategory(ev) {
ev.preventDefault();
ev.stopPropagation();
var target = ev.target.nodeName === 'IMG' ? ev.target.parentElement : ev.target;
var category = target.getAttribute("data-category").trim();
this.model.save({
'current_category': category,
'scroll_position': 0
});
}
2016-11-07 15:43:48 +01:00
});
2016-03-07 18:54:07 +01:00
2017-02-03 13:51:07 +01:00
_converse.ChatBoxView = Backbone.View.extend({
2016-11-07 15:43:48 +01:00
length: 200,
tagName: 'div',
2016-11-30 17:27:20 +01:00
className: 'chatbox hidden',
2017-07-22 22:21:05 +02:00
is_chatroom: false, // Leaky abstraction from MUC
2016-03-07 18:54:07 +01:00
2016-11-07 15:43:48 +01:00
events: {
'click .close-chatbox-button': 'close',
'keypress .chat-textarea': 'keyPressed',
2017-07-22 22:21:05 +02:00
'click .send-button': 'onFormSubmitted',
'click .toggle-smiley': 'toggleEmojiMenu',
'click .toggle-smiley ul.emoji-picker li': 'insertEmoji',
2016-11-07 15:43:48 +01:00
'click .toggle-clear': 'clearMessages',
'click .toggle-call': 'toggleCall',
'click .new-msgs-indicator': 'viewUnreadMessages'
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
this.markScrolled = _.debounce(this.markScrolled, 100);
2017-07-22 22:21:05 +02:00
this.createEmojiPicker();
2016-11-07 15:43:48 +01:00
this.model.messages.on('add', this.onMessageAdded, this);
this.model.on('show', this.show, this);
this.model.on('destroy', this.hide, this);
// TODO check for changed fullname as well
this.model.on('change:chat_state', this.sendChatState, this);
this.model.on('change:chat_status', this.onChatStatusChanged, this);
this.model.on('change:image', this.renderAvatar, this);
this.model.on('change:status', this.onStatusChanged, this);
this.model.on('showHelpMessages', this.showHelpMessages, this);
this.model.on('sendMessage', this.sendMessage, this);
this.render().fetchMessages();
2017-02-03 13:51:07 +01:00
_converse.emit('chatBoxInitialized', this);
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
render: function render() {
this.$el.attr('id', this.model.get('box_id')).html(tpl_chatbox(_.extend(this.model.toJSON(), {
show_toolbar: _converse.show_toolbar,
show_textarea: true,
show_send_button: _converse.show_send_button,
title: this.model.get('fullname'),
unread_msgs: __('You have unread messages'),
info_close: __('Close this chat box'),
label_personal_message: __('Personal message'),
label_send: __('Send')
})));
2016-11-07 15:43:48 +01:00
this.$content = this.$el.find('.chat-content');
this.renderToolbar().renderAvatar();
2017-02-03 13:51:07 +01:00
_converse.emit('chatBoxOpened', this);
2016-11-07 15:43:48 +01:00
utils.refreshWebkit();
return this.showStatusMessage();
},
2017-07-22 22:21:05 +02:00
createEmojiPicker: function createEmojiPicker() {
if (_.isUndefined(_converse.emojipicker)) {
_converse.emojipicker = new _converse.EmojiPicker();
_converse.emojipicker.fetch();
}
this.emoji_picker_view = new _converse.EmojiPickerView({
'model': _converse.emojipicker
});
},
afterMessagesFetched: function afterMessagesFetched() {
this.insertIntoDOM();
this.scrollDown();
// We only start listening for the scroll event after
// cached messages have been fetched
this.$content.on('scroll', this.markScrolled.bind(this));
2017-07-22 22:21:05 +02:00
_converse.emit('afterMessagesFetched', this);
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
fetchMessages: function fetchMessages() {
2016-11-07 15:43:48 +01:00
this.model.messages.fetch({
'add': true,
'success': this.afterMessagesFetched.bind(this),
2017-07-22 22:21:05 +02:00
'error': this.afterMessagesFetched.bind(this)
2016-11-07 15:43:48 +01:00
});
return this;
},
2017-07-22 22:21:05 +02:00
insertIntoDOM: function insertIntoDOM() {
2016-11-07 15:43:48 +01:00
/* This method gets overridden in src/converse-controlbox.js if
* the controlbox plugin is active.
*/
var container = document.querySelector('#conversejs');
if (this.el.parentNode !== container) {
container.insertBefore(this.el, container.firstChild);
}
2016-11-07 15:43:48 +01:00
return this;
},
2017-07-22 22:21:05 +02:00
clearStatusNotification: function clearStatusNotification() {
2016-11-07 15:43:48 +01:00
this.$content.find('div.chat-event').remove();
},
2017-07-22 22:21:05 +02:00
showStatusNotification: function showStatusNotification(message, keep_old, permanent) {
2016-11-07 15:43:48 +01:00
if (!keep_old) {
this.clearStatusNotification();
}
var $el = $('<div class="chat-info"></div>').text(message);
if (!permanent) {
$el.addClass('chat-event');
}
this.$content.append($el);
this.scrollDown();
},
2017-07-22 22:21:05 +02:00
addSpinner: function addSpinner() {
2017-04-04 17:26:06 +02:00
if (_.isNull(this.el.querySelector('.spinner'))) {
2017-06-23 20:25:33 +02:00
this.$content.prepend(tpl_spinner);
2016-11-07 15:43:48 +01:00
}
},
2017-07-22 22:21:05 +02:00
clearSpinner: function clearSpinner() {
2016-11-07 15:43:48 +01:00
if (this.$content.children(':first').is('span.spinner')) {
this.$content.children(':first').remove();
}
},
2017-07-22 22:21:05 +02:00
insertDayIndicator: function insertDayIndicator(date, prepend) {
2016-11-07 15:43:48 +01:00
/* Appends (or prepends if "prepend" is truthy) an indicator
* into the chat area, showing the day as given by the
* passed in date.
*
* Parameters:
* (String) date - An ISO8601 date string.
*/
var day_date = moment(date).startOf('day');
2017-07-22 22:21:05 +02:00
var insert = prepend ? this.$content.prepend : this.$content.append;
insert.call(this.$content, tpl_new_day({
2016-11-07 15:43:48 +01:00
isodate: day_date.format(),
datestring: day_date.format("dddd MMM Do YYYY")
}));
},
2017-07-22 22:21:05 +02:00
insertMessage: function insertMessage(attrs, prepend) {
var _this2 = this;
2016-11-07 15:43:48 +01:00
/* Helper method which appends a message (or prepends if the
* 2nd parameter is set to true) to the end of the chat box's
* content area.
*
* Parameters:
* (Object) attrs: An object containing the message attributes.
*/
var insert = prepend ? this.$content.prepend : this.$content.append;
2017-07-22 22:21:05 +02:00
_.flow(function ($el) {
insert.call(_this2.$content, $el);
return $el;
}, this.scrollDown.bind(this))(this.renderMessage(attrs));
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
showMessage: function showMessage(attrs) {
2016-11-07 15:43:48 +01:00
/* 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.
*
* The message to show may either be newer than the newest
* message, or older than the oldest message.
*
* Parameters:
* (Object) attrs: An object containing the message
* attributes.
2016-11-07 15:43:48 +01:00
*/
2017-07-22 22:21:05 +02:00
var current_msg_date = moment(attrs.time) || moment;
var $first_msg = this.$content.find('.chat-message:first'),
first_msg_date = $first_msg.data('isodate');
2016-11-07 15:43:48 +01:00
if (!first_msg_date) {
// This is the first received message, so we insert a
// date indicator before it.
this.insertDayIndicator(current_msg_date);
this.insertMessage(attrs);
return;
}
2017-07-22 22:21:05 +02:00
var last_msg_date = this.$content.find('.chat-message:last').data('isodate');
if (current_msg_date.isAfter(last_msg_date) || current_msg_date.isSame(last_msg_date)) {
2016-11-07 15:43:48 +01:00
// The new message is after the last message
if (current_msg_date.isAfter(last_msg_date, 'day')) {
// Append a new day indicator
this.insertDayIndicator(current_msg_date);
}
this.insertMessage(attrs);
return;
}
2017-07-22 22:21:05 +02:00
if (current_msg_date.isBefore(first_msg_date) || current_msg_date.isSame(first_msg_date)) {
2016-11-07 15:43:48 +01:00
// The message is before the first, but on the same day.
// We need to prepend the message immediately before the
// first message (so that it'll still be after the day
// indicator).
2016-11-07 15:43:48 +01:00
this.insertMessage(attrs, 'prepend');
if (current_msg_date.isBefore(first_msg_date, 'day')) {
// This message is also on a different day, so
// we prepend a day indicator.
2016-11-07 15:43:48 +01:00
this.insertDayIndicator(current_msg_date, 'prepend');
}
return;
}
// Find the correct place to position the message
current_msg_date = current_msg_date.format();
2017-07-22 22:21:05 +02:00
var msg_dates = _.map(this.$content.find('.chat-message'), function (el) {
2016-11-07 15:43:48 +01:00
return $(el).data('isodate');
});
msg_dates.push(current_msg_date);
msg_dates.sort();
2017-07-22 22:21:05 +02:00
var idx = msg_dates.indexOf(current_msg_date) - 1;
var $latest_message = this.$content.find(".chat-message[data-isodate=\"" + msg_dates[idx] + "\"]:last");
_.flow(function ($el) {
$el.insertAfter($latest_message);
return $el;
}, this.scrollDown.bind(this))(this.renderMessage(attrs));
},
getExtraMessageTemplateAttributes: function getExtraMessageTemplateAttributes() {
/* Provides a hook for sending more attributes to the
* message template.
*
* Parameters:
* (Object) attrs: An object containing message attributes.
*/
2016-11-07 15:43:48 +01:00
return {};
},
2017-07-22 22:21:05 +02:00
getExtraMessageClasses: function getExtraMessageClasses(attrs) {
return attrs.delayed && 'delayed' || '';
},
2017-07-22 22:21:05 +02:00
renderMessage: function renderMessage(attrs) {
2016-11-07 15:43:48 +01:00
/* Renders a chat message based on the passed in attributes.
*
* Parameters:
* (Object) attrs: An object containing the message attributes.
*
* Returns:
* The DOM element representing the message.
*/
2017-07-22 22:21:05 +02:00
var text = attrs.message,
2016-11-07 15:43:48 +01:00
fullname = this.model.get('fullname') || attrs.fullname,
2017-07-22 22:21:05 +02:00
template = void 0,
username = void 0;
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
var match = text.match(/^\/(.*?)(?: (.*))?$/);
if (match && match[1] === 'me') {
2016-11-07 15:43:48 +01:00
text = text.replace(/^\/me/, '');
template = tpl_action;
if (attrs.sender === 'me') {
2017-03-05 10:45:18 +01:00
fullname = _converse.xmppstatus.get('fullname') || attrs.fullname;
2017-07-22 22:21:05 +02:00
username = _.isNil(fullname) ? _converse.bare_jid : fullname;
} else {
username = attrs.fullname;
}
2017-07-22 22:21:05 +02:00
} else {
template = tpl_message;
2016-11-07 15:43:48 +01:00
username = attrs.sender === 'me' && __('me') || fullname;
}
this.$content.find('div.chat-event').remove();
if (text.length > 8000) {
text = text.substring(0, 10) + '...';
2017-07-22 22:21:05 +02:00
this.showStatusNotification(__("A very large message has been received." + "This might be due to an attack meant to degrade the chat performance." + "Output has been shortened."), true, true);
}
var msg_time = moment(attrs.time) || moment;
var $msg = $(template(_.extend(this.getExtraMessageTemplateAttributes(attrs), {
'msgid': attrs.msgid,
'sender': attrs.sender,
'time': msg_time.format(_converse.time_format),
'isodate': msg_time.format(),
'username': username,
'extra_classes': this.getExtraMessageClasses(attrs)
})));
var msg_content = $msg[0].querySelector('.chat-msg-content');
msg_content.innerHTML = utils.addEmoji(_converse, emojione, utils.addHyperlinks(xss.filterXSS(text, { 'whiteList': {} })));
utils.renderImageURLs(msg_content);
2016-11-07 15:43:48 +01:00
return $msg;
},
2017-07-22 22:21:05 +02:00
showHelpMessages: function showHelpMessages(msgs, type, spinner) {
var _this3 = this;
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
_.each(msgs, function (msg) {
_this3.$content.append($(tpl_help_message({
'type': type || 'info',
'message': msgs
2017-04-23 19:02:44 +02:00
})));
2017-07-22 22:21:05 +02:00
});
2016-11-07 15:43:48 +01:00
if (spinner === true) {
2017-06-23 20:25:33 +02:00
this.$content.append(tpl_spinner);
2016-11-07 15:43:48 +01:00
} else if (spinner === false) {
this.$content.find('span.spinner').remove();
}
return this.scrollDown();
},
2017-07-22 22:21:05 +02:00
handleChatStateMessage: function handleChatStateMessage(message) {
2017-02-03 13:51:07 +01:00
if (message.get('chat_state') === _converse.COMPOSING) {
2017-04-04 17:26:06 +02:00
if (message.get('sender') === 'me') {
this.showStatusNotification(__('Typing from another device'));
} else {
2017-07-22 22:21:05 +02:00
this.showStatusNotification(message.get('fullname') + ' ' + __('is typing'));
2017-04-04 17:26:06 +02:00
}
2016-11-07 15:43:48 +01:00
this.clear_status_timeout = window.setTimeout(this.clearStatusNotification.bind(this), 30000);
2017-02-03 13:51:07 +01:00
} else if (message.get('chat_state') === _converse.PAUSED) {
2017-04-04 17:26:06 +02:00
if (message.get('sender') === 'me') {
this.showStatusNotification(__('Stopped typing on the other device'));
} else {
2017-07-22 22:21:05 +02:00
this.showStatusNotification(message.get('fullname') + ' ' + __('has stopped typing'));
2017-04-04 17:26:06 +02:00
}
2017-02-03 13:51:07 +01:00
} else if (_.includes([_converse.INACTIVE, _converse.ACTIVE], message.get('chat_state'))) {
2016-11-07 15:43:48 +01:00
this.$content.find('div.chat-event').remove();
2017-02-03 13:51:07 +01:00
} else if (message.get('chat_state') === _converse.GONE) {
2017-07-22 22:21:05 +02:00
this.showStatusNotification(message.get('fullname') + ' ' + __('has gone away'));
2016-11-07 15:43:48 +01:00
}
},
2017-07-22 22:21:05 +02:00
shouldShowOnTextMessage: function shouldShowOnTextMessage() {
2016-11-07 15:43:48 +01:00
return !this.$el.is(':visible');
},
2017-07-22 22:21:05 +02:00
handleTextMessage: function handleTextMessage(message) {
2016-11-07 15:43:48 +01:00
this.showMessage(_.clone(message.attributes));
2017-06-23 20:25:33 +02:00
if (utils.isNewMessage(message) && message.get('sender') === 'me') {
2016-11-07 15:43:48 +01:00
// We remove the "scrolled" flag so that the chat area
// gets scrolled down. We always want to scroll down
// when the user writes a message as opposed to when a
// message is received.
this.model.set('scrolled', false);
2017-06-23 20:25:33 +02:00
} else {
if (utils.isNewMessage(message) && this.model.get('scrolled', true)) {
this.$el.find('.new-msgs-indicator').removeClass('hidden');
}
2016-11-07 15:43:48 +01:00
}
if (this.shouldShowOnTextMessage()) {
this.show();
} else {
this.scrollDown();
}
},
2017-07-22 22:21:05 +02:00
handleErrorMessage: function handleErrorMessage(message) {
var $message = $("[data-msgid=" + message.get('msgid') + "]");
2016-11-07 15:43:48 +01:00
if ($message.length) {
$message.after($('<div class="chat-info chat-error"></div>').text(message.get('message')));
this.scrollDown();
}
},
2017-07-22 22:21:05 +02:00
onMessageAdded: function onMessageAdded(message) {
2016-11-07 15:43:48 +01:00
/* Handler that gets called when a new message object is created.
*
* Parameters:
* (Object) message - The message Backbone object that was added.
*/
if (!_.isUndefined(this.clear_status_timeout)) {
2016-11-07 15:43:48 +01:00
window.clearTimeout(this.clear_status_timeout);
delete this.clear_status_timeout;
}
if (message.get('type') === 'error') {
this.handleErrorMessage(message);
} else if (!message.get('message')) {
this.handleChatStateMessage(message);
} else {
this.handleTextMessage(message);
}
2017-06-23 20:25:33 +02:00
_converse.emit('messageAdded', {
'message': message,
'chatbox': this.model
});
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
createMessageStanza: function createMessageStanza(message) {
2016-11-07 15:43:48 +01:00
return $msg({
2017-07-22 22:21:05 +02:00
from: _converse.connection.jid,
to: this.model.get('jid'),
type: 'chat',
id: message.get('msgid')
}).c('body').t(message.get('message')).up().c(_converse.ACTIVE, { 'xmlns': Strophe.NS.CHATSTATES }).up();
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
sendMessage: function sendMessage(message) {
2016-11-07 15:43:48 +01:00
/* Responsible for sending off a text message.
*
* Parameters:
* (Message) message - The chat message
*/
// TODO: We might want to send to specfic resources.
// Especially in the OTR case.
var messageStanza = this.createMessageStanza(message);
2017-02-03 13:51:07 +01:00
_converse.connection.send(messageStanza);
if (_converse.forward_messages) {
2016-11-07 15:43:48 +01:00
// Forward the message, so that other connected resources are also aware of it.
2017-07-22 22:21:05 +02:00
_converse.connection.send($msg({ to: _converse.bare_jid, type: 'chat', id: message.get('msgid') }).c('forwarded', { xmlns: 'urn:xmpp:forward:0' }).c('delay', { xmns: 'urn:xmpp:delay', stamp: new Date().getTime() }).up().cnode(messageStanza.tree()));
2016-11-07 15:43:48 +01:00
}
},
2017-07-22 22:21:05 +02:00
onMessageSubmitted: function onMessageSubmitted(text) {
2016-11-07 15:43:48 +01:00
/* This method gets called once the user has typed a message
* and then pressed enter in a chat box.
*
* Parameters:
* (string) text - The chat message text.
*/
2017-02-03 13:51:07 +01:00
if (!_converse.connection.authenticated) {
2017-07-22 22:21:05 +02:00
return this.showHelpMessages(['Sorry, the connection has been lost, ' + 'and your message could not be sent'], 'error');
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
var match = text.replace(/^\s*/, "").match(/^\/(.*)\s*$/);
2016-11-07 15:43:48 +01:00
if (match) {
if (match[1] === "clear") {
return this.clearMessages();
2017-07-22 22:21:05 +02:00
} else if (match[1] === "help") {
var msgs = ["<strong>/help</strong>:" + __('Show this menu'), "<strong>/me</strong>:" + __('Write in the third person'), "<strong>/clear</strong>:" + __('Remove messages')];
2016-11-07 15:43:48 +01:00
this.showHelpMessages(msgs);
return;
}
}
2017-02-03 13:51:07 +01:00
var fullname = _converse.xmppstatus.get('fullname');
2017-07-22 22:21:05 +02:00
fullname = _.isEmpty(fullname) ? _converse.bare_jid : fullname;
2016-11-07 15:43:48 +01:00
var message = this.model.messages.create({
fullname: fullname,
sender: 'me',
time: moment().format(),
message: text
});
this.sendMessage(message);
},
2017-07-22 22:21:05 +02:00
sendChatState: function sendChatState() {
2016-11-07 15:43:48 +01:00
/* Sends a message with the status of the user in this chat session
* as taken from the 'chat_state' attribute of the chat box.
* See XEP-0085 Chat State Notifications.
*/
2017-07-22 22:21:05 +02:00
_converse.connection.send($msg({ 'to': this.model.get('jid'), 'type': 'chat' }).c(this.model.get('chat_state'), { 'xmlns': Strophe.NS.CHATSTATES }).up().c('no-store', { 'xmlns': Strophe.NS.HINTS }).up().c('no-permanent-store', { 'xmlns': Strophe.NS.HINTS }));
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
setChatState: function setChatState(state, no_save) {
2016-11-07 15:43:48 +01:00
/* Mutator for setting the chat state of this chat session.
* Handles clearing of any chat state notification timeouts and
* setting new ones if necessary.
* Timeouts are set when the state being set is COMPOSING or PAUSED.
* After the timeout, COMPOSING will become PAUSED and PAUSED will become INACTIVE.
* See XEP-0085 Chat State Notifications.
*
* Parameters:
* (string) state - The chat state (consts ACTIVE, COMPOSING, PAUSED, INACTIVE, GONE)
* (Boolean) no_save - Just do the cleanup or setup but don't actually save the state.
*/
if (!_.isUndefined(this.chat_state_timeout)) {
2016-11-07 15:43:48 +01:00
window.clearTimeout(this.chat_state_timeout);
delete this.chat_state_timeout;
}
2017-02-03 13:51:07 +01:00
if (state === _converse.COMPOSING) {
2017-07-22 22:21:05 +02:00
this.chat_state_timeout = window.setTimeout(this.setChatState.bind(this), _converse.TIMEOUTS.PAUSED, _converse.PAUSED);
2017-02-03 13:51:07 +01:00
} else if (state === _converse.PAUSED) {
2017-07-22 22:21:05 +02:00
this.chat_state_timeout = window.setTimeout(this.setChatState.bind(this), _converse.TIMEOUTS.INACTIVE, _converse.INACTIVE);
2016-11-07 15:43:48 +01:00
}
if (!no_save && this.model.get('chat_state') !== state) {
this.model.set('chat_state', state);
}
return this;
},
2017-07-22 22:21:05 +02:00
onFormSubmitted: function onFormSubmitted(ev) {
2017-04-04 17:26:06 +02:00
ev.preventDefault();
var textarea = this.el.querySelector('.chat-textarea'),
message = textarea.value;
textarea.value = '';
textarea.focus();
if (message !== '') {
this.onMessageSubmitted(message);
_converse.emit('messageSend', message);
}
this.setChatState(_converse.ACTIVE);
},
2017-07-22 22:21:05 +02:00
keyPressed: function keyPressed(ev) {
/* Event handler for when a key is pressed in a chat box textarea.
*/
if (ev.keyCode === KEY.ENTER) {
this.onFormSubmitted(ev);
} else {
// Set chat state to composing if keyCode is not a forward-slash
// (which would imply an internal command and not a message).
this.setChatState(_converse.COMPOSING, ev.keyCode === KEY.FORWARD_SLASH);
}
},
clearMessages: function clearMessages(ev) {
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2016-11-07 15:43:48 +01:00
var result = confirm(__("Are you sure you want to clear the messages from this chat box?"));
if (result === true) {
this.$content.empty();
this.model.messages.reset();
this.model.messages.browserStorage._clear();
}
return this;
},
2017-07-22 22:21:05 +02:00
insertIntoTextArea: function insertIntoTextArea(value) {
2016-11-07 15:43:48 +01:00
var $textbox = this.$el.find('textarea.chat-textarea');
var existing = $textbox.val();
2017-07-22 22:21:05 +02:00
if (existing && existing[existing.length - 1] !== ' ') {
2016-11-07 15:43:48 +01:00
existing = existing + ' ';
}
2017-07-22 22:21:05 +02:00
$textbox.focus().val(existing + value + ' ');
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
insertEmoji: function insertEmoji(ev) {
2016-11-07 15:43:48 +01:00
ev.stopPropagation();
2017-07-22 22:21:05 +02:00
this.toggleEmojiMenu();
var target = ev.target.nodeName === 'IMG' ? ev.target.parentElement : ev.target;
this.insertIntoTextArea(target.getAttribute('data-emoji'));
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
toggleEmojiMenu: function toggleEmojiMenu(ev) {
if (!_.isUndefined(ev)) {
ev.stopPropagation();
if (ev.target.classList.contains('emoji-category-picker') || ev.target.classList.contains('emoji-skintone-picker') || ev.target.classList.contains('emoji-category')) {
return;
}
}
var elements = _.difference(document.querySelectorAll('.toolbar-menu'), [this.emoji_picker_view.el]);
utils.slideInAllElements(elements).then(_.partial(utils.slideToggleElement, this.emoji_picker_view.el));
},
toggleCall: function toggleCall(ev) {
2016-11-07 15:43:48 +01:00
ev.stopPropagation();
2017-02-03 13:51:07 +01:00
_converse.emit('callButtonClicked', {
connection: _converse.connection,
2016-11-07 15:43:48 +01:00
model: this.model
});
},
2017-07-22 22:21:05 +02:00
onChatStatusChanged: function onChatStatusChanged(item) {
var chat_status = item.get('chat_status');
var fullname = item.get('fullname');
fullname = _.isEmpty(fullname) ? item.get('jid') : fullname;
2016-11-07 15:43:48 +01:00
if (this.$el.is(':visible')) {
if (chat_status === 'offline') {
2017-07-22 22:21:05 +02:00
this.showStatusNotification(fullname + ' ' + __('has gone offline'));
2016-11-07 15:43:48 +01:00
} else if (chat_status === 'away') {
2017-07-22 22:21:05 +02:00
this.showStatusNotification(fullname + ' ' + __('has gone away'));
} else if (chat_status === 'dnd') {
this.showStatusNotification(fullname + ' ' + __('is busy'));
2016-11-07 15:43:48 +01:00
} else if (chat_status === 'online') {
this.$el.find('div.chat-event').remove();
}
}
},
2017-07-22 22:21:05 +02:00
onStatusChanged: function onStatusChanged(item) {
2016-11-07 15:43:48 +01:00
this.showStatusMessage();
2017-02-03 13:51:07 +01:00
_converse.emit('contactStatusMessageChanged', {
2016-11-07 15:43:48 +01:00
'contact': item.attributes,
'message': item.get('status')
});
},
2017-07-22 22:21:05 +02:00
showStatusMessage: function showStatusMessage(msg) {
2016-11-07 15:43:48 +01:00
msg = msg || this.model.get('status');
if (_.isString(msg)) {
2016-11-07 15:43:48 +01:00
this.$el.find('p.user-custom-message').text(msg).attr('title', msg);
}
return this;
},
2017-07-22 22:21:05 +02:00
close: function close(ev) {
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2017-02-03 13:51:07 +01:00
if (_converse.connection.connected) {
2016-11-07 15:43:48 +01:00
// Immediately sending the chat state, because the
// model is going to be destroyed afterwards.
2017-02-03 13:51:07 +01:00
this.model.set('chat_state', _converse.INACTIVE);
2016-11-07 15:43:48 +01:00
this.sendChatState();
}
2017-04-23 19:02:44 +02:00
try {
this.model.destroy();
} catch (e) {
2017-07-05 11:59:55 +02:00
_converse.log(e, Strophe.LogLevel.ERROR);
2017-04-23 19:02:44 +02:00
}
2016-11-07 15:43:48 +01:00
this.remove();
2017-02-03 13:51:07 +01:00
_converse.emit('chatBoxClosed', this);
2016-11-07 15:43:48 +01:00
return this;
},
2017-07-22 22:21:05 +02:00
getToolbarOptions: function getToolbarOptions(options) {
2016-11-07 15:43:48 +01:00
return _.extend(options || {}, {
'label_clear': __('Clear all messages'),
'label_insert_smiley': __('Insert a smiley'),
'label_start_call': __('Start a call'),
2017-02-03 13:51:07 +01:00
'show_call_button': _converse.visible_toolbar_buttons.call,
'show_clear_button': _converse.visible_toolbar_buttons.clear,
2017-07-22 22:21:05 +02:00
'use_emoji': _converse.visible_toolbar_buttons.emoji
2016-11-07 15:43:48 +01:00
});
},
2017-07-22 22:21:05 +02:00
renderToolbar: function renderToolbar(toolbar, options) {
if (!_converse.show_toolbar) {
return;
}
toolbar = toolbar || tpl_toolbar;
2017-07-22 22:21:05 +02:00
options = _.assign(this.model.toJSON(), this.getToolbarOptions(options || {}));
this.el.querySelector('.chat-toolbar').innerHTML = toolbar(options);
var toggle = this.el.querySelector('.toggle-smiley');
toggle.innerHTML = '';
toggle.appendChild(this.emoji_picker_view.render().el);
2016-11-07 15:43:48 +01:00
return this;
},
2017-07-22 22:21:05 +02:00
renderAvatar: function renderAvatar() {
2016-11-07 15:43:48 +01:00
if (!this.model.get('image')) {
return;
}
2017-02-03 13:51:07 +01:00
var width = _converse.chatview_avatar_width;
var height = _converse.chatview_avatar_height;
2017-07-22 22:21:05 +02:00
var img_src = "data:" + this.model.get('image_type') + ";base64," + this.model.get('image'),
canvas = $(tpl_avatar({
2017-07-22 22:21:05 +02:00
'width': width,
'height': height
})).get(0);
2016-11-07 15:43:48 +01:00
if (!(canvas.getContext && canvas.getContext('2d'))) {
return this;
}
var ctx = canvas.getContext('2d');
2017-07-22 22:21:05 +02:00
var img = new Image(); // Create new Image object
2016-11-07 15:43:48 +01:00
img.onload = function () {
2017-07-22 22:21:05 +02:00
var ratio = img.width / img.height;
2016-11-07 15:43:48 +01:00
if (ratio < 1) {
2017-07-22 22:21:05 +02:00
ctx.drawImage(img, 0, 0, width, height * (1 / ratio));
2016-11-07 15:43:48 +01:00
} else {
2017-07-22 22:21:05 +02:00
ctx.drawImage(img, 0, 0, width, height * ratio);
2016-11-07 15:43:48 +01:00
}
};
img.src = img_src;
this.$el.find('.chat-title').before(canvas);
return this;
},
2017-07-22 22:21:05 +02:00
focus: function focus() {
2016-11-07 15:43:48 +01:00
this.$el.find('.chat-textarea').focus();
2017-02-03 13:51:07 +01:00
_converse.emit('chatBoxFocused', this);
2016-11-07 15:43:48 +01:00
return this;
},
2017-07-22 22:21:05 +02:00
hide: function hide() {
2016-11-30 17:27:20 +01:00
this.el.classList.add('hidden');
2016-11-07 15:43:48 +01:00
utils.refreshWebkit();
return this;
},
2017-07-22 22:21:05 +02:00
afterShown: function afterShown(focus) {
2017-06-23 20:25:33 +02:00
if (utils.isPersistableModel(this.model)) {
2016-11-07 15:43:48 +01:00
this.model.save();
}
2017-02-03 13:51:07 +01:00
this.setChatState(_converse.ACTIVE);
2016-11-07 15:43:48 +01:00
this.scrollDown();
if (focus) {
this.focus();
}
},
2017-07-22 22:21:05 +02:00
_show: function _show(focus) {
2016-11-07 15:43:48 +01:00
/* Inner show method that gets debounced */
if (this.$el.is(':visible') && this.$el.css('opacity') === "1") {
2017-07-22 22:21:05 +02:00
if (focus) {
this.focus();
}
2016-11-07 15:43:48 +01:00
return;
}
utils.fadeIn(this.el, _.bind(this.afterShown, this, focus));
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
show: function show(focus) {
if (_.isUndefined(this.debouncedShow)) {
2016-11-07 15:43:48 +01:00
/* We wrap the method in a debouncer and set it on the
* instance, so that we have it debounced per instance.
* Debouncing it on the class-level is too broad.
*/
2017-07-22 22:21:05 +02:00
this.debouncedShow = _.debounce(this._show, 250, { 'leading': true });
2016-11-07 15:43:48 +01:00
}
this.debouncedShow.apply(this, arguments);
return this;
},
2017-07-22 22:21:05 +02:00
hideNewMessagesIndicator: function hideNewMessagesIndicator() {
2017-03-05 10:45:18 +01:00
var new_msgs_indicator = this.el.querySelector('.new-msgs-indicator');
if (!_.isNull(new_msgs_indicator)) {
new_msgs_indicator.classList.add('hidden');
}
},
2017-07-22 22:21:05 +02:00
markScrolled: function markScrolled(ev) {
2016-11-07 15:43:48 +01:00
/* Called when the chat content is scrolled up or down.
* We want to record when the user has scrolled away from
* the bottom, so that we don't automatically scroll away
* from what the user is reading when new messages are
* received.
*/
2017-07-22 22:21:05 +02:00
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2017-03-05 10:45:18 +01:00
if (this.model.get('auto_scrolled')) {
this.model.set({
'scrolled': false,
'auto_scrolled': false
});
return;
}
2017-06-23 20:25:33 +02:00
var scrolled = true;
2017-07-22 22:21:05 +02:00
var is_at_bottom = this.$content.scrollTop() + this.$content.innerHeight() >= this.$content[0].scrollHeight - 10;
2016-11-07 15:43:48 +01:00
if (is_at_bottom) {
2017-06-23 20:25:33 +02:00
scrolled = false;
this.onScrolledDown();
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
utils.safeSave(this.model, { 'scrolled': scrolled });
},
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
viewUnreadMessages: function viewUnreadMessages() {
this.model.save('scrolled', false);
2016-11-07 15:43:48 +01:00
this.scrollDown();
},
2017-07-22 22:21:05 +02:00
_scrollDown: function _scrollDown() {
2017-02-03 13:51:07 +01:00
/* Inner method that gets debounced */
2016-11-07 15:43:48 +01:00
if (this.$content.is(':visible') && !this.model.get('scrolled')) {
this.$content.scrollTop(this.$content[0].scrollHeight);
2017-06-23 20:25:33 +02:00
this.onScrolledDown();
2017-07-22 22:21:05 +02:00
this.model.save({ 'auto_scrolled': true });
2016-11-07 15:43:48 +01:00
}
2017-02-03 13:51:07 +01:00
},
2017-07-22 22:21:05 +02:00
onScrolledDown: function onScrolledDown() {
2017-06-23 20:25:33 +02:00
this.hideNewMessagesIndicator();
if (_converse.windowState !== 'hidden') {
this.model.clearUnreadMsgCounter();
}
2017-07-22 22:21:05 +02:00
_converse.emit('chatBoxScrolledDown', { 'chatbox': this.model });
2017-06-23 20:25:33 +02:00
},
2017-07-22 22:21:05 +02:00
scrollDown: function scrollDown() {
2017-02-03 13:51:07 +01:00
if (_.isUndefined(this.debouncedScrollDown)) {
/* We wrap the method in a debouncer and set it on the
* instance, so that we have it debounced per instance.
* Debouncing it on the class-level is too broad.
*/
this.debouncedScrollDown = _.debounce(this._scrollDown, 250);
2017-02-03 13:51:07 +01:00
}
this.debouncedScrollDown.apply(this, arguments);
2016-11-07 15:43:48 +01:00
return this;
2017-06-23 20:25:33 +02:00
},
2017-07-22 22:21:05 +02:00
onWindowStateChanged: function onWindowStateChanged(state) {
2017-06-23 20:25:33 +02:00
if (this.model.get('num_unread', 0) && !this.model.newMessageWillBeHidden()) {
this.model.clearUnreadMsgCounter();
}
2016-11-07 15:43:48 +01:00
}
});
}
});
return converse;
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-chatview.js.map;
2016-11-07 15:43:48 +01:00
define('tpl!add_contact_dropdown', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<dl class="add-converse-contact dropdown">\n <dt id="xmpp-contact-search" class="fancy-dropdown">\n <a class="toggle-xmpp-contact-form icon-plus" href="#" title="' +
2017-02-13 17:16:13 +01:00
__e(label_click_to_chat) +
'"> ' +
2017-02-13 17:16:13 +01:00
__e(label_add_contact) +
2017-07-22 22:21:05 +02:00
'</a>\n </dt>\n <dd class="search-xmpp">\n <div class="contact-form-container collapsed"></div>\n <ul></ul>\n </dd>\n</dl>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!add_contact_form', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-07-22 22:21:05 +02:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-07-22 22:21:05 +02:00
__p += '<form class="pure-form add-xmpp-contact">\n ';
if (error_message) { ;
__p += '\n <span class="pure-form-message error">' +
__e(error_message) +
'</span>\n ';
} ;
__p += '\n <input type="text"\n name="identifier"\n value="' +
__e(value) +
'"\n class="username ';
if (error_message) { ;
__p += ' error ';
} ;
__p += '"\n placeholder="' +
2017-02-13 17:16:13 +01:00
__e(label_contact_username) +
2017-07-22 22:21:05 +02:00
'"/>\n <button class="pure-button button-primary" type="submit">' +
2017-02-13 17:16:13 +01:00
__e(label_add) +
2017-07-22 22:21:05 +02:00
'</button>\n</form>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!change_status_message', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
2017-06-23 20:25:33 +02:00
__p += '<fieldset>\n <span class="input-button-group">\n <input type="text" class="custom-xmpp-status" value="' +
2017-02-13 17:16:13 +01:00
__e(status_message) +
'" placeholder="' +
__e(label_custom_status) +
'"/>\n <input type="submit" class="pure-button button-primary" value="' +
2017-02-13 17:16:13 +01:00
__e(label_save) +
2017-06-23 20:25:33 +02:00
'"/>\n </span>\n</fieldset>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!chat_status', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<div class="xmpp-status">\n <a class="choose-xmpp-status ' +
2017-02-13 17:16:13 +01:00
__e(chat_status) +
' icon-' +
2017-02-13 17:16:13 +01:00
__e(chat_status) +
'" data-value="' +
2017-02-13 17:16:13 +01:00
__e(status_message) +
'" href="#" title="' +
2017-02-13 17:16:13 +01:00
__e(desc_change_status) +
'">\n ' +
2017-02-13 17:16:13 +01:00
__e(status_message) +
'\n </a>\n <a class="change-xmpp-status-message icon-pencil" href="#" title="' +
2017-02-13 17:16:13 +01:00
__e(desc_custom_status) +
2016-11-07 15:43:48 +01:00
'"></a>\n</div>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-11-07 15:43:48 +01:00
define('tpl!choose_status', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '';
with (obj) {
__p += '<dl id="target" class="dropdown">\n <dt id="fancy-xmpp-status-select" class="fancy-dropdown"></dt>\n <dd><ul class="xmpp-status-menu"></ul></dd>\n</dl>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!contacts_panel', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-06-23 20:25:33 +02:00
__p += '<form class="pure-form set-xmpp-status" id="set-xmpp-status" action="" method="post">\n <select id="select-xmpp-status">\n <option value="online">' +
2017-02-13 17:16:13 +01:00
__e(label_online) +
2017-06-23 20:25:33 +02:00
'</option>\n <option value="dnd">' +
2017-02-13 17:16:13 +01:00
__e(label_busy) +
2017-06-23 20:25:33 +02:00
'</option>\n <option value="away">' +
2017-02-13 17:16:13 +01:00
__e(label_away) +
2017-06-23 20:25:33 +02:00
'</option>\n ';
if (include_offline_state) { ;
2017-06-23 20:25:33 +02:00
__p += '\n <option value="offline">' +
2017-02-13 17:16:13 +01:00
__e(label_offline) +
2017-06-23 20:25:33 +02:00
'</option>\n ';
} ;
2017-06-23 20:25:33 +02:00
__p += '\n ';
if (allow_logout) { ;
2017-06-23 20:25:33 +02:00
__p += '\n <option value="logout">' +
2017-02-13 17:16:13 +01:00
__e(label_logout) +
2017-06-23 20:25:33 +02:00
'</option>\n ';
} ;
2017-06-23 20:25:33 +02:00
__p += '\n </select>\n</form>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!contacts_tab', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-06-23 20:25:33 +02:00
__p += '<a class="s contacts-tab\n ';
if (is_current) { ;
__p += ' current ';
} ;
2017-06-23 20:25:33 +02:00
__p += '\n ';
if (num_unread) { ;
__p += ' unread-msgs ';
} ;
__p += '"\n data-id="users" href="#users">\n ' +
2017-02-13 17:16:13 +01:00
__e(label_contacts) +
2017-06-23 20:25:33 +02:00
'\n ';
if (num_unread) { ;
__p += '\n <span class="msgs-indicator">' +
__e( num_unread ) +
'</span>\n ';
} ;
__p += '\n</a>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!controlbox', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-06-23 20:25:33 +02:00
__p += '<div class="flyout box-flyout">\n <div class="chat-head controlbox-head">\n <ul id="controlbox-tabs"></ul>\n ';
if (!sticky_controlbox) { ;
__p += '\n <a class="chatbox-btn close-chatbox-button icon-close"></a>\n ';
} ;
__p += '\n </div>\n <div class="controlbox-panes"></div>\n</div>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-11-07 15:43:48 +01:00
define('tpl!controlbox_toggle', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<span class="conn-feedback">' +
2017-02-13 17:16:13 +01:00
__e(label_toggle) +
2016-11-07 15:43:48 +01:00
'</span>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!login_panel', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
__p += '<form class="pure-form pure-form-stacked converse-form" id="converse-login" method="post">\n ';
if (auto_login) { ;
__p += '\n <span class="spinner login-submit"/>\n ';
} ;
__p += '\n ';
if (!auto_login) { ;
__p += '\n ';
if (authentication == LOGIN || authentication == EXTERNAL) { ;
__p += '\n <label>' +
2017-02-13 17:16:13 +01:00
__e(label_username) +
'</label>\n <input type="text" name="jid" placeholder="' +
2017-02-13 17:16:13 +01:00
__e(placeholder_username) +
2016-11-07 15:43:48 +01:00
'">\n ';
if (authentication !== EXTERNAL) { ;
__p += '\n <label>' +
2017-02-13 17:16:13 +01:00
__e(label_password) +
'</label>\n <input type="password" name="password" placeholder="' +
2017-02-13 17:16:13 +01:00
__e(placeholder_password) +
2016-11-07 15:43:48 +01:00
'">\n ';
} ;
__p += '\n <input class="pure-button button-primary" type="submit" value="' +
2017-02-13 17:16:13 +01:00
__e(label_login) +
2016-11-07 15:43:48 +01:00
'">\n <span class="conn-feedback"></span>\n ';
} ;
__p += '\n ';
if (authentication == ANONYMOUS) { ;
2017-02-13 17:16:13 +01:00
__p += '\n <input class="pure-button button-primary login-anon" type="submit" value="' +
__e(label_anon_login) +
2016-11-07 15:43:48 +01:00
'"/>\n ';
} ;
__p += '\n ';
if (authentication == PREBIND) { ;
__p += '\n <p>Disconnected.</p>\n ';
} ;
__p += '\n ';
} ;
__p += '\n</form>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-11-07 15:43:48 +01:00
define('tpl!login_tab', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
2017-04-04 17:26:06 +02:00
__p += '<li><a class="current" data-id="login" href="#login-dialog">' +
2017-02-13 17:16:13 +01:00
__e(label_sign_in) +
2016-11-07 15:43:48 +01:00
'</a></li>\n';
}
return __p
};});
define('tpl!search_contact', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<li>\n <form class="search-xmpp-contact">\n <input type="text"\n name="identifier"\n class="username"\n placeholder="' +
2017-02-13 17:16:13 +01:00
__e(label_contact_name) +
'"/>\n <button type="submit">' +
2017-02-13 17:16:13 +01:00
__e(label_search) +
2016-11-07 15:43:48 +01:00
'</button>\n </form>\n</li>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!status_option', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<li>\n <a href="#" class="' +
2017-02-13 17:16:13 +01:00
__e( value ) +
'" data-value="' +
2017-02-13 17:16:13 +01:00
__e( value ) +
'">\n <span class="icon-' +
2017-02-13 17:16:13 +01:00
__e( value ) +
'"></span>\n ' +
2017-02-13 17:16:13 +01:00
__e( text ) +
2016-11-07 15:43:48 +01:00
'\n </a>\n</li>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!group_header', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<a href="#" class="group-toggle icon-' +
2017-02-13 17:16:13 +01:00
__e(toggle_state) +
'" title="' +
2017-02-13 17:16:13 +01:00
__e(desc_group_toggle) +
'">' +
2017-02-13 17:16:13 +01:00
__e(label_group) +
2016-11-07 15:43:48 +01:00
'</a>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!pending_contact', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
if (allow_chat_pending_contacts) { ;
__p += '\n<a class="open-chat"href="#">\n';
} ;
__p += '\n<span class="pending-contact-name" title="Name: ' +
2017-02-13 17:16:13 +01:00
__e(fullname) +
'\nJID: ' +
2017-02-13 17:16:13 +01:00
__e(jid) +
'">' +
2017-02-13 17:16:13 +01:00
__e(fullname) +
2016-11-07 15:43:48 +01:00
'</span> \n';
if (allow_chat_pending_contacts) { ;
__p += '\n</a>\n';
} ;
__p += '\n<a class="remove-xmpp-contact icon-remove" title="' +
2017-02-13 17:16:13 +01:00
__e(desc_remove) +
2016-11-07 15:43:48 +01:00
'" href="#"></a>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!requesting_contact', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
if (allow_chat_pending_contacts) { ;
__p += '\n<a class="open-chat"href="#">\n';
} ;
__p += '\n<span class="req-contact-name" title="Name: ' +
2017-02-13 17:16:13 +01:00
__e(fullname) +
'\nJID: ' +
2017-02-13 17:16:13 +01:00
__e(jid) +
'">' +
2017-02-13 17:16:13 +01:00
__e(fullname) +
2016-11-07 15:43:48 +01:00
'</span>\n';
if (allow_chat_pending_contacts) { ;
__p += '\n</a>\n';
} ;
__p += '\n<span class="request-actions">\n <a class="accept-xmpp-request icon-checkmark" aria-label="' +
__e(desc_accept) +
'" title="' +
2017-02-13 17:16:13 +01:00
__e(desc_accept) +
'" href="#"></a>\n <a class="decline-xmpp-request icon-close" aria-label="' +
__e(desc_decline) +
'" title="' +
2017-02-13 17:16:13 +01:00
__e(desc_decline) +
2016-11-07 15:43:48 +01:00
'" href="#"></a>\n</span>\n';
}
return __p
};});
define('tpl!roster', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-04-04 17:26:06 +02:00
var __t, __p = '';
with (obj) {
2017-06-23 20:25:33 +02:00
__p += '<dl class="roster-contacts"></dl>\n';
2017-04-04 17:26:06 +02:00
}
return __p
};});
define('tpl!roster_filter', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-02-13 17:16:13 +01:00
__p += '<form class="pure-form roster-filter-form input-button-group">\n <input value="' +
((__t = (filter_text)) == null ? '' : __t) +
2017-06-23 20:25:33 +02:00
'" class="roster-filter roster-filter-' +
((__t = (filter_type)) == null ? '' : __t) +
'"\n placeholder="' +
((__t = (placeholder)) == null ? '' : __t) +
2017-06-23 20:25:33 +02:00
'">\n <select class="state-type state-type-' +
((__t = (filter_type)) == null ? '' : __t) +
'">\n <option value="">' +
((__t = (label_any)) == null ? '' : __t) +
2017-06-23 20:25:33 +02:00
'</option>\n <option ';
if (chat_state === 'unread_messages') { ;
__p += ' selected="selected" ';
} ;
__p += '\n value="unread_messages">' +
((__t = (label_unread_messages)) == null ? '' : __t) +
2016-11-07 15:43:48 +01:00
'</option>\n <option ';
if (chat_state === 'online') { ;
__p += ' selected="selected" ';
} ;
__p += '\n value="online">' +
((__t = (label_online)) == null ? '' : __t) +
2016-11-07 15:43:48 +01:00
'</option>\n <option ';
if (chat_state === 'chat') { ;
__p += ' selected="selected" ';
} ;
__p += '\n value="chat">' +
((__t = (label_chatty)) == null ? '' : __t) +
2016-11-07 15:43:48 +01:00
'</option>\n <option ';
if (chat_state === 'dnd') { ;
__p += ' selected="selected" ';
} ;
__p += '\n value="dnd">' +
((__t = (label_busy)) == null ? '' : __t) +
2016-11-07 15:43:48 +01:00
'</option>\n <option ';
if (chat_state === 'away') { ;
__p += ' selected="selected" ';
} ;
__p += '\n value="away">' +
((__t = (label_away)) == null ? '' : __t) +
2016-11-07 15:43:48 +01:00
'</option>\n <option ';
if (chat_state === 'xa') { ;
__p += ' selected="selected" ';
} ;
__p += '\n value="xa">' +
((__t = (label_xa)) == null ? '' : __t) +
2016-11-07 15:43:48 +01:00
'</option>\n <option ';
if (chat_state === 'offline') { ;
__p += ' selected="selected" ';
} ;
__p += '\n value="offline">' +
((__t = (label_offline)) == null ? '' : __t) +
2016-11-07 15:43:48 +01:00
'</option>\n </select>\n <select class="filter-type">\n <option ';
if (filter_type === 'contacts') { ;
__p += ' selected="selected" ';
} ;
__p += '\n value="contacts">' +
((__t = (label_contacts)) == null ? '' : __t) +
2016-11-07 15:43:48 +01:00
'</option>\n <option ';
if (filter_type === 'groups') { ;
__p += ' selected="selected" ';
} ;
__p += '\n value="groups">' +
((__t = (label_groups)) == null ? '' : __t) +
2016-11-07 15:43:48 +01:00
'</option>\n <option ';
if (filter_type === 'state') { ;
__p += ' selected="selected" ';
} ;
__p += '\n value="state">' +
((__t = (label_state)) == null ? '' : __t) +
2016-11-07 15:43:48 +01:00
'</option>\n </select>\n</form>\n';
2016-03-16 12:49:35 +01:00
}
return __p
};});
define('tpl!roster_item', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-06-23 20:25:33 +02:00
__p += '<a class="open-chat ';
if (num_unread) { ;
__p += ' unread-msgs ';
} ;
__p += '"\n title="' +
2017-02-13 17:16:13 +01:00
__e(title_fullname) +
': ' +
2017-02-13 17:16:13 +01:00
__e(fullname) +
2017-06-23 20:25:33 +02:00
' JID: ' +
2017-02-13 17:16:13 +01:00
__e(jid) +
2017-06-23 20:25:33 +02:00
' ' +
2017-02-13 17:16:13 +01:00
__e(desc_chat) +
2017-06-23 20:25:33 +02:00
'"\n href="#">\n <div class="avatar avatar-' +
__e(chat_status) +
'">\n <span class="status-icon icon-' +
2017-02-13 17:16:13 +01:00
__e(chat_status) +
'" title="' +
2017-02-13 17:16:13 +01:00
__e(desc_status) +
2017-06-23 20:25:33 +02:00
'"></span>\n </div>\n ';
if (num_unread) { ;
__p += '\n <span class="msgs-indicator">' +
__e( num_unread ) +
'</span>\n ';
} ;
__p += '\n <span class="contact-name ';
if (num_unread) { ;
__p += ' unread-msgs ';
} ;
__p += '">' +
2017-02-13 17:16:13 +01:00
__e(fullname) +
2017-06-23 20:25:33 +02:00
'</span>\n</a>\n';
if (allow_contact_removal) { ;
__p += '\n<a class="remove-xmpp-contact icon-remove" title="' +
2017-02-13 17:16:13 +01:00
__e(desc_remove) +
2016-11-07 15:43:48 +01:00
'" href="#"></a>\n';
} ;
2017-06-23 20:25:33 +02:00
__p += '\n\n\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-05-03 17:37:10 +02:00
2017-07-22 22:21:05 +02:00
2016-03-16 12:49:35 +01:00
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
2017-02-13 17:16:13 +01:00
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
2016-03-16 12:49:35 +01:00
// Licensed under the Mozilla Public License (MPLv2)
//
2017-04-23 19:02:44 +02:00
/*global define */
2016-03-16 12:49:35 +01:00
(function (root, factory) {
2017-07-22 22:21:05 +02:00
define('converse-rosterview',["jquery.noconflict", "converse-core", "tpl!group_header", "tpl!pending_contact", "tpl!requesting_contact", "tpl!roster", "tpl!roster_filter", "tpl!roster_item"], factory);
})(undefined, function ($, converse, tpl_group_header, tpl_pending_contact, tpl_requesting_contact, tpl_roster, tpl_roster_filter, tpl_roster_item) {
2016-03-16 12:49:35 +01:00
"use strict";
2017-07-22 22:21:05 +02:00
var _converse$env = converse.env,
Backbone = _converse$env.Backbone,
utils = _converse$env.utils,
Strophe = _converse$env.Strophe,
$iq = _converse$env.$iq,
b64_sha1 = _converse$env.b64_sha1,
sizzle = _converse$env.sizzle,
_ = _converse$env._;
2017-02-03 13:51:07 +01:00
2017-04-23 19:02:44 +02:00
converse.plugins.add('converse-rosterview', {
2016-03-07 18:54:07 +01:00
2016-03-16 12:49:35 +01:00
overrides: {
// Overrides mentioned here will be picked up by converse.js's
// plugin architecture they will replace existing methods on the
// relevant objects or classes.
//
// New functions which don't exist yet can also be added.
2017-07-22 22:21:05 +02:00
afterReconnected: function afterReconnected() {
2016-11-07 15:43:48 +01:00
this.__super__.afterReconnected.apply(this, arguments);
},
2017-07-22 22:21:05 +02:00
_tearDown: function _tearDown() {
2016-12-13 20:46:07 +01:00
/* Remove the rosterview when tearing down. It gets created
* anew when reconnecting or logging in.
*/
this.__super__._tearDown.apply(this, arguments);
if (!_.isUndefined(this.rosterview)) {
this.rosterview.remove();
}
},
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
RosterGroups: {
2017-07-22 22:21:05 +02:00
comparator: function comparator() {
2016-11-07 15:43:48 +01:00
// RosterGroupsComparator only gets set later (once i18n is
// set up), so we need to wrap it in this nameless function.
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2017-02-03 13:51:07 +01:00
return _converse.RosterGroupsComparator.apply(this, arguments);
2016-03-07 18:54:07 +01:00
}
}
2016-03-16 12:49:35 +01:00
},
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-03-16 12:49:35 +01:00
/* The initialize function gets called as soon as the plugin is
* loaded by converse.js's plugin machinery.
*/
2017-02-03 13:51:07 +01:00
var _converse = this._converse,
2017-06-23 20:25:33 +02:00
__ = _converse.__,
___ = _converse.___;
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
2017-07-05 11:59:55 +02:00
_converse.api.settings.update({
2017-02-03 13:51:07 +01:00
allow_chat_pending_contacts: true,
2016-11-07 15:43:48 +01:00
allow_contact_removal: true,
2017-07-22 22:21:05 +02:00
show_toolbar: true
2016-03-16 12:49:35 +01:00
});
2016-03-07 18:54:07 +01:00
2016-11-07 15:43:48 +01:00
var STATUSES = {
'dnd': __('This contact is busy'),
'online': __('This contact is online'),
'offline': __('This contact is offline'),
'unavailable': __('This contact is unavailable'),
'xa': __('This contact is away for an extended period'),
'away': __('This contact is away')
};
var LABEL_CONTACTS = __('Contacts');
var LABEL_GROUPS = __('Groups');
2017-07-22 22:21:05 +02:00
var HEADER_CURRENT_CONTACTS = __('My contacts');
2016-11-07 15:43:48 +01:00
var HEADER_PENDING_CONTACTS = __('Pending contacts');
var HEADER_REQUESTING_CONTACTS = __('Contact requests');
var HEADER_UNGROUPED = __('Ungrouped');
var HEADER_WEIGHTS = {};
HEADER_WEIGHTS[HEADER_REQUESTING_CONTACTS] = 0;
2017-07-22 22:21:05 +02:00
HEADER_WEIGHTS[HEADER_CURRENT_CONTACTS] = 1;
HEADER_WEIGHTS[HEADER_UNGROUPED] = 2;
HEADER_WEIGHTS[HEADER_PENDING_CONTACTS] = 3;
2016-03-07 18:54:07 +01:00
2017-02-03 13:51:07 +01:00
_converse.RosterGroupsComparator = function (a, b) {
2016-11-07 15:43:48 +01:00
/* Groups are sorted alphabetically, ignoring case.
* However, Ungrouped, Requesting Contacts and Pending Contacts
* appear last and in that order.
*/
a = a.get('name');
b = b.get('name');
var special_groups = _.keys(HEADER_WEIGHTS);
var a_is_special = _.includes(special_groups, a);
var b_is_special = _.includes(special_groups, b);
2017-07-22 22:21:05 +02:00
if (!a_is_special && !b_is_special) {
return a.toLowerCase() < b.toLowerCase() ? -1 : a.toLowerCase() > b.toLowerCase() ? 1 : 0;
2016-11-07 15:43:48 +01:00
} else if (a_is_special && b_is_special) {
2017-07-22 22:21:05 +02:00
return HEADER_WEIGHTS[a] < HEADER_WEIGHTS[b] ? -1 : HEADER_WEIGHTS[a] > HEADER_WEIGHTS[b] ? 1 : 0;
2016-11-07 15:43:48 +01:00
} else if (!a_is_special && b_is_special) {
2017-07-22 22:21:05 +02:00
return b === HEADER_REQUESTING_CONTACTS ? 1 : -1;
2016-11-07 15:43:48 +01:00
} else if (a_is_special && !b_is_special) {
2017-07-22 22:21:05 +02:00
return a === HEADER_REQUESTING_CONTACTS ? -1 : 1;
2016-11-07 15:43:48 +01:00
}
};
2016-03-07 18:54:07 +01:00
2017-02-03 13:51:07 +01:00
_converse.RosterFilter = Backbone.Model.extend({
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-11-07 15:43:48 +01:00
this.set({
'filter_text': '',
'filter_type': 'contacts',
'chat_state': ''
});
2017-07-22 22:21:05 +02:00
}
2016-11-07 15:43:48 +01:00
});
2016-03-07 18:54:07 +01:00
2017-02-03 13:51:07 +01:00
_converse.RosterFilterView = Backbone.View.extend({
2016-11-07 15:43:48 +01:00
tagName: 'span',
events: {
"keydown .roster-filter": "liveFilter",
2017-02-13 17:16:13 +01:00
"submit form.roster-filter-form": "submitFilter",
2016-11-07 15:43:48 +01:00
"click .onX": "clearFilter",
"mousemove .x": "toggleX",
"change .filter-type": "changeTypeFilter",
"change .state-type": "changeChatStateFilter"
2016-03-16 12:49:35 +01:00
},
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2017-02-13 17:16:13 +01:00
this.model.on('change:filter_type', this.render, this);
2017-04-23 19:02:44 +02:00
this.model.on('change:filter_text', this.renderClearButton, this);
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
render: function render() {
this.el.innerHTML = tpl_roster_filter(_.extend(this.model.toJSON(), {
placeholder: __('Filter'),
label_contacts: LABEL_CONTACTS,
label_groups: LABEL_GROUPS,
label_state: __('State'),
label_any: __('Any'),
label_unread_messages: __('Unread'),
label_online: __('Online'),
label_chatty: __('Chatty'),
label_busy: __('Busy'),
label_away: __('Away'),
label_xa: __('Extended Away'),
label_offline: __('Offline')
}));
2017-02-13 17:16:13 +01:00
this.renderClearButton();
return this.$el;
},
2017-07-22 22:21:05 +02:00
renderClearButton: function renderClearButton() {
2017-04-23 19:02:44 +02:00
var roster_filter = this.el.querySelector('.roster-filter');
if (_.isNull(roster_filter)) {
return;
}
roster_filter.classList[this.tog(roster_filter.value)]('x');
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
tog: function tog(v) {
return v ? 'add' : 'remove';
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
toggleX: function toggleX(ev) {
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2016-11-07 15:43:48 +01:00
var el = ev.target;
2017-07-22 22:21:05 +02:00
el.classList[this.tog(el.offsetWidth - 18 < ev.clientX - el.getBoundingClientRect().left)]('onX');
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
changeChatStateFilter: function changeChatStateFilter(ev) {
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2016-11-07 15:43:48 +01:00
this.model.save({
2017-04-04 17:26:06 +02:00
'chat_state': this.el.querySelector('.state-type').value
2016-11-07 15:43:48 +01:00
});
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
changeTypeFilter: function changeTypeFilter(ev) {
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2016-11-07 15:43:48 +01:00
var type = ev.target.value;
if (type === 'state') {
this.model.save({
'filter_type': type,
2017-04-04 17:26:06 +02:00
'chat_state': this.el.querySelector('.state-type').value
2016-11-07 15:43:48 +01:00
});
} else {
this.model.save({
'filter_type': type,
2017-04-04 17:26:06 +02:00
'filter_text': this.el.querySelector('.roster-filter').value
2016-11-07 15:43:48 +01:00
});
2016-03-16 12:49:35 +01:00
}
},
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
liveFilter: _.debounce(function (ev) {
this.model.save({
2017-04-04 17:26:06 +02:00
'filter_type': this.el.querySelector('.filter-type').value,
'filter_text': this.el.querySelector('.roster-filter').value
2016-11-07 15:43:48 +01:00
});
}, 250),
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
submitFilter: function submitFilter(ev) {
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2017-02-13 17:16:13 +01:00
this.liveFilter();
this.render();
},
2017-07-22 22:21:05 +02:00
isActive: function isActive() {
2016-11-07 15:43:48 +01:00
/* Returns true if the filter is enabled (i.e. if the user
* has added values to the filter).
*/
2017-07-22 22:21:05 +02:00
if (this.model.get('filter_type') === 'state' || this.model.get('filter_text')) {
2016-11-07 15:43:48 +01:00
return true;
}
return false;
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
show: function show() {
if (this.$el.is(':visible')) {
return this;
}
2016-11-07 15:43:48 +01:00
this.$el.show();
return this;
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
hide: function hide() {
if (!this.$el.is(':visible')) {
return this;
}
2017-04-04 17:26:06 +02:00
if (this.el.querySelector('.roster-filter').value.length > 0) {
2016-11-07 15:43:48 +01:00
// Don't hide if user is currently filtering.
2016-03-16 12:49:35 +01:00
return;
}
2016-11-07 15:43:48 +01:00
this.model.save({
'filter_text': '',
'chat_state': ''
});
this.$el.hide();
return this;
},
2017-07-22 22:21:05 +02:00
clearFilter: function clearFilter(ev) {
2016-11-07 15:43:48 +01:00
if (ev && ev.preventDefault) {
ev.preventDefault();
$(ev.target).removeClass('x onX').val('');
2016-03-16 12:49:35 +01:00
}
2016-11-07 15:43:48 +01:00
this.model.save({
'filter_text': ''
2016-06-20 21:11:43 +02:00
});
2016-11-07 15:43:48 +01:00
}
});
2017-02-03 13:51:07 +01:00
_converse.RosterView = Backbone.Overview.extend({
2016-11-07 15:43:48 +01:00
tagName: 'div',
id: 'converse-roster',
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2017-02-03 13:51:07 +01:00
_converse.roster.on("add", this.onContactAdd, this);
_converse.roster.on('change', this.onContactChange, this);
_converse.roster.on("destroy", this.update, this);
_converse.roster.on("remove", this.update, this);
2016-11-07 15:43:48 +01:00
this.model.on("add", this.onGroupAdd, this);
this.model.on("reset", this.reset, this);
2017-02-03 13:51:07 +01:00
_converse.on('rosterGroupsFetched', this.positionFetchedGroups, this);
_converse.on('rosterContactsFetched', this.update, this);
2016-11-07 15:43:48 +01:00
this.createRosterFilter();
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
render: function render() {
2017-04-04 17:26:06 +02:00
this.renderRoster();
2016-11-07 15:43:48 +01:00
this.$el.html(this.filter_view.render());
2017-02-03 13:51:07 +01:00
if (!_converse.allow_contact_requests) {
2016-11-07 15:43:48 +01:00
// XXX: if we ever support live editing of config then
// we'll need to be able to remove this class on the fly.
2017-04-04 17:26:06 +02:00
this.el.classList.add('no-contact-requests');
2016-11-07 15:43:48 +01:00
}
return this;
2016-07-26 08:00:30 +02:00
},
2017-07-22 22:21:05 +02:00
renderRoster: function renderRoster() {
2017-04-04 17:26:06 +02:00
this.$roster = $(tpl_roster());
this.roster = this.$roster[0];
},
2017-07-22 22:21:05 +02:00
createRosterFilter: function createRosterFilter() {
2016-11-07 15:43:48 +01:00
// Create a model on which we can store filter properties
2017-02-03 13:51:07 +01:00
var model = new _converse.RosterFilter();
2017-07-22 22:21:05 +02:00
model.id = b64_sha1("_converse.rosterfilter" + _converse.bare_jid);
2016-11-07 15:43:48 +01:00
model.browserStorage = new Backbone.BrowserStorage.local(this.filter.id);
2017-07-22 22:21:05 +02:00
this.filter_view = new _converse.RosterFilterView({ 'model': model });
2016-11-07 15:43:48 +01:00
this.filter_view.model.on('change', this.updateFilter, this);
this.filter_view.model.fetch();
},
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
updateFilter: _.debounce(function () {
/* Filter the roster again.
* Called whenever the filter settings have been changed or
* when contacts have been added, removed or changed.
*
2016-11-07 15:43:48 +01:00
* Debounced so that it doesn't get called for every
* contact fetched from browser storage.
*/
2016-11-07 15:43:48 +01:00
var type = this.filter_view.model.get('filter_type');
if (type === 'state') {
this.filter(this.filter_view.model.get('chat_state'), type);
} else {
this.filter(this.filter_view.model.get('filter_text'), type);
2016-03-16 12:49:35 +01:00
}
2016-11-07 15:43:48 +01:00
}, 100),
2016-03-07 18:54:07 +01:00
2016-11-07 15:43:48 +01:00
update: _.debounce(function () {
2017-04-04 17:26:06 +02:00
if (_.isNull(this.roster.parentElement)) {
2016-11-07 15:43:48 +01:00
this.$el.append(this.$roster.show());
2016-09-16 14:35:02 +02:00
}
2016-11-07 15:43:48 +01:00
return this.showHideFilter();
2017-02-03 13:51:07 +01:00
}, _converse.animate ? 100 : 0),
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
showHideFilter: function showHideFilter() {
2016-11-07 15:43:48 +01:00
if (!this.$el.is(':visible')) {
return;
2016-03-16 12:49:35 +01:00
}
2017-07-22 22:21:05 +02:00
if (_converse.roster.length >= 10) {
2016-11-07 15:43:48 +01:00
this.filter_view.show();
} else if (!this.filter_view.isActive()) {
this.filter_view.hide();
2016-03-16 12:49:35 +01:00
}
2016-11-07 15:43:48 +01:00
return this;
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
filter: function filter(query, type) {
2016-11-07 15:43:48 +01:00
// First we make sure the filter is restored to its
// original state
_.each(this.getAll(), function (view) {
if (view.model.contacts.length > 0) {
view.show().filter('');
}
});
// Now we can filter
query = query.toLowerCase();
if (type === 'groups') {
_.each(this.getAll(), function (view, idx) {
if (!_.includes(view.model.get('name').toLowerCase(), query.toLowerCase())) {
2016-11-07 15:43:48 +01:00
view.hide();
} else if (view.model.contacts.length > 0) {
view.show();
}
});
} else {
_.each(this.getAll(), function (view) {
view.filter(query, type);
});
2016-03-16 12:49:35 +01:00
}
},
2017-07-22 22:21:05 +02:00
reset: function reset() {
2017-02-03 13:51:07 +01:00
_converse.roster.reset();
2016-11-07 15:43:48 +01:00
this.removeAll();
2017-04-04 17:26:06 +02:00
this.renderRoster();
2016-11-07 15:43:48 +01:00
this.render().update();
return this;
},
2017-07-22 22:21:05 +02:00
onGroupAdd: function onGroupAdd(group) {
var view = new _converse.RosterGroupView({ model: group });
2016-11-07 15:43:48 +01:00
this.add(group.get('name'), view.render());
this.positionGroup(view);
2016-07-26 08:00:30 +02:00
},
2017-07-22 22:21:05 +02:00
onContactAdd: function onContactAdd(contact) {
2016-11-07 15:43:48 +01:00
this.addRosterContact(contact).update();
this.updateFilter();
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
onContactChange: function onContactChange(contact) {
2016-11-07 15:43:48 +01:00
this.updateChatBox(contact).update();
if (_.has(contact.changed, 'subscription')) {
if (contact.changed.subscription === 'from') {
this.addContactToGroup(contact, HEADER_PENDING_CONTACTS);
} else if (_.includes(['both', 'to'], contact.get('subscription'))) {
2016-11-07 15:43:48 +01:00
this.addExistingContact(contact);
}
}
if (_.has(contact.changed, 'ask') && contact.changed.ask === 'subscribe') {
this.addContactToGroup(contact, HEADER_PENDING_CONTACTS);
}
if (_.has(contact.changed, 'subscription') && contact.changed.requesting === 'true') {
this.addContactToGroup(contact, HEADER_REQUESTING_CONTACTS);
}
this.updateFilter();
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
updateChatBox: function updateChatBox(contact) {
2017-02-03 13:51:07 +01:00
var chatbox = _converse.chatboxes.get(contact.get('jid')),
2016-11-07 15:43:48 +01:00
changes = {};
if (!chatbox) {
return this;
}
if (_.has(contact.changed, 'chat_status')) {
changes.chat_status = contact.get('chat_status');
}
if (_.has(contact.changed, 'status')) {
changes.status = contact.get('status');
2016-03-16 12:49:35 +01:00
}
2016-11-07 15:43:48 +01:00
chatbox.save(changes);
return this;
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
positionFetchedGroups: function positionFetchedGroups() {
2016-11-07 15:43:48 +01:00
/* Instead of throwing an add event for each group
* fetched, we wait until they're all fetched and then
* we position them.
* Works around the problem of positionGroup not
* working when all groups besides the one being
* positioned aren't already in inserted into the
* roster DOM element.
*/
2017-02-03 13:51:07 +01:00
var that = this;
2016-11-07 15:43:48 +01:00
this.model.sort();
this.model.each(function (group, idx) {
2017-02-03 13:51:07 +01:00
var view = that.get(group.get('name'));
2016-11-07 15:43:48 +01:00
if (!view) {
2017-07-22 22:21:05 +02:00
view = new _converse.RosterGroupView({ model: group });
2017-02-03 13:51:07 +01:00
that.add(group.get('name'), view.render());
2016-03-16 12:49:35 +01:00
}
2016-11-07 15:43:48 +01:00
if (idx === 0) {
2017-02-03 13:51:07 +01:00
that.$roster.append(view.$el);
2016-11-07 15:43:48 +01:00
} else {
2017-02-03 13:51:07 +01:00
that.appendGroup(view);
2016-03-16 12:49:35 +01:00
}
2017-02-03 13:51:07 +01:00
});
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
positionGroup: function positionGroup(view) {
2016-11-07 15:43:48 +01:00
/* Place the group's DOM element in the correct alphabetical
* position amongst the other groups in the roster.
*/
2016-11-07 15:43:48 +01:00
var $groups = this.$roster.find('.roster-group'),
index = $groups.length ? this.model.indexOf(view.model) : 0;
if (index === 0) {
this.$roster.prepend(view.$el);
2017-07-22 22:21:05 +02:00
} else if (index === this.model.length - 1) {
2016-11-07 15:43:48 +01:00
this.appendGroup(view);
} else {
$($groups.eq(index)).before(view.$el);
}
return this;
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
appendGroup: function appendGroup(view) {
2016-11-07 15:43:48 +01:00
/* Add the group at the bottom of the roster
*/
2016-11-07 15:43:48 +01:00
var $last = this.$roster.find('.roster-group').last();
var $siblings = $last.siblings('dd');
if ($siblings.length > 0) {
$siblings.last().after(view.$el);
} else {
$last.after(view.$el);
2016-03-16 12:49:35 +01:00
}
return this;
},
2017-07-22 22:21:05 +02:00
getGroup: function getGroup(name) {
2016-11-07 15:43:48 +01:00
/* Returns the group as specified by name.
* Creates the group if it doesn't exist.
*/
2017-07-22 22:21:05 +02:00
var view = this.get(name);
2016-11-07 15:43:48 +01:00
if (view) {
return view.model;
}
2017-07-22 22:21:05 +02:00
return this.model.create({ name: name, id: b64_sha1(name) });
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
addContactToGroup: function addContactToGroup(contact, name) {
2016-11-07 15:43:48 +01:00
this.getGroup(name).contacts.add(contact);
},
2017-07-22 22:21:05 +02:00
addExistingContact: function addExistingContact(contact) {
var groups = void 0;
2017-02-03 13:51:07 +01:00
if (_converse.roster_groups) {
2016-11-07 15:43:48 +01:00
groups = contact.get('groups');
if (groups.length === 0) {
groups = [HEADER_UNGROUPED];
2016-03-16 12:49:35 +01:00
}
2016-08-12 22:52:21 +02:00
} else {
2016-11-07 15:43:48 +01:00
groups = [HEADER_CURRENT_CONTACTS];
2016-03-16 12:49:35 +01:00
}
2016-11-07 15:43:48 +01:00
_.each(groups, _.bind(this.addContactToGroup, this, contact));
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
addRosterContact: function addRosterContact(contact) {
2016-11-07 15:43:48 +01:00
if (contact.get('subscription') === 'both' || contact.get('subscription') === 'to') {
this.addExistingContact(contact);
} else {
2017-07-22 22:21:05 +02:00
if (contact.get('ask') === 'subscribe' || contact.get('subscription') === 'from') {
2016-11-07 15:43:48 +01:00
this.addContactToGroup(contact, HEADER_PENDING_CONTACTS);
} else if (contact.get('requesting') === true) {
this.addContactToGroup(contact, HEADER_REQUESTING_CONTACTS);
}
2016-03-16 12:49:35 +01:00
}
return this;
2016-11-07 15:43:48 +01:00
}
});
2016-03-07 18:54:07 +01:00
2017-02-03 13:51:07 +01:00
_converse.RosterContactView = Backbone.View.extend({
2016-11-07 15:43:48 +01:00
tagName: 'dd',
events: {
"click .accept-xmpp-request": "acceptRequest",
"click .decline-xmpp-request": "declineRequest",
"click .open-chat": "openChat",
"click .remove-xmpp-contact": "removeContact"
2016-07-28 18:06:31 +02:00
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-11-07 15:43:48 +01:00
this.model.on("change", this.render, this);
this.model.on("remove", this.remove, this);
this.model.on("destroy", this.remove, this);
this.model.on("open", this.openChat, this);
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
render: function render() {
var that = this;
2016-11-07 15:43:48 +01:00
if (!this.mayBeShown()) {
this.$el.hide();
return this;
}
var item = this.model,
ask = item.get('ask'),
chat_status = item.get('chat_status'),
2017-07-22 22:21:05 +02:00
requesting = item.get('requesting'),
2016-11-07 15:43:48 +01:00
subscription = item.get('subscription');
2017-07-22 22:21:05 +02:00
var classes_to_remove = ['current-xmpp-contact', 'pending-xmpp-contact', 'requesting-xmpp-contact'].concat(_.keys(STATUSES));
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
_.each(classes_to_remove, function (cls) {
if (_.includes(that.el.className, cls)) {
that.el.classList.remove(cls);
}
});
2016-11-07 15:43:48 +01:00
this.$el.addClass(chat_status).data('status', chat_status);
2017-07-22 22:21:05 +02:00
if (ask === 'subscribe' || subscription === 'from') {
2016-11-07 15:43:48 +01:00
/* ask === 'subscribe'
* Means we have asked to subscribe to them.
*
* subscription === 'from'
* They are subscribed to use, but not vice versa.
* We assume that there is a pending subscription
* from us to them (otherwise we're in a state not
* supported by converse.js).
*
* So in both cases the user is a "pending" contact.
*/
2017-04-04 17:26:06 +02:00
this.el.classList.add('pending-xmpp-contact');
2017-07-22 22:21:05 +02:00
this.$el.html(tpl_pending_contact(_.extend(item.toJSON(), {
'desc_remove': __(___('Click to remove %1$s as a contact'), item.get('fullname')),
'allow_chat_pending_contacts': _converse.allow_chat_pending_contacts
})));
2016-11-07 15:43:48 +01:00
} else if (requesting === true) {
2017-04-04 17:26:06 +02:00
this.el.classList.add('requesting-xmpp-contact');
2017-07-22 22:21:05 +02:00
this.$el.html(tpl_requesting_contact(_.extend(item.toJSON(), {
'desc_accept': __(___("Click to accept the contact request from %1$s"), item.get('fullname')),
'desc_decline': __(___("Click to decline the contact request from %1$s"), item.get('fullname')),
'allow_chat_pending_contacts': _converse.allow_chat_pending_contacts
})));
2016-11-07 15:43:48 +01:00
} else if (subscription === 'both' || subscription === 'to') {
2017-04-04 17:26:06 +02:00
this.el.classList.add('current-xmpp-contact');
2017-07-22 22:21:05 +02:00
this.el.classList.remove(_.without(['both', 'to'], subscription)[0]);
2017-06-23 20:25:33 +02:00
this.el.classList.add(subscription);
this.renderRosterItem(item);
2016-11-07 15:43:48 +01:00
}
return this;
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
renderRosterItem: function renderRosterItem(item) {
2017-06-23 20:25:33 +02:00
var chat_status = item.get('chat_status');
2017-07-22 22:21:05 +02:00
this.$el.html(tpl_roster_item(_.extend(item.toJSON(), {
'desc_status': STATUSES[chat_status || 'offline'],
'desc_chat': __('Click to chat with this contact'),
'desc_remove': __(___('Click to remove %1$s as a contact'), item.get('fullname')),
'title_fullname': __('Name'),
'allow_contact_removal': _converse.allow_contact_removal,
'num_unread': item.get('num_unread') || 0
})));
2017-06-23 20:25:33 +02:00
return this;
},
2017-07-22 22:21:05 +02:00
isGroupCollapsed: function isGroupCollapsed() {
2016-11-07 15:43:48 +01:00
/* Check whether the group in which this contact appears is
* collapsed.
*/
// XXX: this sucks and is fragile.
// It's because I tried to do the "right thing"
// and use definition lists to represent roster groups.
// If roster group items were inside the group elements, we
// would simplify things by not having to check whether the
// group is collapsed or not.
var name = this.$el.prevAll('dt:first').data('group');
2017-07-22 22:21:05 +02:00
var group = _.head(_converse.rosterview.model.where({ 'name': name.toString() }));
2017-02-03 13:51:07 +01:00
if (group.get('state') === _converse.CLOSED) {
2016-11-07 15:43:48 +01:00
return true;
}
return false;
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
mayBeShown: function mayBeShown() {
2016-11-07 15:43:48 +01:00
/* Return a boolean indicating whether this contact should
* generally be visible in the roster.
*
* It doesn't check for the more specific case of whether
* the group it's in is collapsed (see isGroupCollapsed).
*/
var chatStatus = this.model.get('chat_status');
2017-07-22 22:21:05 +02:00
if (_converse.show_only_online_users && chatStatus !== 'online' || _converse.hide_offline_users && chatStatus === 'offline') {
2016-11-07 15:43:48 +01:00
// If pending or requesting, show
2017-07-22 22:21:05 +02:00
if (this.model.get('ask') === 'subscribe' || this.model.get('subscription') === 'from' || this.model.get('requesting') === true) {
2016-11-07 15:43:48 +01:00
return true;
2016-03-16 12:49:35 +01:00
}
2016-11-07 15:43:48 +01:00
return false;
2016-03-16 12:49:35 +01:00
}
2016-11-07 15:43:48 +01:00
return true;
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
openChat: function openChat(ev) {
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2017-06-23 20:25:33 +02:00
return _converse.chatboxviews.showChat(this.model.attributes, true);
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
removeContact: function removeContact(ev) {
var _this = this;
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
if (ev && ev.preventDefault) {
ev.preventDefault();
}
if (!_converse.allow_contact_removal) {
return;
}
2016-11-07 15:43:48 +01:00
var result = confirm(__("Are you sure you want to remove this contact?"));
if (result === true) {
2017-07-22 22:21:05 +02:00
var iq = $iq({ type: 'set' }).c('query', { xmlns: Strophe.NS.ROSTER }).c('item', { jid: this.model.get('jid'), subscription: "remove" });
_converse.connection.sendIQ(iq, function (iq) {
_this.model.destroy();
_this.remove();
}, function (err) {
alert(__("Sorry, there was an error while trying to remove " + name + " as a contact."));
_converse.log(err, Strophe.LogLevel.ERROR);
});
2016-03-16 12:49:35 +01:00
}
},
2017-07-22 22:21:05 +02:00
acceptRequest: function acceptRequest(ev) {
var _this2 = this;
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
if (ev && ev.preventDefault) {
ev.preventDefault();
}
_converse.roster.sendContactAddIQ(this.model.get('jid'), this.model.get('fullname'), [], function () {
_this2.model.authorize().subscribe();
});
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
declineRequest: function declineRequest(ev) {
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2016-11-07 15:43:48 +01:00
var result = confirm(__("Are you sure you want to decline this contact request?"));
if (result === true) {
this.model.unauthorize().destroy();
2016-03-16 12:49:35 +01:00
}
return this;
2016-11-07 15:43:48 +01:00
}
});
2017-02-03 13:51:07 +01:00
_converse.RosterGroupView = Backbone.Overview.extend({
2016-11-07 15:43:48 +01:00
tagName: 'dt',
className: 'roster-group',
events: {
"click a.group-toggle": "toggle"
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-11-07 15:43:48 +01:00
this.model.contacts.on("add", this.addContact, this);
this.model.contacts.on("change:subscription", this.onContactSubscriptionChange, this);
this.model.contacts.on("change:requesting", this.onContactRequestChange, this);
this.model.contacts.on("change:chat_status", function (contact) {
// This might be optimized by instead of first sorting,
// finding the correct position in positionContact
this.model.contacts.sort();
this.positionContact(contact).render();
}, this);
this.model.contacts.on("destroy", this.onRemove, this);
this.model.contacts.on("remove", this.onRemove, this);
2017-02-03 13:51:07 +01:00
_converse.roster.on('change:groups', this.onContactGroupChange, this);
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
render: function render() {
2017-04-04 17:26:06 +02:00
this.el.setAttribute('data-group', this.model.get('name'));
2017-04-23 19:02:44 +02:00
var html = tpl_group_header({
label_group: this.model.get('name'),
desc_group_toggle: this.model.get('description'),
toggle_state: this.model.get('state')
});
this.el.innerHTML = html;
2016-11-07 15:43:48 +01:00
return this;
},
2017-07-22 22:21:05 +02:00
addContact: function addContact(contact) {
var view = new _converse.RosterContactView({ model: contact });
2016-11-07 15:43:48 +01:00
this.add(contact.get('id'), view);
view = this.positionContact(contact).render();
if (view.mayBeShown()) {
2017-02-03 13:51:07 +01:00
if (this.model.get('state') === _converse.CLOSED) {
2017-07-22 22:21:05 +02:00
if (view.$el[0].style.display !== "none") {
view.$el.hide();
}
if (!this.$el.is(':visible')) {
this.$el.show();
}
2016-03-16 12:49:35 +01:00
} else {
2017-07-22 22:21:05 +02:00
if (this.$el[0].style.display !== "block") {
this.show();
}
2016-03-16 12:49:35 +01:00
}
2016-11-07 15:43:48 +01:00
}
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
positionContact: function positionContact(contact) {
2016-11-07 15:43:48 +01:00
/* Place the contact's DOM element in the correct alphabetical
* position amongst the other contacts in this group.
*/
var view = this.get(contact.get('id'));
var index = this.model.contacts.indexOf(contact);
view.$el.detach();
if (index === 0) {
this.$el.after(view.$el);
2017-07-22 22:21:05 +02:00
} else if (index === this.model.contacts.length - 1) {
2016-11-07 15:43:48 +01:00
this.$el.nextUntil('dt').last().after(view.$el);
} else {
this.$el.nextUntil('dt').eq(index).before(view.$el);
}
return view;
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
show: function show() {
2016-11-07 15:43:48 +01:00
this.$el.show();
_.each(this.getAll(), function (view) {
if (view.mayBeShown() && !view.isGroupCollapsed()) {
view.$el.show();
}
});
2016-03-16 12:49:35 +01:00
return this;
},
2017-07-22 22:21:05 +02:00
hide: function hide() {
2016-11-07 15:43:48 +01:00
this.$el.nextUntil('dt').addBack().hide();
},
2017-07-22 22:21:05 +02:00
filter: function filter(q, type) {
var _this3 = this;
2016-11-07 15:43:48 +01:00
/* Filter the group's contacts based on the query "q".
* The query is matched against the contact's full name.
* If all contacts are filtered out (i.e. hidden), then the
* group must be filtered out as well.
*/
2017-07-22 22:21:05 +02:00
var matches = void 0;
2016-11-07 15:43:48 +01:00
if (q.length === 0) {
2017-02-03 13:51:07 +01:00
if (this.model.get('state') === _converse.OPENED) {
2016-11-07 15:43:48 +01:00
this.model.contacts.each(function (item) {
2017-07-22 22:21:05 +02:00
var view = _this3.get(item.get('id'));
2016-11-07 15:43:48 +01:00
if (view.mayBeShown() && !view.isGroupCollapsed()) {
view.$el.show();
}
2017-07-22 22:21:05 +02:00
});
2016-11-07 15:43:48 +01:00
}
this.showIfNecessary();
} else {
q = q.toLowerCase();
if (type === 'state') {
if (this.model.get('name') === HEADER_REQUESTING_CONTACTS) {
// When filtering by chat state, we still want to
// show requesting contacts, even though they don't
// have the state in question.
2017-07-22 22:21:05 +02:00
matches = this.model.contacts.filter(function (contact) {
return utils.contains.not('chat_status', q)(contact) && !contact.get('requesting');
});
2017-06-23 20:25:33 +02:00
} else if (q === 'unread_messages') {
2017-07-22 22:21:05 +02:00
matches = this.model.contacts.filter({ 'num_unread': 0 });
2016-11-07 15:43:48 +01:00
} else {
2017-07-22 22:21:05 +02:00
matches = this.model.contacts.filter(utils.contains.not('chat_status', q));
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
} else {
matches = this.model.contacts.filter(utils.contains.not('fullname', q));
2016-11-07 15:43:48 +01:00
}
if (matches.length === this.model.contacts.length) {
// hide the whole group
this.hide();
} else {
_.each(matches, function (item) {
2017-07-22 22:21:05 +02:00
_this3.get(item.get('id')).$el.hide();
});
if (this.model.get('state') === _converse.OPENED) {
_.each(this.model.contacts.reject(utils.contains.not('fullname', q)), function (item) {
_this3.get(item.get('id')).$el.show();
});
}
2016-11-07 15:43:48 +01:00
this.showIfNecessary();
}
}
},
2017-07-22 22:21:05 +02:00
showIfNecessary: function showIfNecessary() {
2016-11-07 15:43:48 +01:00
if (!this.$el.is(':visible') && this.model.contacts.length > 0) {
this.$el.show();
2016-03-16 12:49:35 +01:00
}
},
2017-07-22 22:21:05 +02:00
toggle: function toggle(ev) {
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2016-11-07 15:43:48 +01:00
var $el = $(ev.target);
if ($el.hasClass("icon-opened")) {
this.$el.nextUntil('dt').slideUp();
2017-07-22 22:21:05 +02:00
this.model.save({ state: _converse.CLOSED });
2016-11-07 15:43:48 +01:00
$el.removeClass("icon-opened").addClass("icon-closed");
2016-06-20 21:11:43 +02:00
} else {
2016-11-07 15:43:48 +01:00
$el.removeClass("icon-closed").addClass("icon-opened");
2017-07-22 22:21:05 +02:00
this.model.save({ state: _converse.OPENED });
this.filter(_converse.rosterview.$('.roster-filter').val() || '', _converse.rosterview.$('.filter-type').val());
2016-06-20 21:11:43 +02:00
}
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
onContactGroupChange: function onContactGroupChange(contact) {
var in_this_group = _.includes(contact.get('groups'), this.model.get('name'));
2016-11-07 15:43:48 +01:00
var cid = contact.get('id');
var in_this_overview = !this.get(cid);
if (in_this_group && !in_this_overview) {
this.model.contacts.remove(cid);
} else if (!in_this_group && in_this_overview) {
this.addContact(contact);
}
},
2017-07-22 22:21:05 +02:00
onContactSubscriptionChange: function onContactSubscriptionChange(contact) {
if (this.model.get('name') === HEADER_PENDING_CONTACTS && contact.get('subscription') !== 'from') {
2016-11-07 15:43:48 +01:00
this.model.contacts.remove(contact.get('id'));
}
2016-06-20 21:11:43 +02:00
},
2017-07-22 22:21:05 +02:00
onContactRequestChange: function onContactRequestChange(contact) {
if (this.model.get('name') === HEADER_REQUESTING_CONTACTS && !contact.get('requesting')) {
2016-11-07 15:43:48 +01:00
/* We suppress events, otherwise the remove event will
* also cause the contact's view to be removed from the
* "Pending Contacts" group.
*/
2017-07-22 22:21:05 +02:00
this.model.contacts.remove(contact.get('id'), { 'silent': true });
2016-11-07 15:43:48 +01:00
// Since we suppress events, we make sure the view and
// contact are removed from this group.
this.get(contact.get('id')).remove();
this.onRemove(contact);
2016-03-16 12:49:35 +01:00
}
},
2017-07-22 22:21:05 +02:00
onRemove: function onRemove(contact) {
2016-11-07 15:43:48 +01:00
this.remove(contact.get('id'));
if (this.model.contacts.length === 0) {
this.$el.hide();
2016-03-16 12:49:35 +01:00
}
}
});
2016-11-07 15:43:48 +01:00
/* -------- Event Handlers ----------- */
2017-07-22 22:21:05 +02:00
var onChatBoxMaximized = function onChatBoxMaximized(chatboxview) {
2017-06-23 20:25:33 +02:00
/* When a chat box gets maximized, the num_unread counter needs
* to be cleared, but if chatbox is scrolled up, then num_unread should not be cleared.
*/
var chatbox = chatboxview.model;
if (chatbox.get('type') !== 'chatroom') {
2017-07-22 22:21:05 +02:00
var contact = _.head(_converse.roster.where({ 'jid': chatbox.get('jid') }));
2017-06-23 20:25:33 +02:00
if (!_.isUndefined(contact) && !chatbox.isScrolledUp()) {
2017-07-22 22:21:05 +02:00
contact.save({ 'num_unread': 0 });
2017-06-23 20:25:33 +02:00
}
}
};
2017-07-22 22:21:05 +02:00
var onMessageReceived = function onMessageReceived(data) {
2017-06-23 20:25:33 +02:00
/* Given a newly received message, update the unread counter on
* the relevant roster contact.
*/
var chatbox = data.chatbox;
2017-07-22 22:21:05 +02:00
2017-06-23 20:25:33 +02:00
if (_.isUndefined(chatbox)) {
return;
}
if (_.isNull(data.stanza.querySelector('body'))) {
return; // The message has no text
}
2017-07-22 22:21:05 +02:00
if (chatbox.get('type') !== 'chatroom' && utils.isNewMessage(data.stanza) && chatbox.newMessageWillBeHidden()) {
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var contact = _.head(_converse.roster.where({ 'jid': chatbox.get('jid') }));
2017-06-23 20:25:33 +02:00
if (!_.isUndefined(contact)) {
2017-07-22 22:21:05 +02:00
contact.save({ 'num_unread': contact.get('num_unread') + 1 });
2017-06-23 20:25:33 +02:00
}
}
};
2017-07-22 22:21:05 +02:00
var onChatBoxScrolledDown = function onChatBoxScrolledDown(data) {
2017-06-23 20:25:33 +02:00
var chatbox = data.chatbox;
2017-07-22 22:21:05 +02:00
2017-06-23 20:25:33 +02:00
if (_.isUndefined(chatbox)) {
return;
}
2017-07-22 22:21:05 +02:00
var contact = _.head(_converse.roster.where({ 'jid': chatbox.get('jid') }));
2017-06-23 20:25:33 +02:00
if (!_.isUndefined(contact)) {
2017-07-22 22:21:05 +02:00
contact.save({ 'num_unread': 0 });
2017-06-23 20:25:33 +02:00
}
};
2017-07-22 22:21:05 +02:00
var initRoster = function initRoster() {
2016-11-07 15:43:48 +01:00
/* Create an instance of RosterView once the RosterGroups
* collection has been created (in converse-core.js)
*/
2017-02-03 13:51:07 +01:00
_converse.rosterview = new _converse.RosterView({
'model': _converse.rostergroups
2016-11-07 15:43:48 +01:00
});
2017-02-03 13:51:07 +01:00
_converse.rosterview.render();
2016-11-07 15:43:48 +01:00
};
2017-06-23 20:25:33 +02:00
_converse.api.listen.on('rosterInitialized', initRoster);
_converse.api.listen.on('rosterReadyAfterReconnection', initRoster);
_converse.api.listen.on('message', onMessageReceived);
_converse.api.listen.on('chatBoxMaximized', onChatBoxMaximized);
_converse.api.listen.on('chatBoxScrolledDown', onChatBoxScrolledDown);
2016-03-16 12:49:35 +01:00
}
});
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-rosterview.js.map;
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
2017-02-03 13:51:07 +01:00
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
// Licensed under the Mozilla Public License (MPLv2)
//
2017-04-23 19:02:44 +02:00
/*global define */
(function (root, factory) {
2017-07-22 22:21:05 +02:00
define('converse-controlbox',["jquery.noconflict", "converse-core", "tpl!add_contact_dropdown", "tpl!add_contact_form", "tpl!change_status_message", "tpl!chat_status", "tpl!choose_status", "tpl!contacts_panel", "tpl!contacts_tab", "tpl!controlbox", "tpl!controlbox_toggle", "tpl!login_panel", "tpl!login_tab", "tpl!search_contact", "tpl!status_option", "converse-chatview", "converse-rosterview"], factory);
})(undefined, function ($, converse, tpl_add_contact_dropdown, tpl_add_contact_form, tpl_change_status_message, tpl_chat_status, tpl_choose_status, tpl_contacts_panel, tpl_contacts_tab, tpl_controlbox, tpl_controlbox_toggle, tpl_login_panel, tpl_login_tab, tpl_search_contact, tpl_status_option) {
2016-03-07 18:54:07 +01:00
"use strict";
2016-11-07 15:43:48 +01:00
var USERS_PANEL_ID = 'users';
2017-06-23 20:25:33 +02:00
var CHATBOX_TYPE = 'chatbox';
2017-07-22 22:21:05 +02:00
var _converse$env = converse.env,
Strophe = _converse$env.Strophe,
Backbone = _converse$env.Backbone,
utils = _converse$env.utils,
_ = _converse$env._,
fp = _converse$env.fp,
moment = _converse$env.moment;
2016-03-07 18:54:07 +01:00
2016-11-07 15:43:48 +01:00
2017-02-03 13:51:07 +01:00
converse.plugins.add('converse-controlbox', {
2016-03-07 18:54:07 +01:00
2016-03-16 12:49:35 +01:00
overrides: {
// Overrides mentioned here will be picked up by converse.js's
// plugin architecture they will replace existing methods on the
// relevant objects or classes.
//
// New functions which don't exist yet can also be added.
2016-07-26 08:00:30 +02:00
2017-07-22 22:21:05 +02:00
initChatBoxes: function initChatBoxes() {
2017-06-23 20:25:33 +02:00
this.__super__.initChatBoxes.apply(this, arguments);
2016-11-07 15:43:48 +01:00
this.controlboxtoggle = new this.ControlBoxToggle();
},
2017-07-22 22:21:05 +02:00
initConnection: function initConnection() {
2016-11-07 15:43:48 +01:00
this.__super__.initConnection.apply(this, arguments);
if (this.connection) {
this.addControlBox();
2016-07-26 08:00:30 +02:00
}
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
_tearDown: function _tearDown() {
2016-11-07 15:43:48 +01:00
this.__super__._tearDown.apply(this, arguments);
if (this.rosterview) {
// Removes roster groups
this.rosterview.model.off().reset();
this.rosterview.each(function (groupview) {
groupview.removeAll();
groupview.remove();
2016-05-03 17:37:10 +02:00
});
2016-11-07 15:43:48 +01:00
this.rosterview.removeAll().remove();
}
},
2017-07-22 22:21:05 +02:00
clearSession: function clearSession() {
2016-11-07 15:43:48 +01:00
this.__super__.clearSession.apply(this, arguments);
2017-06-23 20:25:33 +02:00
var controlbox = this.chatboxes.get('controlbox');
2017-07-22 22:21:05 +02:00
if (controlbox && controlbox.collection && controlbox.collection.browserStorage) {
controlbox.save({ 'connected': false });
2016-11-07 15:43:48 +01:00
}
},
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
ChatBoxes: {
2017-07-22 22:21:05 +02:00
chatBoxMayBeShown: function chatBoxMayBeShown(chatbox) {
return this.__super__.chatBoxMayBeShown.apply(this, arguments) && chatbox.get('id') !== 'controlbox';
2016-05-03 17:37:10 +02:00
},
2017-07-22 22:21:05 +02:00
onChatBoxesFetched: function onChatBoxesFetched(collection, resp) {
2017-03-05 10:45:18 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
this.__super__.onChatBoxesFetched.apply(this, arguments);
2017-03-05 10:45:18 +01:00
if (!_.includes(_.map(collection, 'id'), 'controlbox')) {
_converse.addControlBox();
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
this.get('controlbox').save({ connected: true });
}
2016-11-07 15:43:48 +01:00
},
2016-05-03 17:37:10 +02:00
2016-11-07 15:43:48 +01:00
ChatBoxViews: {
2017-07-22 22:21:05 +02:00
onChatBoxAdded: function onChatBoxAdded(item) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
if (item.get('box_id') === 'controlbox') {
var view = this.get(item.get('id'));
if (view) {
view.model = item;
view.initialize();
return view;
} else {
2017-07-22 22:21:05 +02:00
view = new _converse.ControlBoxView({ model: item });
2016-11-07 15:43:48 +01:00
return this.add(item.get('id'), view);
}
2016-05-03 17:37:10 +02:00
} else {
2016-11-07 15:43:48 +01:00
return this.__super__.onChatBoxAdded.apply(this, arguments);
2016-05-03 17:37:10 +02:00
}
},
2017-07-22 22:21:05 +02:00
closeAllChatBoxes: function closeAllChatBoxes() {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
this.each(function (view) {
2017-07-22 22:21:05 +02:00
if (view.model.get('id') === 'controlbox' && (_converse.disconnection_cause !== _converse.LOGOUT || _converse.show_controlbox_by_default)) {
return;
2016-11-07 15:43:48 +01:00
}
view.close();
2016-05-03 17:37:10 +02:00
});
2016-11-07 15:43:48 +01:00
return this;
},
2017-07-22 22:21:05 +02:00
getChatBoxWidth: function getChatBoxWidth(view) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
var controlbox = this.get('controlbox');
if (view.model.get('id') === 'controlbox') {
/* We return the width of the controlbox or its toggle,
* depending on which is visible.
*/
if (!controlbox || !controlbox.$el.is(':visible')) {
2017-02-03 13:51:07 +01:00
return _converse.controlboxtoggle.$el.outerWidth(true);
2016-11-07 15:43:48 +01:00
} else {
return controlbox.$el.outerWidth(true);
}
} else {
return this.__super__.getChatBoxWidth.apply(this, arguments);
2016-05-24 10:23:19 +02:00
}
2016-11-07 15:43:48 +01:00
}
},
2016-05-24 10:23:19 +02:00
2016-11-07 15:43:48 +01:00
ChatBox: {
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-11-07 15:43:48 +01:00
if (this.get('id') === 'controlbox') {
2017-07-22 22:21:05 +02:00
this.set({ 'time_opened': moment(0).valueOf() });
2016-11-07 15:43:48 +01:00
} else {
this.__super__.initialize.apply(this, arguments);
2016-05-03 17:37:10 +02:00
}
2017-07-22 22:21:05 +02:00
}
2016-11-07 15:43:48 +01:00
},
2016-05-03 17:37:10 +02:00
2016-11-07 15:43:48 +01:00
ChatBoxView: {
2017-07-22 22:21:05 +02:00
insertIntoDOM: function insertIntoDOM() {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2017-02-03 13:51:07 +01:00
this.$el.insertAfter(_converse.chatboxviews.get("controlbox").$el);
2016-11-07 15:43:48 +01:00
return this;
2016-05-03 17:37:10 +02:00
}
2016-11-07 15:43:48 +01:00
}
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-11-07 15:43:48 +01:00
/* The initialize function gets called as soon as the plugin is
* loaded by converse.js's plugin machinery.
*/
2017-02-03 13:51:07 +01:00
var _converse = this._converse,
__ = _converse.__;
2017-07-22 22:21:05 +02:00
2017-07-05 11:59:55 +02:00
_converse.api.settings.update({
2016-11-07 15:43:48 +01:00
allow_logout: true,
default_domain: undefined,
show_controlbox_by_default: false,
sticky_controlbox: false,
xhr_user_search: false,
xhr_user_search_url: ''
2016-05-03 17:37:10 +02:00
});
2016-11-07 15:43:48 +01:00
var LABEL_CONTACTS = __('Contacts');
2017-02-03 13:51:07 +01:00
_converse.addControlBox = function () {
return _converse.chatboxes.add({
2016-11-07 15:43:48 +01:00
id: 'controlbox',
box_id: 'controlbox',
2017-07-22 22:21:05 +02:00
type: 'controlbox',
2017-02-03 13:51:07 +01:00
closed: !_converse.show_controlbox_by_default
2016-11-07 15:43:48 +01:00
});
};
2017-02-03 13:51:07 +01:00
_converse.ControlBoxView = _converse.ChatBoxView.extend({
2016-05-03 17:37:10 +02:00
tagName: 'div',
2016-11-07 15:43:48 +01:00
className: 'chatbox',
id: 'controlbox',
events: {
'click a.close-chatbox-button': 'close',
2017-07-22 22:21:05 +02:00
'click ul#controlbox-tabs li a': 'switchTab'
2016-11-07 15:43:48 +01:00
},
2016-05-03 17:37:10 +02:00
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2017-02-03 13:51:07 +01:00
this.$el.insertAfter(_converse.controlboxtoggle.$el);
2016-11-07 15:43:48 +01:00
this.model.on('change:connected', this.onConnected, this);
this.model.on('destroy', this.hide, this);
this.model.on('hide', this.hide, this);
this.model.on('show', this.show, this);
this.model.on('change:closed', this.ensureClosedState, this);
this.render();
if (this.model.get('connected')) {
this.insertRoster();
}
},
2017-07-22 22:21:05 +02:00
render: function render() {
if (this.model.get('connected')) {
if (_.isUndefined(this.model.get('closed'))) {
this.model.set('closed', !_converse.show_controlbox_by_default);
}
2016-11-07 15:43:48 +01:00
}
if (!this.model.get('closed')) {
this.show();
} else {
this.hide();
}
2017-07-22 22:21:05 +02:00
this.el.innerHTML = tpl_controlbox(_.extend(this.model.toJSON(), {
'sticky_controlbox': _converse.sticky_controlbox
}));
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (!_converse.connection.connected || !_converse.connection.authenticated || _converse.connection.disconnecting) {
2016-11-07 15:43:48 +01:00
this.renderLoginPanel();
2017-07-22 22:21:05 +02:00
} else if (this.model.get('connected') && (!this.contactspanel || !this.contactspanel.$el.is(':visible'))) {
2016-11-07 15:43:48 +01:00
this.renderContactsPanel();
2016-05-03 17:37:10 +02:00
}
return this;
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
onConnected: function onConnected() {
2016-11-07 15:43:48 +01:00
if (this.model.get('connected')) {
this.render().insertRoster();
2017-03-05 10:45:18 +01:00
this.model.save();
2016-11-07 15:43:48 +01:00
}
},
2017-07-22 22:21:05 +02:00
insertRoster: function insertRoster() {
2016-11-07 15:43:48 +01:00
/* Place the rosterview inside the "Contacts" panel.
2016-05-03 17:37:10 +02:00
*/
2017-02-03 13:51:07 +01:00
this.contactspanel.$el.append(_converse.rosterview.$el);
2016-11-07 15:43:48 +01:00
return this;
},
2017-07-22 22:21:05 +02:00
renderLoginPanel: function renderLoginPanel() {
2017-02-03 13:51:07 +01:00
this.loginpanel = new _converse.LoginPanel({
2016-11-07 15:43:48 +01:00
'$parent': this.$el.find('.controlbox-panes'),
'model': this
});
this.loginpanel.render();
return this;
},
2017-07-22 22:21:05 +02:00
renderContactsPanel: function renderContactsPanel() {
2016-11-07 15:43:48 +01:00
if (_.isUndefined(this.model.get('active-panel'))) {
2017-07-22 22:21:05 +02:00
this.model.save({ 'active-panel': USERS_PANEL_ID });
2016-11-07 15:43:48 +01:00
}
2017-02-03 13:51:07 +01:00
this.contactspanel = new _converse.ContactsPanel({
2016-11-07 15:43:48 +01:00
'$parent': this.$el.find('.controlbox-panes')
});
2017-06-23 20:25:33 +02:00
this.contactspanel.insertIntoDOM();
2017-02-03 13:51:07 +01:00
_converse.xmppstatusview = new _converse.XMPPStatusView({
'model': _converse.xmppstatus
2016-11-07 15:43:48 +01:00
});
2017-02-03 13:51:07 +01:00
_converse.xmppstatusview.render();
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
close: function close(ev) {
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2017-06-23 20:25:33 +02:00
if (_converse.sticky_controlbox) {
return;
}
2017-02-03 13:51:07 +01:00
if (_converse.connection.connected && !_converse.connection.disconnecting) {
2017-07-22 22:21:05 +02:00
this.model.save({ 'closed': true });
2016-11-07 15:43:48 +01:00
} else {
this.model.trigger('hide');
2016-03-07 18:54:07 +01:00
}
2017-02-03 13:51:07 +01:00
_converse.emit('controlBoxClosed', this);
2016-11-07 15:43:48 +01:00
return this;
},
2017-07-22 22:21:05 +02:00
ensureClosedState: function ensureClosedState() {
2016-11-07 15:43:48 +01:00
if (this.model.get('closed')) {
this.hide();
} else {
this.show();
2016-03-16 12:49:35 +01:00
}
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
hide: function hide(callback) {
2017-06-23 20:25:33 +02:00
if (_converse.sticky_controlbox) {
return;
}
2016-11-30 17:27:20 +01:00
this.$el.addClass('hidden');
utils.refreshWebkit();
2017-02-03 13:51:07 +01:00
_converse.emit('chatBoxClosed', this);
if (!_converse.connection.connected) {
_converse.controlboxtoggle.render();
2016-11-30 17:27:20 +01:00
}
2017-02-03 13:51:07 +01:00
_converse.controlboxtoggle.show(callback);
2016-11-07 15:43:48 +01:00
return this;
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
onControlBoxToggleHidden: function onControlBoxToggleHidden() {
2016-11-07 15:43:48 +01:00
var that = this;
2016-11-30 17:27:20 +01:00
utils.fadeIn(this.el, function () {
2017-02-03 13:51:07 +01:00
_converse.controlboxtoggle.updateOnlineCount();
2016-11-07 15:43:48 +01:00
utils.refreshWebkit();
2017-03-05 10:45:18 +01:00
that.model.set('closed', false);
2017-02-03 13:51:07 +01:00
_converse.emit('controlBoxOpened', that);
2016-11-07 15:43:48 +01:00
});
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
show: function show() {
_converse.controlboxtoggle.hide(this.onControlBoxToggleHidden.bind(this));
2016-11-07 15:43:48 +01:00
return this;
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
switchTab: function switchTab(ev) {
2016-11-07 15:43:48 +01:00
// TODO: automatically focus the relevant input
2017-07-22 22:21:05 +02:00
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2016-11-07 15:43:48 +01:00
var $tab = $(ev.target),
$sibling = $tab.parent().siblings('li').children('a'),
$tab_panel = $($tab.attr('href'));
$($sibling.attr('href')).addClass('hidden');
$sibling.removeClass('current');
$tab.addClass('current');
$tab_panel.removeClass('hidden');
2017-04-04 17:26:06 +02:00
if (!_.isUndefined(_converse.chatboxes.browserStorage)) {
2017-07-22 22:21:05 +02:00
this.model.save({ 'active-panel': $tab.data('id') });
2016-11-07 15:43:48 +01:00
}
return this;
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
showHelpMessages: function showHelpMessages() {
/* Override showHelpMessages in ChatBoxView, for now do nothing.
*
* Parameters:
* (Array) msgs: Array of messages
*/
2016-11-07 15:43:48 +01:00
return;
}
});
2017-02-03 13:51:07 +01:00
_converse.LoginPanel = Backbone.View.extend({
2016-11-07 15:43:48 +01:00
tagName: 'div',
id: "login-dialog",
className: 'controlbox-pane',
events: {
'submit form#converse-login': 'authenticate'
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
initialize: function initialize(cfg) {
cfg.$parent.html(this.$el.html(tpl_login_panel({
'ANONYMOUS': _converse.ANONYMOUS,
'EXTERNAL': _converse.EXTERNAL,
'LOGIN': _converse.LOGIN,
'PREBIND': _converse.PREBIND,
'auto_login': _converse.auto_login,
'authentication': _converse.authentication,
'label_username': __('XMPP Username:'),
'label_password': __('Password:'),
'label_anon_login': __('Click here to log in anonymously'),
'label_login': __('Log In'),
'placeholder_username': (_converse.locked_domain || _converse.default_domain) && __('Username') || __('user@server'),
'placeholder_password': __('password')
})));
2016-11-07 15:43:48 +01:00
this.$tabs = cfg.$parent.parent().find('#controlbox-tabs');
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
render: function render() {
this.$tabs.append(tpl_login_tab({ label_sign_in: __('Sign in') }));
2016-11-07 15:43:48 +01:00
this.$el.find('input#jid').focus();
if (!this.$el.is(':visible')) {
this.$el.show();
2016-03-07 18:54:07 +01:00
}
2016-11-07 15:43:48 +01:00
return this;
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
authenticate: function authenticate(ev) {
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2016-11-07 15:43:48 +01:00
var $form = $(ev.target);
2017-02-03 13:51:07 +01:00
if (_converse.authentication === _converse.ANONYMOUS) {
this.connect($form, _converse.jid, null);
2016-11-07 15:43:48 +01:00
return;
2016-03-16 12:49:35 +01:00
}
2017-07-22 22:21:05 +02:00
var $jid_input = $form.find('input[name=jid]');
var $pw_input = $form.find('input[name=password]');
var password = $pw_input.val();
var jid = $jid_input.val(),
2016-11-07 15:43:48 +01:00
errors = false;
2017-07-22 22:21:05 +02:00
if (!jid || _.filter(jid.split('@')).length < 2) {
2016-11-07 15:43:48 +01:00
errors = true;
$jid_input.addClass('error');
2016-03-16 12:49:35 +01:00
}
2017-07-22 22:21:05 +02:00
if (!password && _converse.authentication !== _converse.EXTERNAL) {
2016-11-07 15:43:48 +01:00
errors = true;
$pw_input.addClass('error');
2016-03-16 12:49:35 +01:00
}
2017-07-22 22:21:05 +02:00
if (errors) {
return;
}
2017-02-03 13:51:07 +01:00
if (_converse.locked_domain) {
jid = Strophe.escapeNode(jid) + '@' + _converse.locked_domain;
} else if (_converse.default_domain && !_.includes(jid, '@')) {
jid = jid + '@' + _converse.default_domain;
2016-11-07 15:43:48 +01:00
}
this.connect($form, jid, password);
return false;
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
connect: function connect($form, jid, password) {
var resource = void 0;
2016-11-07 15:43:48 +01:00
if ($form) {
$form.find('input[type=submit]').hide().after('<span class="spinner login-submit"/>');
}
if (jid) {
resource = Strophe.getResourceFromJid(jid);
if (!resource) {
2017-02-03 13:51:07 +01:00
jid = jid.toLowerCase() + _converse.generateResource();
2016-03-16 12:49:35 +01:00
} else {
2017-07-22 22:21:05 +02:00
jid = Strophe.getBareJidFromJid(jid).toLowerCase() + '/' + resource;
2016-03-16 12:49:35 +01:00
}
2016-11-07 15:43:48 +01:00
}
2017-04-04 17:26:06 +02:00
_converse.connection.reset();
2017-02-03 13:51:07 +01:00
_converse.connection.connect(jid, password, _converse.onConnectStatusChanged);
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
remove: function remove() {
2016-11-07 15:43:48 +01:00
this.$tabs.empty();
this.$el.parent().empty();
}
});
2017-02-03 13:51:07 +01:00
_converse.XMPPStatusView = Backbone.View.extend({
2017-06-23 20:25:33 +02:00
el: "form#set-xmpp-status",
2016-11-07 15:43:48 +01:00
events: {
"click a.choose-xmpp-status": "toggleOptions",
"click #fancy-xmpp-status-select a.change-xmpp-status-message": "renderStatusChangeForm",
2017-06-23 20:25:33 +02:00
"submit": "setStatusMessage",
2016-11-07 15:43:48 +01:00
"click .dropdown dd ul li a": "setStatus"
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-11-07 15:43:48 +01:00
this.model.on("change:status", this.updateStatusUI, this);
this.model.on("change:status_message", this.updateStatusUI, this);
this.model.on("update-status-ui", this.updateStatusUI, this);
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
render: function render() {
2016-11-07 15:43:48 +01:00
// Replace the default dropdown with something nicer
2017-07-22 22:21:05 +02:00
var $select = this.$el.find('select#select-xmpp-status');
var chat_status = this.model.get('status') || 'offline';
var options = $('option', $select);
var options_list = [];
this.$el.html(tpl_choose_status());
2017-07-22 22:21:05 +02:00
this.$el.find('#fancy-xmpp-status-select').html(tpl_chat_status({
'status_message': this.model.get('status_message') || __("I am %1$s", this.getPrettyStatus(chat_status)),
'chat_status': chat_status,
'desc_custom_status': __('Click here to write a custom status message'),
'desc_change_status': __('Click to change your chat status')
}));
2016-11-07 15:43:48 +01:00
// iterate through all the <option> elements and add option values
options.each(function () {
options_list.push(tpl_status_option({
2016-11-07 15:43:48 +01:00
'value': $(this).val(),
'text': this.text
}));
});
2017-07-22 22:21:05 +02:00
var $options_target = this.$el.find("#target dd ul").hide();
2016-11-07 15:43:48 +01:00
$options_target.append(options_list.join(''));
$select.remove();
2016-03-07 18:54:07 +01:00
return this;
},
2017-07-22 22:21:05 +02:00
toggleOptions: function toggleOptions(ev) {
2016-11-07 15:43:48 +01:00
ev.preventDefault();
2017-07-22 22:21:05 +02:00
utils.slideInAllElements(document.querySelectorAll('#conversejs .contact-form-container'));
2016-11-07 15:43:48 +01:00
$(ev.target).parent().parent().siblings('dd').find('ul').toggle('fast');
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
renderStatusChangeForm: function renderStatusChangeForm(ev) {
2016-11-07 15:43:48 +01:00
ev.preventDefault();
2017-02-13 17:16:13 +01:00
var status_message = _converse.xmppstatus.get('status_message') || '';
var input = tpl_change_status_message({
2016-11-07 15:43:48 +01:00
'status_message': status_message,
'label_custom_status': __('Custom status'),
'label_save': __('Save')
});
var $xmppstatus = this.$el.find('.xmpp-status');
$xmppstatus.parent().addClass('no-border');
$xmppstatus.replaceWith(input);
this.$el.find('.custom-xmpp-status').focus().focus();
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
setStatusMessage: function setStatusMessage(ev) {
2016-11-07 15:43:48 +01:00
ev.preventDefault();
this.model.setStatusMessage($(ev.target).find('input').val());
},
2017-07-22 22:21:05 +02:00
setStatus: function setStatus(ev) {
2016-11-07 15:43:48 +01:00
ev.preventDefault();
var $el = $(ev.currentTarget),
value = $el.attr('data-value');
if (value === 'logout') {
this.$el.find(".dropdown dd ul").hide();
2017-02-03 13:51:07 +01:00
_converse.logOut();
2016-03-16 12:49:35 +01:00
} else {
2016-11-07 15:43:48 +01:00
this.model.setStatus(value);
this.$el.find(".dropdown dd ul").hide();
2016-03-07 18:54:07 +01:00
}
},
2017-07-22 22:21:05 +02:00
getPrettyStatus: function getPrettyStatus(stat) {
2016-11-07 15:43:48 +01:00
if (stat === 'chat') {
return __('online');
} else if (stat === 'dnd') {
return __('busy');
} else if (stat === 'xa') {
return __('away for long');
} else if (stat === 'away') {
return __('away');
} else if (stat === 'offline') {
return __('offline');
2016-03-16 12:49:35 +01:00
} else {
2016-11-07 15:43:48 +01:00
return __(stat) || __('online');
2016-03-16 12:49:35 +01:00
}
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
updateStatusUI: function updateStatusUI(model) {
2016-11-07 15:43:48 +01:00
var stat = model.get('status');
// For translators: the %1$s part gets replaced with the status
// Example, I am online
var status_message = model.get('status_message') || __("I am %1$s", this.getPrettyStatus(stat));
2017-07-22 22:21:05 +02:00
this.$el.find('#fancy-xmpp-status-select').removeClass('no-border').html(tpl_chat_status({
'chat_status': stat,
'status_message': status_message,
'desc_custom_status': __('Click here to write a custom status message'),
'desc_change_status': __('Click to change your chat status')
}));
2016-03-07 18:54:07 +01:00
}
});
2017-02-03 13:51:07 +01:00
_converse.ContactsPanel = Backbone.View.extend({
2016-11-07 15:43:48 +01:00
tagName: 'div',
className: 'controlbox-pane',
id: 'users',
2016-03-07 18:54:07 +01:00
events: {
2016-11-07 15:43:48 +01:00
'click a.toggle-xmpp-contact-form': 'toggleContactForm',
'submit form.add-xmpp-contact': 'addContactFromForm',
'submit form.search-xmpp-contact': 'searchContacts',
'click a.subscribe-to-user': 'addContactFromList'
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
initialize: function initialize(cfg) {
2017-06-23 20:25:33 +02:00
this.parent_el = cfg.$parent[0];
this.tab_el = document.createElement('li');
_converse.chatboxes.on('change:num_unread', this.renderTab, this);
_converse.chatboxes.on('add', _.debounce(this.renderTab, 100), this);
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
render: function render() {
2017-06-23 20:25:33 +02:00
this.renderTab();
var widgets = tpl_contacts_panel({
2016-11-07 15:43:48 +01:00
label_online: __('Online'),
label_busy: __('Busy'),
label_away: __('Away'),
label_offline: __('Offline'),
label_logout: __('Log out'),
2017-02-03 13:51:07 +01:00
include_offline_state: _converse.include_offline_state,
allow_logout: _converse.allow_logout
2016-11-07 15:43:48 +01:00
});
2017-06-23 20:25:33 +02:00
if (_converse.allow_contact_requests) {
widgets += tpl_add_contact_dropdown({
label_click_to_chat: __('Click to add new chat contacts'),
label_add_contact: __('Add a contact')
});
}
this.el.innerHTML = widgets;
2017-02-03 13:51:07 +01:00
var controlbox = _converse.chatboxes.get('controlbox');
2017-06-23 20:25:33 +02:00
if (controlbox.get('active-panel') !== USERS_PANEL_ID) {
this.el.classList.add('hidden');
}
return this;
},
2017-07-22 22:21:05 +02:00
renderTab: function renderTab() {
2017-06-23 20:25:33 +02:00
var controlbox = _converse.chatboxes.get('controlbox');
var chats = fp.filter(_.partial(utils.isOfType, CHATBOX_TYPE), _converse.chatboxes.models);
this.tab_el.innerHTML = tpl_contacts_tab({
2016-11-07 15:43:48 +01:00
'label_contacts': LABEL_CONTACTS,
2017-06-23 20:25:33 +02:00
'is_current': controlbox.get('active-panel') === USERS_PANEL_ID,
'num_unread': fp.sum(fp.map(fp.curry(utils.getAttribute)('num_unread'), chats))
});
},
2017-07-22 22:21:05 +02:00
insertIntoDOM: function insertIntoDOM() {
2017-06-23 20:25:33 +02:00
this.parent_el.appendChild(this.render().el);
this.tabs = this.parent_el.parentNode.querySelector('#controlbox-tabs');
this.tabs.appendChild(this.tab_el);
return this;
},
2017-07-22 22:21:05 +02:00
generateAddContactHTML: function generateAddContactHTML() {
var settings = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
2017-06-23 20:25:33 +02:00
2017-02-03 13:51:07 +01:00
if (_converse.xhr_user_search) {
2017-06-23 20:25:33 +02:00
return tpl_search_contact({
2016-11-07 15:43:48 +01:00
label_contact_name: __('Contact name'),
label_search: __('Search')
});
} else {
2017-07-22 22:21:05 +02:00
return tpl_add_contact_form(_.assign({
error_message: null,
2016-11-07 15:43:48 +01:00
label_contact_username: __('e.g. user@example.org'),
2017-07-22 22:21:05 +02:00
label_add: __('Add'),
value: ''
}, settings));
2016-03-07 18:54:07 +01:00
}
},
2017-07-22 22:21:05 +02:00
toggleContactForm: function toggleContactForm(ev) {
2016-11-07 15:43:48 +01:00
ev.preventDefault();
2017-07-22 22:21:05 +02:00
this.el.querySelector('.search-xmpp div').innerHTML = this.generateAddContactHTML();
var dropdown = this.el.querySelector('.contact-form-container');
utils.slideToggleElement(dropdown).then(function () {
if ($(dropdown).is(':visible')) {
$(dropdown).find('input.username').focus();
2016-11-07 15:43:48 +01:00
}
});
},
2017-07-22 22:21:05 +02:00
searchContacts: function searchContacts(ev) {
2016-11-07 15:43:48 +01:00
ev.preventDefault();
2017-07-22 22:21:05 +02:00
$.getJSON(_converse.xhr_user_search_url + "?q=" + $(ev.target).find('input.username').val(), function (data) {
var $ul = $('.search-xmpp ul');
2016-11-07 15:43:48 +01:00
$ul.find('li.found-user').remove();
$ul.find('li.chat-info').remove();
if (!data.length) {
2017-07-22 22:21:05 +02:00
$ul.append("<li class=\"chat-info\">" + __('No users found') + "</li>");
2016-11-07 15:43:48 +01:00
}
$(data).each(function (idx, obj) {
2017-07-22 22:21:05 +02:00
$ul.append($('<li class="found-user"></li>').append($("<a class=\"subscribe-to-user\" href=\"#\" title=\"" + __('Click to add as a chat contact') + "\"></a>").attr('data-recipient', Strophe.getNodeFromJid(obj.id) + "@" + Strophe.getDomainFromJid(obj.id)).text(obj.fullname)));
2016-11-07 15:43:48 +01:00
});
});
},
2017-07-22 22:21:05 +02:00
addContactFromForm: function addContactFromForm(ev) {
2016-11-07 15:43:48 +01:00
ev.preventDefault();
var $input = $(ev.target).find('input');
var jid = $input.val();
2017-07-22 22:21:05 +02:00
if (!jid || _.filter(jid.split('@')).length < 2) {
this.el.querySelector('.search-xmpp div').innerHTML = this.generateAddContactHTML({
error_message: __('Please enter a valid XMPP username'),
label_contact_username: __('e.g. user@example.org'),
label_add: __('Add'),
value: jid
});
2016-11-07 15:43:48 +01:00
return;
2016-05-03 17:37:10 +02:00
}
2017-02-03 13:51:07 +01:00
_converse.roster.addAndSubscribe(jid);
2017-07-22 22:21:05 +02:00
utils.slideIn(this.el.querySelector('.contact-form-container'));
2016-05-03 17:37:10 +02:00
},
2017-07-22 22:21:05 +02:00
addContactFromList: function addContactFromList(ev) {
2016-11-07 15:43:48 +01:00
ev.preventDefault();
var $target = $(ev.target),
jid = $target.attr('data-recipient'),
name = $target.text();
2017-02-03 13:51:07 +01:00
_converse.roster.addAndSubscribe(jid, name);
2016-11-07 15:43:48 +01:00
$target.parent().remove();
2017-07-22 22:21:05 +02:00
utils.slideIn(this.el.querySelector('.contact-form-container'));
2016-11-07 15:43:48 +01:00
}
});
2017-02-03 13:51:07 +01:00
_converse.ControlBoxToggle = Backbone.View.extend({
2016-11-07 15:43:48 +01:00
tagName: 'a',
2016-11-30 17:27:20 +01:00
className: 'toggle-controlbox hidden',
2016-11-07 15:43:48 +01:00
id: 'toggle-controlbox',
events: {
'click': 'onClick'
},
attributes: {
'href': "#"
2016-05-03 17:37:10 +02:00
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2017-02-03 13:51:07 +01:00
_converse.chatboxviews.$el.prepend(this.render());
2016-11-07 15:43:48 +01:00
this.updateOnlineCount();
2016-12-13 20:46:07 +01:00
var that = this;
2017-02-03 13:51:07 +01:00
_converse.on('initialized', function () {
_converse.roster.on("add", that.updateOnlineCount, that);
_converse.roster.on('change', that.updateOnlineCount, that);
_converse.roster.on("destroy", that.updateOnlineCount, that);
_converse.roster.on("remove", that.updateOnlineCount, that);
2016-12-13 20:46:07 +01:00
});
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
render: function render() {
2016-11-07 15:43:48 +01:00
// We let the render method of ControlBoxView decide whether
// the ControlBox or the Toggle must be shown. This prevents
// artifacts (i.e. on page load the toggle is shown only to then
// seconds later be hidden in favor of the control box).
2017-07-22 22:21:05 +02:00
return this.$el.html(tpl_controlbox_toggle({
'label_toggle': __('Toggle chat')
}));
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
updateOnlineCount: _.debounce(function () {
2017-02-03 13:51:07 +01:00
if (_.isUndefined(_converse.roster)) {
2016-11-07 15:43:48 +01:00
return;
}
var $count = this.$('#online-count');
2017-07-22 22:21:05 +02:00
$count.text("(" + _converse.roster.getNumOnlineContacts() + ")");
2016-11-07 15:43:48 +01:00
if (!$count.is(':visible')) {
$count.show();
2016-03-07 18:54:07 +01:00
}
2017-02-03 13:51:07 +01:00
}, _converse.animate ? 100 : 0),
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
hide: function hide(callback) {
2016-11-30 17:27:20 +01:00
this.el.classList.add('hidden');
callback();
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
show: function show(callback) {
2016-11-30 17:27:20 +01:00
utils.fadeIn(this.el, callback);
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
showControlBox: function showControlBox() {
2017-02-03 13:51:07 +01:00
var controlbox = _converse.chatboxes.get('controlbox');
2016-11-07 15:43:48 +01:00
if (!controlbox) {
2017-02-03 13:51:07 +01:00
controlbox = _converse.addControlBox();
2016-11-07 15:43:48 +01:00
}
2017-02-03 13:51:07 +01:00
if (_converse.connection.connected) {
2017-07-22 22:21:05 +02:00
controlbox.save({ closed: false });
2016-11-07 15:43:48 +01:00
} else {
controlbox.trigger('show');
}
},
2017-07-22 22:21:05 +02:00
onClick: function onClick(e) {
2016-11-07 15:43:48 +01:00
e.preventDefault();
if ($("div#controlbox").is(':visible')) {
2017-02-03 13:51:07 +01:00
var controlbox = _converse.chatboxes.get('controlbox');
if (_converse.connection.connected) {
2017-07-22 22:21:05 +02:00
controlbox.save({ closed: true });
2016-11-07 15:43:48 +01:00
} else {
controlbox.trigger('hide');
}
} else {
this.showControlBox();
2016-03-07 18:54:07 +01:00
}
}
2016-03-07 18:54:07 +01:00
});
2016-12-13 20:46:07 +01:00
2017-07-22 22:21:05 +02:00
var disconnect = function disconnect() {
2016-12-13 20:46:07 +01:00
/* Upon disconnection, set connected to `false`, so that if
* we reconnect,
* "onConnected" will be called, to fetch the roster again and
* to send out a presence stanza.
*/
2017-02-03 13:51:07 +01:00
var view = _converse.chatboxviews.get('controlbox');
2017-07-22 22:21:05 +02:00
view.model.set({ connected: false });
2016-12-13 20:46:07 +01:00
view.$('#controlbox-tabs').empty();
view.renderLoginPanel();
};
2017-02-03 13:51:07 +01:00
_converse.on('disconnected', disconnect);
2016-12-13 20:46:07 +01:00
2017-07-22 22:21:05 +02:00
var afterReconnected = function afterReconnected() {
2016-12-13 20:46:07 +01:00
/* After reconnection makes sure the controlbox's is aware.
*/
2017-02-03 13:51:07 +01:00
var view = _converse.chatboxviews.get('controlbox');
2016-12-13 20:46:07 +01:00
if (view.model.get('connected')) {
2017-02-03 13:51:07 +01:00
_converse.chatboxviews.get("controlbox").onConnected();
2016-12-13 20:46:07 +01:00
} else {
2017-07-22 22:21:05 +02:00
view.model.set({ connected: true });
2016-12-13 20:46:07 +01:00
}
};
2017-02-03 13:51:07 +01:00
_converse.on('reconnected', afterReconnected);
2016-11-07 15:43:48 +01:00
}
});
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-controlbox.js.map;
2016-11-07 15:43:48 +01:00
define('tpl!chatarea', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-04-04 17:26:06 +02:00
__p += '<div class="chat-area">\n <div class="chat-content ';
if (show_send_button) { ;
__p += 'chat-content-sendbutton';
} ;
__p += '"></div>\n <div class="new-msgs-indicator hidden">▼ ' +
2017-02-13 17:16:13 +01:00
__e( unread_msgs ) +
2017-07-22 22:21:05 +02:00
' ▼</div>\n <form class="sendXMPPMessage">\n ';
if (show_toolbar) { ;
__p += '\n <ul class="chat-toolbar no-text-select"></ul>\n ';
} ;
2017-04-04 17:26:06 +02:00
__p += '\n <textarea type="text" class="chat-textarea ';
if (show_send_button) { ;
__p += 'chat-textarea-send-button';
} ;
__p += '"\n placeholder="' +
2017-02-13 17:16:13 +01:00
__e(label_message) +
2017-04-04 17:26:06 +02:00
'"/>\n ';
if (show_send_button) { ;
2017-04-23 19:02:44 +02:00
__p += '\n <button type="submit" class="pure-button send-button">' +
__e( label_send ) +
'</button>\n ';
2017-04-04 17:26:06 +02:00
} ;
__p += '\n </form>\n</div>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-11-07 15:43:48 +01:00
define('tpl!chatroom', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '';
with (obj) {
__p += '<div class="flyout box-flyout">\n <div class="chat-head chat-head-chatroom"></div>\n <div class="chat-body chatroom-body"><span class="spinner centered"/></div>\n</div>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-11-07 15:43:48 +01:00
2017-04-23 19:02:44 +02:00
define('tpl!chatroom_disconnect', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<p class="disconnect-msg">' +
__e(disconnect_message) +
'</p>\n';
}
return __p
};});
define('tpl!chatroom_features', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-03-05 10:45:18 +01:00
if (has_features) { ;
__p += '\n<p class="occupants-heading">' +
__e(label_features) +
'</p>\n';
} ;
__p += '\n<ul class="features-list">\n';
if (passwordprotected) { ;
__p += '\n<li class="feature" title="' +
__e( tt_passwordprotected ) +
'"><span class="icon-lock-2"></span>' +
__e( label_passwordprotected ) +
'</li>\n';
} ;
__p += '\n';
if (unsecured) { ;
__p += '\n<li class="feature" title="' +
__e( tt_unsecured ) +
'"><span class="icon-unlocked"></span>' +
__e( label_unsecured ) +
'</li>\n';
} ;
__p += '\n';
if (hidden) { ;
__p += '\n<li class="feature" title="' +
__e( tt_hidden ) +
'"><span class="icon-eye-blocked"></span>' +
__e( label_hidden ) +
'</li>\n';
} ;
__p += '\n';
if (public) { ;
__p += '\n<li class="feature" title="' +
__e( tt_public ) +
'"><span class="icon-eye"></span>' +
__e( label_public ) +
'</li>\n';
} ;
__p += '\n';
if (membersonly) { ;
__p += '\n<li class="feature" title="' +
__e( tt_membersonly ) +
'"><span class="icon-address-book"></span>' +
__e( label_membersonly ) +
'</li>\n';
} ;
__p += '\n';
if (open) { ;
__p += '\n<li class="feature" title="' +
__e( tt_open ) +
'"><span class="icon-globe"></span>' +
__e( label_open ) +
'</li>\n';
} ;
__p += '\n';
if (persistent) { ;
__p += '\n<li class="feature" title="' +
__e( tt_persistent ) +
'"><span class="icon-save"></span>' +
__e( label_persistent ) +
'</li>\n';
} ;
__p += '\n';
if (temporary) { ;
__p += '\n<li class="feature" title="' +
__e( tt_temporary ) +
'"><span class="icon-snowflake"></span>' +
__e( label_temporary ) +
'</li>\n';
} ;
__p += '\n';
if (nonanonymous) { ;
__p += '\n<li class="feature" title="' +
__e( tt_nonanonymous ) +
'"><span class="icon-idcard-dark"></span>' +
__e( label_nonanonymous ) +
'</li>\n';
} ;
__p += '\n';
if (semianonymous) { ;
__p += '\n<li class="feature" title="' +
__e( tt_semianonymous ) +
'"><span class="icon-info"></span>' +
__e( label_semianonymous ) +
'</li>\n';
} ;
__p += '\n';
if (moderated) { ;
__p += '\n<li class="feature" title="' +
__e( tt_moderated ) +
'"><span class="icon-legal"></span>' +
__e( label_moderated ) +
'</li>\n';
} ;
__p += '\n';
if (unmoderated) { ;
__p += '\n<li class="feature" title="' +
__e( tt_unmoderated ) +
'"><span class="icon-info"></span>' +
__e( label_unmoderated ) +
'</li>\n';
} ;
__p += '\n';
if (mam_enabled) { ;
__p += '\n<li class="feature" title="' +
__e( tt_mam_enabled ) +
'"><span class="icon-database"></span>' +
__e( label_mam_enabled ) +
'</li>\n';
} ;
2017-03-05 10:45:18 +01:00
__p += '\n</ul>\n';
}
return __p
};});
define('tpl!chatroom_form', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '';
with (obj) {
__p += '<div class="chatroom-form-container">\n <form class="pure-form pure-form-stacked converse-form chatroom-form">\n <fieldset>\n <span class="spinner centered"/>\n </fieldset>\n </form>\n</div>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!chatroom_head', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
__p += '<a class="chatbox-btn close-chatbox-button icon-close" title="' +
__e(info_close) +
'"></a>\n';
if (affiliation == 'owner') { ;
__p += '\n <a class="chatbox-btn configure-chatroom-button icon-wrench" title="' +
__e(info_configure) +
' "></a>\n';
} ;
2017-03-05 10:45:18 +01:00
__p += '\n<div class="chat-title" title="' +
__e(jid) +
2017-07-22 22:21:05 +02:00
'">\n ';
if (name && name !== Strophe.getNodeFromJid(jid)) { ;
__p += '\n <span class="chatroom-name">' +
__e( name ) +
2017-07-22 22:21:05 +02:00
'</span>\n ';
} else { ;
__p += '\n <span class="chatroom-name">' +
__e( Strophe.getNodeFromJid(jid) ) +
'</span>@' +
__e( Strophe.getDomainFromJid(jid) ) +
'\n ';
} ;
__p += '\n <p class="chatroom-description">' +
__e( description ) +
'<p/>\n</div>\n';
}
return __p
};});
2017-03-05 10:45:18 +01:00
define('tpl!chatroom_invite', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-07-22 22:21:05 +02:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
2017-03-05 10:45:18 +01:00
with (obj) {
2017-07-22 22:21:05 +02:00
__p += '<form class="pure-form room-invite">\n ';
if (error_message) { ;
__p += '\n <span class="pure-form-message error">' +
__e(error_message) +
'</span>\n ';
} ;
__p += '\n <input class="invited-contact" placeholder="' +
2017-03-05 10:45:18 +01:00
__e(label_invitation) +
'" type="text"/>\n</form>\n';
}
return __p
};});
define('tpl!chatroom_nickname_form', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
2017-02-03 13:51:07 +01:00
__p += '<div class="chatroom-form-container">\n <form class="pure-form converse-form chatroom-form converse-centered-form">\n <fieldset>\n <label>' +
2017-02-13 17:16:13 +01:00
__e(heading) +
'</label>\n <p class="validation-message">' +
2017-02-13 17:16:13 +01:00
__e(validation_message) +
'</p>\n <input type="text" required="required" name="nick" class="new-chatroom-nick" placeholder="' +
2017-02-13 17:16:13 +01:00
__e(label_nickname) +
'"/>\n </fieldset>\n <fieldset>\n <input type="submit" class="pure-button button-primary" name="join" value="' +
2017-02-13 17:16:13 +01:00
__e(label_join) +
2016-11-07 15:43:48 +01:00
'"/>\n </fieldset>\n </form>\n</div>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!chatroom_password_form', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<div class="chatroom-form-container">\n <form class="pure-form converse-form chatroom-form">\n <fieldset>\n <legend>' +
2017-02-13 17:16:13 +01:00
__e(heading) +
'</legend>\n <label>' +
2017-02-13 17:16:13 +01:00
__e(label_password) +
'</label>\n <input type="password" name="password"/>\n </fieldset>\n <fieldset>\n <input class="pure-button button-primary" type="submit" value="' +
2017-02-13 17:16:13 +01:00
__e(label_submit) +
2016-11-07 15:43:48 +01:00
'"/>\n </fieldset>\n </form>\n</div>\n';
2016-03-07 18:54:07 +01:00
}
return __p
};});
define('tpl!chatroom_sidebar', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-03-05 10:45:18 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<!-- <div class="occupants"> -->\n<p class="occupants-heading">' +
2017-02-13 17:16:13 +01:00
__e(label_occupants) +
2017-03-05 10:45:18 +01:00
'</p>\n<ul class="occupant-list"></ul>\n<div class="chatroom-features"></div>\n<!-- </div> -->\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-11-07 15:43:48 +01:00
define('tpl!chatroom_toolbar', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-07-22 22:21:05 +02:00
if (use_emoji) { ;
__p += '\n <li class="toggle-smiley icon-happy" title="' +
2017-02-13 17:16:13 +01:00
__e(label_insert_smiley) +
2017-07-22 22:21:05 +02:00
'">\n <ul class="emoji-picker"></ul>\n </li>\n';
} ;
__p += '\n';
if (show_call_button) { ;
__p += '\n<li class="toggle-call"><a class="icon-phone" title="' +
2017-02-13 17:16:13 +01:00
__e(label_start_call) +
2016-11-07 15:43:48 +01:00
'"></a></li>\n';
} ;
__p += '\n';
if (show_occupants_toggle) { ;
__p += '\n<li class="toggle-occupants"><a class="icon-hide-users" title="' +
2017-02-13 17:16:13 +01:00
__e(label_hide_occupants) +
2016-11-07 15:43:48 +01:00
'"></a></li>\n';
} ;
__p += '\n';
if (show_clear_button) { ;
2017-06-23 20:25:33 +02:00
__p += '\n<li class="toggle-clear"><a class="icon-trash" title="' +
2017-02-13 17:16:13 +01:00
__e(label_clear) +
2016-11-07 15:43:48 +01:00
'"></a></li>\n';
} ;
__p += '\n\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-03-07 18:54:07 +01:00
define('tpl!chatrooms_tab', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-06-23 20:25:33 +02:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-06-23 20:25:33 +02:00
__p += '<a class="s rooms-tab\n ';
if (is_current) { ;
__p += ' current ';
} ;
2017-06-23 20:25:33 +02:00
__p += '\n ';
if (num_unread) { ;
__p += ' unread-msgs ';
} ;
__p += '"\n data-id="chatrooms" href="#chatrooms">\n ' +
((__t = (label_rooms)) == null ? '' : __t) +
2017-06-23 20:25:33 +02:00
'\n ';
if (num_unread) { ;
__p += '\n <span class="msgs-indicator">' +
__e( num_unread ) +
'</span>\n ';
} ;
__p += '\n</a>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-03-07 18:54:07 +01:00
define('tpl!info', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<div class="chat-info">' +
2017-02-13 17:16:13 +01:00
__e(message) +
2016-11-07 15:43:48 +01:00
'</div>\n';
}
return __p
};});
define('tpl!occupant', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
__p += '<li class="' +
__e( role ) +
' occupant" id="' +
__e( id ) +
2016-11-07 15:43:48 +01:00
'"\n ';
if (role === "moderator") { ;
__p += '\n title="' +
__e( jid ) +
' ' +
__e( desc_moderator ) +
' ' +
__e( hint_occupant ) +
2016-11-07 15:43:48 +01:00
'"\n ';
} ;
__p += '\n ';
if (role === "occupant") { ;
__p += '\n title="' +
__e( jid ) +
' ' +
__e( desc_occupant ) +
' ' +
__e( hint_occupant ) +
2016-11-07 15:43:48 +01:00
'"\n ';
} ;
__p += '\n ';
if (role === "visitor") { ;
__p += '\n title="' +
__e( jid ) +
' ' +
__e( desc_visitor ) +
' ' +
__e( hint_occupant ) +
'"\n ';
} ;
__p += '\n ';
if (!_.includes(["visitor", "occupant", "moderator"], role)) { ;
__p += '\n title="' +
__e( jid ) +
' ' +
__e( hint_occupant ) +
2017-03-05 10:45:18 +01:00
'"\n ';
} ;
2017-03-05 10:45:18 +01:00
__p += '><div class="occupant-status occupant-' +
__e(show) +
' circle" title="' +
__e(hint_show) +
'"></div>' +
2017-02-13 17:16:13 +01:00
__e(nick) +
2016-11-07 15:43:48 +01:00
'</li>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!room_description', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
__p += '<!-- FIXME: check markup in mockup -->\n<div class="room-info">\n<p class="room-info"><strong>' +
2017-07-22 22:21:05 +02:00
__e(label_jid) +
2017-04-23 19:02:44 +02:00
'</strong> ' +
2017-07-22 22:21:05 +02:00
__e(jid) +
2017-04-23 19:02:44 +02:00
'</p>\n<p class="room-info"><strong>' +
2017-02-13 17:16:13 +01:00
__e(label_desc) +
'</strong> ' +
2017-02-13 17:16:13 +01:00
__e(desc) +
'</p>\n<p class="room-info"><strong>' +
2017-02-13 17:16:13 +01:00
__e(label_occ) +
'</strong> ' +
2017-02-13 17:16:13 +01:00
__e(occ) +
'</p>\n<p class="room-info"><strong>' +
2017-02-13 17:16:13 +01:00
__e(label_features) +
2016-11-07 15:43:48 +01:00
'</strong>\n <ul>\n ';
if (passwordprotected) { ;
__p += '\n <li class="room-info locked">' +
2017-02-13 17:16:13 +01:00
__e(label_requires_auth) +
2016-11-07 15:43:48 +01:00
'</li>\n ';
} ;
__p += '\n ';
if (hidden) { ;
__p += '\n <li class="room-info">' +
2017-02-13 17:16:13 +01:00
__e(label_hidden) +
2016-11-07 15:43:48 +01:00
'</li>\n ';
} ;
__p += '\n ';
if (membersonly) { ;
__p += '\n <li class="room-info">' +
2017-02-13 17:16:13 +01:00
__e(label_requires_invite) +
2016-11-07 15:43:48 +01:00
'</li>\n ';
} ;
__p += '\n ';
if (moderated) { ;
__p += '\n <li class="room-info">' +
2017-02-13 17:16:13 +01:00
__e(label_moderated) +
2016-11-07 15:43:48 +01:00
'</li>\n ';
} ;
__p += '\n ';
if (nonanonymous) { ;
__p += '\n <li class="room-info">' +
2017-02-13 17:16:13 +01:00
__e(label_non_anon) +
2016-11-07 15:43:48 +01:00
'</li>\n ';
} ;
__p += '\n ';
if (open) { ;
__p += '\n <li class="room-info">' +
2017-02-13 17:16:13 +01:00
__e(label_open_room) +
2016-11-07 15:43:48 +01:00
'</li>\n ';
} ;
__p += '\n ';
if (persistent) { ;
__p += '\n <li class="room-info">' +
2017-02-13 17:16:13 +01:00
__e(label_permanent_room) +
2016-11-07 15:43:48 +01:00
'</li>\n ';
} ;
__p += '\n ';
if (publicroom) { ;
__p += '\n <li class="room-info">' +
2017-02-13 17:16:13 +01:00
__e(label_public) +
2016-11-07 15:43:48 +01:00
'</li>\n ';
} ;
__p += '\n ';
if (semianonymous) { ;
__p += '\n <li class="room-info">' +
2017-02-13 17:16:13 +01:00
__e(label_semi_anon) +
2016-11-07 15:43:48 +01:00
'</li>\n ';
} ;
__p += '\n ';
if (temporary) { ;
__p += '\n <li class="room-info">' +
2017-02-13 17:16:13 +01:00
__e(label_temp_room) +
2016-11-07 15:43:48 +01:00
'</li>\n ';
} ;
__p += '\n ';
if (unmoderated) { ;
__p += '\n <li class="room-info">' +
2017-02-13 17:16:13 +01:00
__e(label_unmoderated) +
2016-11-07 15:43:48 +01:00
'</li>\n ';
} ;
__p += '\n </ul>\n</p>\n</div>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!room_item', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
2017-07-22 22:21:05 +02:00
__p += '<dd class="available-chatroom">\n<a class="open-room available-room"\n data-room-jid="' +
2017-02-13 17:16:13 +01:00
__e(jid) +
'"\n title="' +
2017-02-13 17:16:13 +01:00
__e(open_title) +
2017-07-22 22:21:05 +02:00
'"\n href="#">' +
__e(name) +
'</a>\n<a class="right room-info icon-room-info"\n data-room-jid="' +
2017-02-13 17:16:13 +01:00
__e(jid) +
'"\n title="' +
2017-02-13 17:16:13 +01:00
__e(info_title) +
2016-11-07 15:43:48 +01:00
'" href="#">&nbsp;</a>\n</dd>\n';
}
return __p
};});
define('tpl!room_panel', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
__p += '<form class="pure-form pure-form-stacked converse-form add-chatroom" action="" method="post">\n <fieldset>\n <label>' +
((__t = (label_room_name)) == null ? '' : __t) +
'</label>\n <input type="text" name="chatroom" class="new-chatroom-name" placeholder="' +
((__t = (label_room_name)) == null ? '' : __t) +
2016-11-07 15:43:48 +01:00
'"/>\n ';
if (server_input_type != 'hidden') { ;
__p += '\n <label' +
((__t = (server_label_global_attr)) == null ? '' : __t) +
'>' +
((__t = (label_server)) == null ? '' : __t) +
2016-11-07 15:43:48 +01:00
'</label>\n ';
} ;
__p += '\n <input type="' +
((__t = (server_input_type)) == null ? '' : __t) +
'" name="server" class="new-chatroom-server" placeholder="' +
((__t = (label_server)) == null ? '' : __t) +
'"/>\n <input type="submit" class="pure-button button-primary" name="join" value="' +
((__t = (label_join)) == null ? '' : __t) +
'"/>\n <input type="button" class="pure-button button-secondary" name="show" id="show-rooms" value="' +
((__t = (label_show_rooms)) == null ? '' : __t) +
2017-06-23 20:25:33 +02:00
'"/>\n </fieldset>\n</form>\n<div class="rooms-list-container">\n <dl id="available-chatrooms" class="rooms-list"></dl>\n</div>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2017-07-22 22:21:05 +02:00
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
// Licensed under the Mozilla Public License (MPLv2)
//
/* This is a Converse.js plugin which add support for XEP-0030: Service Discovery */
/*global Backbone, define, window, document */
(function (root, factory) {
define('converse-disco',["converse-core", "sizzle", "strophe.disco"], factory);
}(this, function (converse, sizzle) {
const { Backbone, Promise, Strophe, b64_sha1, _ } = converse.env;
converse.plugins.add('converse-disco', {
initialize () {
/* The initialize function gets called as soon as the plugin is
* loaded by converse.js's plugin machinery.
*/
const { _converse } = this;
function onDiscoItems (stanza) {
_.each(stanza.querySelectorAll('query item'), (item) => {
if (item.getAttribute("node")) {
// XXX: ignore nodes for now.
// See: https://xmpp.org/extensions/xep-0030.html#items-nodes
return;
}
const jid = item.getAttribute('jid');
const entities = _converse.disco_entities;
if (_.isUndefined(entities.get(jid))) {
entities.create({'jid': jid});
}
});
}
// Promises exposed by this plugin
_converse.api.promises.add('discoInitialized');
_converse.DiscoEntity = Backbone.Model.extend({
/* A Disco Entity is a JID addressable entity that can be queried
* for features.
*
* See XEP-0030: https://xmpp.org/extensions/xep-0030.html
*/
idAttribute: 'jid',
initialize () {
this.features = new Backbone.Collection();
this.features.browserStorage = new Backbone.BrowserStorage[_converse.storage](
b64_sha1(`converse.features-${this.get('jid')}`)
);
this.features.on('add', this.onFeatureAdded);
this.identities = new Backbone.Collection();
this.identities.browserStorage = new Backbone.BrowserStorage[_converse.storage](
b64_sha1(`converse.identities-${this.get('jid')}`)
);
this.fetchFeatures();
},
onFeatureAdded (feature) {
_converse.emit('serviceDiscovered', feature);
},
fetchFeatures () {
if (this.features.browserStorage.records.length === 0) {
this.queryInfo();
} else {
this.features.fetch({add: true});
this.identities.fetch({add: true});
}
},
queryInfo () {
_converse.connection.disco.info(this.get('jid'), null, this.onInfo.bind(this));
},
queryForItems () {
if (_.isEmpty(this.identities.where({'category': 'server'}))) {
// Don't fetch features and items if this is not a
// server or a conference component.
return;
}
_converse.connection.disco.items(this.get('jid'), null, onDiscoItems);
},
onInfo (stanza) {
_.forEach(stanza.querySelectorAll('identity'), (identity) => {
this.identities.create({
'category': identity.getAttribute('category'),
'type': stanza.getAttribute('type'),
'name': stanza.getAttribute('name')
});
});
if (stanza.querySelector('feature[var="'+Strophe.NS.DISCO_ITEMS+'"]')) {
this.queryForItems();
}
_.forEach(stanza.querySelectorAll('feature'), (feature) => {
this.features.create({
'var': feature.getAttribute('var'),
'from': stanza.getAttribute('from')
});
});
2017-08-08 22:11:13 +02:00
this.trigger('featuresDiscovered');
2017-07-22 22:21:05 +02:00
}
});
_converse.DiscoEntities = Backbone.Collection.extend({
model: _converse.DiscoEntity,
initialize () {
this.browserStorage = new Backbone.BrowserStorage[_converse.storage](
b64_sha1(`converse.disco-entities-${_converse.bare_jid}`)
);
this.fetchEntities().then(
_.partial(_converse.emit, 'discoInitialized'),
_.partial(_converse.emit, 'discoInitialized')
2017-08-08 22:11:13 +02:00
).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
2017-07-22 22:21:05 +02:00
},
fetchEntities () {
return new Promise((resolve, reject) => {
this.fetch({
add: true,
success: function (collection) {
if (collection.length === 0 || !collection.get(_converse.domain)) {
this.create({'jid': _converse.domain});
}
resolve();
}.bind(this),
error () {
reject (new Error("Could not fetch disco entities"));
}
});
});
}
});
function addClientFeatures () {
/* The strophe.disco.js plugin keeps a list of features which
* it will advertise to any #info queries made to it.
*
* See: http://xmpp.org/extensions/xep-0030.html#info
*/
// See http://xmpp.org/registrar/disco-categories.html
_converse.connection.disco.addIdentity('client', 'web', 'Converse.js');
_converse.connection.disco.addFeature(Strophe.NS.BOSH);
_converse.connection.disco.addFeature(Strophe.NS.CHATSTATES);
_converse.connection.disco.addFeature(Strophe.NS.DISCO_INFO);
_converse.connection.disco.addFeature(Strophe.NS.ROSTERX); // Limited support
if (_converse.message_carbons) {
_converse.connection.disco.addFeature(Strophe.NS.CARBONS);
}
_converse.emit('addClientFeatures');
return this;
}
function initializeDisco () {
addClientFeatures();
_converse.disco_entities = new _converse.DiscoEntities();
}
_converse.api.listen.on('reconnected', initializeDisco);
_converse.api.listen.on('connected', initializeDisco);
_converse.api.listen.on('beforeTearDown', () => {
if (_converse.disco_entities) {
_converse.disco_entities.each((entity) => {
entity.features.reset();
entity.features.browserStorage._clear();
});
_converse.disco_entities.reset();
_converse.disco_entities.browserStorage._clear();
}
});
}
});
}));
2016-03-16 12:49:35 +01:00
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
2017-02-13 17:16:13 +01:00
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
2016-03-16 12:49:35 +01:00
// Licensed under the Mozilla Public License (MPLv2)
//
2017-04-23 19:02:44 +02:00
/*global define */
2016-03-16 12:49:35 +01:00
2016-11-07 15:43:48 +01:00
/* This is a Converse.js plugin which add support for multi-user chat rooms, as
* specified in XEP-0045 Multi-user chat.
*/
2016-03-16 12:49:35 +01:00
(function (root, factory) {
2017-07-22 22:21:05 +02:00
define('converse-muc',["jquery.noconflict", "converse-core", "tpl!chatarea", "tpl!chatroom", "tpl!chatroom_disconnect", "tpl!chatroom_features", "tpl!chatroom_form", "tpl!chatroom_head", "tpl!chatroom_invite", "tpl!chatroom_nickname_form", "tpl!chatroom_password_form", "tpl!chatroom_sidebar", "tpl!chatroom_toolbar", "tpl!chatrooms_tab", "tpl!info", "tpl!occupant", "tpl!room_description", "tpl!room_item", "tpl!room_panel", "tpl!spinner", "awesomplete", "converse-chatview", "converse-disco"], factory);
})(undefined, function ($, converse, tpl_chatarea, tpl_chatroom, tpl_chatroom_disconnect, tpl_chatroom_features, tpl_chatroom_form, tpl_chatroom_head, tpl_chatroom_invite, tpl_chatroom_nickname_form, tpl_chatroom_password_form, tpl_chatroom_sidebar, tpl_chatroom_toolbar, tpl_chatrooms_tab, tpl_info, tpl_occupant, tpl_room_description, tpl_room_item, tpl_room_panel, tpl_spinner, Awesomplete) {
2017-06-23 20:25:33 +02:00
2016-03-16 12:49:35 +01:00
"use strict";
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
var ROOMS_PANEL_ID = 'chatrooms';
2017-06-23 20:25:33 +02:00
var CHATROOMS_TYPE = 'chatroom';
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
var _converse$env = converse.env,
Strophe = _converse$env.Strophe,
Backbone = _converse$env.Backbone,
Promise = _converse$env.Promise,
$iq = _converse$env.$iq,
$build = _converse$env.$build,
$msg = _converse$env.$msg,
$pres = _converse$env.$pres,
b64_sha1 = _converse$env.b64_sha1,
sizzle = _converse$env.sizzle,
utils = _converse$env.utils,
_ = _converse$env._,
fp = _converse$env.fp,
moment = _converse$env.moment;
2016-03-16 12:49:35 +01:00
2016-11-07 15:43:48 +01:00
// Add Strophe Namespaces
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
Strophe.addNamespace('MUC_ADMIN', Strophe.NS.MUC + "#admin");
Strophe.addNamespace('MUC_OWNER', Strophe.NS.MUC + "#owner");
Strophe.addNamespace('MUC_REGISTER', "jabber:iq:register");
Strophe.addNamespace('MUC_ROOMCONF', Strophe.NS.MUC + "#roomconfig");
Strophe.addNamespace('MUC_USER', Strophe.NS.MUC + "#user");
2017-07-22 22:21:05 +02:00
var ROOM_FEATURES = ['passwordprotected', 'unsecured', 'hidden', 'public', 'membersonly', 'open', 'persistent', 'temporary', 'nonanonymous', 'semianonymous', 'moderated', 'unmoderated', 'mam_enabled'];
2017-04-04 17:26:06 +02:00
var ROOM_FEATURES_MAP = {
'passwordprotected': 'unsecured',
'unsecured': 'passwordprotected',
'hidden': 'public',
'public': 'hidden',
'membersonly': 'open',
'open': 'membersonly',
'persistent': 'temporary',
'temporary': 'persistent',
'nonanonymous': 'semianonymous',
'semianonymous': 'nonanonymous',
'moderated': 'unmoderated',
'unmoderated': 'moderated'
};
2017-07-22 22:21:05 +02:00
converse.ROOMSTATUS = {
CONNECTED: 0,
CONNECTING: 1,
2017-03-05 10:45:18 +01:00
NICKNAME_REQUIRED: 2,
2017-04-04 17:26:06 +02:00
PASSWORD_REQUIRED: 3,
DISCONNECTED: 4,
ENTERED: 5
};
2017-02-03 13:51:07 +01:00
converse.plugins.add('converse-muc', {
2016-11-07 15:43:48 +01:00
/* Optional dependencies are other plugins which might be
2017-04-04 17:26:06 +02:00
* overridden or relied upon, and therefore need to be loaded before
* this plugin. They are called "optional" because they might not be
* available, in which case any overrides applicable to them will be
* ignored.
2016-11-07 15:43:48 +01:00
*
2017-04-04 17:26:06 +02:00
* It's possible however to make optional dependencies non-optional.
* If the setting "strict_plugin_dependencies" is set to true,
2016-11-07 15:43:48 +01:00
* an error will be raised if the plugin is not found.
*
* NB: These plugins need to have already been loaded via require.js.
*/
optional_dependencies: ["converse-controlbox"],
2016-03-16 12:49:35 +01:00
overrides: {
// Overrides mentioned here will be picked up by converse.js's
// plugin architecture they will replace existing methods on the
// relevant objects or classes.
//
// New functions which don't exist yet can also be added.
2017-07-22 22:21:05 +02:00
_tearDown: function _tearDown() {
var rooms = this.chatboxes.where({ 'type': CHATROOMS_TYPE });
2017-06-23 20:25:33 +02:00
_.each(rooms, function (room) {
2017-07-22 22:21:05 +02:00
utils.safeSave(room, { 'connection_status': converse.ROOMSTATUS.DISCONNECTED });
2017-06-23 20:25:33 +02:00
});
this.__super__._tearDown.call(this, arguments);
},
2016-03-16 12:49:35 +01:00
2017-06-23 20:25:33 +02:00
ChatBoxes: {
2017-07-22 22:21:05 +02:00
model: function model(attrs, options) {
2017-06-23 20:25:33 +02:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2017-06-23 20:25:33 +02:00
if (attrs.type == CHATROOMS_TYPE) {
return new _converse.ChatRoom(attrs, options);
} else {
return this.__super__.model.apply(this, arguments);
}
2017-07-22 22:21:05 +02:00
}
2017-06-23 20:25:33 +02:00
},
2016-11-07 15:43:48 +01:00
ControlBoxView: {
2017-07-22 22:21:05 +02:00
renderRoomsPanel: function renderRoomsPanel() {
2017-06-23 20:25:33 +02:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2017-06-23 20:25:33 +02:00
this.roomspanel = new _converse.RoomsPanel({
'$parent': this.$el.find('.controlbox-panes'),
'model': new (Backbone.Model.extend({
2017-07-22 22:21:05 +02:00
id: b64_sha1("converse.roomspanel" + _converse.bare_jid), // Required by sessionStorage
browserStorage: new Backbone.BrowserStorage[_converse.storage](b64_sha1("converse.roomspanel" + _converse.bare_jid))
2017-06-23 20:25:33 +02:00
}))()
});
this.roomspanel.insertIntoDOM().model.fetch();
if (!this.roomspanel.model.get('nick')) {
this.roomspanel.model.save({
nick: Strophe.getNodeFromJid(_converse.bare_jid)
});
}
_converse.emit('roomsPanelRendered');
},
2017-07-22 22:21:05 +02:00
renderContactsPanel: function renderContactsPanel() {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
this.__super__.renderContactsPanel.apply(this, arguments);
2017-02-03 13:51:07 +01:00
if (_converse.allow_muc) {
2017-06-23 20:25:33 +02:00
this.renderRoomsPanel();
2016-03-16 12:49:35 +01:00
}
},
2017-07-22 22:21:05 +02:00
featureAdded: function featureAdded(feature) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
if (feature.get('var') === Strophe.NS.MUC && _converse.allow_muc) {
this.setMUCDomain(feature.get('from'));
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
},
getMUCDomainFromDisco: function getMUCDomainFromDisco() {
var _this = this;
/* Check whether service discovery for the user's domain
* returned MUC information and use that to automatically
* set the MUC domain for the "Rooms" panel of the
* controlbox.
*/
var _converse = this.__super__._converse;
_converse.api.waitUntil('discoInitialized').then(function () {
_converse.api.listen.on('serviceDiscovered', _this.featureAdded, _this);
2016-11-07 15:43:48 +01:00
// Features could have been added before the controlbox was
// initialized. We're only interested in MUC
2017-07-22 22:21:05 +02:00
var feature = _converse.disco_entities[_converse.domain].features.findWhere({
2016-11-07 15:43:48 +01:00
'var': Strophe.NS.MUC
});
if (feature) {
2017-07-22 22:21:05 +02:00
_this.featureAdded(feature);
}
2017-07-22 22:21:05 +02:00
});
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
onConnected: function onConnected() {
var _converse = this.__super__._converse;
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
this.__super__.onConnected.apply(this, arguments);
if (!this.model.get('connected')) {
return;
}
if (_.isUndefined(_converse.muc_domain)) {
this.getMUCDomainFromDisco();
} else {
this.setMUCDomain(_converse.muc_domain);
2016-11-07 15:43:48 +01:00
}
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
setMUCDomain: function setMUCDomain(domain) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
_converse.muc_domain = domain;
this.roomspanel.model.save({ 'muc_domain': domain });
var $server = this.$el.find('input.new-chatroom-server');
if (!$server.is(':focus')) {
$server.val(this.roomspanel.model.get('muc_domain'));
2016-03-16 12:49:35 +01:00
}
}
},
2016-11-07 15:43:48 +01:00
ChatBoxViews: {
2017-07-22 22:21:05 +02:00
onChatBoxAdded: function onChatBoxAdded(item) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
var view = this.get(item.get('id'));
2017-06-23 20:25:33 +02:00
if (!view && item.get('type') === CHATROOMS_TYPE) {
2017-07-22 22:21:05 +02:00
view = new _converse.ChatRoomView({ 'model': item });
2016-11-07 15:43:48 +01:00
return this.add(item.get('id'), view);
2016-03-16 12:49:35 +01:00
} else {
2016-11-07 15:43:48 +01:00
return this.__super__.onChatBoxAdded.apply(this, arguments);
2016-03-16 12:49:35 +01:00
}
}
}
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-03-16 12:49:35 +01:00
/* The initialize function gets called as soon as the plugin is
* loaded by converse.js's plugin machinery.
*/
2017-02-03 13:51:07 +01:00
var _converse = this._converse,
__ = _converse.__,
___ = _converse.___;
2016-12-13 20:46:07 +01:00
// XXX: Inside plugins, all calls to the translation machinery
// (e.g. utils.__) should only be done in the initialize function.
// If called before, we won't know what language the user wants,
// and it'll fall back to English.
2016-12-13 20:46:07 +01:00
/* http://xmpp.org/extensions/xep-0045.html
* ----------------------------------------
* 100 message Entering a room Inform user that any occupant is allowed to see the user's full JID
* 101 message (out of band) Affiliation change Inform user that his or her affiliation changed while not in the room
* 102 message Configuration change Inform occupants that room now shows unavailable members
* 103 message Configuration change Inform occupants that room now does not show unavailable members
* 104 message Configuration change Inform occupants that a non-privacy-related room configuration change has occurred
* 110 presence Any room presence Inform user that presence refers to one of its own room occupants
* 170 message or initial presence Configuration change Inform occupants that room logging is now enabled
* 171 message Configuration change Inform occupants that room logging is now disabled
* 172 message Configuration change Inform occupants that the room is now non-anonymous
* 173 message Configuration change Inform occupants that the room is now semi-anonymous
* 174 message Configuration change Inform occupants that the room is now fully-anonymous
* 201 presence Entering a room Inform user that a new room has been created
* 210 presence Entering a room Inform user that the service has assigned or modified the occupant's roomnick
* 301 presence Removal from room Inform user that he or she has been banned from the room
* 303 presence Exiting a room Inform all occupants of new room nickname
* 307 presence Removal from room Inform user that he or she has been kicked from the room
* 321 presence Removal from room Inform user that he or she is being removed from the room because of an affiliation change
* 322 presence Removal from room Inform user that he or she is being removed from the room because the room has been changed to members-only and the user is not a member
* 332 presence Removal from room Inform user that he or she is being removed from the room because of a system shutdown
*/
2017-02-03 13:51:07 +01:00
_converse.muc = {
2016-12-13 20:46:07 +01:00
info_messages: {
100: __('This room is not anonymous'),
102: __('This room now shows unavailable members'),
103: __('This room does not show unavailable members'),
104: __('The room configuration has changed'),
170: __('Room logging is now enabled'),
171: __('Room logging is now disabled'),
172: __('This room is now no longer anonymous'),
173: __('This room is now semi-anonymous'),
174: __('This room is now fully-anonymous'),
201: __('A new room has been created')
},
disconnect_messages: {
301: __('You have been banned from this room'),
307: __('You have been kicked from this room'),
321: __("You have been removed from this room because of an affiliation change"),
322: __("You have been removed from this room because the room has changed to members-only and you're not a member"),
332: __("You have been removed from this room because the MUC (Multi-user chat) service is being shut down.")
},
action_info_messages: {
/* XXX: Note the triple underscore function and not double
* underscore.
*
* This is a hack. We can't pass the strings to __ because we
* don't yet know what the variable to interpolate is.
*
* Triple underscore will just return the string again, but we
* can then at least tell gettext to scan for it so that these
* strings are picked up by the translation machinery.
*/
2017-02-13 17:16:13 +01:00
301: ___("%1$s has been banned"),
303: ___("%1$s's nickname has changed"),
307: ___("%1$s has been kicked out"),
321: ___("%1$s has been removed because of an affiliation change"),
322: ___("%1$s has been removed for not being a member")
2016-12-13 20:46:07 +01:00
},
new_nickname_messages: {
2017-02-13 17:16:13 +01:00
210: ___('Your nickname has been automatically set to: %1$s'),
303: ___('Your nickname has been changed to: %1$s')
2016-12-13 20:46:07 +01:00
}
};
2016-11-07 15:43:48 +01:00
// Configuration values for this plugin
// ====================================
// Refer to docs/source/configuration.rst for explanations of these
// configuration settings.
2017-07-05 11:59:55 +02:00
_converse.api.settings.update({
2016-11-07 15:43:48 +01:00
allow_muc: true,
allow_muc_invitations: true,
auto_join_on_invite: false,
auto_join_rooms: [],
auto_list_rooms: false,
hide_muc_server: false,
2016-12-13 20:46:07 +01:00
muc_disable_moderator_commands: false,
2016-11-07 15:43:48 +01:00
muc_domain: undefined,
muc_history_max_stanzas: undefined,
muc_instant_rooms: true,
muc_nickname_from_jid: false,
muc_show_join_leave: true,
2016-11-07 15:43:48 +01:00
visible_toolbar_buttons: {
'toggle_occupants': true
2017-07-22 22:21:05 +02:00
}
2016-03-16 12:49:35 +01:00
});
2017-07-05 11:59:55 +02:00
_converse.api.promises.add('roomsPanelRendered');
2017-06-23 20:25:33 +02:00
_converse.openChatRoom = function (settings, bring_to_foreground) {
2017-06-23 20:25:33 +02:00
/* Opens a chat room, making sure that certain attributes
2016-12-13 20:46:07 +01:00
* are correct, for example that the "type" is set to
* "chatroom".
*/
if (_.isUndefined(settings.jid)) {
throw new Error("openChatRoom needs to be called with a JID");
}
2017-07-22 22:21:05 +02:00
settings.type = CHATROOMS_TYPE;
settings.id = settings.jid;
settings.box_id = b64_sha1(settings.jid);
return _converse.chatboxviews.showChat(settings, bring_to_foreground);
2016-12-13 20:46:07 +01:00
};
2016-03-16 12:49:35 +01:00
2017-06-23 20:25:33 +02:00
_converse.ChatRoom = _converse.ChatBox.extend({
2017-07-22 22:21:05 +02:00
defaults: function defaults() {
return _.assign(_.clone(_converse.ChatBox.prototype.defaults), _.zipObject(ROOM_FEATURES, _.map(ROOM_FEATURES, _.stubFalse)), {
// For group chats, we distinguish between generally unread
// messages and those ones that specifically mention the
// user.
//
// To keep things simple, we reuse `num_unread` from
// _converse.ChatBox to indicate unread messages which
// mention the user and `num_unread_general` to indicate
// generally unread messages (which *includes* mentions!).
'num_unread_general': 0,
'affiliation': null,
'connection_status': converse.ROOMSTATUS.DISCONNECTED,
'name': '',
'description': '',
'features_fetched': false,
'roomconfig': {},
'type': CHATROOMS_TYPE
});
2017-06-23 20:25:33 +02:00
},
2017-07-22 22:21:05 +02:00
isUserMentioned: function isUserMentioned(message) {
2017-06-23 20:25:33 +02:00
/* Returns a boolean to indicate whether the current user
* was mentioned in a message.
*
* Parameters:
* (String): The text message
*/
2017-07-22 22:21:05 +02:00
return new RegExp("\\b" + this.get('nick') + "\\b").test(message);
2017-06-23 20:25:33 +02:00
},
2017-07-22 22:21:05 +02:00
incrementUnreadMsgCounter: function incrementUnreadMsgCounter(stanza) {
2017-06-23 20:25:33 +02:00
/* Given a newly received message, update the unread counter if
* necessary.
*
* Parameters:
* (XMLElement): The <messsage> stanza
*/
2017-07-22 22:21:05 +02:00
var body = stanza.querySelector('body');
2017-06-23 20:25:33 +02:00
if (_.isNull(body)) {
return; // The message has no text
}
if (utils.isNewMessage(stanza) && this.newMessageWillBeHidden()) {
2017-07-22 22:21:05 +02:00
this.save({ 'num_unread_general': this.get('num_unread_general') + 1 });
2017-06-23 20:25:33 +02:00
if (this.isUserMentioned(body.textContent)) {
2017-07-22 22:21:05 +02:00
this.save({ 'num_unread': this.get('num_unread') + 1 });
2017-06-23 20:25:33 +02:00
_converse.incrementMsgCounter();
}
}
},
2017-07-22 22:21:05 +02:00
clearUnreadMsgCounter: function clearUnreadMsgCounter() {
2017-06-23 20:25:33 +02:00
utils.safeSave(this, {
'num_unread': 0,
'num_unread_general': 0
});
}
});
2017-02-03 13:51:07 +01:00
_converse.ChatRoomView = _converse.ChatBoxView.extend({
2016-11-07 15:43:48 +01:00
/* Backbone View which renders a chat room, based upon the view
* for normal one-on-one chat boxes.
*/
length: 300,
2016-03-07 18:54:07 +01:00
tagName: 'div',
2016-11-30 17:27:20 +01:00
className: 'chatbox chatroom hidden',
2016-11-07 15:43:48 +01:00
is_chatroom: true,
2016-03-07 18:54:07 +01:00
events: {
2016-11-07 15:43:48 +01:00
'click .close-chatbox-button': 'close',
2017-04-23 19:02:44 +02:00
'click .configure-chatroom-button': 'getAndRenderConfigurationForm',
2017-07-22 22:21:05 +02:00
'click .toggle-smiley': 'toggleEmojiMenu',
'click .toggle-smiley ul.emoji-picker li': 'insertEmoji',
2016-11-07 15:43:48 +01:00
'click .toggle-clear': 'clearChatRoomMessages',
'click .toggle-call': 'toggleCall',
'click .toggle-occupants a': 'toggleOccupants',
'click .new-msgs-indicator': 'viewUnreadMessages',
'click .occupant': 'onOccupantClicked',
2017-04-04 17:26:06 +02:00
'keypress .chat-textarea': 'keyPressed',
'click .send-button': 'onSendButtonClicked'
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
var _this2 = this;
2016-11-07 15:43:48 +01:00
this.model.messages.on('add', this.onMessageAdded, this);
2016-03-16 12:49:35 +01:00
this.model.on('show', this.show, this);
2016-11-07 15:43:48 +01:00
this.model.on('destroy', this.hide, this);
this.model.on('change:connection_status', this.afterConnected, this);
2016-12-13 20:46:07 +01:00
this.model.on('change:affiliation', this.renderHeading, this);
this.model.on('change:chat_state', this.sendChatState, this);
this.model.on('change:description', this.renderHeading, this);
2016-12-13 20:46:07 +01:00
this.model.on('change:name', this.renderHeading, this);
2017-07-22 22:21:05 +02:00
this.createEmojiPicker();
2016-12-13 20:46:07 +01:00
this.createOccupantsView();
2017-03-05 10:45:18 +01:00
this.render().insertIntoDOM();
this.registerHandlers();
2017-03-05 10:45:18 +01:00
2017-07-22 22:21:05 +02:00
if (this.model.get('connection_status') !== converse.ROOMSTATUS.ENTERED) {
var handler = function handler() {
_this2.join();
_this2.fetchMessages();
_converse.emit('chatRoomOpened', _this2);
};
this.getRoomFeatures().then(handler, handler);
} else {
this.fetchMessages();
2017-06-23 20:25:33 +02:00
_converse.emit('chatRoomOpened', this);
}
},
2017-07-22 22:21:05 +02:00
render: function render() {
2017-04-04 17:26:06 +02:00
this.el.setAttribute('id', this.model.get('box_id'));
this.el.innerHTML = tpl_chatroom();
this.renderHeading();
this.renderChatArea();
2017-07-22 22:21:05 +02:00
if (this.model.get('connection_status') !== converse.ROOMSTATUS.ENTERED) {
2017-04-04 17:26:06 +02:00
this.showSpinner();
}
utils.refreshWebkit();
return this;
},
2017-07-22 22:21:05 +02:00
renderHeading: function renderHeading() {
/* Render the heading UI of the chat room. */
this.el.querySelector('.chat-head-chatroom').innerHTML = this.generateHeadingHTML();
},
2017-07-22 22:21:05 +02:00
renderChatArea: function renderChatArea() {
/* Render the UI container in which chat room messages will
* appear.
*/
if (!this.$('.chat-area').length) {
2017-07-22 22:21:05 +02:00
this.$('.chatroom-body').empty().append(tpl_chatarea({
'label_message': __('Message'),
'label_send': __('Send'),
'show_send_button': _converse.show_send_button,
'show_toolbar': _converse.show_toolbar,
'unread_msgs': __('You have unread messages')
})).append(this.occupantsview.$el);
this.renderToolbar(tpl_chatroom_toolbar);
this.$content = this.$el.find('.chat-content');
}
this.toggleOccupants(null, true);
return this;
2016-12-13 20:46:07 +01:00
},
2017-07-22 22:21:05 +02:00
createOccupantsView: function createOccupantsView() {
2016-12-13 20:46:07 +01:00
/* Create the ChatRoomOccupantsView Backbone.View
*/
var model = new _converse.ChatRoomOccupants();
model.chatroomview = this;
2017-07-22 22:21:05 +02:00
this.occupantsview = new _converse.ChatRoomOccupantsView({ 'model': model });
var id = b64_sha1("converse.occupants" + _converse.bare_jid + this.model.get('jid'));
2016-12-13 20:46:07 +01:00
this.occupantsview.model.browserStorage = new Backbone.BrowserStorage.session(id);
this.occupantsview.render();
2017-07-22 22:21:05 +02:00
this.occupantsview.model.fetch({ add: true });
2017-03-05 10:45:18 +01:00
return this;
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
insertIntoDOM: function insertIntoDOM() {
2017-03-05 10:45:18 +01:00
if (document.querySelector('body').contains(this.el)) {
return;
}
2017-02-03 13:51:07 +01:00
var view = _converse.chatboxviews.get("controlbox");
2016-11-07 15:43:48 +01:00
if (view) {
this.$el.insertAfter(view.$el);
} else {
$('#conversejs').prepend(this.$el);
}
2016-03-16 12:49:35 +01:00
return this;
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
generateHeadingHTML: function generateHeadingHTML() {
/* Returns the heading HTML to be rendered.
2016-12-13 20:46:07 +01:00
*/
2017-07-22 22:21:05 +02:00
return tpl_chatroom_head(_.extend(this.model.toJSON(), {
Strophe: Strophe,
info_close: __('Close and leave this room'),
info_configure: __('Configure this room'),
description: this.model.get('description') || ''
2016-12-13 20:46:07 +01:00
}));
},
2017-07-22 22:21:05 +02:00
afterShown: function afterShown() {
/* Override from converse-chatview, specifically to avoid
* the 'active' chat state from being sent out prematurely.
*
* This is instead done in `afterConnected` below.
2016-12-13 20:46:07 +01:00
*/
2017-06-23 20:25:33 +02:00
if (this.model.collection && this.model.collection.browserStorage) {
// Without a connection, we haven't yet initialized
// localstorage
this.model.save();
}
2017-03-05 10:45:18 +01:00
this.occupantsview.setOccupantsHeight();
2016-12-13 20:46:07 +01:00
},
2017-07-22 22:21:05 +02:00
afterConnected: function afterConnected() {
if (this.model.get('connection_status') === converse.ROOMSTATUS.ENTERED) {
this.setChatState(_converse.ACTIVE);
this.scrollDown();
this.focus();
2016-11-07 15:43:48 +01:00
}
},
2017-07-22 22:21:05 +02:00
getExtraMessageClasses: function getExtraMessageClasses(attrs) {
var extra_classes = _converse.ChatBoxView.prototype.getExtraMessageClasses.apply(this, arguments);
2017-07-22 22:21:05 +02:00
if (this.is_chatroom && attrs.sender === 'them' && this.model.isUserMentioned(attrs.message)) {
// Add special class to mark groupchat messages
// in which we are mentioned.
extra_classes += ' mentioned';
}
return extra_classes;
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
getToolbarOptions: function getToolbarOptions() {
return _.extend(_converse.ChatBoxView.prototype.getToolbarOptions.apply(this, arguments), {
label_hide_occupants: __('Hide the list of occupants'),
show_occupants_toggle: this.is_chatroom && _converse.visible_toolbar_buttons.toggle_occupants
});
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
close: function close(ev) {
2016-12-13 20:46:07 +01:00
/* Close this chat box, which implies leaving the room as
* well.
*/
2016-11-07 15:43:48 +01:00
this.leave();
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
toggleOccupants: function toggleOccupants(ev, preserve_state) {
2016-12-13 20:46:07 +01:00
/* Show or hide the right sidebar containing the chat
* occupants (and the invite widget).
*/
2016-11-07 15:43:48 +01:00
if (ev) {
ev.preventDefault();
ev.stopPropagation();
2016-03-07 18:54:07 +01:00
}
2016-11-07 15:43:48 +01:00
if (preserve_state) {
// Bit of a hack, to make sure that the sidebar's state doesn't change
2017-07-22 22:21:05 +02:00
this.model.set({ hidden_occupants: !this.model.get('hidden_occupants') });
2016-03-16 12:49:35 +01:00
}
2016-11-07 15:43:48 +01:00
if (!this.model.get('hidden_occupants')) {
2017-07-22 22:21:05 +02:00
this.model.save({ hidden_occupants: true });
2016-11-07 15:43:48 +01:00
this.$('.icon-hide-users').removeClass('icon-hide-users').addClass('icon-show-users');
this.$('.occupants').addClass('hidden');
this.$('.chat-area').addClass('full');
this.scrollDown();
} else {
2017-07-22 22:21:05 +02:00
this.model.save({ hidden_occupants: false });
2016-11-07 15:43:48 +01:00
this.$('.icon-show-users').removeClass('icon-show-users').addClass('icon-hide-users');
this.$('.chat-area').removeClass('full');
this.$('div.occupants').removeClass('hidden');
this.scrollDown();
2016-03-16 12:49:35 +01:00
}
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
onOccupantClicked: function onOccupantClicked(ev) {
2016-11-07 15:43:48 +01:00
/* When an occupant is clicked, insert their nickname into
* the chat textarea input.
*/
this.insertIntoTextArea(ev.target.textContent);
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
requestMemberList: function requestMemberList(chatroom_jid, affiliation) {
2016-12-13 20:46:07 +01:00
/* Send an IQ stanza to the server, asking it for the
* member-list of this room.
*
* See: http://xmpp.org/extensions/xep-0045.html#modifymember
*
* Parameters:
* (String) chatroom_jid: The JID of the chatroom for
* which the member-list is being requested
2016-12-13 20:46:07 +01:00
* (String) affiliation: The specific member list to
* fetch. 'admin', 'owner' or 'member'.
*
* Returns:
* A promise which resolves once the list has been
* retrieved.
*/
2017-07-22 22:21:05 +02:00
return new Promise(function (resolve, reject) {
affiliation = affiliation || 'member';
var iq = $iq({ to: chatroom_jid, type: "get" }).c("query", { xmlns: Strophe.NS.MUC_ADMIN }).c("item", { 'affiliation': affiliation });
_converse.connection.sendIQ(iq, resolve, reject);
});
2016-12-13 20:46:07 +01:00
},
2017-07-22 22:21:05 +02:00
parseMemberListIQ: function parseMemberListIQ(iq) {
2016-12-13 20:46:07 +01:00
/* Given an IQ stanza with a member list, create an array of member
* objects.
*/
2017-07-22 22:21:05 +02:00
return _.map($(iq).find("query[xmlns=\"" + Strophe.NS.MUC_ADMIN + "\"] item"), function (item) {
return {
'jid': item.getAttribute('jid'),
'affiliation': item.getAttribute('affiliation')
};
});
2016-12-13 20:46:07 +01:00
},
2017-07-22 22:21:05 +02:00
computeAffiliationsDelta: function computeAffiliationsDelta(exclude_existing, remove_absentees, new_list, old_list) {
2016-12-13 20:46:07 +01:00
/* Given two lists of objects with 'jid', 'affiliation' and
* 'reason' properties, return a new list containing
* those objects that are new, changed or removed
* (depending on the 'remove_absentees' boolean).
*
* The affiliations for new and changed members stay the
* same, for removed members, the affiliation is set to 'none'.
*
* The 'reason' property is not taken into account when
* comparing whether affiliations have been changed.
*
* Parameters:
* (Boolean) exclude_existing: Indicates whether JIDs from
* the new list which are also in the old list
* (regardless of affiliation) should be excluded
* from the delta. One reason to do this
* would be when you want to add a JID only if it
* doesn't have *any* existing affiliation at all.
* (Boolean) remove_absentees: Indicates whether JIDs
* from the old list which are not in the new list
* should be considered removed and therefore be
* included in the delta with affiliation set
* to 'none'.
* (Array) new_list: Array containing the new affiliations
* (Array) old_list: Array containing the old affiliations
*/
var new_jids = _.map(new_list, 'jid');
var old_jids = _.map(old_list, 'jid');
2016-12-13 20:46:07 +01:00
// Get the new affiliations
var delta = _.map(_.difference(new_jids, old_jids), function (jid) {
return new_list[_.indexOf(new_jids, jid)];
});
if (!exclude_existing) {
// Get the changed affiliations
delta = delta.concat(_.filter(new_list, function (item) {
var idx = _.indexOf(old_jids, item.jid);
if (idx >= 0) {
return item.affiliation !== old_list[idx].affiliation;
}
return false;
}));
}
if (remove_absentees) {
// Get the removed affiliations
delta = delta.concat(_.map(_.difference(old_jids, new_jids), function (jid) {
2017-07-22 22:21:05 +02:00
return { 'jid': jid, 'affiliation': 'none' };
2016-12-13 20:46:07 +01:00
}));
}
return delta;
},
2017-07-22 22:21:05 +02:00
sendAffiliationIQ: function sendAffiliationIQ(chatroom_jid, affiliation, member) {
/* Send an IQ stanza specifying an affiliation change.
*
* Paremeters:
* (String) chatroom_jid: JID of the relevant room
* (String) affiliation: affiliation (could also be stored
* on the member object).
* (Object) member: Map containing the member's jid and
* optionally a reason and affiliation.
*/
2017-07-22 22:21:05 +02:00
return new Promise(function (resolve, reject) {
var iq = $iq({ to: chatroom_jid, type: "set" }).c("query", { xmlns: Strophe.NS.MUC_ADMIN }).c("item", {
'affiliation': member.affiliation || affiliation,
'jid': member.jid
});
2017-07-22 22:21:05 +02:00
if (!_.isUndefined(member.reason)) {
iq.c("reason", member.reason);
}
_converse.connection.sendIQ(iq, resolve, reject);
});
},
2017-07-22 22:21:05 +02:00
setAffiliation: function setAffiliation(affiliation, members) {
2016-12-13 20:46:07 +01:00
/* Send IQ stanzas to the server to set an affiliation for
* the provided JIDs.
*
* See: http://xmpp.org/extensions/xep-0045.html#modifymember
*
* XXX: Prosody doesn't accept multiple JIDs' affiliations
* being set in one IQ stanza, so as a workaround we send
* a separate stanza for each JID.
* Related ticket: https://prosody.im/issues/issue/795
*
* Parameters:
* (String) affiliation: The affiliation
2016-12-13 20:46:07 +01:00
* (Object) members: A map of jids, affiliations and
* optionally reasons. Only those entries with the
* same affiliation as being currently set will be
* considered.
*
* Returns:
* A promise which resolves and fails depending on the
* XMPP server response.
*/
members = _.filter(members, function (member) {
2017-07-22 22:21:05 +02:00
return (
// We only want those members who have the right
// affiliation (or none, which implies the provided
// one).
_.isUndefined(member.affiliation) || member.affiliation === affiliation
);
2016-12-13 20:46:07 +01:00
});
2017-07-22 22:21:05 +02:00
var promises = _.map(members, _.partial(this.sendAffiliationIQ, this.model.get('jid'), affiliation));
return Promise.all(promises);
2016-12-13 20:46:07 +01:00
},
2017-07-22 22:21:05 +02:00
setAffiliations: function setAffiliations(members) {
2016-12-13 20:46:07 +01:00
/* Send IQ stanzas to the server to modify the
* affiliations in this room.
*
* See: http://xmpp.org/extensions/xep-0045.html#modifymember
*
* Parameters:
* (Object) members: A map of jids, affiliations and optionally reasons
* (Function) onSuccess: callback for a succesful response
* (Function) onError: callback for an error response
*/
var affiliations = _.uniq(_.map(members, 'affiliation'));
2017-07-22 22:21:05 +02:00
_.each(affiliations, _.partial(this.setAffiliation.bind(this), _, members));
2016-12-13 20:46:07 +01:00
},
2017-07-22 22:21:05 +02:00
marshallAffiliationIQs: function marshallAffiliationIQs() {
2016-12-13 20:46:07 +01:00
/* Marshall a list of IQ stanzas into a map of JIDs and
* affiliations.
*
* Parameters:
* Any amount of XMLElement objects, representing the IQ
* stanzas.
*/
return _.flatMap(arguments, this.parseMemberListIQ);
2016-12-13 20:46:07 +01:00
},
2017-07-22 22:21:05 +02:00
getJidsWithAffiliations: function getJidsWithAffiliations(affiliations) {
var _this3 = this;
2016-12-13 20:46:07 +01:00
/* Returns a map of JIDs that have the affiliations
* as provided.
*/
if (_.isString(affiliations)) {
2016-12-13 20:46:07 +01:00
affiliations = [affiliations];
}
2017-07-22 22:21:05 +02:00
return new Promise(function (resolve, reject) {
var promises = _.map(affiliations, _.partial(_this3.requestMemberList, _this3.model.get('jid')));
Promise.all(promises).then(_.flow(_this3.marshallAffiliationIQs.bind(_this3), resolve), _.flow(_this3.marshallAffiliationIQs.bind(_this3), resolve));
});
2016-12-13 20:46:07 +01:00
},
2017-07-22 22:21:05 +02:00
updateMemberLists: function updateMemberLists(members, affiliations, deltaFunc) {
var _this4 = this;
2016-12-13 20:46:07 +01:00
/* Fetch the lists of users with the given affiliations.
* Then compute the delta between those users and
* the passed in members, and if it exists, send the delta
* to the XMPP server to update the member list.
*
* Parameters:
* (Object) members: Map of member jids and affiliations.
* (String|Array) affiliation: An array of affiliations or
* a string if only one affiliation.
* (Function) deltaFunc: The function to compute the delta
* between old and new member lists.
*
* Returns:
* A promise which is resolved once the list has been
* updated or once it's been established there's no need
* to update the list.
*/
this.getJidsWithAffiliations(affiliations).then(function (old_members) {
2017-07-22 22:21:05 +02:00
_this4.setAffiliations(deltaFunc(members, old_members));
2016-12-13 20:46:07 +01:00
});
},
2017-07-22 22:21:05 +02:00
directInvite: function directInvite(recipient, reason) {
2016-12-13 20:46:07 +01:00
/* Send a direct invitation as per XEP-0249
*
* Parameters:
* (String) recipient - JID of the person being invited
* (String) reason - Optional reason for the invitation
*/
if (this.model.get('membersonly')) {
// When inviting to a members-only room, we first add
// the person to the member list by giving them an
// affiliation of 'member' (if they're not affiliated
// already), otherwise they won't be able to join.
2017-07-22 22:21:05 +02:00
var map = {};map[recipient] = 'member';
2016-12-13 20:46:07 +01:00
var deltaFunc = _.partial(this.computeAffiliationsDelta, true, false);
2017-07-22 22:21:05 +02:00
this.updateMemberLists([{ 'jid': recipient, 'affiliation': 'member', 'reason': reason }], ['member', 'owner', 'admin'], deltaFunc);
2016-12-13 20:46:07 +01:00
}
2016-11-07 15:43:48 +01:00
var attrs = {
2016-12-13 20:46:07 +01:00
'xmlns': 'jabber:x:conference',
'jid': this.model.get('jid')
2016-11-07 15:43:48 +01:00
};
2017-07-22 22:21:05 +02:00
if (reason !== null) {
attrs.reason = reason;
}
if (this.model.get('password')) {
attrs.password = this.model.get('password');
}
2016-11-07 15:43:48 +01:00
var invitation = $msg({
2017-02-03 13:51:07 +01:00
from: _converse.connection.jid,
2016-11-07 15:43:48 +01:00
to: recipient,
2017-02-03 13:51:07 +01:00
id: _converse.connection.getUniqueId()
2016-11-07 15:43:48 +01:00
}).c('x', attrs);
2017-02-03 13:51:07 +01:00
_converse.connection.send(invitation);
_converse.emit('roomInviteSent', {
2016-11-07 15:43:48 +01:00
'room': this,
'recipient': recipient,
'reason': reason
});
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
handleChatStateMessage: function handleChatStateMessage(message) {
2016-11-07 15:43:48 +01:00
/* Override the method on the ChatBoxView base class to
* ignore <gone/> notifications in groupchats.
*
* As laid out in the business rules in XEP-0085
* http://xmpp.org/extensions/xep-0085.html#bizrules-groupchat
*/
if (message.get('fullname') === this.model.get('nick')) {
// Don't know about other servers, but OpenFire sends
// back to you your own chat state notifications.
// We ignore them here...
return;
}
2017-02-03 13:51:07 +01:00
if (message.get('chat_state') !== _converse.GONE) {
_converse.ChatBoxView.prototype.handleChatStateMessage.apply(this, arguments);
2016-11-07 15:43:48 +01:00
}
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
sendChatState: function sendChatState() {
2016-11-07 15:43:48 +01:00
/* Sends a message with the status of the user in this chat session
* as taken from the 'chat_state' attribute of the chat box.
* See XEP-0085 Chat State Notifications.
*/
2017-07-22 22:21:05 +02:00
if (this.model.get('connection_status') !== converse.ROOMSTATUS.ENTERED) {
return;
}
2016-11-07 15:43:48 +01:00
var chat_state = this.model.get('chat_state');
2017-02-03 13:51:07 +01:00
if (chat_state === _converse.GONE) {
2016-11-07 15:43:48 +01:00
// <gone/> is not applicable within MUC context
return;
}
2017-07-22 22:21:05 +02:00
_converse.connection.send($msg({ 'to': this.model.get('jid'), 'type': 'groupchat' }).c(chat_state, { 'xmlns': Strophe.NS.CHATSTATES }).up().c('no-store', { 'xmlns': Strophe.NS.HINTS }).up().c('no-permanent-store', { 'xmlns': Strophe.NS.HINTS }));
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
sendChatRoomMessage: function sendChatRoomMessage(text) {
2016-12-13 20:46:07 +01:00
/* Constuct a message stanza to be sent to this chat room,
* and send it to the server.
*
* Parameters:
* (String) text: The message text to be sent.
*/
2017-02-03 13:51:07 +01:00
var msgid = _converse.connection.getUniqueId();
2016-11-07 15:43:48 +01:00
var msg = $msg({
to: this.model.get('jid'),
2017-02-03 13:51:07 +01:00
from: _converse.connection.jid,
2016-11-07 15:43:48 +01:00
type: 'groupchat',
id: msgid
2017-07-22 22:21:05 +02:00
}).c("body").t(text).up().c("x", { xmlns: "jabber:x:event" }).c(_converse.COMPOSING);
2017-02-03 13:51:07 +01:00
_converse.connection.send(msg);
2016-11-07 15:43:48 +01:00
this.model.messages.create({
fullname: this.model.get('nick'),
sender: 'me',
time: moment().format(),
message: text,
msgid: msgid
2016-03-16 12:49:35 +01:00
});
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
modifyRole: function modifyRole(room, nick, role, reason, onSuccess, onError) {
var item = $build("item", { nick: nick, role: role });
var iq = $iq({ to: room, type: "set" }).c("query", { xmlns: Strophe.NS.MUC_ADMIN }).cnode(item.node);
if (reason !== null) {
iq.c("reason", reason);
}
2017-02-03 13:51:07 +01:00
return _converse.connection.sendIQ(iq.tree(), onSuccess, onError);
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
validateRoleChangeCommand: function validateRoleChangeCommand(command, args) {
2016-11-07 15:43:48 +01:00
/* Check that a command to change a chat room user's role or
* affiliation has anough arguments.
*/
// TODO check if first argument is valid
if (args.length < 1 || args.length > 2) {
2017-07-22 22:21:05 +02:00
this.showStatusNotification(__("Error: the \"" + command + "\" command takes two arguments, the user's nickname and optionally a reason."), true);
2016-11-07 15:43:48 +01:00
return false;
2016-03-07 18:54:07 +01:00
}
2016-11-07 15:43:48 +01:00
return true;
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
clearChatRoomMessages: function clearChatRoomMessages(ev) {
2016-12-13 20:46:07 +01:00
/* Remove all messages from the chat room UI.
*/
2017-07-22 22:21:05 +02:00
if (!_.isUndefined(ev)) {
ev.stopPropagation();
}
2016-11-07 15:43:48 +01:00
var result = confirm(__("Are you sure you want to clear the messages from this room?"));
if (result === true) {
this.$content.empty();
}
return this;
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
onCommandError: function onCommandError() {
2016-12-13 20:46:07 +01:00
this.showStatusNotification(__("Error: could not execute the command"), true);
},
2017-07-22 22:21:05 +02:00
onMessageSubmitted: function onMessageSubmitted(text) {
2016-11-07 15:43:48 +01:00
/* Gets called when the user presses enter to send off a
* message in a chat room.
*
* Parameters:
* (String) text - The message text.
*/
2017-02-03 13:51:07 +01:00
if (_converse.muc_disable_moderator_commands) {
2016-12-13 20:46:07 +01:00
return this.sendChatRoomMessage(text);
}
2016-11-07 15:43:48 +01:00
var match = text.replace(/^\s*/, "").match(/^\/(.*?)(?: (.*))?$/) || [false, '', ''],
2017-02-03 13:51:07 +01:00
args = match[2] && match[2].splitOnce(' ') || [],
command = match[1].toLowerCase();
switch (command) {
2016-11-07 15:43:48 +01:00
case 'admin':
2017-07-22 22:21:05 +02:00
if (!this.validateRoleChangeCommand(command, args)) {
break;
}
this.setAffiliation('admin', [{ 'jid': args[0],
'reason': args[1]
}]).then(null, this.onCommandError.bind(this));
2016-11-07 15:43:48 +01:00
break;
case 'ban':
2017-07-22 22:21:05 +02:00
if (!this.validateRoleChangeCommand(command, args)) {
break;
}
this.setAffiliation('outcast', [{ 'jid': args[0],
'reason': args[1]
}]).then(null, this.onCommandError.bind(this));
2016-11-07 15:43:48 +01:00
break;
case 'clear':
this.clearChatRoomMessages();
break;
case 'deop':
2017-07-22 22:21:05 +02:00
if (!this.validateRoleChangeCommand(command, args)) {
break;
}
this.modifyRole(this.model.get('jid'), args[0], 'occupant', args[1], undefined, this.onCommandError.bind(this));
2016-11-07 15:43:48 +01:00
break;
case 'help':
2017-07-22 22:21:05 +02:00
this.showHelpMessages(["<strong>/admin</strong>: " + __("Change user's affiliation to admin"), "<strong>/ban</strong>: " + __('Ban user from room'), "<strong>/clear</strong>: " + __('Remove messages'), "<strong>/deop</strong>: " + __('Change user role to occupant'), "<strong>/help</strong>: " + __('Show this menu'), "<strong>/kick</strong>: " + __('Kick user from room'), "<strong>/me</strong>: " + __('Write in 3rd person'), "<strong>/member</strong>: " + __('Grant membership to a user'), "<strong>/mute</strong>: " + __("Remove user's ability to post messages"), "<strong>/nick</strong>: " + __('Change your nickname'), "<strong>/op</strong>: " + __('Grant moderator role to user'), "<strong>/owner</strong>: " + __('Grant ownership of this room'), "<strong>/revoke</strong>: " + __("Revoke user's membership"), "<strong>/subject</strong>: " + __('Set room subject'), "<strong>/topic</strong>: " + __('Set room subject (alias for /subject)'), "<strong>/voice</strong>: " + __('Allow muted user to post messages')]);
2016-11-07 15:43:48 +01:00
break;
case 'kick':
2017-07-22 22:21:05 +02:00
if (!this.validateRoleChangeCommand(command, args)) {
break;
}
this.modifyRole(this.model.get('jid'), args[0], 'none', args[1], undefined, this.onCommandError.bind(this));
2016-11-07 15:43:48 +01:00
break;
case 'mute':
2017-07-22 22:21:05 +02:00
if (!this.validateRoleChangeCommand(command, args)) {
break;
}
this.modifyRole(this.model.get('jid'), args[0], 'visitor', args[1], undefined, this.onCommandError.bind(this));
2016-11-07 15:43:48 +01:00
break;
case 'member':
2017-07-22 22:21:05 +02:00
if (!this.validateRoleChangeCommand(command, args)) {
break;
}
this.setAffiliation('member', [{ 'jid': args[0],
'reason': args[1]
}]).then(null, this.onCommandError.bind(this));
2016-11-07 15:43:48 +01:00
break;
case 'nick':
2017-02-03 13:51:07 +01:00
_converse.connection.send($pres({
from: _converse.connection.jid,
2016-11-07 15:43:48 +01:00
to: this.getRoomJIDAndNick(match[2]),
2017-02-03 13:51:07 +01:00
id: _converse.connection.getUniqueId()
2016-11-07 15:43:48 +01:00
}).tree());
break;
case 'owner':
2017-07-22 22:21:05 +02:00
if (!this.validateRoleChangeCommand(command, args)) {
break;
}
this.setAffiliation('owner', [{ 'jid': args[0],
'reason': args[1]
}]).then(null, this.onCommandError.bind(this));
2016-11-07 15:43:48 +01:00
break;
case 'op':
2017-07-22 22:21:05 +02:00
if (!this.validateRoleChangeCommand(command, args)) {
break;
}
this.modifyRole(this.model.get('jid'), args[0], 'moderator', args[1], undefined, this.onCommandError.bind(this));
2016-11-07 15:43:48 +01:00
break;
case 'revoke':
2017-07-22 22:21:05 +02:00
if (!this.validateRoleChangeCommand(command, args)) {
break;
}
this.setAffiliation('none', [{ 'jid': args[0],
'reason': args[1]
}]).then(null, this.onCommandError.bind(this));
2016-11-07 15:43:48 +01:00
break;
case 'topic':
2017-02-03 13:51:07 +01:00
case 'subject':
2017-07-22 22:21:05 +02:00
_converse.connection.send($msg({
to: this.model.get('jid'),
from: _converse.connection.jid,
type: "groupchat"
}).c("subject", { xmlns: "jabber:client" }).t(match[2]).tree());
2016-11-07 15:43:48 +01:00
break;
case 'voice':
2017-07-22 22:21:05 +02:00
if (!this.validateRoleChangeCommand(command, args)) {
break;
}
this.modifyRole(this.model.get('jid'), args[0], 'occupant', args[1], undefined, this.onCommandError.bind(this));
2016-11-07 15:43:48 +01:00
break;
default:
this.sendChatRoomMessage(text);
2017-07-22 22:21:05 +02:00
break;
2016-03-07 18:54:07 +01:00
}
},
2017-07-22 22:21:05 +02:00
handleMUCMessage: function handleMUCMessage(stanza) {
2016-12-13 20:46:07 +01:00
/* Handler for all MUC messages sent to this chat room.
*
* Parameters:
* (XMLElement) stanza: The message stanza.
*/
var configuration_changed = stanza.querySelector("status[code='104']");
var logging_enabled = stanza.querySelector("status[code='170']");
var logging_disabled = stanza.querySelector("status[code='171']");
var room_no_longer_anon = stanza.querySelector("status[code='172']");
var room_now_semi_anon = stanza.querySelector("status[code='173']");
var room_now_fully_anon = stanza.querySelector("status[code='173']");
2017-07-22 22:21:05 +02:00
if (configuration_changed || logging_enabled || logging_disabled || room_no_longer_anon || room_now_semi_anon || room_now_fully_anon) {
2016-12-13 20:46:07 +01:00
this.getRoomFeatures();
}
_.flow(this.showStatusMessages.bind(this), this.onChatRoomMessage.bind(this))(stanza);
2016-11-07 15:43:48 +01:00
return true;
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
getRoomJIDAndNick: function getRoomJIDAndNick(nick) {
2016-12-13 20:46:07 +01:00
/* Utility method to construct the JID for the current user
* as occupant of the room.
*
* This is the room JID, with the user's nick added at the
* end.
*
* For example: room@conference.example.org/nickname
*/
2016-11-07 15:43:48 +01:00
if (nick) {
2017-07-22 22:21:05 +02:00
this.model.save({ 'nick': nick });
2016-03-07 18:54:07 +01:00
} else {
2016-11-07 15:43:48 +01:00
nick = this.model.get('nick');
2016-03-07 18:54:07 +01:00
}
2016-11-07 15:43:48 +01:00
var room = this.model.get('jid');
var node = Strophe.getNodeFromJid(room);
var domain = Strophe.getDomainFromJid(room);
return node + "@" + domain + (nick !== null ? "/" + nick : "");
2016-03-07 18:54:07 +01:00
},
2017-07-22 22:21:05 +02:00
registerHandlers: function registerHandlers() {
2016-12-13 20:46:07 +01:00
/* Register presence and message handlers for this chat
* room
*/
2016-11-07 15:43:48 +01:00
var room_jid = this.model.get('jid');
this.removeHandlers();
2017-07-22 22:21:05 +02:00
this.presence_handler = _converse.connection.addHandler(this.onChatRoomPresence.bind(this), Strophe.NS.MUC, 'presence', null, null, room_jid, { 'ignoreNamespaceFragment': true, 'matchBareFromJid': true });
this.message_handler = _converse.connection.addHandler(this.handleMUCMessage.bind(this), null, 'message', 'groupchat', null, room_jid, { 'matchBareFromJid': true });
},
2017-07-22 22:21:05 +02:00
removeHandlers: function removeHandlers() {
2016-12-13 20:46:07 +01:00
/* Remove the presence and message handlers that were
* registered for this chat room.
*/
2016-11-07 15:43:48 +01:00
if (this.message_handler) {
2017-02-03 13:51:07 +01:00
_converse.connection.deleteHandler(this.message_handler);
2016-11-07 15:43:48 +01:00
delete this.message_handler;
}
if (this.presence_handler) {
2017-02-03 13:51:07 +01:00
_converse.connection.deleteHandler(this.presence_handler);
2016-11-07 15:43:48 +01:00
delete this.presence_handler;
}
return this;
},
2017-07-22 22:21:05 +02:00
join: function join(nick, password) {
2016-12-13 20:46:07 +01:00
/* Join the chat room.
*
* Parameters:
* (String) nick: The user's nickname
* (String) password: Optional password, if required by
* the room.
*/
nick = nick ? nick : this.model.get('nick');
if (!nick) {
return this.checkForReservedNick();
}
2017-07-22 22:21:05 +02:00
if (this.model.get('connection_status') === converse.ROOMSTATUS.ENTERED) {
2016-12-13 20:46:07 +01:00
// We have restored a chat room from session storage,
// so we don't send out a presence stanza again.
return this;
}
2016-11-07 15:43:48 +01:00
var stanza = $pres({
2017-02-03 13:51:07 +01:00
'from': _converse.connection.jid,
2016-11-07 15:43:48 +01:00
'to': this.getRoomJIDAndNick(nick)
2017-07-22 22:21:05 +02:00
}).c("x", { 'xmlns': Strophe.NS.MUC }).c("history", { 'maxstanzas': _converse.muc_history_max_stanzas }).up();
2016-11-07 15:43:48 +01:00
if (password) {
stanza.cnode(Strophe.xmlElement("password", [], password));
}
2017-07-22 22:21:05 +02:00
this.model.save('connection_status', converse.ROOMSTATUS.CONNECTING);
2017-02-03 13:51:07 +01:00
_converse.connection.send(stanza);
2016-12-13 20:46:07 +01:00
return this;
},
2017-07-22 22:21:05 +02:00
sendUnavailablePresence: function sendUnavailablePresence(exit_msg) {
2017-06-23 20:25:33 +02:00
var presence = $pres({
type: "unavailable",
from: _converse.connection.jid,
to: this.getRoomJIDAndNick()
});
if (exit_msg !== null) {
presence.c("status", exit_msg);
2017-04-23 19:02:44 +02:00
}
2017-06-23 20:25:33 +02:00
_converse.connection.sendPresence(presence);
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
leave: function leave(exit_msg) {
2016-12-13 20:46:07 +01:00
/* Leave the chat room.
*
* Parameters:
* (String) exit_msg: Optional message to indicate your
* reason for leaving.
*/
this.hide();
this.occupantsview.model.reset();
this.occupantsview.model.browserStorage._clear();
2017-06-23 20:25:33 +02:00
if (_converse.connection.connected) {
this.sendUnavailablePresence(exit_msg);
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
utils.safeSave(this.model, { 'connection_status': converse.ROOMSTATUS.DISCONNECTED });
2017-06-23 20:25:33 +02:00
this.removeHandlers();
_converse.ChatBoxView.prototype.close.apply(this, arguments);
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
renderConfigurationForm: function renderConfigurationForm(stanza) {
var _this5 = this;
2016-12-13 20:46:07 +01:00
/* Renders a form given an IQ stanza containing the current
* room configuration.
*
* Returns a promise which resolves once the user has
* either submitted the form, or canceled it.
*
* Parameters:
2017-06-23 20:25:33 +02:00
* (XMLElement) stanza: The IQ stanza containing the room
* config.
2016-12-13 20:46:07 +01:00
*/
2017-07-22 22:21:05 +02:00
var $body = this.$('.chatroom-body');
2016-12-13 20:46:07 +01:00
$body.children().addClass('hidden');
// Remove any existing forms
$body.find('form.chatroom-form').remove();
$body.append(tpl_chatroom_form());
2016-12-13 20:46:07 +01:00
2017-07-22 22:21:05 +02:00
var $form = $body.find('form.chatroom-form');
var $fieldset = $form.children('fieldset:first');
var $stanza = $(stanza),
2016-11-07 15:43:48 +01:00
$fields = $stanza.find('field'),
title = $stanza.find('title').text(),
instructions = $stanza.find('instructions').text();
$fieldset.find('span.spinner').remove();
$fieldset.append($('<legend>').text(title));
if (instructions && instructions !== title) {
$fieldset.append($('<p class="instructions">').text(instructions));
2016-03-07 18:54:07 +01:00
}
2016-11-07 15:43:48 +01:00
_.each($fields, function (field) {
$fieldset.append(utils.xForm2webForm($(field), $stanza));
});
$form.append('<fieldset></fieldset>');
$fieldset = $form.children('fieldset:last');
2017-07-22 22:21:05 +02:00
$fieldset.append("<input type=\"submit\" class=\"pure-button button-primary\" value=\"" + __('Save') + "\"/>");
$fieldset.append("<input type=\"button\" class=\"pure-button button-cancel\" value=\"" + __('Cancel') + "\"/>");
2016-12-13 20:46:07 +01:00
$fieldset.find('input[type=button]').on('click', function (ev) {
ev.preventDefault();
2017-07-22 22:21:05 +02:00
_this5.cancelConfiguration();
2016-12-13 20:46:07 +01:00
});
$form.on('submit', function (ev) {
ev.preventDefault();
2017-07-22 22:21:05 +02:00
_this5.saveConfiguration(ev.target).then(_this5.getRoomFeatures.bind(_this5));
2016-12-13 20:46:07 +01:00
});
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
sendConfiguration: function sendConfiguration(config, onSuccess, onError) {
2016-12-13 20:46:07 +01:00
/* Send an IQ stanza with the room configuration.
*
* Parameters:
* (Array) config: The room configuration
* (Function) onSuccess: Callback upon succesful IQ response
* The first parameter passed in is IQ containing the
* room configuration.
* The second is the response IQ from the server.
* (Function) onError: Callback upon error IQ response
* The first parameter passed in is IQ containing the
* room configuration.
* The second is the response IQ from the server.
*/
2017-07-22 22:21:05 +02:00
var iq = $iq({ to: this.model.get('jid'), type: "set" }).c("query", { xmlns: Strophe.NS.MUC_OWNER }).c("x", { xmlns: Strophe.NS.XFORM, type: "submit" });
_.each(config || [], function (node) {
iq.cnode(node).up();
});
2016-12-13 20:46:07 +01:00
onSuccess = _.isUndefined(onSuccess) ? _.noop : _.partial(onSuccess, iq.nodeTree);
onError = _.isUndefined(onError) ? _.noop : _.partial(onError, iq.nodeTree);
2017-02-03 13:51:07 +01:00
return _converse.connection.sendIQ(iq, onSuccess, onError);
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
saveConfiguration: function saveConfiguration(form) {
var _this6 = this;
2016-11-07 15:43:48 +01:00
2016-12-13 20:46:07 +01:00
/* Submit the room configuration form by sending an IQ
* stanza to the server.
*
* Returns a promise which resolves once the XMPP server
* has return a response IQ.
*
* Parameters:
* (HTMLElement) form: The configuration form DOM element.
*/
2017-07-22 22:21:05 +02:00
return new Promise(function (resolve, reject) {
var $inputs = $(form).find(':input:not([type=button]):not([type=submit])'),
configArray = [];
$inputs.each(function () {
configArray.push(utils.webForm2xForm(this));
2016-11-07 15:43:48 +01:00
});
2017-07-22 22:21:05 +02:00
_this6.sendConfiguration(configArray, resolve, reject);
_this6.$el.find('div.chatroom-form-container').hide(function (el) {
$(el).remove();
_this6.renderAfterTransition();
});
});
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
autoConfigureChatRoom: function autoConfigureChatRoom() {
var _this7 = this;
2016-03-16 12:49:35 +01:00
2016-11-30 17:27:20 +01:00
/* Automatically configure room based on the
* 'roomconfig' data on this view's model.
2016-12-13 20:46:07 +01:00
*
* Returns a promise which resolves once a response IQ has
* been received.
*
* Parameters:
* (XMLElement) stanza: IQ stanza from the server,
* containing the configuration.
2016-11-30 17:27:20 +01:00
*/
2017-07-22 22:21:05 +02:00
var that = this;
return new Promise(function (resolve, reject) {
_this7.fetchRoomConfiguration().then(function (stanza) {
var configArray = [],
fields = stanza.querySelectorAll('field'),
config = that.model.get('roomconfig');
var count = fields.length;
_.each(fields, function (field) {
var fieldname = field.getAttribute('var').replace('muc#roomconfig_', ''),
type = field.getAttribute('type');
var value = void 0;
if (fieldname in config) {
switch (type) {
case 'boolean':
value = config[fieldname] ? 1 : 0;
break;
case 'list-multi':
// TODO: we don't yet handle "list-multi" types
value = field.innerHTML;
break;
default:
value = config[fieldname];
}
field.innerHTML = $build('value').t(value);
2017-04-23 19:02:44 +02:00
}
2017-07-22 22:21:05 +02:00
configArray.push(field);
if (! --count) {
that.sendConfiguration(configArray, resolve, reject);
}
});
2017-04-23 19:02:44 +02:00
});
2016-11-30 17:27:20 +01:00
});
},
2017-07-22 22:21:05 +02:00
cancelConfiguration: function cancelConfiguration() {
var _this8 = this;
2016-11-30 17:27:20 +01:00
2016-12-13 20:46:07 +01:00
/* Remove the configuration form without submitting and
* return to the chat view.
*/
2017-07-22 22:21:05 +02:00
this.$el.find('div.chatroom-form-container').hide(function (el) {
$(el).remove();
_this8.renderAfterTransition();
});
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
fetchRoomConfiguration: function fetchRoomConfiguration(handler) {
var _this9 = this,
_arguments = arguments;
2016-05-03 17:37:10 +02:00
2016-12-13 20:46:07 +01:00
/* Send an IQ stanza to fetch the room configuration data.
* Returns a promise which resolves once the response IQ
* has been received.
*
* Parameters:
* (Function) handler: The handler for the response IQ
*/
2017-07-22 22:21:05 +02:00
return new Promise(function (resolve, reject) {
_converse.connection.sendIQ($iq({
'to': _this9.model.get('jid'),
2016-11-30 17:27:20 +01:00
'type': "get"
2017-07-22 22:21:05 +02:00
}).c("query", { xmlns: Strophe.NS.MUC_OWNER }), function (iq) {
2016-12-13 20:46:07 +01:00
if (handler) {
2017-07-22 22:21:05 +02:00
handler.apply(_this9, _arguments);
2016-12-13 20:46:07 +01:00
}
2017-07-22 22:21:05 +02:00
resolve(iq);
}, reject // errback
);
});
},
parseRoomFeatures: function parseRoomFeatures(iq) {
/* See http://xmpp.org/extensions/xep-0045.html#disco-roominfo
*
* <identity
* category='conference'
* name='A Dark Cave'
* type='text'/>
* <feature var='http://jabber.org/protocol/muc'/>
* <feature var='muc_passwordprotected'/>
* <feature var='muc_hidden'/>
* <feature var='muc_temporary'/>
* <feature var='muc_open'/>
* <feature var='muc_unmoderated'/>
* <feature var='muc_nonanonymous'/>
* <feature var='urn:xmpp:mam:0'/>
*/
var features = {
'features_fetched': true,
'name': iq.querySelector('identity').getAttribute('name')
};
_.each(iq.querySelectorAll('feature'), function (field) {
var fieldname = field.getAttribute('var');
if (!fieldname.startsWith('muc_')) {
if (fieldname === Strophe.NS.MAM) {
features.mam_enabled = true;
}
return;
}
features[fieldname.replace('muc_', '')] = true;
});
var desc_field = iq.querySelector('field[var="muc#roominfo_description"] value');
if (!_.isNull(desc_field)) {
features.description = desc_field.textContent;
}
this.model.save(features);
2016-12-13 20:46:07 +01:00
},
2017-07-22 22:21:05 +02:00
getRoomFeatures: function getRoomFeatures() {
var _this10 = this;
2016-12-13 20:46:07 +01:00
/* Fetch the room disco info, parse it and then
* save it on the Backbone.Model of this chat rooms.
*/
2017-07-22 22:21:05 +02:00
return new Promise(function (resolve, reject) {
_converse.connection.disco.info(_this10.model.get('jid'), null, _.flow(_this10.parseRoomFeatures.bind(_this10), resolve), function () {
reject(new Error("Could not parse the room features"));
}, 5000);
});
2016-12-13 20:46:07 +01:00
},
2017-07-22 22:21:05 +02:00
getAndRenderConfigurationForm: function getAndRenderConfigurationForm(ev) {
2016-12-13 20:46:07 +01:00
/* Start the process of configuring a chat room, either by
* rendering a configuration form, or by auto-configuring
* based on the "roomconfig" data stored on the
* Backbone.Model.
*
* Stores the new configuration on the Backbone.Model once
* completed.
*
* Paremeters:
* (Event) ev: DOM event that might be passed in if this
* method is called due to a user action. In this
* case, auto-configure won't happen, regardless of
* the settings.
*/
2017-04-23 19:02:44 +02:00
this.showSpinner();
2017-07-22 22:21:05 +02:00
this.fetchRoomConfiguration().then(this.renderConfigurationForm.bind(this));
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
submitNickname: function submitNickname(ev) {
2016-12-13 20:46:07 +01:00
/* Get the nickname value from the form and then join the
* chat room with it.
*/
2016-11-07 15:43:48 +01:00
ev.preventDefault();
2017-06-23 20:25:33 +02:00
var nick_el = ev.target.nick;
var nick = nick_el.value;
2016-07-28 18:06:31 +02:00
if (!nick) {
2017-06-23 20:25:33 +02:00
nick_el.classList.add('error');
2016-11-07 15:43:48 +01:00
return;
2017-07-22 22:21:05 +02:00
} else {
2017-06-23 20:25:33 +02:00
nick_el.classList.remove('error');
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
this.$el.find('.chatroom-form-container').replaceWith(tpl_spinner);
2016-11-07 15:43:48 +01:00
this.join(nick);
},
2017-07-22 22:21:05 +02:00
checkForReservedNick: function checkForReservedNick() {
2016-11-07 15:43:48 +01:00
/* User service-discovery to ask the XMPP server whether
* this user has a reserved nickname for this room.
* If so, we'll use that, otherwise we render the nickname
* form.
*/
this.showSpinner();
2017-07-22 22:21:05 +02:00
_converse.connection.sendIQ($iq({
'to': this.model.get('jid'),
'from': _converse.connection.jid,
'type': "get"
}).c("query", {
'xmlns': Strophe.NS.DISCO_INFO,
'node': 'x-roomuser-item'
}), this.onNickNameFound.bind(this), this.onNickNameNotFound.bind(this));
2016-12-13 20:46:07 +01:00
return this;
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
onNickNameFound: function onNickNameFound(iq) {
2016-11-07 15:43:48 +01:00
/* We've received an IQ response from the server which
* might contain the user's reserved nickname.
2016-12-13 20:46:07 +01:00
* If no nickname is found we either render a form for
* them to specify one, or we try to join the room with the
* node of the user's JID.
*
* Parameters:
* (XMLElement) iq: The received IQ stanza
2016-11-07 15:43:48 +01:00
*/
2017-07-22 22:21:05 +02:00
var nick = $(iq).find('query[node="x-roomuser-item"] identity').attr('name');
2016-11-07 15:43:48 +01:00
if (!nick) {
this.onNickNameNotFound();
} else {
this.join(nick);
}
},
2017-07-22 22:21:05 +02:00
onNickNameNotFound: function onNickNameNotFound(message) {
2017-02-03 13:51:07 +01:00
if (_converse.muc_nickname_from_jid) {
2016-11-07 15:43:48 +01:00
// We try to enter the room with the node part of
// the user's JID.
2017-02-03 13:51:07 +01:00
this.join(Strophe.unescapeNode(Strophe.getNodeFromJid(_converse.bare_jid)));
2016-07-28 18:06:31 +02:00
} else {
2016-11-07 15:43:48 +01:00
this.renderNicknameForm(message);
2016-07-28 18:06:31 +02:00
}
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
getDefaultNickName: function getDefaultNickName() {
2016-11-07 15:43:48 +01:00
/* The default nickname (used when muc_nickname_from_jid is true)
* is the node part of the user's JID.
* We put this in a separate method so that it can be
* overridden by plugins.
*/
2017-02-03 13:51:07 +01:00
return Strophe.unescapeNode(Strophe.getNodeFromJid(_converse.bare_jid));
},
2017-07-22 22:21:05 +02:00
onNicknameClash: function onNicknameClash(presence) {
2016-11-07 15:43:48 +01:00
/* When the nickname is already taken, we either render a
* form for the user to choose a new nickname, or we
* try to make the nickname unique by adding an integer to
* it. So john will become john-2, and then john-3 and so on.
*
* Which option is take depends on the value of
* muc_nickname_from_jid.
*/
2017-02-03 13:51:07 +01:00
if (_converse.muc_nickname_from_jid) {
2016-11-07 15:43:48 +01:00
var nick = presence.getAttribute('from').split('/')[1];
if (nick === this.getDefaultNickName()) {
this.join(nick + '-2');
} else {
2017-07-22 22:21:05 +02:00
var del = nick.lastIndexOf("-");
var num = nick.substring(del + 1, nick.length);
this.join(nick.substring(0, del + 1) + String(Number(num) + 1));
2016-11-07 15:43:48 +01:00
}
2016-06-20 21:11:43 +02:00
} else {
2017-07-22 22:21:05 +02:00
this.renderNicknameForm(__("The nickname you chose is reserved or " + "currently in use, please choose a different one."));
2016-06-20 21:11:43 +02:00
}
},
2017-07-22 22:21:05 +02:00
renderNicknameForm: function renderNicknameForm(message) {
2016-11-07 15:43:48 +01:00
/* Render a form which allows the user to choose their
* nickname.
*/
this.$('.chatroom-body').children().addClass('hidden');
this.$('span.centered.spinner').remove();
if (!_.isString(message)) {
2016-11-07 15:43:48 +01:00
message = '';
}
2017-07-22 22:21:05 +02:00
this.$('.chatroom-body').append(tpl_chatroom_nickname_form({
heading: __('Please choose your nickname'),
label_nickname: __('Nickname'),
label_join: __('Enter room'),
validation_message: message
}));
this.model.save('connection_status', converse.ROOMSTATUS.NICKNAME_REQUIRED);
2016-11-07 15:43:48 +01:00
this.$('.chatroom-form').on('submit', this.submitNickname.bind(this));
},
2017-07-22 22:21:05 +02:00
submitPassword: function submitPassword(ev) {
2016-11-07 15:43:48 +01:00
ev.preventDefault();
var password = this.$el.find('.chatroom-form').find('input[type=password]').val();
2017-06-23 20:25:33 +02:00
this.$el.find('.chatroom-form-container').replaceWith(tpl_spinner);
2016-11-07 15:43:48 +01:00
this.join(this.model.get('nick'), password);
},
2017-07-22 22:21:05 +02:00
renderPasswordForm: function renderPasswordForm() {
2016-11-07 15:43:48 +01:00
this.$('.chatroom-body').children().addClass('hidden');
this.$('span.centered.spinner').remove();
2017-07-22 22:21:05 +02:00
this.$('.chatroom-body').append(tpl_chatroom_password_form({
heading: __('This chatroom requires a password'),
label_password: __('Password: '),
label_submit: __('Submit')
}));
this.model.save('connection_status', converse.ROOMSTATUS.PASSWORD_REQUIRED);
2016-11-07 15:43:48 +01:00
this.$('.chatroom-form').on('submit', this.submitPassword.bind(this));
},
2017-07-22 22:21:05 +02:00
showDisconnectMessage: function showDisconnectMessage(msg) {
2016-11-07 15:43:48 +01:00
this.$('.chat-area').addClass('hidden');
this.$('.occupants').addClass('hidden');
this.$('span.centered.spinner').remove();
2017-04-23 19:02:44 +02:00
this.$('.chatroom-body').append(tpl_chatroom_disconnect({
'disconnect_message': msg
}));
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
getMessageFromStatus: function getMessageFromStatus(stat, stanza, is_self) {
2016-12-13 20:46:07 +01:00
/* Parameters:
* (XMLElement) stat: A <status> element.
* (Boolean) is_self: Whether the element refers to the
* current user.
* (XMLElement) stanza: The original stanza received.
*/
2017-07-22 22:21:05 +02:00
var code = stat.getAttribute('code');
if (code === '110') {
return;
}
if (code in _converse.muc.info_messages) {
2017-02-03 13:51:07 +01:00
return _converse.muc.info_messages[code];
}
2017-07-22 22:21:05 +02:00
var nick = void 0;
if (!is_self) {
if (code in _converse.muc.action_info_messages) {
nick = Strophe.getResourceFromJid(stanza.getAttribute('from'));
return __(_converse.muc.action_info_messages[code], nick);
2016-11-30 17:27:20 +01:00
}
} else if (code in _converse.muc.new_nickname_messages) {
if (is_self && code === "210") {
nick = Strophe.getResourceFromJid(stanza.getAttribute('from'));
} else if (is_self && code === "303") {
nick = stanza.querySelector('x item').getAttribute('nick');
}
return __(_converse.muc.new_nickname_messages[code], nick);
2016-11-30 17:27:20 +01:00
}
return;
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
saveAffiliationAndRole: function saveAffiliationAndRole(pres) {
2016-12-13 20:46:07 +01:00
/* Parse the presence stanza for the current user's
* affiliation.
*
* Parameters:
* (XMLElement) pres: A <presence> stanza.
*/
2017-07-22 22:21:05 +02:00
var item = sizzle("x[xmlns=\"" + Strophe.NS.MUC_USER + "\"] item", pres).pop();
2017-07-05 11:59:55 +02:00
var is_self = pres.querySelector("status[code='110']");
if (is_self && !_.isNil(item)) {
2016-12-13 20:46:07 +01:00
var affiliation = item.getAttribute('affiliation');
var role = item.getAttribute('role');
if (affiliation) {
2017-07-22 22:21:05 +02:00
this.model.save({ 'affiliation': affiliation });
2016-12-13 20:46:07 +01:00
}
if (role) {
2017-07-22 22:21:05 +02:00
this.model.save({ 'role': role });
2016-12-13 20:46:07 +01:00
}
}
},
2017-07-22 22:21:05 +02:00
parseXUserElement: function parseXUserElement(x, stanza, is_self) {
2016-11-30 17:27:20 +01:00
/* Parse the passed-in <x xmlns='http://jabber.org/protocol/muc#user'>
* element and construct a map containing relevant
* information.
*/
2016-12-13 20:46:07 +01:00
// 1. Get notification messages based on the <status> elements.
2016-11-30 17:27:20 +01:00
var statuses = x.querySelectorAll('status');
2016-12-13 20:46:07 +01:00
var mapper = _.partial(this.getMessageFromStatus, _, stanza, is_self);
var notification = {};
var messages = _.reject(_.map(statuses, mapper), _.isUndefined);
if (messages.length) {
notification.messages = messages;
}
2016-12-13 20:46:07 +01:00
// 2. Get disconnection messages based on the <status> elements
var codes = _.invokeMap(statuses, Element.prototype.getAttribute, 'code');
2017-02-03 13:51:07 +01:00
var disconnection_codes = _.intersection(codes, _.keys(_converse.muc.disconnect_messages));
2016-11-30 17:27:20 +01:00
var disconnected = is_self && disconnection_codes.length > 0;
if (disconnected) {
notification.disconnected = true;
2017-02-03 13:51:07 +01:00
notification.disconnection_message = _converse.muc.disconnect_messages[disconnection_codes[0]];
2016-11-30 17:27:20 +01:00
}
2016-12-13 20:46:07 +01:00
// 3. Find the reason and actor from the <item> element
var item = x.querySelector('item');
// By using querySelector above, we assume here there is
// one <item> per <x xmlns='http://jabber.org/protocol/muc#user'>
// element. This appears to be a safe assumption, since
// each <x/> element pertains to a single user.
if (!_.isNull(item)) {
var reason = item.querySelector('reason');
if (reason) {
notification.reason = reason ? reason.textContent : undefined;
}
var actor = item.querySelector('actor');
if (actor) {
notification.actor = actor ? actor.getAttribute('nick') : undefined;
}
}
2016-11-30 17:27:20 +01:00
return notification;
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
displayNotificationsforUser: function displayNotificationsforUser(notification) {
var _this11 = this;
2016-11-07 15:43:48 +01:00
2016-11-30 17:27:20 +01:00
/* Given the notification object generated by
* parseXUserElement, display any relevant messages and
* information to the user.
2016-11-07 15:43:48 +01:00
*/
2016-11-30 17:27:20 +01:00
if (notification.disconnected) {
this.showDisconnectMessage(notification.disconnection_message);
if (notification.actor) {
2017-04-04 17:26:06 +02:00
this.showDisconnectMessage(__(___('This action was done by %1$s.'), notification.actor));
2016-11-07 15:43:48 +01:00
}
2016-11-30 17:27:20 +01:00
if (notification.reason) {
2017-04-04 17:26:06 +02:00
this.showDisconnectMessage(__(___('The reason given is: "%1$s".'), notification.reason));
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
this.model.save('connection_status', converse.ROOMSTATUS.DISCONNECTED);
2016-11-07 15:43:48 +01:00
return;
}
2016-11-30 17:27:20 +01:00
_.each(notification.messages, function (message) {
2017-07-22 22:21:05 +02:00
_this11.$content.append(tpl_info({ 'message': message }));
2016-11-30 17:27:20 +01:00
});
if (notification.reason) {
2017-07-22 22:21:05 +02:00
this.showStatusNotification(__("The reason given is: \"" + notification.reason + "\""), true);
2016-11-07 15:43:48 +01:00
}
2016-11-30 17:27:20 +01:00
if (notification.messages.length) {
this.scrollDown();
}
2016-11-30 17:27:20 +01:00
},
2017-07-22 22:21:05 +02:00
getJoinLeaveMessages: function getJoinLeaveMessages(stanza) {
/* Parse the given stanza and return notification messages
* for join/leave events.
*/
// XXX: some mangling required to make the returned
// result look like the structure returned by
// parseXUserElement. Not nice...
var nick = Strophe.getResourceFromJid(stanza.getAttribute('from'));
2017-03-05 10:45:18 +01:00
var stat = stanza.querySelector('status');
if (stanza.getAttribute('type') === 'unavailable') {
if (!_.isNull(stat) && stat.textContent) {
2017-07-22 22:21:05 +02:00
return [{ 'messages': [__(nick + ' has left the room. "' + stat.textContent + '"')] }];
} else {
2017-07-22 22:21:05 +02:00
return [{ 'messages': [__(nick + ' has left the room')] }];
}
}
2017-07-22 22:21:05 +02:00
if (!this.occupantsview.model.find({ 'nick': nick })) {
2017-03-05 10:45:18 +01:00
// Only show join message if we don't already have the
// occupant model. Doing so avoids showing duplicate
// join messages.
if (!_.isNull(stat) && stat.textContent) {
2017-07-22 22:21:05 +02:00
return [{ 'messages': [__(nick + ' has joined the room. "' + stat.textContent + '"')] }];
2017-03-05 10:45:18 +01:00
} else {
2017-07-22 22:21:05 +02:00
return [{ 'messages': [__(nick + ' has joined the room.')] }];
2017-03-05 10:45:18 +01:00
}
}
},
2017-07-22 22:21:05 +02:00
showStatusMessages: function showStatusMessages(stanza) {
2016-11-30 17:27:20 +01:00
/* Check for status codes and communicate their purpose to the user.
* See: http://xmpp.org/registrar/mucstatus.html
2016-12-13 20:46:07 +01:00
*
* Parameters:
* (XMLElement) stanza: The message or presence stanza
* containing the status codes.
2016-11-30 17:27:20 +01:00
*/
2017-07-22 22:21:05 +02:00
var elements = sizzle("x[xmlns=\"" + Strophe.NS.MUC_USER + "\"]", stanza);
2016-12-13 20:46:07 +01:00
var is_self = stanza.querySelectorAll("status[code='110']").length;
var iteratee = _.partial(this.parseXUserElement.bind(this), _, stanza, is_self);
var notifications = _.reject(_.map(elements, iteratee), _.isEmpty);
2017-07-22 22:21:05 +02:00
if (_.isEmpty(notifications) && _converse.muc_show_join_leave && stanza.nodeName === 'presence' && this.model.get('connection_status') === converse.ROOMSTATUS.ENTERED) {
notifications = this.getJoinLeaveMessages(stanza);
}
2016-11-30 17:27:20 +01:00
_.each(notifications, this.displayNotificationsforUser.bind(this));
2016-12-13 20:46:07 +01:00
return stanza;
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
showErrorMessage: function showErrorMessage(presence) {
// We didn't enter the room, so we must remove it from the MUC add-on
var error = presence.querySelector('error');
if (error.getAttribute('type') === 'auth') {
if (!_.isNull(error.querySelector('not-authorized'))) {
2016-11-07 15:43:48 +01:00
this.renderPasswordForm();
} else if (!_.isNull(error.querySelector('registration-required'))) {
2017-04-23 19:02:44 +02:00
this.showDisconnectMessage(__('You are not on the member list of this room.'));
} else if (!_.isNull(error.querySelector('forbidden'))) {
2017-04-23 19:02:44 +02:00
this.showDisconnectMessage(__('You have been banned from this room.'));
2016-11-07 15:43:48 +01:00
}
} else if (error.getAttribute('type') === 'modify') {
if (!_.isNull(error.querySelector('jid-malformed'))) {
2017-04-23 19:02:44 +02:00
this.showDisconnectMessage(__('No nickname was specified.'));
2016-11-07 15:43:48 +01:00
}
} else if (error.getAttribute('type') === 'cancel') {
if (!_.isNull(error.querySelector('not-allowed'))) {
2017-04-23 19:02:44 +02:00
this.showDisconnectMessage(__('You are not allowed to create new rooms.'));
} else if (!_.isNull(error.querySelector('not-acceptable'))) {
2017-04-23 19:02:44 +02:00
this.showDisconnectMessage(__("Your nickname doesn't conform to this room's policies."));
} else if (!_.isNull(error.querySelector('conflict'))) {
2016-11-07 15:43:48 +01:00
this.onNicknameClash(presence);
} else if (!_.isNull(error.querySelector('item-not-found'))) {
2017-04-23 19:02:44 +02:00
this.showDisconnectMessage(__("This room does not (yet) exist."));
} else if (!_.isNull(error.querySelector('service-unavailable'))) {
2017-04-23 19:02:44 +02:00
this.showDisconnectMessage(__("This room has reached its maximum number of occupants."));
2016-11-07 15:43:48 +01:00
}
}
},
2017-07-22 22:21:05 +02:00
renderAfterTransition: function renderAfterTransition() {
2017-03-05 10:45:18 +01:00
/* Rerender the room after some kind of transition. For
* example after the spinner has been removed or after a
* form has been submitted and removed.
*/
2017-07-22 22:21:05 +02:00
if (this.model.get('connection_status') == converse.ROOMSTATUS.NICKNAME_REQUIRED) {
2017-03-05 10:45:18 +01:00
this.renderNicknameForm();
2017-07-22 22:21:05 +02:00
} else if (this.model.get('connection_status') == converse.ROOMSTATUS.PASSWORD_REQUIRED) {
2017-04-04 17:26:06 +02:00
this.renderPasswordForm();
2017-03-05 10:45:18 +01:00
} else {
this.$el.find('.chat-area').removeClass('hidden');
this.$el.find('.occupants').removeClass('hidden');
this.occupantsview.setOccupantsHeight();
this.scrollDown();
}
},
2017-07-22 22:21:05 +02:00
showSpinner: function showSpinner() {
2017-04-04 17:26:06 +02:00
this.$('.chatroom-body').children().addClass('hidden');
2017-06-23 20:25:33 +02:00
this.$el.find('.chatroom-body').prepend(tpl_spinner);
2017-04-04 17:26:06 +02:00
},
2017-07-22 22:21:05 +02:00
hideSpinner: function hideSpinner() {
2016-11-07 15:43:48 +01:00
/* Check if the spinner is being shown and if so, hide it.
* Also make sure then that the chat area and occupants
* list are both visible.
2016-07-28 18:06:31 +02:00
*/
2017-03-05 10:45:18 +01:00
var spinner = this.el.querySelector('.spinner');
if (!_.isNull(spinner)) {
spinner.parentNode.removeChild(spinner);
this.renderAfterTransition();
2016-11-07 15:43:48 +01:00
}
return this;
2016-07-28 18:06:31 +02:00
},
2017-07-22 22:21:05 +02:00
onOwnChatRoomPresence: function onOwnChatRoomPresence(pres) {
2017-04-23 19:02:44 +02:00
/* Handles a received presence relating to the current
* user.
*
* For locked rooms (which are by definition "new"), the
* room will either be auto-configured or created instantly
* (with default config) or a configuration room will be
* rendered.
2016-12-13 20:46:07 +01:00
*
2017-04-23 19:02:44 +02:00
* If the room is not locked, then the room will be
* auto-configured only if applicable and if the current
* user is the room's owner.
*
* Parameters:
* (XMLElement) pres: The stanza
2016-12-13 20:46:07 +01:00
*/
2017-04-23 19:02:44 +02:00
this.saveAffiliationAndRole(pres);
var locked_room = pres.querySelector("status[code='201']");
if (locked_room) {
if (this.model.get('auto_configure')) {
this.autoConfigureChatRoom().then(this.getRoomFeatures.bind(this));
} else if (_converse.muc_instant_rooms) {
// Accept default configuration
this.saveConfiguration().then(this.getRoomFeatures.bind(this));
} else {
this.getAndRenderConfigurationForm();
return; // We haven't yet entered the room, so bail here.
}
} else if (!this.model.get('features_fetched')) {
// The features for this room weren't fetched.
2017-06-23 20:25:33 +02:00
// That must mean it's a new room without locking
2017-04-23 19:02:44 +02:00
// (in which case Prosody doesn't send a 201 status),
// otherwise the features would have been fetched in
// the "initialize" method already.
if (this.model.get('affiliation') === 'owner' && this.model.get('auto_configure')) {
this.autoConfigureChatRoom().then(this.getRoomFeatures.bind(this));
} else {
this.getRoomFeatures();
}
}
2017-07-22 22:21:05 +02:00
this.model.save('connection_status', converse.ROOMSTATUS.ENTERED);
2016-12-13 20:46:07 +01:00
},
2017-07-22 22:21:05 +02:00
onChatRoomPresence: function onChatRoomPresence(pres) {
2016-12-13 20:46:07 +01:00
/* Handles all MUC presence stanzas.
*
* Parameters:
* (XMLElement) pres: The stanza
*/
if (pres.getAttribute('type') === 'error') {
2017-07-22 22:21:05 +02:00
this.model.save('connection_status', converse.ROOMSTATUS.DISCONNECTED);
2016-11-07 15:43:48 +01:00
this.showErrorMessage(pres);
2016-12-13 20:46:07 +01:00
return true;
}
var is_self = pres.querySelector("status[code='110']");
2017-04-23 19:02:44 +02:00
if (is_self && pres.getAttribute('type') !== 'unavailable') {
this.onOwnChatRoomPresence(pres);
2016-12-13 20:46:07 +01:00
}
this.hideSpinner().showStatusMessages(pres);
2017-03-05 10:45:18 +01:00
// This must be called after showStatusMessages so that
// "join" messages are correctly shown.
2016-11-07 15:43:48 +01:00
this.occupantsview.updateOccupantsOnPresence(pres);
2017-07-22 22:21:05 +02:00
if (this.model.get('role') !== 'none' && this.model.get('connection_status') === converse.ROOMSTATUS.CONNECTING) {
this.model.save('connection_status', converse.ROOMSTATUS.CONNECTED);
2016-12-13 20:46:07 +01:00
}
2016-11-07 15:43:48 +01:00
return true;
},
2017-07-22 22:21:05 +02:00
setChatRoomSubject: function setChatRoomSubject(sender, subject) {
// For translators: the %1$s and %2$s parts will get
// replaced by the user and topic text respectively
2016-11-07 15:43:48 +01:00
// Example: Topic set by JC Brand to: Hello World!
2017-07-22 22:21:05 +02:00
this.$content.append(tpl_info({ 'message': __('Topic set by %1$s to: %2$s', sender, subject) }));
2016-11-07 15:43:48 +01:00
this.scrollDown();
},
2017-07-22 22:21:05 +02:00
isDuplicateBasedOnTime: function isDuplicateBasedOnTime(message) {
/* Checks whether a received messages is actually a
* duplicate based on whether it has a "ts" attribute
* with a unix timestamp.
*
* This is used for better integration with Slack's XMPP
* gateway, which doesn't use message IDs but instead the
* aforementioned "ts" attributes.
*/
var entity = _converse.disco_entities.get(_converse.domain);
if (entity.identities.where({ 'name': "Slack-XMPP" })) {
var ts = message.getAttribute('ts');
if (_.isNull(ts)) {
return false;
} else {
return this.model.messages.where({
'sender': 'me',
'message': this.model.getMessageBody(message)
}).filter(function (msg) {
return Math.abs(moment(msg.get('time')).diff(moment.unix(ts))) < 5000;
}).length > 0;
}
}
return false;
},
isDuplicate: function isDuplicate(message) {
var msgid = message.getAttribute('id'),
jid = message.getAttribute('from'),
resource = Strophe.getResourceFromJid(jid),
sender = resource && Strophe.unescapeNode(resource) || '';
if (msgid) {
return this.model.messages.filter(
// Some bots (like HAL in the prosody chatroom)
// respond to commands with the same ID as the
// original message. So we also check the sender.
function (msg) {
return msg.get('msgid') === msgid && msg.get('fullname') === sender;
}).length > 0;
}
return this.isDuplicateBasedOnTime(message);
},
onChatRoomMessage: function onChatRoomMessage(message) {
2016-12-13 20:46:07 +01:00
/* Given a <message> stanza, create a message
* Backbone.Model if appropriate.
*
* Parameters:
* (XMLElement) msg: The received message stanza
*/
2017-02-03 13:51:07 +01:00
var original_stanza = message,
2017-07-22 22:21:05 +02:00
forwarded = message.querySelector('forwarded');
var delay = void 0;
2017-02-03 13:51:07 +01:00
if (!_.isNull(forwarded)) {
message = forwarded.querySelector('message');
delay = forwarded.querySelector('delay');
}
var jid = message.getAttribute('from'),
2016-11-07 15:43:48 +01:00
resource = Strophe.getResourceFromJid(jid),
sender = resource && Strophe.unescapeNode(resource) || '',
2017-07-22 22:21:05 +02:00
subject = _.propertyOf(message.querySelector('subject'))('textContent');
if (this.isDuplicate(message)) {
2016-11-07 15:43:48 +01:00
return true;
}
if (subject) {
this.setChatRoomSubject(sender, subject);
}
if (sender === '') {
return true;
}
2017-06-23 20:25:33 +02:00
this.model.incrementUnreadMsgCounter(original_stanza);
2017-02-03 13:51:07 +01:00
this.model.createMessage(message, delay, original_stanza);
2016-11-07 15:43:48 +01:00
if (sender !== this.model.get('nick')) {
// We only emit an event if it's not our own message
2017-07-22 22:21:05 +02:00
_converse.emit('message', { 'stanza': original_stanza, 'chatbox': this.model });
2016-08-12 22:52:21 +02:00
}
2016-11-07 15:43:48 +01:00
return true;
}
});
2017-02-03 13:51:07 +01:00
_converse.ChatRoomOccupant = Backbone.Model.extend({
2017-07-22 22:21:05 +02:00
initialize: function initialize(attributes) {
2016-11-07 15:43:48 +01:00
this.set(_.extend({
2017-07-22 22:21:05 +02:00
'id': _converse.connection.getUniqueId()
2016-11-07 15:43:48 +01:00
}, attributes));
}
});
2017-02-03 13:51:07 +01:00
_converse.ChatRoomOccupantView = Backbone.View.extend({
2016-11-07 15:43:48 +01:00
tagName: 'li',
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-11-07 15:43:48 +01:00
this.model.on('change', this.render, this);
this.model.on('destroy', this.destroy, this);
},
2017-07-22 22:21:05 +02:00
render: function render() {
2017-03-05 10:45:18 +01:00
var show = this.model.get('show') || 'online';
2017-07-22 22:21:05 +02:00
var new_el = tpl_occupant(_.extend({ 'jid': '',
'show': show,
'hint_show': _converse.PRETTY_CHAT_STATUS[show],
'hint_occupant': __("Click to mention " + this.model.get('nick') + " in your message."),
'desc_moderator': __('This user is a moderator.'),
'desc_occupant': __('This user can send messages in this room.'),
'desc_visitor': __('This user can NOT send messages in this room.')
}, this.model.toJSON()));
2016-11-07 15:43:48 +01:00
var $parents = this.$el.parents();
if ($parents.length) {
this.$el.replaceWith(new_el);
2017-07-22 22:21:05 +02:00
this.setElement($parents.first().children("#" + this.model.get('id')), true);
2016-11-07 15:43:48 +01:00
this.delegateEvents();
} else {
this.$el.replaceWith(new_el);
this.setElement(new_el, true);
}
2016-11-07 15:43:48 +01:00
return this;
},
2017-07-22 22:21:05 +02:00
destroy: function destroy() {
2016-11-07 15:43:48 +01:00
this.$el.remove();
}
});
2017-02-03 13:51:07 +01:00
_converse.ChatRoomOccupants = Backbone.Collection.extend({
model: _converse.ChatRoomOccupant
2016-11-07 15:43:48 +01:00
});
2017-02-03 13:51:07 +01:00
_converse.ChatRoomOccupantsView = Backbone.Overview.extend({
2016-11-07 15:43:48 +01:00
tagName: 'div',
className: 'occupants',
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-11-07 15:43:48 +01:00
this.model.on("add", this.onOccupantAdded, this);
this.chatroomview = this.model.chatroomview;
2017-03-05 10:45:18 +01:00
this.chatroomview.model.on('change:open', this.renderInviteWidget, this);
this.chatroomview.model.on('change:affiliation', this.renderInviteWidget, this);
2017-04-04 17:26:06 +02:00
this.chatroomview.model.on('change:hidden', this.onFeatureChanged, this);
this.chatroomview.model.on('change:mam_enabled', this.onFeatureChanged, this);
this.chatroomview.model.on('change:membersonly', this.onFeatureChanged, this);
this.chatroomview.model.on('change:moderated', this.onFeatureChanged, this);
this.chatroomview.model.on('change:nonanonymous', this.onFeatureChanged, this);
this.chatroomview.model.on('change:open', this.onFeatureChanged, this);
this.chatroomview.model.on('change:passwordprotected', this.onFeatureChanged, this);
this.chatroomview.model.on('change:persistent', this.onFeatureChanged, this);
this.chatroomview.model.on('change:public', this.onFeatureChanged, this);
this.chatroomview.model.on('change:semianonymous', this.onFeatureChanged, this);
this.chatroomview.model.on('change:temporary', this.onFeatureChanged, this);
this.chatroomview.model.on('change:unmoderated', this.onFeatureChanged, this);
this.chatroomview.model.on('change:unsecured', this.onFeatureChanged, this);
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
render: function render() {
this.el.innerHTML = tpl_chatroom_sidebar(_.extend(this.chatroomview.model.toJSON(), {
'allow_muc_invitations': _converse.allow_muc_invitations,
'label_occupants': __('Occupants')
}));
2017-02-03 13:51:07 +01:00
if (_converse.allow_muc_invitations) {
2017-07-22 22:21:05 +02:00
_converse.api.waitUntil('rosterContactsFetched').then(this.renderInviteWidget.bind(this));
}
return this.renderRoomFeatures();
},
2017-07-22 22:21:05 +02:00
renderInviteWidget: function renderInviteWidget() {
2017-03-05 10:45:18 +01:00
var form = this.el.querySelector('form.room-invite');
if (this.shouldInviteWidgetBeShown()) {
if (_.isNull(form)) {
var heading = this.el.querySelector('.occupants-heading');
form = tpl_chatroom_invite({
2017-07-22 22:21:05 +02:00
'error_message': null,
'label_invitation': __('Invite')
2017-03-05 10:45:18 +01:00
});
heading.insertAdjacentHTML('afterend', form);
this.initInviteWidget();
}
} else {
if (!_.isNull(form)) {
form.remove();
}
}
return this;
},
2017-07-22 22:21:05 +02:00
renderRoomFeatures: function renderRoomFeatures() {
2017-03-05 10:45:18 +01:00
var picks = _.pick(this.chatroomview.model.attributes, ROOM_FEATURES),
2017-07-22 22:21:05 +02:00
iteratee = function iteratee(a, v) {
return a || v;
},
2017-03-05 10:45:18 +01:00
el = this.el.querySelector('.chatroom-features');
2017-07-22 22:21:05 +02:00
el.innerHTML = tpl_chatroom_features(_.extend(this.chatroomview.model.toJSON(), {
'has_features': _.reduce(_.values(picks), iteratee),
'label_features': __('Features'),
'label_hidden': __('Hidden'),
'label_mam_enabled': __('Message archiving'),
'label_membersonly': __('Members only'),
'label_moderated': __('Moderated'),
'label_nonanonymous': __('Non-anonymous'),
'label_open': __('Open'),
'label_passwordprotected': __('Password protected'),
'label_persistent': __('Persistent'),
'label_public': __('Public'),
'label_semianonymous': __('Semi-anonymous'),
'label_temporary': __('Temporary'),
'label_unmoderated': __('Unmoderated'),
'label_unsecured': __('Unsecured'),
'tt_hidden': __('This room is not publicly searchable'),
'tt_mam_enabled': __('Messages are archived on the server'),
'tt_membersonly': __('This room is restricted to members only'),
'tt_moderated': __('This room is being moderated'),
'tt_nonanonymous': __('All other room occupants can see your XMPP username'),
'tt_open': __('Anyone can join this room'),
'tt_passwordprotected': __('This room requires a password before entry'),
'tt_persistent': __('This room persists even if it\'s unoccupied'),
'tt_public': __('This room is publicly searchable'),
'tt_semianonymous': __('Only moderators can see your XMPP username'),
'tt_temporary': __('This room will disappear once the last person leaves'),
'tt_unmoderated': __('This room is not being moderated'),
'tt_unsecured': __('This room does not require a password upon entry')
}));
2017-03-05 10:45:18 +01:00
this.setOccupantsHeight();
2016-03-16 12:49:35 +01:00
return this;
},
2017-07-22 22:21:05 +02:00
onFeatureChanged: function onFeatureChanged(model) {
2017-04-04 17:26:06 +02:00
/* When a feature has been changed, it's logical opposite
* must be set to the opposite value.
*
* So for example, if "temporary" was set to "false", then
* "persistent" will be set to "true" in this method.
*
* Additionally a debounced render method is called to make
* sure the features widget gets updated.
*/
if (_.isUndefined(this.debouncedRenderRoomFeatures)) {
2017-07-22 22:21:05 +02:00
this.debouncedRenderRoomFeatures = _.debounce(this.renderRoomFeatures, 100, { 'leading': false });
2017-04-04 17:26:06 +02:00
}
2017-04-23 19:02:44 +02:00
var changed_features = {};
2017-04-04 17:26:06 +02:00
_.each(_.keys(model.changed), function (k) {
if (!_.isNil(ROOM_FEATURES_MAP[k])) {
changed_features[ROOM_FEATURES_MAP[k]] = !model.changed[k];
}
});
2017-07-22 22:21:05 +02:00
this.chatroomview.model.save(changed_features, { 'silent': true });
2017-04-04 17:26:06 +02:00
this.debouncedRenderRoomFeatures();
},
2017-07-22 22:21:05 +02:00
setOccupantsHeight: function setOccupantsHeight() {
2017-03-05 10:45:18 +01:00
var el = this.el.querySelector('.chatroom-features');
2017-07-22 22:21:05 +02:00
this.el.querySelector('.occupant-list').style.cssText = "height: calc(100% - " + el.offsetHeight + "px - 5em);";
2017-03-05 10:45:18 +01:00
},
2017-07-22 22:21:05 +02:00
onOccupantAdded: function onOccupantAdded(item) {
2016-11-07 15:43:48 +01:00
var view = this.get(item.get('id'));
if (!view) {
2017-07-22 22:21:05 +02:00
view = this.add(item.get('id'), new _converse.ChatRoomOccupantView({ model: item }));
2016-11-07 15:43:48 +01:00
} else {
delete view.model; // Remove ref to old model to help garbage collection
view.model = item;
view.initialize();
}
2016-11-07 15:43:48 +01:00
this.$('.occupant-list').append(view.render().$el);
},
2017-07-22 22:21:05 +02:00
parsePresence: function parsePresence(pres) {
2016-11-07 15:43:48 +01:00
var id = Strophe.getResourceFromJid(pres.getAttribute("from"));
var data = {
nick: id,
type: pres.getAttribute("type"),
states: []
};
_.each(pres.childNodes, function (child) {
switch (child.nodeName) {
case "status":
data.status = child.textContent || null;
break;
case "show":
2017-03-05 10:45:18 +01:00
data.show = child.textContent || 'online';
2016-11-07 15:43:48 +01:00
break;
case "x":
if (child.getAttribute("xmlns") === Strophe.NS.MUC_USER) {
_.each(child.childNodes, function (item) {
switch (item.nodeName) {
case "item":
data.affiliation = item.getAttribute("affiliation");
data.role = item.getAttribute("role");
data.jid = item.getAttribute("jid");
data.nick = item.getAttribute("nick") || data.nick;
break;
case "status":
if (item.getAttribute("code")) {
data.states.push(item.getAttribute("code"));
}
}
});
}
}
2016-11-07 15:43:48 +01:00
});
return data;
},
2017-07-22 22:21:05 +02:00
findOccupant: function findOccupant(data) {
2016-11-07 15:43:48 +01:00
/* Try to find an existing occupant based on the passed in
* data object.
*
* If we have a JID, we use that as lookup variable,
* otherwise we use the nick. We don't always have both,
* but should have at least one or the other.
*/
var jid = Strophe.getBareJidFromJid(data.jid);
if (jid !== null) {
2017-07-22 22:21:05 +02:00
return this.model.where({ 'jid': jid }).pop();
2016-07-28 18:06:31 +02:00
} else {
2017-07-22 22:21:05 +02:00
return this.model.where({ 'nick': data.nick }).pop();
2016-07-28 18:06:31 +02:00
}
},
2017-07-22 22:21:05 +02:00
updateOccupantsOnPresence: function updateOccupantsOnPresence(pres) {
2016-12-13 20:46:07 +01:00
/* Given a presence stanza, update the occupant models
* based on its contents.
*
* Parameters:
* (XMLElement) pres: The presence stanza
*/
2016-11-07 15:43:48 +01:00
var data = this.parsePresence(pres);
if (data.type === 'error') {
return true;
}
2016-11-07 15:43:48 +01:00
var occupant = this.findOccupant(data);
2017-07-22 22:21:05 +02:00
if (data.type === 'unavailable') {
if (occupant) {
occupant.destroy();
}
} else {
var jid = Strophe.getBareJidFromJid(data.jid);
var attributes = _.extend(data, {
'jid': jid ? jid : undefined,
'resource': data.jid ? Strophe.getResourceFromJid(data.jid) : undefined
});
if (occupant) {
occupant.save(attributes);
} else {
this.model.create(attributes);
}
}
},
2017-07-22 22:21:05 +02:00
promptForInvite: function promptForInvite(suggestion) {
var reason = prompt(__(___('You are about to invite %1$s to the chat room "%2$s". '), suggestion.text.label, this.model.get('id')) + __("You may optionally include a message, explaining the reason for the invitation."));
if (reason !== null) {
this.chatroomview.directInvite(suggestion.text.value, reason);
}
2017-07-22 22:21:05 +02:00
var form = suggestion.target.form,
error = form.querySelector('.pure-form-message.error');
if (!_.isNull(error)) {
error.parentNode.removeChild(error);
}
suggestion.target.value = '';
},
2017-07-22 22:21:05 +02:00
inviteFormSubmitted: function inviteFormSubmitted(evt) {
evt.preventDefault();
2017-07-22 22:21:05 +02:00
var el = evt.target.querySelector('input.invited-contact'),
jid = el.value;
if (!jid || _.filter(jid.split('@')).length < 2) {
evt.target.outerHTML = tpl_chatroom_invite({
'error_message': __('Please enter a valid XMPP username'),
'label_invitation': __('Invite')
});
this.initInviteWidget();
return;
}
this.promptForInvite({
'target': el,
'text': {
2017-07-22 22:21:05 +02:00
'label': jid,
'value': jid
} });
},
2017-07-22 22:21:05 +02:00
shouldInviteWidgetBeShown: function shouldInviteWidgetBeShown() {
return _converse.allow_muc_invitations && (this.chatroomview.model.get('open') || this.chatroomview.model.get('affiliation') === "owner");
2017-03-05 10:45:18 +01:00
},
2017-07-22 22:21:05 +02:00
initInviteWidget: function initInviteWidget() {
var form = this.el.querySelector('form.room-invite');
2017-03-05 10:45:18 +01:00
if (_.isNull(form)) {
return;
}
form.addEventListener('submit', this.inviteFormSubmitted.bind(this));
var el = this.el.querySelector('input.invited-contact');
var list = _converse.roster.map(function (item) {
2017-07-22 22:21:05 +02:00
var label = item.get('fullname') || item.get('jid');
return { 'label': label, 'value': item.get('jid') };
});
var awesomplete = new Awesomplete(el, {
'minChars': 1,
'list': list
});
el.addEventListener('awesomplete-selectcomplete', this.promptForInvite.bind(this));
2016-11-07 15:43:48 +01:00
}
});
2017-02-03 13:51:07 +01:00
_converse.RoomsPanel = Backbone.View.extend({
2016-11-07 15:43:48 +01:00
/* Backbone View which renders the "Rooms" tab and accompanying
* panel in the control box.
*
* In this panel, chat rooms can be listed, joined and new rooms
* can be created.
*/
tagName: 'div',
className: 'controlbox-pane',
id: 'chatrooms',
events: {
2017-06-23 20:25:33 +02:00
'submit form.add-chatroom': 'openChatRoom',
2016-11-07 15:43:48 +01:00
'click input#show-rooms': 'showRooms',
2017-06-23 20:25:33 +02:00
'click a.open-room': 'openChatRoom',
2016-11-07 15:43:48 +01:00
'click a.room-info': 'toggleRoomInfo',
'change input[name=server]': 'setDomain',
'change input[name=nick]': 'setNick'
},
2017-07-22 22:21:05 +02:00
initialize: function initialize(cfg) {
2017-06-23 20:25:33 +02:00
this.parent_el = cfg.$parent[0];
this.tab_el = document.createElement('li');
2016-11-07 15:43:48 +01:00
this.model.on('change:muc_domain', this.onDomainChange, this);
this.model.on('change:nick', this.onNickChange, this);
2017-06-23 20:25:33 +02:00
_converse.chatboxes.on('change:num_unread', this.renderTab, this);
_converse.chatboxes.on('add', _.debounce(this.renderTab, 100), this);
},
2017-07-22 22:21:05 +02:00
render: function render() {
2017-06-23 20:25:33 +02:00
this.el.innerHTML = tpl_room_panel({
'server_input_type': _converse.hide_muc_server && 'hidden' || 'text',
'server_label_global_attr': _converse.hide_muc_server && ' hidden' || '',
'label_room_name': __('Room name'),
'label_nickname': __('Nickname'),
'label_server': __('Server'),
'label_join': __('Join Room'),
'label_show_rooms': __('Show rooms')
});
this.renderTab();
2017-02-03 13:51:07 +01:00
var controlbox = _converse.chatboxes.get('controlbox');
2016-11-07 15:43:48 +01:00
if (controlbox.get('active-panel') !== ROOMS_PANEL_ID) {
2017-06-23 20:25:33 +02:00
this.el.classList.add('hidden');
2016-11-07 15:43:48 +01:00
}
return this;
},
2017-07-22 22:21:05 +02:00
renderTab: function renderTab() {
2017-06-23 20:25:33 +02:00
var controlbox = _converse.chatboxes.get('controlbox');
2017-07-22 22:21:05 +02:00
var chatrooms = fp.filter(_.partial(utils.isOfType, CHATROOMS_TYPE), _converse.chatboxes.models);
2017-06-23 20:25:33 +02:00
this.tab_el.innerHTML = tpl_chatrooms_tab({
'label_rooms': __('Rooms'),
'is_current': controlbox.get('active-panel') === ROOMS_PANEL_ID,
'num_unread': fp.sum(fp.map(fp.curry(utils.getAttribute)('num_unread'), chatrooms))
});
},
2017-07-22 22:21:05 +02:00
insertIntoDOM: function insertIntoDOM() {
2017-06-23 20:25:33 +02:00
this.parent_el.appendChild(this.render().el);
this.tabs = this.parent_el.parentNode.querySelector('#controlbox-tabs');
this.tabs.appendChild(this.tab_el);
return this;
},
2017-07-22 22:21:05 +02:00
onDomainChange: function onDomainChange(model) {
2016-11-07 15:43:48 +01:00
var $server = this.$el.find('input.new-chatroom-server');
$server.val(model.get('muc_domain'));
2017-02-03 13:51:07 +01:00
if (_converse.auto_list_rooms) {
2016-11-07 15:43:48 +01:00
this.updateRoomsList();
}
},
2017-07-22 22:21:05 +02:00
onNickChange: function onNickChange(model) {
2016-11-07 15:43:48 +01:00
var $nick = this.$el.find('input.new-chatroom-nick');
$nick.val(model.get('nick'));
},
2017-07-22 22:21:05 +02:00
informNoRoomsFound: function informNoRoomsFound() {
2016-11-07 15:43:48 +01:00
var $available_chatrooms = this.$el.find('#available-chatrooms');
// For translators: %1$s is a variable and will be replaced with the XMPP server name
2017-07-22 22:21:05 +02:00
$available_chatrooms.html("<dt>" + __('No rooms on %1$s', this.model.get('muc_domain')) + "</dt>");
2016-11-07 15:43:48 +01:00
$('input#show-rooms').show().siblings('span.spinner').remove();
},
2017-07-22 22:21:05 +02:00
onRoomsFound: function onRoomsFound(iq) {
2016-11-07 15:43:48 +01:00
/* Handle the IQ stanza returned from the server, containing
* all its public rooms.
*/
2017-07-22 22:21:05 +02:00
var $available_chatrooms = this.$el.find('#available-chatrooms');
2016-11-07 15:43:48 +01:00
this.rooms = $(iq).find('query').find('item');
if (this.rooms.length) {
// For translators: %1$s is a variable and will be
// replaced with the XMPP server name
2017-07-22 22:21:05 +02:00
$available_chatrooms.html("<dt>" + __('Rooms on %1$s', this.model.get('muc_domain')) + "</dt>");
var fragment = document.createDocumentFragment();
for (var i = 0; i < this.rooms.length; i++) {
var name = Strophe.unescapeNode($(this.rooms[i]).attr('name') || $(this.rooms[i]).attr('jid'));
var jid = $(this.rooms[i]).attr('jid');
fragment.appendChild($(tpl_room_item({
'name': name,
'jid': jid,
'open_title': __('Click to open this room'),
'info_title': __('Show more information on this room')
}))[0]);
2016-11-07 15:43:48 +01:00
}
$available_chatrooms.append(fragment);
$('input#show-rooms').show().siblings('span.spinner').remove();
} else {
this.informNoRoomsFound();
2016-07-28 18:06:31 +02:00
}
2016-11-07 15:43:48 +01:00
return true;
2016-07-28 18:06:31 +02:00
},
2017-07-22 22:21:05 +02:00
updateRoomsList: function updateRoomsList() {
2016-11-07 15:43:48 +01:00
/* Send and IQ stanza to the server asking for all rooms
2016-07-28 18:06:31 +02:00
*/
2017-07-22 22:21:05 +02:00
_converse.connection.sendIQ($iq({
to: this.model.get('muc_domain'),
from: _converse.connection.jid,
type: "get"
}).c("query", { xmlns: Strophe.NS.DISCO_ITEMS }), this.onRoomsFound.bind(this), this.informNoRoomsFound.bind(this));
2016-07-28 18:06:31 +02:00
},
2017-07-22 22:21:05 +02:00
showRooms: function showRooms() {
2016-11-07 15:43:48 +01:00
var $available_chatrooms = this.$el.find('#available-chatrooms');
var $server = this.$el.find('input.new-chatroom-server');
var server = $server.val();
if (!server) {
$server.addClass('error');
return;
2016-08-12 22:52:21 +02:00
}
2016-11-07 15:43:48 +01:00
this.$el.find('input.new-chatroom-name').removeClass('error');
$server.removeClass('error');
$available_chatrooms.empty();
2017-06-23 20:25:33 +02:00
$('input#show-rooms').hide().after(tpl_spinner);
2017-07-22 22:21:05 +02:00
this.model.save({ muc_domain: server });
2016-11-07 15:43:48 +01:00
this.updateRoomsList();
2016-08-12 22:52:21 +02:00
},
2017-07-22 22:21:05 +02:00
insertRoomInfo: function insertRoomInfo(el, stanza) {
2016-11-07 15:43:48 +01:00
/* Insert room info (based on returned #disco IQ stanza)
2016-12-13 20:46:07 +01:00
*
* Parameters:
* (HTMLElement) el: The HTML DOM element that should
* contain the info.
* (XMLElement) stanza: The IQ stanza containing the room
* info.
2016-08-12 22:52:21 +02:00
*/
2016-11-07 15:43:48 +01:00
var $stanza = $(stanza);
// All MUC features found here: http://xmpp.org/registrar/disco-features.html
2017-07-22 22:21:05 +02:00
el.querySelector('span.spinner').outerHTML = tpl_room_description({
'jid': stanza.getAttribute('from'),
'desc': $stanza.find('field[var="muc#roominfo_description"] value').text(),
'occ': $stanza.find('field[var="muc#roominfo_occupants"] value').text(),
'hidden': $stanza.find('feature[var="muc_hidden"]').length,
'membersonly': $stanza.find('feature[var="muc_membersonly"]').length,
'moderated': $stanza.find('feature[var="muc_moderated"]').length,
'nonanonymous': $stanza.find('feature[var="muc_nonanonymous"]').length,
'open': $stanza.find('feature[var="muc_open"]').length,
'passwordprotected': $stanza.find('feature[var="muc_passwordprotected"]').length,
'persistent': $stanza.find('feature[var="muc_persistent"]').length,
'publicroom': $stanza.find('feature[var="muc_public"]').length,
'semianonymous': $stanza.find('feature[var="muc_semianonymous"]').length,
'temporary': $stanza.find('feature[var="muc_temporary"]').length,
'unmoderated': $stanza.find('feature[var="muc_unmoderated"]').length,
'label_desc': __('Description:'),
'label_jid': __('Room Address (JID):'),
'label_occ': __('Occupants:'),
'label_features': __('Features:'),
'label_requires_auth': __('Requires authentication'),
'label_hidden': __('Hidden'),
'label_requires_invite': __('Requires an invitation'),
'label_moderated': __('Moderated'),
'label_non_anon': __('Non-anonymous'),
'label_open_room': __('Open room'),
'label_permanent_room': __('Permanent room'),
'label_public': __('Public'),
'label_semi_anon': __('Semi-anonymous'),
'label_temp_room': __('Temporary room'),
'label_unmoderated': __('Unmoderated')
});
2016-08-12 22:52:21 +02:00
},
2017-07-22 22:21:05 +02:00
toggleRoomInfo: function toggleRoomInfo(ev) {
2016-11-07 15:43:48 +01:00
/* Show/hide extra information about a room in the listing.
2016-08-12 22:52:21 +02:00
*/
2016-11-07 15:43:48 +01:00
var target = ev.target,
$parent = $(target).parent('dd'),
$div = $parent.find('div.room-info');
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
if ($div.length) {
$div.remove();
2016-08-12 22:52:21 +02:00
} else {
2016-11-07 15:43:48 +01:00
$parent.find('span.spinner').remove();
2017-06-23 20:25:33 +02:00
$parent.append(tpl_spinner);
2017-07-22 22:21:05 +02:00
_converse.connection.disco.info($(target).attr('data-room-jid'), null, _.partial(this.insertRoomInfo, $parent[0]));
2016-08-12 22:52:21 +02:00
}
},
2017-07-22 22:21:05 +02:00
parseRoomDataFromEvent: function parseRoomDataFromEvent(ev) {
var name = void 0,
$name = void 0,
server = void 0,
$server = void 0,
jid = void 0;
2016-11-07 15:43:48 +01:00
if (ev.type === 'click') {
name = $(ev.target).text();
jid = $(ev.target).attr('data-room-jid');
} else {
2017-07-22 22:21:05 +02:00
var _$name = this.$el.find('input.new-chatroom-name');
var _$server = this.$el.find('input.new-chatroom-server');
var _server = _$server.val();
name = _$name.val().trim();
_$name.val(''); // Clear the input
if (name && _server) {
jid = Strophe.escapeNode(name.toLowerCase()) + '@' + _server.toLowerCase();
_$name.removeClass('error');
_$server.removeClass('error');
this.model.save({ muc_domain: _server });
2016-11-07 15:43:48 +01:00
} else {
2017-07-22 22:21:05 +02:00
if (!name) {
_$name.addClass('error');
}
if (!_server) {
_$server.addClass('error');
}
2016-11-07 15:43:48 +01:00
return;
}
}
2017-06-23 20:25:33 +02:00
return {
2016-11-07 15:43:48 +01:00
'jid': jid,
'name': name || Strophe.unescapeNode(Strophe.getNodeFromJid(jid))
2017-07-22 22:21:05 +02:00
};
2017-06-23 20:25:33 +02:00
},
2017-07-22 22:21:05 +02:00
openChatRoom: function openChatRoom(ev) {
2017-06-23 20:25:33 +02:00
ev.preventDefault();
2017-08-08 17:44:01 +02:00
var data = this.parseRoomDataFromEvent(ev);
if (!_.isUndefined(data)) {
_converse.openChatRoom(data);
}
},
2017-07-22 22:21:05 +02:00
setDomain: function setDomain(ev) {
this.model.save({ muc_domain: ev.target.value });
},
2017-07-22 22:21:05 +02:00
setNick: function setNick(ev) {
this.model.save({ nick: ev.target.value });
2016-11-07 15:43:48 +01:00
}
});
2016-12-13 20:46:07 +01:00
/************************ End of ChatRoomView **********************/
2017-02-03 13:51:07 +01:00
_converse.onDirectMUCInvitation = function (message) {
2016-12-13 20:46:07 +01:00
/* A direct MUC invitation to join a room has been received
* See XEP-0249: Direct MUC invitations.
*
* Parameters:
* (XMLElement) message: The message stanza containing the
* invitation.
*/
2016-11-07 15:43:48 +01:00
var $message = $(message),
$x = $message.children('x[xmlns="jabber:x:conference"]'),
from = Strophe.getBareJidFromJid($message.attr('from')),
room_jid = $x.attr('jid'),
2017-07-22 22:21:05 +02:00
reason = $x.attr('reason');
var contact = _converse.roster.get(from),
result = void 0;
2017-02-03 13:51:07 +01:00
if (_converse.auto_join_on_invite) {
2016-11-07 15:43:48 +01:00
result = true;
} else {
// Invite request might come from someone not your roster list
2017-07-22 22:21:05 +02:00
contact = contact ? contact.get('fullname') : Strophe.getNodeFromJid(from);
2016-11-07 15:43:48 +01:00
if (!reason) {
2017-07-22 22:21:05 +02:00
result = confirm(__(___("%1$s has invited you to join a chat room: %2$s"), contact, room_jid));
2016-11-07 15:43:48 +01:00
} else {
2017-07-22 22:21:05 +02:00
result = confirm(__(___('%1$s has invited you to join a chat room: %2$s, and left the following reason: "%3$s"'), contact, room_jid, reason));
2016-11-07 15:43:48 +01:00
}
}
if (result === true) {
2017-06-23 20:25:33 +02:00
var chatroom = _converse.openChatRoom({
2016-11-07 15:43:48 +01:00
'jid': room_jid,
'password': $x.attr('password')
});
2017-07-22 22:21:05 +02:00
if (chatroom.get('connection_status') === converse.ROOMSTATUS.DISCONNECTED) {
2017-02-03 13:51:07 +01:00
_converse.chatboxviews.get(room_jid).join();
2016-11-07 15:43:48 +01:00
}
}
};
2017-02-03 13:51:07 +01:00
if (_converse.allow_muc_invitations) {
2017-07-22 22:21:05 +02:00
var registerDirectInvitationHandler = function registerDirectInvitationHandler() {
_converse.connection.addHandler(function (message) {
_converse.onDirectMUCInvitation(message);
return true;
}, 'jabber:x:conference', 'message');
2016-12-13 20:46:07 +01:00
};
2017-02-03 13:51:07 +01:00
_converse.on('connected', registerDirectInvitationHandler);
_converse.on('reconnected', registerDirectInvitationHandler);
2016-12-13 20:46:07 +01:00
}
2017-07-22 22:21:05 +02:00
function autoJoinRooms() {
2016-12-13 20:46:07 +01:00
/* Automatically join chat rooms, based on the
* "auto_join_rooms" configuration setting, which is an array
* of strings (room JIDs) or objects (with room JID and other
* settings).
*/
2017-02-03 13:51:07 +01:00
_.each(_converse.auto_join_rooms, function (room) {
if (_.isString(room)) {
2017-02-03 13:51:07 +01:00
_converse.api.rooms.open(room);
} else if (_.isObject(room)) {
2017-02-03 13:51:07 +01:00
_converse.api.rooms.open(room.jid, room.nick);
2016-11-07 15:43:48 +01:00
} else {
2017-07-22 22:21:05 +02:00
_converse.log('Invalid room criteria specified for "auto_join_rooms"', Strophe.LogLevel.ERROR);
2016-11-07 15:43:48 +01:00
}
});
2017-07-22 22:21:05 +02:00
}
2017-02-03 13:51:07 +01:00
_converse.on('chatBoxesFetched', autoJoinRooms);
_converse.getChatRoom = function (jid, attrs, fetcher) {
2016-11-07 15:43:48 +01:00
jid = jid.toLowerCase();
return _converse.getViewForChatBox(fetcher(_.extend({
2016-11-07 15:43:48 +01:00
'id': jid,
'jid': jid,
'name': Strophe.unescapeNode(Strophe.getNodeFromJid(jid)),
2017-06-23 20:25:33 +02:00
'type': CHATROOMS_TYPE,
2016-11-07 15:43:48 +01:00
'box_id': b64_sha1(jid)
}, attrs)));
};
2016-11-07 15:43:48 +01:00
/* We extend the default converse.js API to add methods specific to MUC
* chat rooms.
*/
2017-02-03 13:51:07 +01:00
_.extend(_converse.api, {
2016-11-07 15:43:48 +01:00
'rooms': {
2017-07-22 22:21:05 +02:00
'close': function close(jids) {
if (_.isUndefined(jids)) {
2017-02-03 13:51:07 +01:00
_converse.chatboxviews.each(function (view) {
2016-11-07 15:43:48 +01:00
if (view.is_chatroom && view.model) {
view.close();
}
});
} else if (_.isString(jids)) {
2017-02-03 13:51:07 +01:00
var view = _converse.chatboxviews.get(jids);
2017-07-22 22:21:05 +02:00
if (view) {
view.close();
}
2016-11-07 15:43:48 +01:00
} else {
_.each(jids, function (jid) {
2017-02-03 13:51:07 +01:00
var view = _converse.chatboxviews.get(jid);
2017-07-22 22:21:05 +02:00
if (view) {
view.close();
}
2016-11-07 15:43:48 +01:00
});
}
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
'open': function open(jids, attrs) {
if (_.isString(attrs)) {
2017-07-22 22:21:05 +02:00
attrs = { 'nick': attrs };
} else if (_.isUndefined(attrs)) {
2016-11-07 15:43:48 +01:00
attrs = {};
}
2016-11-30 17:27:20 +01:00
if (_.isUndefined(attrs.maximize)) {
attrs.maximize = false;
}
2017-02-03 13:51:07 +01:00
if (!attrs.nick && _converse.muc_nickname_from_jid) {
attrs.nick = Strophe.getNodeFromJid(_converse.bare_jid);
}
if (_.isUndefined(jids)) {
2016-11-07 15:43:48 +01:00
throw new TypeError('rooms.open: You need to provide at least one JID');
} else if (_.isString(jids)) {
2017-06-23 20:25:33 +02:00
return _converse.getChatRoom(jids, attrs, _converse.openChatRoom);
}
2017-06-23 20:25:33 +02:00
return _.map(jids, _.partial(_converse.getChatRoom, _, attrs, _converse.openChatRoom));
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
'get': function get(jids, attrs, create) {
if (_.isString(attrs)) {
2017-07-22 22:21:05 +02:00
attrs = { 'nick': attrs };
} else if (_.isUndefined(attrs)) {
2016-11-07 15:43:48 +01:00
attrs = {};
}
if (_.isUndefined(jids)) {
2016-11-07 15:43:48 +01:00
var result = [];
2017-02-03 13:51:07 +01:00
_converse.chatboxes.each(function (chatbox) {
2017-06-23 20:25:33 +02:00
if (chatbox.get('type') === CHATROOMS_TYPE) {
result.push(_converse.getViewForChatBox(chatbox));
2016-11-07 15:43:48 +01:00
}
});
return result;
}
2017-02-03 13:51:07 +01:00
var fetcher = _.partial(_converse.chatboxviews.getChatBox.bind(_converse.chatboxviews), _, create);
2016-11-07 15:43:48 +01:00
if (!attrs.nick) {
2017-02-03 13:51:07 +01:00
attrs.nick = Strophe.getNodeFromJid(_converse.bare_jid);
2016-11-07 15:43:48 +01:00
}
if (_.isString(jids)) {
return _converse.getChatRoom(jids, attrs, fetcher);
}
return _.map(jids, _.partial(_converse.getChatRoom, _, attrs, fetcher));
}
2016-11-07 15:43:48 +01:00
}
});
2016-12-13 20:46:07 +01:00
2017-07-22 22:21:05 +02:00
/* Event handlers */
_converse.on('addClientFeatures', function () {
if (_converse.allow_muc) {
_converse.connection.disco.addFeature(Strophe.NS.MUC);
}
if (_converse.allow_muc_invitations) {
_converse.connection.disco.addFeature('jabber:x:conference'); // Invites
}
});
_converse.on('reconnected', function reconnectToChatRooms() {
2016-12-13 20:46:07 +01:00
/* Upon a reconnection event from converse, join again
* all the open chat rooms.
*/
2017-02-03 13:51:07 +01:00
_converse.chatboxviews.each(function (view) {
2017-06-23 20:25:33 +02:00
if (view.model.get('type') === CHATROOMS_TYPE) {
2017-07-22 22:21:05 +02:00
view.model.save('connection_status', converse.ROOMSTATUS.DISCONNECTED);
2017-06-23 20:25:33 +02:00
view.registerHandlers();
2016-12-13 20:46:07 +01:00
view.join();
2017-06-23 20:25:33 +02:00
view.fetchMessages();
2016-12-13 20:46:07 +01:00
}
});
2017-07-22 22:21:05 +02:00
});
2016-12-13 20:46:07 +01:00
2017-07-22 22:21:05 +02:00
function disconnectChatRooms() {
2016-12-13 20:46:07 +01:00
/* When disconnecting, or reconnecting, mark all chat rooms as
* disconnected, so that they will be properly entered again
* when fetched from session storage.
*/
2017-02-03 13:51:07 +01:00
_converse.chatboxes.each(function (model) {
2017-06-23 20:25:33 +02:00
if (model.get('type') === CHATROOMS_TYPE) {
2017-07-22 22:21:05 +02:00
model.save('connection_status', converse.ROOMSTATUS.DISCONNECTED);
2016-12-13 20:46:07 +01:00
}
});
2017-07-22 22:21:05 +02:00
}
2017-02-03 13:51:07 +01:00
_converse.on('reconnecting', disconnectChatRooms);
_converse.on('disconnecting', disconnectChatRooms);
2016-11-07 15:43:48 +01:00
}
});
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-muc.js.map;
2016-07-28 18:06:31 +02:00
define('tpl!chatroom_bookmark_form', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<div class="chatroom-form-container">\n <form class="pure-form converse-form chatroom-form">\n <fieldset>\n <legend>' +
2017-02-13 17:16:13 +01:00
__e(heading) +
'</legend>\n <label>' +
2017-02-13 17:16:13 +01:00
__e(label_name) +
'</label>\n <input type="text" name="name" required="required"/>\n <label>' +
2017-02-13 17:16:13 +01:00
__e(label_autojoin) +
'</label>\n <input type="checkbox" name="autojoin"/>\n <label>' +
2017-02-13 17:16:13 +01:00
__e(label_nick) +
'</label>\n <input type="text" name="nick" value="' +
2017-02-13 17:16:13 +01:00
__e(default_nick) +
'"/>\n </fieldset>\n <fieldset>\n <input class="pure-button button-primary" type="submit" value="' +
2017-02-13 17:16:13 +01:00
__e(label_submit) +
'"/>\n <input class="pure-button button-cancel" type="button" value="' +
2017-02-13 17:16:13 +01:00
__e(label_cancel) +
2016-11-07 15:43:48 +01:00
'"/>\n </fieldset>\n </form>\n</div>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!chatroom_bookmark_toggle', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
__p += '<a class="chatbox-btn toggle-bookmark icon-pushpin\n ';
if (bookmarked) {;
__p += '\n button-on\n ';
} ;
__p += '" title="' +
2017-02-13 17:16:13 +01:00
__e(info_toggle_bookmark) +
2016-12-13 20:46:07 +01:00
'"></a>\n';
2016-12-13 20:46:07 +01:00
}
return __p
};});
define('tpl!bookmark', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-06-23 20:25:33 +02:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-06-23 20:25:33 +02:00
__p += '<dd class="available-chatroom" data-room-jid="' +
__e(jid) +
'">\n<a class="open-room" data-room-jid="' +
2017-02-13 17:16:13 +01:00
__e(jid) +
'" title="' +
2017-02-13 17:16:13 +01:00
__e(open_title) +
'" href="#">' +
2017-02-13 17:16:13 +01:00
__e(name) +
2017-06-23 20:25:33 +02:00
'</a>\n<a class="right remove-bookmark icon-pushpin ';
if (bookmarked) { ;
__p += ' button-on ';
} ;
__p += '"\n data-room-jid="' +
2017-02-13 17:16:13 +01:00
__e(jid) +
'" data-bookmark-name="' +
2017-02-13 17:16:13 +01:00
__e(name) +
2017-06-23 20:25:33 +02:00
'"\n title="' +
__e(info_remove_bookmark) +
'" href="#">&nbsp;</a>\n<a class="right room-info icon-room-info" data-room-jid="' +
2017-02-13 17:16:13 +01:00
__e(jid) +
2017-06-23 20:25:33 +02:00
'"\n title="' +
2017-02-13 17:16:13 +01:00
__e(info_title) +
2016-11-07 15:43:48 +01:00
'" href="#">&nbsp;</a>\n</dd>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!bookmarks_list', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
2017-06-23 20:25:33 +02:00
__p += '<a href="#" class="rooms-toggle bookmarks-toggle icon-' +
2017-02-13 17:16:13 +01:00
__e(toggle_state) +
'" title="' +
2017-02-13 17:16:13 +01:00
__e(desc_bookmarks) +
'">' +
2017-02-13 17:16:13 +01:00
__e(label_bookmarks) +
2016-11-07 15:43:48 +01:00
'</a>\n<dl class="bookmarks rooms-list"></dl>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
2017-02-03 13:51:07 +01:00
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
2016-11-07 15:43:48 +01:00
// Licensed under the Mozilla Public License (MPLv2)
//
2017-04-23 19:02:44 +02:00
/*global define */
2016-11-07 15:43:48 +01:00
/* This is a Converse.js plugin which add support for bookmarks specified
* in XEP-0048.
*/
(function (root, factory) {
2017-07-22 22:21:05 +02:00
define('converse-bookmarks',["jquery.noconflict", "utils", "converse-core", "converse-muc", "tpl!chatroom_bookmark_form", "tpl!chatroom_bookmark_toggle", "tpl!bookmark", "tpl!bookmarks_list"], factory);
})(undefined, function ($, utils, converse, muc, tpl_chatroom_bookmark_form, tpl_chatroom_bookmark_toggle, tpl_bookmark, tpl_bookmarks_list) {
var _converse$env = converse.env,
Backbone = _converse$env.Backbone,
Promise = _converse$env.Promise,
Strophe = _converse$env.Strophe,
$iq = _converse$env.$iq,
b64_sha1 = _converse$env.b64_sha1,
sizzle = _converse$env.sizzle,
_ = _converse$env._;
2016-07-28 18:06:31 +02:00
2017-02-03 13:51:07 +01:00
converse.plugins.add('converse-bookmarks', {
2016-11-07 15:43:48 +01:00
overrides: {
// Overrides mentioned here will be picked up by converse.js's
// plugin architecture they will replace existing methods on the
// relevant objects or classes.
//
// New functions which don't exist yet can also be added.
2017-07-22 22:21:05 +02:00
clearSession: function clearSession() {
2016-11-07 15:43:48 +01:00
this.__super__.clearSession.apply(this, arguments);
if (!_.isUndefined(this.bookmarks)) {
2017-07-05 11:59:55 +02:00
this.bookmarks.reset();
2016-11-07 15:43:48 +01:00
this.bookmarks.browserStorage._clear();
2017-07-22 22:21:05 +02:00
window.sessionStorage.removeItem(this.bookmarks.fetched_flag);
}
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
ChatRoomView: {
events: {
'click .toggle-bookmark': 'toggleBookmark'
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-11-07 15:43:48 +01:00
this.__super__.initialize.apply(this, arguments);
this.model.on('change:bookmarked', this.onBookmarked, this);
this.setBookmarkState();
},
2017-07-22 22:21:05 +02:00
generateHeadingHTML: function generateHeadingHTML() {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse,
__ = _converse.__,
html = this.__super__.generateHeadingHTML.apply(this, arguments);
2017-07-22 22:21:05 +02:00
2017-02-03 13:51:07 +01:00
if (_converse.allow_bookmarks) {
2016-12-13 20:46:07 +01:00
var div = document.createElement('div');
div.innerHTML = html;
2017-07-22 22:21:05 +02:00
var bookmark_button = tpl_chatroom_bookmark_toggle(_.assignIn(this.model.toJSON(), {
info_toggle_bookmark: __('Bookmark this room'),
bookmarked: this.model.get('bookmarked')
}));
2016-12-13 20:46:07 +01:00
var close_button = div.querySelector('.close-chatbox-button');
close_button.insertAdjacentHTML('afterend', bookmark_button);
return div.innerHTML;
2016-11-30 17:27:20 +01:00
}
2016-12-13 20:46:07 +01:00
return html;
},
2017-07-22 22:21:05 +02:00
checkForReservedNick: function checkForReservedNick() {
2016-11-07 15:43:48 +01:00
/* Check if the user has a bookmark with a saved nickanme
* for this room, and if so use it.
* Otherwise delegate to the super method.
*/
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2017-02-03 13:51:07 +01:00
if (_.isUndefined(_converse.bookmarks) || !_converse.allow_bookmarks) {
2016-11-30 17:27:20 +01:00
return this.__super__.checkForReservedNick.apply(this, arguments);
2016-11-07 15:43:48 +01:00
}
2017-07-22 22:21:05 +02:00
var model = _converse.bookmarks.findWhere({ 'jid': this.model.get('jid') });
2016-11-07 15:43:48 +01:00
if (!_.isUndefined(model) && model.get('nick')) {
this.join(model.get('nick'));
} else {
2016-12-13 20:46:07 +01:00
return this.__super__.checkForReservedNick.apply(this, arguments);
}
},
2017-07-22 22:21:05 +02:00
onBookmarked: function onBookmarked() {
2016-11-07 15:43:48 +01:00
if (this.model.get('bookmarked')) {
this.$('.icon-pushpin').addClass('button-on');
} else {
this.$('.icon-pushpin').removeClass('button-on');
}
},
2017-07-22 22:21:05 +02:00
setBookmarkState: function setBookmarkState() {
2016-11-07 15:43:48 +01:00
/* Set whether the room is bookmarked or not.
2016-07-28 18:06:31 +02:00
*/
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2017-02-03 13:51:07 +01:00
if (!_.isUndefined(_converse.bookmarks)) {
2017-07-22 22:21:05 +02:00
var models = _converse.bookmarks.where({ 'jid': this.model.get('jid') });
2016-11-07 15:43:48 +01:00
if (!models.length) {
this.model.save('bookmarked', false);
} else {
this.model.save('bookmarked', true);
}
2016-07-28 18:06:31 +02:00
}
},
2017-07-22 22:21:05 +02:00
renderBookmarkForm: function renderBookmarkForm() {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse,
__ = _converse.__,
$body = this.$('.chatroom-body');
2017-07-22 22:21:05 +02:00
2016-11-07 15:43:48 +01:00
$body.children().addClass('hidden');
2016-12-13 20:46:07 +01:00
// Remove any existing forms
$body.find('form.chatroom-form').remove();
2017-07-22 22:21:05 +02:00
$body.append(tpl_chatroom_bookmark_form({
heading: __('Bookmark this room'),
label_name: __('The name for this bookmark:'),
label_autojoin: __('Would you like this room to be automatically joined upon startup?'),
label_nick: __('What should your nickname for this room be?'),
default_nick: this.model.get('nick'),
label_submit: __('Save'),
label_cancel: __('Cancel')
}));
2016-11-07 15:43:48 +01:00
this.$('.chatroom-form').submit(this.onBookmarkFormSubmitted.bind(this));
this.$('.chatroom-form .button-cancel').on('click', this.cancelConfiguration.bind(this));
},
2017-07-22 22:21:05 +02:00
onBookmarkFormSubmitted: function onBookmarkFormSubmitted(ev) {
2016-11-07 15:43:48 +01:00
ev.preventDefault();
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
var $form = $(ev.target),
that = this;
2017-02-03 13:51:07 +01:00
_converse.bookmarks.createBookmark({
2016-11-07 15:43:48 +01:00
'jid': this.model.get('jid'),
'autojoin': $form.find('input[name="autojoin"]').prop('checked'),
2017-07-22 22:21:05 +02:00
'name': $form.find('input[name=name]').val(),
'nick': $form.find('input[name=nick]').val()
});
this.$el.find('div.chatroom-form-container').hide(function () {
$(this).remove();
that.renderAfterTransition();
});
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
toggleBookmark: function toggleBookmark(ev) {
2016-11-07 15:43:48 +01:00
if (ev) {
ev.preventDefault();
ev.stopPropagation();
}
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
var models = _converse.bookmarks.where({ 'jid': this.model.get('jid') });
2016-11-07 15:43:48 +01:00
if (!models.length) {
this.renderBookmarkForm();
} else {
_.forEach(models, function (model) {
2016-11-07 15:43:48 +01:00
model.destroy();
});
this.$('.icon-pushpin').removeClass('button-on');
}
}
}
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-11-07 15:43:48 +01:00
/* The initialize function gets called as soon as the plugin is
* loaded by converse.js's plugin machinery.
*/
2017-02-03 13:51:07 +01:00
var _converse = this._converse,
__ = _converse.__,
___ = _converse.___;
2016-11-30 17:27:20 +01:00
// Configuration values for this plugin
// ====================================
// Refer to docs/source/configuration.rst for explanations of these
// configuration settings.
2017-07-05 11:59:55 +02:00
_converse.api.settings.update({
2017-06-23 20:25:33 +02:00
allow_bookmarks: true,
hide_open_bookmarks: false
2016-11-30 17:27:20 +01:00
});
2017-07-05 11:59:55 +02:00
// Promises exposed by this plugin
_converse.api.promises.add('bookmarksInitialized');
2016-11-07 15:43:48 +01:00
// Pure functions on the _converse object
_.extend(_converse, {
removeBookmarkViaEvent: function removeBookmarkViaEvent(ev) {
/* Remove a bookmark as determined by the passed in
* event.
*/
ev.preventDefault();
var name = ev.target.getAttribute('data-bookmark-name');
var jid = ev.target.getAttribute('data-room-jid');
if (confirm(__(___("Are you sure you want to remove the bookmark \"%1$s\"?"), name))) {
_.invokeMap(_converse.bookmarks.where({ 'jid': jid }), Backbone.Model.prototype.destroy);
}
},
addBookmarkViaEvent: function addBookmarkViaEvent(ev) {
/* Add a bookmark as determined by the passed in
* event.
*/
ev.preventDefault();
var jid = ev.target.getAttribute('data-room-jid');
var chatroom = _converse.openChatRoom({ 'jid': jid }, true);
_converse.chatboxviews.get(jid).renderBookmarkForm();
}
});
2017-02-03 13:51:07 +01:00
_converse.Bookmark = Backbone.Model;
2016-11-07 15:43:48 +01:00
2017-02-03 13:51:07 +01:00
_converse.BookmarksList = Backbone.Model.extend({
2016-11-07 15:43:48 +01:00
defaults: {
2017-07-22 22:21:05 +02:00
"toggle-state": _converse.OPENED
}
});
2017-02-03 13:51:07 +01:00
_converse.Bookmarks = Backbone.Collection.extend({
model: _converse.Bookmark,
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
this.on('add', _.flow(this.openBookmarkedRoom, this.markRoomAsBookmarked));
2016-11-07 15:43:48 +01:00
this.on('remove', this.markRoomAsUnbookmarked, this);
this.on('remove', this.sendBookmarkStanza, this);
2017-07-22 22:21:05 +02:00
var cache_key = "converse.room-bookmarks" + _converse.bare_jid;
this.fetched_flag = b64_sha1(cache_key + 'fetched');
this.browserStorage = new Backbone.BrowserStorage[_converse.storage](b64_sha1(cache_key));
},
2017-07-22 22:21:05 +02:00
openBookmarkedRoom: function openBookmarkedRoom(bookmark) {
2016-11-07 15:43:48 +01:00
if (bookmark.get('autojoin')) {
2017-02-03 13:51:07 +01:00
_converse.api.rooms.open(bookmark.get('jid'), bookmark.get('nick'));
2016-11-07 15:43:48 +01:00
}
return bookmark;
},
2017-07-22 22:21:05 +02:00
fetchBookmarks: function fetchBookmarks() {
var deferred = utils.getWrappedPromise();
if (this.browserStorage.records.length > 0) {
2016-11-07 15:43:48 +01:00
this.fetch({
'success': _.bind(this.onCachedBookmarksFetched, this, deferred),
2017-07-22 22:21:05 +02:00
'error': _.bind(this.onCachedBookmarksFetched, this, deferred)
2016-11-07 15:43:48 +01:00
});
2017-07-22 22:21:05 +02:00
} else if (!window.sessionStorage.getItem(this.fetched_flag)) {
// There aren't any cached bookmarks and the
// `fetched_flag` is off, so we query the XMPP server.
2016-11-07 15:43:48 +01:00
// If nothing is returned from the XMPP server, we set
2017-07-22 22:21:05 +02:00
// the `fetched_flag` to avoid calling the server again.
2016-11-07 15:43:48 +01:00
this.fetchBookmarksFromServer(deferred);
} else {
deferred.resolve();
}
2017-07-22 22:21:05 +02:00
return deferred.promise;
},
2017-07-22 22:21:05 +02:00
onCachedBookmarksFetched: function onCachedBookmarksFetched(deferred) {
2016-11-07 15:43:48 +01:00
return deferred.resolve();
},
2017-07-22 22:21:05 +02:00
createBookmark: function createBookmark(options) {
2017-02-03 13:51:07 +01:00
_converse.bookmarks.create(options);
_converse.bookmarks.sendBookmarkStanza();
},
2017-07-22 22:21:05 +02:00
sendBookmarkStanza: function sendBookmarkStanza() {
2016-11-07 15:43:48 +01:00
var stanza = $iq({
2017-07-22 22:21:05 +02:00
'type': 'set',
'from': _converse.connection.jid
}).c('pubsub', { 'xmlns': Strophe.NS.PUBSUB }).c('publish', { 'node': 'storage:bookmarks' }).c('item', { 'id': 'current' }).c('storage', { 'xmlns': 'storage:bookmarks' });
2016-11-07 15:43:48 +01:00
this.each(function (model) {
stanza = stanza.c('conference', {
'name': model.get('name'),
'autojoin': model.get('autojoin'),
2017-07-22 22:21:05 +02:00
'jid': model.get('jid')
2016-11-07 15:43:48 +01:00
}).c('nick').t(model.get('nick')).up().up();
});
stanza.up().up().up();
2017-07-22 22:21:05 +02:00
stanza.c('publish-options').c('x', { 'xmlns': Strophe.NS.XFORM, 'type': 'submit' }).c('field', { 'var': 'FORM_TYPE', 'type': 'hidden' }).c('value').t('http://jabber.org/protocol/pubsub#publish-options').up().up().c('field', { 'var': 'pubsub#persist_items' }).c('value').t('true').up().up().c('field', { 'var': 'pubsub#access_model' }).c('value').t('whitelist');
2017-02-03 13:51:07 +01:00
_converse.connection.sendIQ(stanza, null, this.onBookmarkError.bind(this));
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
onBookmarkError: function onBookmarkError(iq) {
2017-07-05 11:59:55 +02:00
_converse.log("Error while trying to add bookmark", Strophe.LogLevel.ERROR);
2017-02-03 13:51:07 +01:00
_converse.log(iq);
2016-11-07 15:43:48 +01:00
// We remove all locally cached bookmarks and fetch them
// again from the server.
this.reset();
this.fetchBookmarksFromServer(null);
window.alert(__("Sorry, something went wrong while trying to save your bookmark."));
},
2017-07-22 22:21:05 +02:00
fetchBookmarksFromServer: function fetchBookmarksFromServer(deferred) {
2016-11-07 15:43:48 +01:00
var stanza = $iq({
2017-02-03 13:51:07 +01:00
'from': _converse.connection.jid,
2017-07-22 22:21:05 +02:00
'type': 'get'
}).c('pubsub', { 'xmlns': Strophe.NS.PUBSUB }).c('items', { 'node': 'storage:bookmarks' });
_converse.connection.sendIQ(stanza, _.bind(this.onBookmarksReceived, this, deferred), _.bind(this.onBookmarksReceivedError, this, deferred));
},
2017-07-22 22:21:05 +02:00
markRoomAsBookmarked: function markRoomAsBookmarked(bookmark) {
2017-02-03 13:51:07 +01:00
var room = _converse.chatboxes.get(bookmark.get('jid'));
2016-11-07 15:43:48 +01:00
if (!_.isUndefined(room)) {
room.save('bookmarked', true);
}
},
2017-07-22 22:21:05 +02:00
markRoomAsUnbookmarked: function markRoomAsUnbookmarked(bookmark) {
2017-02-03 13:51:07 +01:00
var room = _converse.chatboxes.get(bookmark.get('jid'));
2016-11-07 15:43:48 +01:00
if (!_.isUndefined(room)) {
room.save('bookmarked', false);
}
},
2017-07-22 22:21:05 +02:00
onBookmarksReceived: function onBookmarksReceived(deferred, iq) {
var bookmarks = $(iq).find('items[node="storage:bookmarks"] item[id="current"] storage conference');
2016-11-07 15:43:48 +01:00
var that = this;
_.forEach(bookmarks, function (bookmark) {
2016-11-07 15:43:48 +01:00
that.create({
'jid': bookmark.getAttribute('jid'),
'name': bookmark.getAttribute('name'),
'autojoin': bookmark.getAttribute('autojoin') === 'true',
'nick': bookmark.querySelector('nick').textContent
});
});
2016-11-07 15:43:48 +01:00
if (!_.isUndefined(deferred)) {
return deferred.resolve();
}
},
2017-07-22 22:21:05 +02:00
onBookmarksReceivedError: function onBookmarksReceivedError(deferred, iq) {
window.sessionStorage.setItem(this.fetched_flag, true);
2017-07-05 11:59:55 +02:00
_converse.log('Error while fetching bookmarks', Strophe.LogLevel.ERROR);
2017-07-22 22:21:05 +02:00
_converse.log(iq, Strophe.LogLevel.DEBUG);
2017-07-05 11:59:55 +02:00
if (!_.isNil(deferred)) {
2016-11-07 15:43:48 +01:00
return deferred.reject();
}
}
});
2017-02-03 13:51:07 +01:00
_converse.BookmarksView = Backbone.View.extend({
2016-11-07 15:43:48 +01:00
tagName: 'div',
2017-06-23 20:25:33 +02:00
className: 'bookmarks-list, rooms-list-container',
2016-11-07 15:43:48 +01:00
events: {
'click .add-bookmark': 'addBookmark',
'click .bookmarks-toggle': 'toggleBookmarksList',
'click .remove-bookmark': 'removeBookmark'
2016-11-07 15:43:48 +01:00
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-11-07 15:43:48 +01:00
this.model.on('add', this.renderBookmarkListElement, this);
this.model.on('remove', this.removeBookmarkListElement, this);
2017-06-23 20:25:33 +02:00
_converse.chatboxes.on('add', this.renderBookmarkListElement, this);
_converse.chatboxes.on('remove', this.renderBookmarkListElement, this);
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
var cachekey = "converse.room-bookmarks" + _converse.bare_jid + "-list-model";
2017-02-03 13:51:07 +01:00
this.list_model = new _converse.BookmarksList();
2016-11-07 15:43:48 +01:00
this.list_model.id = cachekey;
2017-07-22 22:21:05 +02:00
this.list_model.browserStorage = new Backbone.BrowserStorage[_converse.storage](b64_sha1(cachekey));
2016-11-07 15:43:48 +01:00
this.list_model.fetch();
this.render();
},
2017-07-22 22:21:05 +02:00
render: function render() {
this.$el.html(tpl_bookmarks_list({
2016-11-07 15:43:48 +01:00
'toggle_state': this.list_model.get('toggle-state'),
'desc_bookmarks': __('Click to toggle the bookmarks list'),
2017-06-23 20:25:33 +02:00
'label_bookmarks': __('Bookmarks')
2016-11-07 15:43:48 +01:00
})).hide();
2017-02-03 13:51:07 +01:00
if (this.list_model.get('toggle-state') !== _converse.OPENED) {
2016-11-07 15:43:48 +01:00
this.$('.bookmarks').hide();
}
this.model.each(this.renderBookmarkListElement.bind(this));
2017-02-03 13:51:07 +01:00
var controlboxview = _converse.chatboxviews.get('controlbox');
2016-11-07 15:43:48 +01:00
if (!_.isUndefined(controlboxview)) {
this.$el.prependTo(controlboxview.$('#chatrooms'));
}
2016-11-07 15:43:48 +01:00
return this.$el;
},
removeBookmark: _converse.removeBookmarkViaEvent,
addBookmark: _converse.addBookmarkViaEvent,
2017-07-22 22:21:05 +02:00
renderBookmarkListElement: function renderBookmarkListElement(item) {
2017-06-23 20:25:33 +02:00
if (item instanceof _converse.ChatBox) {
2017-07-22 22:21:05 +02:00
item = _.head(this.model.where({ 'jid': item.get('jid') }));
2017-06-23 20:25:33 +02:00
if (_.isNil(item)) {
// A chat box has been closed, but we don't have a
// bookmark for it, so nothing further to do here.
return;
}
}
2017-07-22 22:21:05 +02:00
if (_converse.hide_open_bookmarks && _converse.chatboxes.where({ 'jid': item.get('jid') }).length) {
2017-06-23 20:25:33 +02:00
// A chat box has been opened, and we don't show
// bookmarks for open chats, so we remove it.
this.removeBookmarkListElement(item);
return;
}
var list_el = this.el.querySelector('.bookmarks');
var div = document.createElement('div');
div.innerHTML = tpl_bookmark({
'bookmarked': true,
'info_leave_room': __('Leave this room'),
'info_remove': __('Remove this bookmark'),
'info_remove_bookmark': __('Unbookmark this room'),
'info_title': __('Show more information on this room'),
'jid': item.get('jid'),
'name': item.get('name'),
'open_title': __('Click to open this room')
});
2017-07-22 22:21:05 +02:00
var el = _.head(sizzle(".available-chatroom[data-room-jid=\"" + item.get('jid') + "\"]", list_el));
2017-06-23 20:25:33 +02:00
if (el) {
el.innerHTML = div.firstChild.innerHTML;
} else {
list_el.appendChild(div.firstChild);
}
this.show();
},
2017-07-22 22:21:05 +02:00
show: function show() {
2016-11-07 15:43:48 +01:00
if (!this.$el.is(':visible')) {
this.$el.show();
}
},
2017-07-22 22:21:05 +02:00
hide: function hide() {
2017-06-23 20:25:33 +02:00
this.$el.hide();
},
2017-07-22 22:21:05 +02:00
removeBookmarkListElement: function removeBookmarkListElement(item) {
2017-06-23 20:25:33 +02:00
var list_el = this.el.querySelector('.bookmarks');
2017-07-22 22:21:05 +02:00
var el = _.head(sizzle(".available-chatroom[data-room-jid=\"" + item.get('jid') + "\"]", list_el));
2017-06-23 20:25:33 +02:00
if (el) {
list_el.removeChild(el);
}
if (list_el.childElementCount === 0) {
this.hide();
2016-11-07 15:43:48 +01:00
}
},
2017-07-22 22:21:05 +02:00
toggleBookmarksList: function toggleBookmarksList(ev) {
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2016-11-07 15:43:48 +01:00
var $el = $(ev.target);
if ($el.hasClass("icon-opened")) {
this.$('.bookmarks').slideUp('fast');
2017-07-22 22:21:05 +02:00
this.list_model.save({ 'toggle-state': _converse.CLOSED });
2016-11-07 15:43:48 +01:00
$el.removeClass("icon-opened").addClass("icon-closed");
} else {
$el.removeClass("icon-closed").addClass("icon-opened");
this.$('.bookmarks').slideDown('fast');
2017-07-22 22:21:05 +02:00
this.list_model.save({ 'toggle-state': _converse.OPENED });
}
}
2016-03-16 12:49:35 +01:00
});
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
var initBookmarks = function initBookmarks() {
2017-02-03 13:51:07 +01:00
if (!_converse.allow_bookmarks) {
2016-11-30 17:27:20 +01:00
return;
}
2017-02-03 13:51:07 +01:00
_converse.bookmarks = new _converse.Bookmarks();
2017-07-22 22:21:05 +02:00
_converse.bookmarks.fetchBookmarks().then(function () {
_converse.bookmarksview = new _converse.BookmarksView({ 'model': _converse.bookmarks });
2017-06-23 20:25:33 +02:00
_converse.emit('bookmarksInitialized');
2016-11-07 15:43:48 +01:00
});
};
2017-07-22 22:21:05 +02:00
Promise.all([_converse.api.waitUntil('chatBoxesFetched'), _converse.api.waitUntil('roomsPanelRendered')]).then(initBookmarks);
var afterReconnection = function afterReconnection() {
2017-02-03 13:51:07 +01:00
if (!_converse.allow_bookmarks) {
2016-11-30 17:27:20 +01:00
return;
}
2017-02-03 13:51:07 +01:00
if (_.isUndefined(_converse.bookmarksview)) {
2016-11-07 15:43:48 +01:00
initBookmarks();
} else {
2017-02-03 13:51:07 +01:00
_converse.bookmarksview.render();
2016-11-07 15:43:48 +01:00
}
};
2017-02-03 13:51:07 +01:00
_converse.on('reconnected', afterReconnection);
2016-03-16 12:49:35 +01:00
}
});
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-bookmarks.js.map;
2017-06-23 20:25:33 +02:00
define('tpl!rooms_list', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<a href="#" class="rooms-toggle open-rooms-toggle icon-' +
__e(toggle_state) +
'" title="' +
__e(desc_rooms) +
'">' +
__e(label_rooms) +
'</a>\n<dl class="rooms-list open-rooms-list"></dl>\n';
}
return __p
};});
define('tpl!rooms_list_item', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
__p += '<dd class="available-chatroom ';
if (num_unread_general) { ;
__p += ' unread-msgs ';
} ;
__p += '" data-room-jid="' +
__e(jid) +
'">\n';
if (num_unread) { ;
__p += '\n <span class="msgs-indicator">' +
__e( num_unread ) +
'</span>\n';
} ;
2017-07-22 22:21:05 +02:00
__p += '\n<a class="open-room"\n data-room-jid="' +
2017-06-23 20:25:33 +02:00
__e(jid) +
2017-07-22 22:21:05 +02:00
'"\n title="' +
2017-06-23 20:25:33 +02:00
__e(open_title) +
'" href="#">' +
__e(name) +
'</a>\n<a class="right close-room icon-leave"\n data-room-jid="' +
__e(jid) +
2017-07-22 22:21:05 +02:00
'"\n data-room-name="' +
__e(name) +
'"\n title="' +
2017-06-23 20:25:33 +02:00
__e(info_leave_room) +
2017-07-22 22:21:05 +02:00
'" href="#">&nbsp;</a>\n\n';
if (allow_bookmarks) { ;
__p += '\n<a class="right icon-pushpin ';
2017-06-23 20:25:33 +02:00
if (bookmarked) { ;
__p += ' remove-bookmark button-on ';
} else { ;
__p += ' add-bookmark ';
2017-06-23 20:25:33 +02:00
} ;
__p += '"\n data-room-jid="' +
__e(jid) +
'" data-bookmark-name="' +
__e(name) +
'"\n title="';
if (bookmarked) { ;
__p += ' ' +
2017-06-23 20:25:33 +02:00
__e(info_remove_bookmark) +
' ';
} else { ;
__p += ' ' +
__e(info_add_bookmark) +
' ';
} ;
__p += '"\n href="#">&nbsp;</a>\n';
2017-07-22 22:21:05 +02:00
} ;
__p += '\n<a class="right room-info icon-room-info" data-room-jid="' +
2017-06-23 20:25:33 +02:00
__e(jid) +
'"\n title="' +
__e(info_title) +
'" href="#">&nbsp;</a>\n</dd>\n';
}
return __p
};});
2017-07-22 22:21:05 +02:00
2017-06-23 20:25:33 +02:00
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
// Licensed under the Mozilla Public License (MPLv2)
//
/*global define */
/* This is a non-core Converse.js plugin which shows a list of currently open
* rooms in the "Rooms Panel" of the ControlBox.
*/
(function (root, factory) {
2017-07-22 22:21:05 +02:00
define('converse-roomslist',["utils", "converse-core", "converse-muc", "tpl!rooms_list", "tpl!rooms_list_item"], factory);
})(undefined, function (utils, converse, muc, tpl_rooms_list, tpl_rooms_list_item) {
var _converse$env = converse.env,
Backbone = _converse$env.Backbone,
Promise = _converse$env.Promise,
b64_sha1 = _converse$env.b64_sha1,
sizzle = _converse$env.sizzle,
_ = _converse$env._;
2017-06-23 20:25:33 +02:00
converse.plugins.add('converse-roomslist', {
/* Optional dependencies are other plugins which might be
* overridden or relied upon, and therefore need to be loaded before
* this plugin. They are called "optional" because they might not be
* available, in which case any overrides applicable to them will be
* ignored.
*
* It's possible however to make optional dependencies non-optional.
* If the setting "strict_plugin_dependencies" is set to true,
* an error will be raised if the plugin is not found.
*
* NB: These plugins need to have already been loaded via require.js.
*/
optional_dependencies: ["converse-bookmarks"],
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2017-06-23 20:25:33 +02:00
/* The initialize function gets called as soon as the plugin is
* loaded by converse.js's plugin machinery.
*/
var _converse = this._converse,
__ = _converse.__,
___ = _converse.___;
2017-07-22 22:21:05 +02:00
2017-06-23 20:25:33 +02:00
_converse.RoomsList = Backbone.Model.extend({
defaults: {
2017-07-22 22:21:05 +02:00
"toggle-state": _converse.OPENED
}
2017-06-23 20:25:33 +02:00
});
_converse.RoomsListView = Backbone.View.extend({
tagName: 'div',
className: 'open-rooms-list rooms-list-container',
events: {
'click .add-bookmark': 'addBookmark',
2017-06-23 20:25:33 +02:00
'click .close-room': 'closeRoom',
'click .open-rooms-toggle': 'toggleRoomsList',
'click .remove-bookmark': 'removeBookmark'
2017-06-23 20:25:33 +02:00
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
this.toggleRoomsList = _.debounce(this.toggleRoomsList, 600, { 'leading': true });
2017-06-23 20:25:33 +02:00
this.model.on('add', this.renderRoomsListElement, this);
this.model.on('change:bookmarked', this.renderRoomsListElement, this);
this.model.on('change:name', this.renderRoomsListElement, this);
this.model.on('change:num_unread', this.renderRoomsListElement, this);
this.model.on('change:num_unread_general', this.renderRoomsListElement, this);
this.model.on('remove', this.removeRoomsListElement, this);
2017-07-22 22:21:05 +02:00
var cachekey = "converse.roomslist" + _converse.bare_jid;
2017-06-23 20:25:33 +02:00
this.list_model = new _converse.RoomsList();
this.list_model.id = cachekey;
2017-07-22 22:21:05 +02:00
this.list_model.browserStorage = new Backbone.BrowserStorage[_converse.storage](b64_sha1(cachekey));
2017-06-23 20:25:33 +02:00
this.list_model.fetch();
this.render();
},
2017-07-22 22:21:05 +02:00
render: function render() {
this.el.innerHTML = tpl_rooms_list({
2017-06-23 20:25:33 +02:00
'toggle_state': this.list_model.get('toggle-state'),
'desc_rooms': __('Click to toggle the rooms list'),
'label_rooms': __('Open Rooms')
2017-07-22 22:21:05 +02:00
});
2017-06-23 20:25:33 +02:00
this.hide();
if (this.list_model.get('toggle-state') !== _converse.OPENED) {
2017-07-22 22:21:05 +02:00
this.el.querySelector('.open-rooms-list').classList.add('collapsed');
2017-06-23 20:25:33 +02:00
}
this.model.each(this.renderRoomsListElement.bind(this));
var controlboxview = _converse.chatboxviews.get('controlbox');
2017-07-22 22:21:05 +02:00
if (!_.isUndefined(controlboxview) && !document.body.contains(this.el)) {
2017-06-23 20:25:33 +02:00
var container = controlboxview.el.querySelector('#chatrooms');
if (!_.isNull(container)) {
container.insertBefore(this.el, container.firstChild);
}
}
return this.el;
},
2017-07-22 22:21:05 +02:00
hide: function hide() {
2017-06-23 20:25:33 +02:00
this.el.classList.add('hidden');
},
2017-07-22 22:21:05 +02:00
show: function show() {
2017-06-23 20:25:33 +02:00
this.el.classList.remove('hidden');
},
2017-07-22 22:21:05 +02:00
closeRoom: function closeRoom(ev) {
2017-06-23 20:25:33 +02:00
ev.preventDefault();
2017-07-22 22:21:05 +02:00
var name = ev.target.getAttribute('data-room-name');
var jid = ev.target.getAttribute('data-room-jid');
2017-06-23 20:25:33 +02:00
if (confirm(__(___("Are you sure you want to leave the room \"%1$s\"?"), name))) {
_converse.chatboxviews.get(jid).leave();
}
},
2017-07-22 22:21:05 +02:00
renderRoomsListElement: function renderRoomsListElement(item) {
2017-06-23 20:25:33 +02:00
if (item.get('type') !== 'chatroom') {
return;
}
this.removeRoomsListElement(item);
2017-07-22 22:21:05 +02:00
var name = void 0,
bookmark = void 0;
2017-06-23 20:25:33 +02:00
if (item.get('bookmarked')) {
2017-07-22 22:21:05 +02:00
bookmark = _.head(_converse.bookmarksview.model.where({ 'jid': item.get('jid') }));
2017-06-23 20:25:33 +02:00
name = bookmark.get('name');
} else {
name = item.get('name');
}
var div = document.createElement('div');
div.innerHTML = tpl_rooms_list_item(_.extend(item.toJSON(), {
2017-07-22 22:21:05 +02:00
'allow_bookmarks': _converse.allow_bookmarks,
2017-06-23 20:25:33 +02:00
'info_leave_room': __('Leave this room'),
'info_remove_bookmark': __('Unbookmark this room'),
'info_add_bookmark': __('Bookmark this room'),
2017-06-23 20:25:33 +02:00
'info_title': __('Show more information on this room'),
'name': name,
'open_title': __('Click to open this room')
}));
this.el.querySelector('.open-rooms-list').appendChild(div.firstChild);
this.show();
},
removeBookmark: _converse.removeBookmarkViaEvent,
addBookmark: _converse.addBookmarkViaEvent,
2017-07-22 22:21:05 +02:00
removeRoomsListElement: function removeRoomsListElement(item) {
2017-06-23 20:25:33 +02:00
var list_el = this.el.querySelector('.open-rooms-list');
2017-07-22 22:21:05 +02:00
var el = _.head(sizzle(".available-chatroom[data-room-jid=\"" + item.get('jid') + "\"]", list_el));
2017-06-23 20:25:33 +02:00
if (el) {
list_el.removeChild(el);
}
if (list_el.childElementCount === 0) {
this.hide();
}
},
2017-07-22 22:21:05 +02:00
toggleRoomsList: function toggleRoomsList(ev) {
var _this = this;
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2017-06-23 20:25:33 +02:00
var el = ev.target;
if (el.classList.contains("icon-opened")) {
2017-07-22 22:21:05 +02:00
utils.slideIn(this.el.querySelector('.open-rooms-list')).then(function () {
_this.list_model.save({ 'toggle-state': _converse.CLOSED });
el.classList.remove("icon-opened");
el.classList.add("icon-closed");
});
2017-06-23 20:25:33 +02:00
} else {
2017-07-22 22:21:05 +02:00
utils.slideOut(this.el.querySelector('.open-rooms-list')).then(function () {
_this.list_model.save({ 'toggle-state': _converse.OPENED });
el.classList.remove("icon-closed");
el.classList.add("icon-opened");
});
2017-06-23 20:25:33 +02:00
}
}
});
2017-07-22 22:21:05 +02:00
var initRoomsListView = function initRoomsListView() {
_converse.rooms_list_view = new _converse.RoomsListView({ 'model': _converse.chatboxes });
2017-06-23 20:25:33 +02:00
};
2017-07-05 11:59:55 +02:00
2017-07-22 22:21:05 +02:00
Promise.all([_converse.api.waitUntil('chatBoxesFetched'), _converse.api.waitUntil('roomsPanelRendered')]).then(function () {
if (_converse.allow_bookmarks) {
_converse.api.waitUntil('bookmarksInitialized').then(initRoomsListView);
} else {
initRoomsListView();
}
});
2017-06-23 20:25:33 +02:00
2017-07-22 22:21:05 +02:00
var afterReconnection = function afterReconnection() {
2017-06-23 20:25:33 +02:00
if (_.isUndefined(_converse.rooms_list_view)) {
initRoomsListView();
} else {
_converse.rooms_list_view.render();
}
};
_converse.api.listen.on('reconnected', afterReconnection);
}
});
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-roomslist.js.map;
2017-06-23 20:25:33 +02:00
2016-03-16 12:49:35 +01:00
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
2017-02-13 17:16:13 +01:00
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
2016-03-16 12:49:35 +01:00
// Licensed under the Mozilla Public License (MPLv2)
//
/*global define */
// XEP-0059 Result Set Management
2016-03-16 12:49:35 +01:00
(function (root, factory) {
2017-07-22 22:21:05 +02:00
define('converse-mam',["jquery.noconflict", "converse-core", "converse-disco", "converse-chatview", // Could be made a soft dependency
"converse-muc", // Could be made a soft dependency
"strophe.rsm"], factory);
})(undefined, function ($, converse) {
2016-03-16 12:49:35 +01:00
"use strict";
2017-07-22 22:21:05 +02:00
var _converse$env = converse.env,
2017-08-08 22:11:13 +02:00
Promise = _converse$env.Promise,
2017-07-22 22:21:05 +02:00
Strophe = _converse$env.Strophe,
$iq = _converse$env.$iq,
_ = _converse$env._,
moment = _converse$env.moment;
2016-03-16 12:49:35 +01:00
var RSM_ATTRIBUTES = ['max', 'first', 'last', 'after', 'before', 'index', 'count'];
// XEP-0313 Message Archive Management
var MAM_ATTRIBUTES = ['with', 'start', 'end'];
2016-03-16 12:49:35 +01:00
2017-08-08 22:11:13 +02:00
function checkMAMSupport(_converse) {
/* Returns a promise which resolves when MAM is supported
* for this user, or which rejects if not.
*/
return _converse.api.waitUntil('discoInitialized').then(function () {
return new Promise(function (resolve, reject) {
function fulfillPromise(entity) {
if (entity.features.findWhere({ 'var': Strophe.NS.MAM })) {
resolve(true);
} else {
resolve(false);
}
}
var entity = _converse.disco_entities.get(_converse.bare_jid);
if (_.isUndefined(entity)) {
entity = _converse.disco_entities.create({ 'jid': _converse.bare_jid });
entity.on('featuresDiscovered', _.partial(fulfillPromise, entity));
} else {
fulfillPromise(entity);
}
});
});
}
2017-02-03 13:51:07 +01:00
converse.plugins.add('converse-mam', {
2016-03-16 12:49:35 +01:00
overrides: {
// Overrides mentioned here will be picked up by converse.js's
// plugin architecture they will replace existing methods on the
// relevant objects or classes.
//
// New functions which don't exist yet can also be added.
2016-06-20 21:11:43 +02:00
ChatBox: {
2017-07-22 22:21:05 +02:00
getMessageAttributes: function getMessageAttributes($message, $delay, original_stanza) {
2016-09-16 14:35:02 +02:00
var attrs = this.__super__.getMessageAttributes.apply(this, arguments);
2017-07-22 22:21:05 +02:00
attrs.archive_id = $(original_stanza).find("result[xmlns=\"" + Strophe.NS.MAM + "\"]").attr('id');
2016-06-20 21:11:43 +02:00
return attrs;
2016-03-16 12:49:35 +01:00
}
},
ChatBoxView: {
2017-07-22 22:21:05 +02:00
render: function render() {
2016-09-16 14:35:02 +02:00
var result = this.__super__.render.apply(this, arguments);
if (!this.disable_mam) {
this.$content.on('scroll', _.debounce(this.onScroll.bind(this), 100));
}
return result;
},
2017-07-22 22:21:05 +02:00
fetchArchivedMessagesIfNecessary: function fetchArchivedMessagesIfNecessary() {
2017-08-08 22:11:13 +02:00
var _this = this;
2016-12-13 20:46:07 +01:00
2017-08-08 22:11:13 +02:00
/* Check if archived messages should be fetched, and if so, do so. */
if (this.disable_mam || this.model.get('mam_initialized')) {
2017-07-22 22:21:05 +02:00
return;
}
2017-08-08 22:11:13 +02:00
var _converse = this.__super__._converse;
this.addSpinner();
checkMAMSupport(_converse).then(function (supported) {
// Success
if (supported) {
_this.fetchArchivedMessages();
} else {
_this.clearSpinner();
}
_this.model.save({ 'mam_initialized': true });
}, function () {
// Error
_this.clearSpinner();
_converse.log("Error or timeout while checking for MAM support", Strophe.LogLevel.ERROR);
}).catch(function (msg) {
_this.clearSpinner();
_converse.log(msg, Strophe.LogLevel.FATAL);
});
},
2017-07-22 22:21:05 +02:00
fetchArchivedMessages: function fetchArchivedMessages(options) {
2017-08-08 22:11:13 +02:00
var _this2 = this;
/* Fetch archived chat messages from the XMPP server.
*
* Then, upon receiving them, call onMessage on the chat
* box, so that they are displayed inside it.
*/
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2017-08-08 22:11:13 +02:00
if (!_converse.disco_entities.get(_converse.bare_jid).features.findWhere({ 'var': Strophe.NS.MAM })) {
2017-07-22 22:21:05 +02:00
_converse.log("Attempted to fetch archived messages but this " + "user's server doesn't support XEP-0313", Strophe.LogLevel.WARN);
return;
}
if (this.disable_mam) {
return;
}
this.addSpinner();
2017-07-22 22:21:05 +02:00
_converse.queryForArchivedMessages(_.extend({
'before': '', // Page backwards from the most recent message
'max': _converse.archived_messages_page_size,
'with': this.model.get('jid')
}, options), function (messages) {
// Success
2017-08-08 22:11:13 +02:00
_this2.clearSpinner();
2017-07-22 22:21:05 +02:00
if (messages.length) {
_.each(messages, _converse.chatboxes.onMessage.bind(_converse.chatboxes));
}
}, function () {
// Error
2017-08-08 22:11:13 +02:00
_this2.clearSpinner();
2017-07-22 22:21:05 +02:00
_converse.log("Error or timeout while trying to fetch " + "archived messages", Strophe.LogLevel.ERROR);
});
},
2017-07-22 22:21:05 +02:00
onScroll: function onScroll(ev) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
if ($(ev.target).scrollTop() === 0 && this.model.messages.length) {
this.fetchArchivedMessages({
2017-07-22 22:21:05 +02:00
'before': this.model.messages.at(0).get('archive_id')
});
}
2017-07-22 22:21:05 +02:00
}
},
ChatRoomView: {
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2017-06-23 23:22:52 +02:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2017-06-23 23:22:52 +02:00
this.__super__.initialize.apply(this, arguments);
2017-07-22 22:21:05 +02:00
this.model.on('change:mam_enabled', this.fetchArchivedMessagesIfNecessary, this);
this.model.on('change:connection_status', this.fetchArchivedMessagesIfNecessary, this);
2017-06-23 23:22:52 +02:00
},
2017-07-22 22:21:05 +02:00
render: function render() {
2016-09-16 14:35:02 +02:00
var result = this.__super__.render.apply(this, arguments);
if (!this.disable_mam) {
this.$content.on('scroll', _.debounce(this.onScroll.bind(this), 100));
}
return result;
},
2017-07-22 22:21:05 +02:00
handleMUCMessage: function handleMUCMessage(stanza) {
2017-03-05 10:45:18 +01:00
/* MAM (message archive management XEP-0313) messages are
* ignored, since they're handled separately.
*/
2017-07-22 22:21:05 +02:00
var is_mam = $(stanza).find("[xmlns=\"" + Strophe.NS.MAM + "\"]").length > 0;
2017-03-05 10:45:18 +01:00
if (is_mam) {
return true;
}
return this.__super__.handleMUCMessage.apply(this, arguments);
},
2017-07-22 22:21:05 +02:00
fetchArchivedMessagesIfNecessary: function fetchArchivedMessagesIfNecessary() {
if (this.model.get('connection_status') !== converse.ROOMSTATUS.ENTERED || !this.model.get('mam_enabled') || this.model.get('mam_initialized')) {
2017-03-05 10:45:18 +01:00
2017-07-22 22:21:05 +02:00
return;
}
this.fetchArchivedMessages();
this.model.save({ 'mam_initialized': true });
},
fetchArchivedMessages: function fetchArchivedMessages(options) {
/* Fetch archived chat messages for this Chat Room
*
* Then, upon receiving them, call onChatRoomMessage
* so that they are displayed inside it.
*/
this.addSpinner();
2017-06-23 23:22:52 +02:00
var that = this;
2017-07-22 22:21:05 +02:00
var _converse = this.__super__._converse;
_converse.api.archive.query(_.extend({
'groupchat': true,
'before': '', // Page backwards from the most recent message
'with': this.model.get('jid'),
'max': _converse.archived_messages_page_size
}, options), function (messages) {
that.clearSpinner();
if (messages.length) {
_.each(messages, that.onChatRoomMessage.bind(that));
}
2017-07-22 22:21:05 +02:00
}, function () {
that.clearSpinner();
_converse.log("Error while trying to fetch archived messages", Strophe.LogLevel.WARN);
});
}
2016-03-16 12:49:35 +01:00
}
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-03-16 12:49:35 +01:00
/* The initialize function gets called as soon as the plugin is
2017-02-03 13:51:07 +01:00
* loaded by Converse.js's plugin machinery.
2016-03-16 12:49:35 +01:00
*/
2017-02-03 13:51:07 +01:00
var _converse = this._converse;
2017-07-22 22:21:05 +02:00
2017-07-05 11:59:55 +02:00
_converse.api.settings.update({
2017-03-05 10:45:18 +01:00
archived_messages_page_size: '50',
2017-02-13 17:16:13 +01:00
message_archiving: undefined, // Supported values are 'always', 'never', 'roster' (https://xmpp.org/extensions/xep-0313.html#prefs)
2017-07-22 22:21:05 +02:00
message_archiving_timeout: 8000 // Time (in milliseconds) to wait before aborting MAM request
2016-03-16 12:49:35 +01:00
});
2017-02-03 13:51:07 +01:00
_converse.queryForArchivedMessages = function (options, callback, errback) {
/* Do a MAM (XEP-0313) query for archived messages.
*
* Parameters:
* (Object) options - Query parameters, either MAM-specific or also for Result Set Management.
* (Function) callback - A function to call whenever we receive query-relevant stanza.
* (Function) errback - A function to call when an error stanza is received.
*
* The options parameter can also be an instance of
* Strophe.RSM to enable easy querying between results pages.
*
* The callback function may be called multiple times, first
* for the initial IQ result and then for each message
* returned. The last time the callback is called, a
* Strophe.RSM object is returned on which "next" or "previous"
* can be called before passing it in again to this method, to
* get the next or previous page in the result set.
*/
2017-07-22 22:21:05 +02:00
var date = void 0;
if (_.isFunction(options)) {
callback = options;
errback = callback;
2016-03-16 12:49:35 +01:00
}
2017-02-03 13:51:07 +01:00
var queryid = _converse.connection.getUniqueId();
2017-07-22 22:21:05 +02:00
var attrs = { 'type': 'set' };
if (!_.isUndefined(options) && options.groupchat) {
if (!options['with']) {
2017-07-22 22:21:05 +02:00
// eslint-disable-line dot-notation
throw new Error('You need to specify a "with" value containing ' + 'the chat room JID, when querying groupchat messages.');
2016-03-16 12:49:35 +01:00
}
2017-07-22 22:21:05 +02:00
attrs.to = options['with']; // eslint-disable-line dot-notation
2016-03-16 12:49:35 +01:00
}
2017-07-22 22:21:05 +02:00
var stanza = $iq(attrs).c('query', { 'xmlns': Strophe.NS.MAM, 'queryid': queryid });
if (!_.isUndefined(options)) {
2017-07-22 22:21:05 +02:00
stanza.c('x', { 'xmlns': Strophe.NS.XFORM, 'type': 'submit' }).c('field', { 'var': 'FORM_TYPE', 'type': 'hidden' }).c('value').t(Strophe.NS.MAM).up().up();
if (options['with'] && !options.groupchat) {
2017-07-22 22:21:05 +02:00
// eslint-disable-line dot-notation
stanza.c('field', { 'var': 'with' }).c('value').t(options['with']).up().up(); // eslint-disable-line dot-notation
}
_.each(['start', 'end'], function (t) {
if (options[t]) {
date = moment(options[t]);
if (date.isValid()) {
2017-07-22 22:21:05 +02:00
stanza.c('field', { 'var': t }).c('value').t(date.format()).up().up();
} else {
2017-07-22 22:21:05 +02:00
throw new TypeError("archive.query: invalid date provided for: " + t);
}
}
});
stanza.up();
if (options instanceof Strophe.RSM) {
stanza.cnode(options.toXML());
} else if (_.intersection(RSM_ATTRIBUTES, _.keys(options)).length) {
stanza.cnode(new Strophe.RSM(options).toXML());
}
2016-03-16 12:49:35 +01:00
}
2016-07-28 18:06:31 +02:00
2017-07-22 22:21:05 +02:00
var messages = [];
2017-06-23 23:22:52 +02:00
var message_handler = _converse.connection.addHandler(function (message) {
var result = message.querySelector('result');
if (!_.isNull(result) && result.getAttribute('queryid') === queryid) {
messages.push(message);
}
return true;
}, Strophe.NS.MAM);
2017-07-22 22:21:05 +02:00
_converse.connection.sendIQ(stanza, function (iq) {
_converse.connection.deleteHandler(message_handler);
if (_.isFunction(callback)) {
var set = iq.querySelector('set');
var rsm = new Strophe.RSM({ xml: set });
_.extend(rsm, _.pick(options, _.concat(MAM_ATTRIBUTES, ['max'])));
callback(messages, rsm);
}
}, function () {
_converse.connection.deleteHandler(message_handler);
if (_.isFunction(errback)) {
errback.apply(this, arguments);
}
}, _converse.message_archiving_timeout);
2016-03-16 12:49:35 +01:00
};
2017-02-03 13:51:07 +01:00
_.extend(_converse.api, {
/* Extend default converse.js API to add methods specific to MAM
2016-03-16 12:49:35 +01:00
*/
'archive': {
2017-02-03 13:51:07 +01:00
'query': _converse.queryForArchivedMessages.bind(_converse)
2016-03-16 12:49:35 +01:00
}
});
2016-03-16 12:49:35 +01:00
2017-02-03 13:51:07 +01:00
_converse.onMAMError = function (iq) {
if ($(iq).find('feature-not-implemented').length) {
2017-07-22 22:21:05 +02:00
_converse.log("Message Archive Management (XEP-0313) not supported by this server", Strophe.LogLevel.WARN);
} else {
2017-07-22 22:21:05 +02:00
_converse.log("An error occured while trying to set archiving preferences.", Strophe.LogLevel.ERROR);
2017-02-03 13:51:07 +01:00
_converse.log(iq);
}
};
2017-02-03 13:51:07 +01:00
_converse.onMAMPreferences = function (feature, iq) {
/* Handle returned IQ stanza containing Message Archive
* Management (XEP-0313) preferences.
*
* XXX: For now we only handle the global default preference.
* The XEP also provides for per-JID preferences, which is
* currently not supported in converse.js.
*
* Per JID preferences will be set in chat boxes, so it'll
* probbaly be handled elsewhere in any case.
*/
2017-07-22 22:21:05 +02:00
var $prefs = $(iq).find("prefs[xmlns=\"" + Strophe.NS.MAM + "\"]");
var default_pref = $prefs.attr('default');
2017-07-22 22:21:05 +02:00
var stanza = void 0;
2017-02-03 13:51:07 +01:00
if (default_pref !== _converse.message_archiving) {
2017-07-22 22:21:05 +02:00
stanza = $iq({ 'type': 'set' }).c('prefs', { 'xmlns': Strophe.NS.MAM, 'default': _converse.message_archiving });
$prefs.children().each(function (idx, child) {
stanza.cnode(child).up();
});
2017-02-03 13:51:07 +01:00
_converse.connection.sendIQ(stanza, _.partial(function (feature, iq) {
2017-07-22 22:21:05 +02:00
// XXX: Strictly speaking, the server should respond with the updated prefs
// (see example 18: https://xmpp.org/extensions/xep-0313.html#config)
// but Prosody doesn't do this, so we don't rely on it.
feature.save({ 'preferences': { 'default': _converse.message_archiving } });
}, feature), _converse.onMAMError);
} else {
2017-07-22 22:21:05 +02:00
feature.save({ 'preferences': { 'default': _converse.message_archiving } });
}
};
2017-07-22 22:21:05 +02:00
/* Event handlers */
_converse.on('serviceDiscovered', function (feature) {
var prefs = feature.get('preferences') || {};
2017-07-22 22:21:05 +02:00
if (feature.get('var') === Strophe.NS.MAM && prefs['default'] !== _converse.message_archiving && // eslint-disable-line dot-notation
!_.isUndefined(_converse.message_archiving)) {
// Ask the server for archiving preferences
2017-07-22 22:21:05 +02:00
_converse.connection.sendIQ($iq({ 'type': 'get' }).c('prefs', { 'xmlns': Strophe.NS.MAM }), _.partial(_converse.onMAMPreferences, feature), _.partial(_converse.onMAMError, feature));
}
2017-07-22 22:21:05 +02:00
});
_converse.on('addClientFeatures', function () {
_converse.connection.disco.addFeature(Strophe.NS.MAM);
});
_converse.on('afterMessagesFetched', function (chatboxview) {
2017-08-08 22:11:13 +02:00
chatboxview.fetchArchivedMessagesIfNecessary();
2017-07-22 22:21:05 +02:00
});
}
});
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-mam.js.map;
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
2017-02-13 17:16:13 +01:00
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
// Licensed under the Mozilla Public License (MPLv2)
//
/*global define */
(function (root, factory) {
define('converse-vcard',["converse-core", "strophe.vcard"], factory);
2017-07-22 22:21:05 +02:00
})(undefined, function (converse) {
"use strict";
2017-07-22 22:21:05 +02:00
var _converse$env = converse.env,
Strophe = _converse$env.Strophe,
_ = _converse$env._,
moment = _converse$env.moment,
sizzle = _converse$env.sizzle;
2017-02-03 13:51:07 +01:00
converse.plugins.add('converse-vcard', {
overrides: {
// Overrides mentioned here will be picked up by converse.js's
// plugin architecture they will replace existing methods on the
// relevant objects or classes.
//
// New functions which don't exist yet can also be added.
RosterContacts: {
2017-07-22 22:21:05 +02:00
createRequestingContact: function createRequestingContact(presence) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
var bare_jid = Strophe.getBareJidFromJid(presence.getAttribute('from'));
2017-07-22 22:21:05 +02:00
_converse.getVCard(bare_jid, _.partial(_converse.createRequestingContactFromVCard, presence), function (iq, jid) {
_converse.log("Error while retrieving vcard for " + jid, Strophe.LogLevel.ERROR);
_converse.createRequestingContactFromVCard(presence, iq, jid);
});
}
}
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
/* The initialize function gets called as soon as the plugin is
* loaded by converse.js's plugin machinery.
*/
2017-02-03 13:51:07 +01:00
var _converse = this._converse;
2017-07-22 22:21:05 +02:00
2017-07-05 11:59:55 +02:00
_converse.api.settings.update({
2017-07-22 22:21:05 +02:00
use_vcards: true
});
2017-02-03 13:51:07 +01:00
_converse.createRequestingContactFromVCard = function (presence, iq, jid, fullname, img, img_type, url) {
var bare_jid = Strophe.getBareJidFromJid(jid);
2017-07-22 22:21:05 +02:00
if (!fullname) {
var nick_el = sizzle("nick[xmlns=\"" + Strophe.NS.NICK + "\"]", presence);
fullname = nick_el.length ? nick_el[0].textContent : bare_jid;
}
var user_data = {
jid: bare_jid,
subscription: 'none',
ask: null,
requesting: true,
2017-07-22 22:21:05 +02:00
fullname: fullname,
image: img,
image_type: img_type,
url: url,
vcard_updated: moment().format()
};
2017-02-03 13:51:07 +01:00
_converse.roster.create(user_data);
_converse.emit('contactRequest', user_data);
};
2017-02-03 13:51:07 +01:00
_converse.onVCardError = function (jid, iq, errback) {
var contact = _converse.roster.get(jid);
if (contact) {
contact.save({ 'vcard_updated': moment().format() });
}
2017-07-22 22:21:05 +02:00
if (errback) {
errback(iq, jid);
}
};
2017-02-03 13:51:07 +01:00
_converse.onVCardData = function (jid, iq, callback) {
2017-07-22 22:21:05 +02:00
var vcard = iq.querySelector('vCard'),
img_type = _.get(vcard.querySelector('TYPE'), 'textContent'),
img = _.get(vcard.querySelector('BINVAL'), 'textContent'),
url = _.get(vcard.querySelector('URL'), 'textContent'),
fullname = _.get(vcard.querySelector('FN'), 'textContent');
2017-07-22 22:21:05 +02:00
if (jid) {
2017-02-03 13:51:07 +01:00
var contact = _converse.roster.get(jid);
if (contact) {
contact.save({
'fullname': fullname || _.get(contact, 'fullname', jid),
'image_type': img_type,
'image': img,
'url': url,
'vcard_updated': moment().format()
});
}
}
if (callback) {
callback(iq, jid, fullname, img, img_type, url);
}
};
2017-02-03 13:51:07 +01:00
_converse.getVCard = function (jid, callback, errback) {
/* Request the VCard of another user.
*
* Parameters:
* (String) jid - The Jabber ID of the user whose VCard
* is being requested.
* (Function) callback - A function to call once the VCard is
* returned.
* (Function) errback - A function to call if an error occured
* while trying to fetch the VCard.
*/
2017-02-03 13:51:07 +01:00
if (!_converse.use_vcards) {
2017-07-22 22:21:05 +02:00
if (callback) {
callback(null, jid);
}
} else {
2017-07-22 22:21:05 +02:00
_converse.connection.vcard.get(_.partial(_converse.onVCardData, jid, _, callback), jid, _.partial(_converse.onVCardError, jid, _, errback));
}
};
2017-07-22 22:21:05 +02:00
/* Event handlers */
_converse.on('addClientFeatures', function () {
if (_converse.use_vcards) {
_converse.connection.disco.addFeature(Strophe.NS.VCARD);
}
});
var updateVCardForChatBox = function updateVCardForChatBox(chatbox) {
if (!_converse.use_vcards) {
return;
}
2016-03-16 12:49:35 +01:00
var jid = chatbox.model.get('jid'),
2017-02-03 13:51:07 +01:00
contact = _converse.roster.get(jid);
2017-07-22 22:21:05 +02:00
if (contact && !contact.get('vcard_updated')) {
_converse.getVCard(jid, function (iq, jid, fullname, image, image_type, url) {
chatbox.model.save({
'fullname': fullname || jid,
'url': url,
'image_type': image_type,
'image': image
});
}, function () {
_converse.log("updateVCardForChatBox: Error occured while fetching vcard", Strophe.LogLevel.ERROR);
});
2016-03-16 12:49:35 +01:00
}
};
2017-02-03 13:51:07 +01:00
_converse.on('chatBoxInitialized', updateVCardForChatBox);
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
var onContactAdd = function onContactAdd(contact) {
if (!contact.get('vcard_updated')) {
// This will update the vcard, which triggers a change
// request which will rerender the roster contact.
2017-02-03 13:51:07 +01:00
_converse.getVCard(contact.get('jid'));
}
};
2017-02-03 13:51:07 +01:00
_converse.on('initialized', function () {
_converse.roster.on("add", onContactAdd);
});
2017-07-22 22:21:05 +02:00
_converse.on('statusInitialized', function fetchOwnVCard() {
2017-02-03 13:51:07 +01:00
if (_converse.xmppstatus.get('fullname') === undefined) {
2017-07-22 22:21:05 +02:00
_converse.getVCard(null, // No 'to' attr when getting one's own vCard
function (iq, jid, fullname) {
_converse.xmppstatus.save({ 'fullname': fullname });
});
2016-03-16 12:49:35 +01:00
}
2017-07-22 22:21:05 +02:00
});
}
});
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-vcard.js.map;
2016-11-07 15:43:48 +01:00
define('tpl!toolbar_otr', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
if (allow_otr) { ;
2017-07-22 22:21:05 +02:00
__p += '\n <li class="toggle-toolbar-menu toggle-otr ' +
2017-02-13 17:16:13 +01:00
__e(otr_status_class) +
'" title="' +
2017-02-13 17:16:13 +01:00
__e(otr_tooltip) +
'">\n <span class="chat-toolbar-text">' +
2017-02-13 17:16:13 +01:00
__e(otr_translated_status) +
2016-11-07 15:43:48 +01:00
'</span>\n ';
if (otr_status == UNENCRYPTED) { ;
__p += '\n <span class="icon-unlocked"></span>\n ';
} ;
__p += '\n ';
if (otr_status == UNVERIFIED) { ;
__p += '\n <span class="icon-lock"></span>\n ';
} ;
__p += '\n ';
if (otr_status == VERIFIED) { ;
__p += '\n <span class="icon-lock"></span>\n ';
} ;
__p += '\n ';
if (otr_status == FINISHED) { ;
__p += '\n <span class="icon-unlocked"></span>\n ';
} ;
2017-07-22 22:21:05 +02:00
__p += '\n <ul class="toolbar-menu collapsed">\n ';
if (otr_status == UNENCRYPTED) { ;
__p += '\n <li><a class="start-otr" href="#">' +
2017-02-13 17:16:13 +01:00
__e(label_start_encrypted_conversation) +
2016-11-07 15:43:48 +01:00
'</a></li>\n ';
} ;
__p += '\n ';
if (otr_status != UNENCRYPTED) { ;
__p += '\n <li><a class="start-otr" href="#">' +
2017-02-13 17:16:13 +01:00
__e(label_refresh_encrypted_conversation) +
'</a></li>\n <li><a class="end-otr" href="#">' +
2017-02-13 17:16:13 +01:00
__e(label_end_encrypted_conversation) +
'</a></li>\n <li><a class="auth-otr" data-scheme="smp" href="#">' +
2017-02-13 17:16:13 +01:00
__e(label_verify_with_smp) +
2016-11-07 15:43:48 +01:00
'</a></li>\n ';
} ;
__p += '\n ';
if (otr_status == UNVERIFIED) { ;
__p += '\n <li><a class="auth-otr" data-scheme="fingerprint" href="#">' +
2017-02-13 17:16:13 +01:00
__e(label_verify_with_fingerprints) +
2016-11-07 15:43:48 +01:00
'</a></li>\n ';
} ;
__p += '\n <li><a href="http://www.cypherpunks.ca/otr/help/3.2.0/levels.php" target="_blank" rel="noopener">' +
2017-02-13 17:16:13 +01:00
__e(label_whats_this) +
2016-11-07 15:43:48 +01:00
'</a></li>\n </ul>\n </li>\n';
} ;
__p += '\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
2017-02-13 17:16:13 +01:00
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
// Licensed under the Mozilla Public License (MPLv2)
//
2017-04-23 19:02:44 +02:00
/*global define, window, crypto, CryptoJS */
/* This is a Converse.js plugin which add support Off-the-record (OTR)
* encryption of one-on-one chat messages.
*/
(function (root, factory) {
2017-07-22 22:21:05 +02:00
define('converse-otr',["jquery.noconflict", "converse-chatview", "tpl!toolbar_otr", 'otr'], factory);
})(undefined, function ($, converse, tpl_toolbar_otr, otr) {
"use strict";
2017-07-22 22:21:05 +02:00
var _converse$env = converse.env,
Strophe = _converse$env.Strophe,
utils = _converse$env.utils,
b64_sha1 = _converse$env.b64_sha1,
_ = _converse$env._;
var HAS_CSPRNG = !_.isUndefined(crypto) && (_.isFunction(crypto.randomBytes) || _.isFunction(crypto.getRandomValues));
var HAS_CRYPTO = HAS_CSPRNG && !_.isUndefined(otr.OTR) && !_.isUndefined(otr.DSA);
var UNENCRYPTED = 0;
2017-07-22 22:21:05 +02:00
var UNVERIFIED = 1;
var VERIFIED = 2;
var FINISHED = 3;
2017-07-22 22:21:05 +02:00
var OTR_TRANSLATED_MAPPING = {}; // Populated in initialize
var OTR_CLASS_MAPPING = {};
OTR_CLASS_MAPPING[UNENCRYPTED] = 'unencrypted';
OTR_CLASS_MAPPING[UNVERIFIED] = 'unverified';
OTR_CLASS_MAPPING[VERIFIED] = 'verified';
OTR_CLASS_MAPPING[FINISHED] = 'finished';
2017-02-03 13:51:07 +01:00
converse.plugins.add('converse-otr', {
overrides: {
// Overrides mentioned here will be picked up by converse.js's
// plugin architecture they will replace existing methods on the
// relevant objects or classes.
//
// New functions which don't exist yet can also be added.
ChatBox: {
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-09-16 14:35:02 +02:00
this.__super__.initialize.apply(this, arguments);
if (this.get('box_id') !== 'controlbox') {
2017-07-22 22:21:05 +02:00
this.save({ 'otr_status': this.get('otr_status') || UNENCRYPTED });
}
},
2017-07-22 22:21:05 +02:00
shouldPlayNotification: function shouldPlayNotification($message) {
/* Don't play a notification if this is an OTR message but
* encryption is not yet set up. That would mean that the
* OTR session is still being established, so there are no
* "visible" OTR messages being exchanged.
*/
2017-07-22 22:21:05 +02:00
return this.__super__.shouldPlayNotification.apply(this, arguments) && !(utils.isOTRMessage($message[0]) && !_.includes([UNVERIFIED, VERIFIED], this.get('otr_status')));
},
2017-07-22 22:21:05 +02:00
createMessage: function createMessage(message, delay, original_stanza) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse,
text = _.propertyOf(message.querySelector('body'))('textContent');
2017-07-22 22:21:05 +02:00
if (!text || !_converse.allow_otr) {
2016-09-16 14:35:02 +02:00
return this.__super__.createMessage.apply(this, arguments);
}
2017-06-23 20:25:33 +02:00
if (utils.isNewMessage(original_stanza)) {
if (text.match(/^\?OTRv23?/)) {
return this.initiateOTR(text);
} else if (_.includes([UNVERIFIED, VERIFIED], this.get('otr_status'))) {
return this.otr.receiveMsg(text);
} else if (text.match(/^\?OTR/)) {
if (!this.otr) {
return this.initiateOTR(text);
} else {
2017-06-23 20:25:33 +02:00
return this.otr.receiveMsg(text);
}
}
}
2017-06-23 20:25:33 +02:00
// Normal unencrypted message (or archived message)
return this.__super__.createMessage.apply(this, arguments);
},
2017-07-22 22:21:05 +02:00
generatePrivateKey: function generatePrivateKey(instance_tag) {
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
var key = new otr.DSA();
var jid = _converse.connection.jid;
2017-07-22 22:21:05 +02:00
if (_converse.cache_otr_key) {
this.save({
'otr_priv_key': key.packPrivate(),
'otr_instance_tag': instance_tag
});
}
return key;
},
2017-07-22 22:21:05 +02:00
getSession: function getSession(callback) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse,
__ = _converse.__;
2017-07-22 22:21:05 +02:00
var instance_tag = void 0,
saved_key = void 0,
encrypted_key = void 0;
2017-02-03 13:51:07 +01:00
if (_converse.cache_otr_key) {
2017-04-23 19:02:44 +02:00
encrypted_key = this.get('otr_priv_key');
if (_.isString(encrypted_key)) {
instance_tag = this.get('otr_instance_tag');
2017-07-22 22:21:05 +02:00
saved_key = otr.DSA.parsePrivate(encrypted_key);
2017-04-23 19:02:44 +02:00
if (saved_key && instance_tag) {
this.trigger('showHelpMessages', [__('Re-establishing encrypted session')]);
callback({
'key': saved_key,
'instance_tag': instance_tag
});
return; // Our work is done here
}
}
}
// We need to generate a new key and instance tag
2017-07-22 22:21:05 +02:00
this.trigger('showHelpMessages', [__('Generating private key.'), __('Your browser might become unresponsive.')], null, true // show spinner
);
var that = this;
window.setTimeout(function () {
callback({
'key': that.generatePrivateKey(instance_tag),
2017-04-23 19:02:44 +02:00
'instance_tag': otr.OTR.makeInstanceTag()
});
}, 500);
},
2017-07-22 22:21:05 +02:00
updateOTRStatus: function updateOTRStatus(state) {
switch (state) {
case otr.OTR.CONST.STATUS_AKE_SUCCESS:
if (this.otr.msgstate === otr.OTR.CONST.MSGSTATE_ENCRYPTED) {
2017-07-22 22:21:05 +02:00
this.save({ 'otr_status': UNVERIFIED });
}
break;
case otr.OTR.CONST.STATUS_END_OTR:
if (this.otr.msgstate === otr.OTR.CONST.MSGSTATE_FINISHED) {
2017-07-22 22:21:05 +02:00
this.save({ 'otr_status': FINISHED });
} else if (this.otr.msgstate === otr.OTR.CONST.MSGSTATE_PLAINTEXT) {
2017-07-22 22:21:05 +02:00
this.save({ 'otr_status': UNENCRYPTED });
}
break;
}
},
2017-07-22 22:21:05 +02:00
onSMP: function onSMP(type, data) {
// Event handler for SMP (Socialist's Millionaire Protocol)
// used by OTR (off-the-record).
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse,
__ = _converse.__;
2017-07-22 22:21:05 +02:00
switch (type) {
case 'question':
2017-07-22 22:21:05 +02:00
this.otr.smpSecret(prompt(__('Authentication request from %1$s\n\nYour chat contact is attempting to verify your identity, by asking you the question below.\n\n%2$s', [this.get('fullname'), data])));
break;
case 'trust':
if (data === true) {
2017-07-22 22:21:05 +02:00
this.save({ 'otr_status': VERIFIED });
} else {
2017-07-22 22:21:05 +02:00
this.trigger('showHelpMessages', [__("Could not verify this user's identify.")], 'error');
this.save({ 'otr_status': UNVERIFIED });
}
break;
default:
throw new TypeError('ChatBox.onSMP: Unknown type for SMP');
}
},
2017-07-22 22:21:05 +02:00
initiateOTR: function initiateOTR(query_msg) {
var _this = this;
// Sets up an OTR object through which we can send and receive
// encrypted messages.
//
// If 'query_msg' is passed in, it means there is an alread incoming
// query message from our contact. Otherwise, it is us who will
// send the query message to them.
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse,
__ = _converse.__;
2017-07-22 22:21:05 +02:00
this.save({ 'otr_status': UNENCRYPTED });
this.getSession(function (session) {
2017-07-22 22:21:05 +02:00
var _converse = _this.__super__._converse;
_this.otr = new otr.OTR({
fragment_size: 140,
send_interval: 200,
priv: session.key,
instance_tag: session.instance_tag,
2017-07-22 22:21:05 +02:00
debug: _this.debug
});
_this.otr.on('status', _this.updateOTRStatus.bind(_this));
_this.otr.on('smp', _this.onSMP.bind(_this));
_this.otr.on('ui', function (msg) {
_this.trigger('showReceivedOTRMessage', msg);
});
_this.otr.on('io', function (msg) {
_this.trigger('sendMessage', new _converse.Message({ message: msg }));
});
2017-07-22 22:21:05 +02:00
_this.otr.on('error', function (msg) {
_this.trigger('showOTRError', msg);
});
_this.trigger('showHelpMessages', [__('Exchanging private key with contact.')]);
if (query_msg) {
2017-07-22 22:21:05 +02:00
_this.otr.receiveMsg(query_msg);
} else {
2017-07-22 22:21:05 +02:00
_this.otr.sendQueryMsg();
}
2017-07-22 22:21:05 +02:00
});
},
2017-07-22 22:21:05 +02:00
endOTR: function endOTR() {
if (this.otr) {
this.otr.endOtr();
}
2017-07-22 22:21:05 +02:00
this.save({ 'otr_status': UNENCRYPTED });
}
},
2017-07-22 22:21:05 +02:00
ChatBoxView: {
events: {
'click .toggle-otr': 'toggleOTRMenu',
'click .start-otr': 'startOTRFromToolbar',
'click .end-otr': 'endOTR',
'click .auth-otr': 'authOTR'
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2016-09-16 14:35:02 +02:00
this.__super__.initialize.apply(this, arguments);
this.model.on('change:otr_status', this.onOTRStatusChanged, this);
this.model.on('showOTRError', this.showOTRError, this);
this.model.on('showSentOTRMessage', function (text) {
2017-07-22 22:21:05 +02:00
this.showMessage({ 'message': text, 'sender': 'me' });
}, this);
this.model.on('showReceivedOTRMessage', function (text) {
2017-07-22 22:21:05 +02:00
this.showMessage({ 'message': text, 'sender': 'them' });
}, this);
2017-07-22 22:21:05 +02:00
if (_.includes([UNVERIFIED, VERIFIED], this.model.get('otr_status')) || _converse.use_otr_by_default) {
this.model.initiateOTR();
}
},
2017-07-22 22:21:05 +02:00
createMessageStanza: function createMessageStanza() {
2016-09-16 14:35:02 +02:00
var stanza = this.__super__.createMessageStanza.apply(this, arguments);
2016-06-20 21:11:43 +02:00
if (this.model.get('otr_status') !== UNENCRYPTED || utils.isOTRMessage(stanza.nodeTree)) {
// OTR messages aren't carbon copied
2017-07-22 22:21:05 +02:00
stanza.c('private', { 'xmlns': Strophe.NS.CARBONS }).up().c('no-store', { 'xmlns': Strophe.NS.HINTS }).up().c('no-permanent-store', { 'xmlns': Strophe.NS.HINTS }).up().c('no-copy', { 'xmlns': Strophe.NS.HINTS });
}
return stanza;
},
2017-07-22 22:21:05 +02:00
onMessageSubmitted: function onMessageSubmitted(text) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2017-02-03 13:51:07 +01:00
if (!_converse.connection.authenticated) {
2017-07-22 22:21:05 +02:00
return this.showHelpMessages(['Sorry, the connection has been lost, ' + 'and your message could not be sent'], 'error');
}
var match = text.replace(/^\s*/, "").match(/^\/(.*)\s*$/);
if (match) {
2017-07-22 22:21:05 +02:00
if (_converse.allow_otr && match[1] === "endotr") {
return this.endOTR();
2017-07-22 22:21:05 +02:00
} else if (_converse.allow_otr && match[1] === "otr") {
return this.model.initiateOTR();
}
}
if (_.includes([UNVERIFIED, VERIFIED], this.model.get('otr_status'))) {
// Off-the-record encryption is active
this.model.otr.sendMsg(text);
this.model.trigger('showSentOTRMessage', text);
} else {
2016-09-16 14:35:02 +02:00
this.__super__.onMessageSubmitted.apply(this, arguments);
}
},
2017-07-22 22:21:05 +02:00
onOTRStatusChanged: function onOTRStatusChanged() {
this.renderToolbar().informOTRChange();
},
2017-07-22 22:21:05 +02:00
informOTRChange: function informOTRChange() {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse,
__ = _converse.__,
data = this.model.toJSON(),
msgs = [];
2017-07-22 22:21:05 +02:00
if (data.otr_status === UNENCRYPTED) {
msgs.push(__("Your messages are not encrypted anymore"));
} else if (data.otr_status === UNVERIFIED) {
msgs.push(__("Your messages are now encrypted but your contact's identity has not been verified."));
} else if (data.otr_status === VERIFIED) {
msgs.push(__("Your contact's identify has been verified."));
} else if (data.otr_status === FINISHED) {
msgs.push(__("Your contact has ended encryption on their end, you should do the same."));
}
return this.showHelpMessages(msgs, 'info', false);
},
2017-07-22 22:21:05 +02:00
showOTRError: function showOTRError(msg) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse,
__ = _converse.__;
2017-07-22 22:21:05 +02:00
if (msg === 'Message cannot be sent at this time.') {
2017-07-22 22:21:05 +02:00
this.showHelpMessages([__('Your message could not be sent')], 'error');
} else if (msg === 'Received an unencrypted message.') {
2017-07-22 22:21:05 +02:00
this.showHelpMessages([__('We received an unencrypted message')], 'error');
} else if (msg === 'Received an unreadable encrypted message.') {
2017-07-22 22:21:05 +02:00
this.showHelpMessages([__('We received an unreadable encrypted message')], 'error');
} else {
2017-07-22 22:21:05 +02:00
this.showHelpMessages(["Encryption error occured: " + msg], 'error');
}
2017-07-22 22:21:05 +02:00
_converse.log("OTR ERROR:" + msg, Strophe.LogLevel.ERROR);
},
2017-07-22 22:21:05 +02:00
startOTRFromToolbar: function startOTRFromToolbar(ev) {
ev.stopPropagation();
this.model.initiateOTR();
},
2017-07-22 22:21:05 +02:00
endOTR: function endOTR(ev) {
if (!_.isUndefined(ev)) {
ev.preventDefault();
ev.stopPropagation();
}
this.model.endOTR();
},
2017-07-22 22:21:05 +02:00
authOTR: function authOTR(ev) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse,
__ = _converse.__,
2017-07-22 22:21:05 +02:00
_$$data = $(ev.target).data(),
scheme = _$$data.scheme;
var result = void 0,
question = void 0,
answer = void 0;
if (scheme === 'fingerprint') {
2017-07-22 22:21:05 +02:00
result = confirm(__('Here are the fingerprints, please confirm them with %1$s, outside of this chat.\n\nFingerprint for you, %2$s: %3$s\n\nFingerprint for %1$s: %4$s\n\nIf you have confirmed that the fingerprints match, click OK, otherwise click Cancel.', [this.model.get('fullname'), _converse.xmppstatus.get('fullname') || _converse.bare_jid, this.model.otr.priv.fingerprint(), this.model.otr.their_priv_pk.fingerprint()]));
if (result === true) {
2017-07-22 22:21:05 +02:00
this.model.save({ 'otr_status': VERIFIED });
} else {
2017-07-22 22:21:05 +02:00
this.model.save({ 'otr_status': UNVERIFIED });
}
} else if (scheme === 'smp') {
alert(__('You will be prompted to provide a security question and then an answer to that question.\n\nYour contact will then be prompted the same question and if they type the exact same answer (case sensitive), their identity will be verified.'));
question = prompt(__('What is your security question?'));
if (question) {
answer = prompt(__('What is the answer to the security question?'));
this.model.otr.smpSecret(answer, question);
}
} else {
this.showHelpMessages([__('Invalid authentication scheme provided')], 'error');
}
},
2017-07-22 22:21:05 +02:00
toggleOTRMenu: function toggleOTRMenu(ev) {
ev.stopPropagation();
2017-07-22 22:21:05 +02:00
var menu = this.el.querySelector('.toggle-otr ul');
var elements = _.difference(document.querySelectorAll('.toolbar-menu'), [menu]);
utils.slideInAllElements(elements).then(_.partial(utils.slideToggleElement, menu));
},
2017-07-22 22:21:05 +02:00
getOTRTooltip: function getOTRTooltip() {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse,
__ = _converse.__,
data = this.model.toJSON();
2017-07-22 22:21:05 +02:00
if (data.otr_status === UNENCRYPTED) {
return __('Your messages are not encrypted. Click here to enable OTR encryption.');
} else if (data.otr_status === UNVERIFIED) {
return __('Your messages are encrypted, but your contact has not been verified.');
} else if (data.otr_status === VERIFIED) {
return __('Your messages are encrypted and your contact verified.');
} else if (data.otr_status === FINISHED) {
return __('Your contact has closed their end of the private session, you should do the same');
}
},
2017-07-22 22:21:05 +02:00
renderToolbar: function renderToolbar(toolbar, options) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse,
__ = _converse.__;
2017-07-22 22:21:05 +02:00
2017-02-03 13:51:07 +01:00
if (!_converse.show_toolbar) {
return;
}
var data = this.model.toJSON();
options = _.extend(options || {}, {
FINISHED: FINISHED,
UNENCRYPTED: UNENCRYPTED,
UNVERIFIED: UNVERIFIED,
VERIFIED: VERIFIED,
// FIXME: Leaky abstraction MUC
2017-02-03 13:51:07 +01:00
allow_otr: _converse.allow_otr && !this.is_chatroom,
label_end_encrypted_conversation: __('End encrypted conversation'),
label_refresh_encrypted_conversation: __('Refresh encrypted conversation'),
label_start_encrypted_conversation: __('Start encrypted conversation'),
label_verify_with_fingerprints: __('Verify with fingerprints'),
label_verify_with_smp: __('Verify with SMP'),
label_whats_this: __("What\'s this?"),
otr_status_class: OTR_CLASS_MAPPING[data.otr_status],
otr_tooltip: this.getOTRTooltip(),
2017-07-22 22:21:05 +02:00
otr_translated_status: OTR_TRANSLATED_MAPPING[data.otr_status]
});
2016-11-07 15:43:48 +01:00
this.__super__.renderToolbar.apply(this, arguments);
2017-07-22 22:21:05 +02:00
this.$el.find('.chat-toolbar').append(tpl_toolbar_otr(_.extend(this.model.toJSON(), options || {})));
return this;
}
}
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
/* The initialize function gets called as soon as the plugin is
* loaded by converse.js's plugin machinery.
*/
2017-02-03 13:51:07 +01:00
var _converse = this._converse,
__ = _converse.__;
2017-07-22 22:21:05 +02:00
2017-07-05 11:59:55 +02:00
_converse.api.settings.update({
allow_otr: true,
cache_otr_key: false,
use_otr_by_default: false
});
2017-02-03 13:51:07 +01:00
// Translation aware constants
// ---------------------------
// We can only call the __ translation method *after* converse.js
// has been initialized and with it the i18n machinery. That's why
// we do it here in the "initialize" method and not at the top of
// the module.
OTR_TRANSLATED_MAPPING[UNENCRYPTED] = __('unencrypted');
OTR_TRANSLATED_MAPPING[UNVERIFIED] = __('unverified');
OTR_TRANSLATED_MAPPING[VERIFIED] = __('verified');
OTR_TRANSLATED_MAPPING[FINISHED] = __('finished');
// Only allow OTR if we have the capability
2017-02-03 13:51:07 +01:00
_converse.allow_otr = _converse.allow_otr && HAS_CRYPTO;
// Only use OTR by default if allow OTR is enabled to begin with
2017-02-03 13:51:07 +01:00
_converse.use_otr_by_default = _converse.use_otr_by_default && _converse.allow_otr;
}
});
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-otr.js.map;
2016-11-07 15:43:48 +01:00
define('tpl!register_panel', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-04-04 17:26:06 +02:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
__p += '<form id="converse-register" class="pure-form converse-form">\n <span class="reg-feedback"></span>\n <label>' +
2017-02-13 17:16:13 +01:00
__e(label_domain) +
2017-04-04 17:26:06 +02:00
'</label>\n ';
if (default_domain) { ;
__p += '\n ' +
__e(default_domain) +
'\n ';
} ;
__p += '\n ';
if (!default_domain) { ;
__p += '\n <input type="text" name="domain" placeholder="' +
2017-02-13 17:16:13 +01:00
__e(domain_placeholder) +
2017-04-04 17:26:06 +02:00
'">\n <p class="form-help">' +
2017-02-13 17:16:13 +01:00
__e(help_providers) +
' <a href="' +
2017-02-13 17:16:13 +01:00
__e(href_providers) +
'" class="url" target="_blank" rel="noopener">' +
2017-02-13 17:16:13 +01:00
__e(help_providers_link) +
2017-04-04 17:26:06 +02:00
'</a>.</p>\n <input class="pure-button button-primary" type="submit" value="' +
2017-02-13 17:16:13 +01:00
__e(label_register) +
2017-04-04 17:26:06 +02:00
'">\n ';
} ;
__p += '\n</form>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-11-07 15:43:48 +01:00
define('tpl!register_tab', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
2017-04-04 17:26:06 +02:00
__p += '<li><a class="s" data-id="register" href="#register">' +
2017-02-13 17:16:13 +01:00
__e(label_register) +
2016-11-07 15:43:48 +01:00
'</a></li>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!registration_form', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<p class="provider-title">' +
2017-02-13 17:16:13 +01:00
__e(domain) +
2017-06-23 20:25:33 +02:00
'</p>\n<p class="title">' +
2017-02-13 17:16:13 +01:00
__e(title) +
'</p>\n<p class="instructions">' +
2017-02-13 17:16:13 +01:00
__e(instructions) +
2016-11-07 15:43:48 +01:00
'</p>\n';
}
return __p
};});
define('tpl!registration_request', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-04-04 17:26:06 +02:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-04-04 17:26:06 +02:00
__p += '<span class="spinner login-submit"></span>\n<p class="info">' +
2017-02-13 17:16:13 +01:00
__e(info_message) +
2017-04-04 17:26:06 +02:00
'</p>\n';
if (cancel) { ;
__p += '\n <button class="pure-button button-cancel hor_centered">' +
2017-02-13 17:16:13 +01:00
__e(cancel) +
2016-11-07 15:43:48 +01:00
'</button>\n';
2017-04-04 17:26:06 +02:00
} ;
__p += '\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-11-07 15:43:48 +01:00
2017-07-22 22:21:05 +02:00
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
2017-02-13 17:16:13 +01:00
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
// Licensed under the Mozilla Public License (MPLv2)
//
2017-04-23 19:02:44 +02:00
/*global define */
/* This is a Converse.js plugin which add support for in-band registration
* as specified in XEP-0077.
*/
(function (root, factory) {
2017-07-22 22:21:05 +02:00
define('converse-register',["jquery.noconflict", "converse-core", "tpl!form_username", "tpl!register_panel", "tpl!register_tab", "tpl!registration_form", "tpl!registration_request", "tpl!spinner", "converse-controlbox"], factory);
})(undefined, function ($, converse, tpl_form_username, tpl_register_panel, tpl_register_tab, tpl_registration_form, tpl_registration_request, tpl_spinner) {
2016-11-07 15:43:48 +01:00
"use strict";
2016-11-07 15:43:48 +01:00
// Strophe methods for building stanzas
2017-07-22 22:21:05 +02:00
var _converse$env = converse.env,
Strophe = _converse$env.Strophe,
Backbone = _converse$env.Backbone,
utils = _converse$env.utils,
$iq = _converse$env.$iq,
_ = _converse$env._;
2017-04-04 17:26:06 +02:00
// Add Strophe Namespaces
2017-07-22 22:21:05 +02:00
Strophe.addNamespace('REGISTER', 'jabber:iq:register');
// Add Strophe Statuses
var i = 0;
_.each(_.keys(Strophe.Status), function (key) {
i = Math.max(i, Strophe.Status[key]);
});
2017-07-22 22:21:05 +02:00
Strophe.Status.REGIFAIL = i + 1;
Strophe.Status.REGISTERED = i + 2;
Strophe.Status.CONFLICT = i + 3;
Strophe.Status.NOTACCEPTABLE = i + 5;
2017-02-03 13:51:07 +01:00
converse.plugins.add('converse-register', {
overrides: {
// Overrides mentioned here will be picked up by converse.js's
// plugin architecture they will replace existing methods on the
// relevant objects or classes.
//
// New functions which don't exist yet can also be added.
ControlBoxView: {
2017-07-22 22:21:05 +02:00
switchTab: function switchTab(ev) {
2017-04-04 17:26:06 +02:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2017-04-04 17:26:06 +02:00
var result = this.__super__.switchTab.apply(this, arguments);
2017-07-22 22:21:05 +02:00
if (_converse.registration_domain && ev.target.getAttribute('data-id') === "register" && !this.model.get('registration_form_rendered')) {
2017-04-04 17:26:06 +02:00
this.registerpanel.fetchRegistrationForm(_converse.registration_domain);
}
return result;
},
2017-07-22 22:21:05 +02:00
renderRegistrationPanel: function renderRegistrationPanel() {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2017-02-03 13:51:07 +01:00
if (_converse.allow_registration) {
this.registerpanel = new _converse.RegisterPanel({
'$parent': this.$el.find('.controlbox-panes'),
2017-04-04 17:26:06 +02:00
'model': this.model
2016-05-03 17:37:10 +02:00
});
2016-11-07 15:43:48 +01:00
this.registerpanel.render().$el.addClass('hidden');
}
return this;
2017-06-23 20:25:33 +02:00
},
2017-07-22 22:21:05 +02:00
renderLoginPanel: function renderLoginPanel() {
2017-06-23 20:25:33 +02:00
/* Also render a registration panel, when rendering the
* login panel.
*/
this.__super__.renderLoginPanel.apply(this, arguments);
this.renderRegistrationPanel();
return this;
}
}
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
/* The initialize function gets called as soon as the plugin is
* loaded by converse.js's plugin machinery.
*/
2017-02-03 13:51:07 +01:00
var _converse = this._converse,
__ = _converse.__;
// Add new templates
_converse.templates.form_username = tpl_form_username;
_converse.templates.register_panel = tpl_register_panel;
_converse.templates.register_tab = tpl_register_tab;
_converse.templates.registration_form = tpl_registration_form;
_converse.templates.registration_request = tpl_registration_request;
2017-07-05 11:59:55 +02:00
_converse.api.settings.update({
allow_registration: true,
2017-07-22 22:21:05 +02:00
domain_placeholder: __(" e.g. conversejs.org"), // Placeholder text shown in the domain input on the registration form
providers_link: 'https://xmpp.net/directory.php' // Link to XMPP providers shown on registration page
});
2017-02-03 13:51:07 +01:00
_converse.RegisterPanel = Backbone.View.extend({
tagName: 'div',
id: "register",
className: 'controlbox-pane',
events: {
'submit form#converse-register': 'onProviderChosen'
},
2017-07-22 22:21:05 +02:00
initialize: function initialize(cfg) {
this.reset();
this.$parent = cfg.$parent;
this.$tabs = cfg.$parent.parent().find('#controlbox-tabs');
this.registerHooks();
},
2017-07-22 22:21:05 +02:00
render: function render() {
2017-04-04 17:26:06 +02:00
this.model.set('registration_form_rendered', false);
2017-07-22 22:21:05 +02:00
this.$parent.append(this.$el.html(tpl_register_panel({
'default_domain': _converse.registration_domain,
'label_domain': __("Your XMPP provider's domain name:"),
'label_register': __('Fetch registration form'),
'help_providers': __('Tip: A list of public XMPP providers is available'),
'help_providers_link': __('here'),
'href_providers': _converse.providers_link,
'domain_placeholder': _converse.domain_placeholder
})));
this.$tabs.append(tpl_register_tab({ label_register: __('Register') }));
return this;
},
2017-07-22 22:21:05 +02:00
registerHooks: function registerHooks() {
var _this = this;
/* Hook into Strophe's _connect_cb, so that we can send an IQ
* requesting the registration fields.
*/
2017-02-03 13:51:07 +01:00
var conn = _converse.connection;
var connect_cb = conn._connect_cb.bind(conn);
conn._connect_cb = function (req, callback, raw) {
2017-07-22 22:21:05 +02:00
if (!_this._registering) {
connect_cb(req, callback, raw);
} else {
2017-07-22 22:21:05 +02:00
if (_this.getRegistrationFields(req, callback, raw)) {
_this._registering = false;
}
}
2017-07-22 22:21:05 +02:00
};
},
2017-07-22 22:21:05 +02:00
getRegistrationFields: function getRegistrationFields(req, _callback, raw) {
/* Send an IQ stanza to the XMPP server asking for the
* registration fields.
* Parameters:
* (Strophe.Request) req - The current request
* (Function) callback
*/
2017-02-03 13:51:07 +01:00
var conn = _converse.connection;
conn.connected = true;
var body = conn._proto._reqToData(req);
2017-07-22 22:21:05 +02:00
if (!body) {
return;
}
if (conn._proto._connect_cb(body) === Strophe.Status.CONNFAIL) {
return false;
}
var register = body.getElementsByTagName("register");
var mechanisms = body.getElementsByTagName("mechanism");
if (register.length === 0 && mechanisms.length === 0) {
conn._proto._no_auth_received(_callback);
return false;
}
if (register.length === 0) {
2017-07-22 22:21:05 +02:00
conn._changeConnectStatus(Strophe.Status.REGIFAIL, __("Sorry, the given provider does not support in " + "band account registration. Please try with a " + "different provider."));
return true;
}
// Send an IQ stanza to get all required data fields
conn._addSysHandler(this.onRegistrationFields.bind(this), null, "iq", null, null);
2017-07-22 22:21:05 +02:00
conn.send($iq({ type: "get" }).c("query", { xmlns: Strophe.NS.REGISTER }).tree());
2017-04-04 17:26:06 +02:00
conn.connected = false;
return true;
},
2017-07-22 22:21:05 +02:00
onRegistrationFields: function onRegistrationFields(stanza) {
/* Handler for Registration Fields Request.
*
* Parameters:
* (XMLElement) elem - The query stanza.
*/
if (stanza.getElementsByTagName("query").length !== 1) {
2017-02-03 13:51:07 +01:00
_converse.connection._changeConnectStatus(Strophe.Status.REGIFAIL, "unknown");
return false;
}
this.setFields(stanza);
this.renderRegistrationForm(stanza);
return false;
},
2017-07-22 22:21:05 +02:00
reset: function reset(settings) {
var defaults = {
fields: {},
urls: [],
title: "",
instructions: "",
registered: false,
_registering: false,
domain: null,
form_type: null
};
_.extend(this, defaults);
if (settings) {
_.extend(this, _.pick(settings, _.keys(defaults)));
}
},
2017-07-22 22:21:05 +02:00
onProviderChosen: function onProviderChosen(ev) {
/* Callback method that gets called when the user has chosen an
* XMPP provider.
*
* Parameters:
* (Submit Event) ev - Form submission event.
*/
2017-07-22 22:21:05 +02:00
if (ev && ev.preventDefault) {
ev.preventDefault();
}
var $form = $(ev.target),
$domain_input = $form.find('input[name=domain]'),
domain = $domain_input.val();
if (!domain) {
$domain_input.addClass('error');
return;
}
2017-04-04 17:26:06 +02:00
$form.find('input[type=submit]').hide();
this.fetchRegistrationForm(domain, __('Cancel'));
},
2017-07-22 22:21:05 +02:00
fetchRegistrationForm: function fetchRegistrationForm(domain_name, cancel_label) {
2017-04-04 17:26:06 +02:00
/* This is called with a domain name based on which, it fetches a
* registration form from the requested domain.
*
* Parameters:
* (Domain name) domain_name - XMPP server domain
*/
this.renderRegistrationRequest(cancel_label);
this.reset({
2017-04-04 17:26:06 +02:00
domain: Strophe.getDomainFromJid(domain_name),
_registering: true
});
2017-02-03 13:51:07 +01:00
_converse.connection.connect(this.domain, "", this.onRegistering.bind(this));
return false;
},
2017-07-22 22:21:05 +02:00
renderRegistrationRequest: function renderRegistrationRequest(cancel_label) {
2017-04-04 17:26:06 +02:00
var form = this.el.querySelector('#converse-register');
2017-07-22 22:21:05 +02:00
var markup = tpl_registration_request({
'cancel': cancel_label,
'info_message': _converse.__('Requesting a registration form from the XMPP server')
});
form.appendChild(utils.createFragmentFromText(markup));
2017-04-04 17:26:06 +02:00
if (!_converse.registration_domain) {
var cancel_button = document.querySelector('button.button-cancel');
cancel_button.addEventListener('click', this.cancelRegistration.bind(this));
}
},
2017-07-22 22:21:05 +02:00
giveFeedback: function giveFeedback(message, klass) {
this.$('.reg-feedback').attr('class', 'reg-feedback').text(message);
if (klass) {
$('.reg-feedback').addClass(klass);
}
},
2017-07-22 22:21:05 +02:00
onRegistering: function onRegistering(status, error) {
var that = void 0;
2017-02-03 13:51:07 +01:00
_converse.log('onRegistering');
2017-07-22 22:21:05 +02:00
if (_.includes([Strophe.Status.DISCONNECTED, Strophe.Status.CONNFAIL, Strophe.Status.REGIFAIL, Strophe.Status.NOTACCEPTABLE, Strophe.Status.CONFLICT], status)) {
_converse.log("Problem during registration: Strophe.Status is: " + status, Strophe.LogLevel.ERROR);
this.cancelRegistration();
if (error) {
this.giveFeedback(error, 'error');
} else {
2017-07-22 22:21:05 +02:00
this.giveFeedback(__('Something went wrong while establishing a connection with "%1$s". Are you sure it exists?', this.domain), 'error');
}
} else if (status === Strophe.Status.REGISTERED) {
2017-02-03 13:51:07 +01:00
_converse.log("Registered successfully.");
_converse.connection.reset();
that = this;
this.$('form').hide(function () {
2017-06-23 20:25:33 +02:00
$(this).replaceWith(tpl_spinner);
if (that.fields.password && that.fields.username) {
// automatically log the user in
2017-07-22 22:21:05 +02:00
_converse.connection.connect(that.fields.username.toLowerCase() + '@' + that.domain.toLowerCase(), that.fields.password, _converse.onConnectStatusChanged);
_converse.chatboxviews.get('controlbox').switchTab({ 'target': that.$tabs.find('.current') });
2017-02-03 13:51:07 +01:00
_converse.giveFeedback(__('Now logging you in'));
} else {
2017-02-03 13:51:07 +01:00
_converse.chatboxviews.get('controlbox').renderLoginPanel();
_converse.giveFeedback(__('Registered successfully'));
}
that.reset();
});
}
},
2017-07-22 22:21:05 +02:00
renderRegistrationForm: function renderRegistrationForm(stanza) {
var _this2 = this;
/* Renders the registration form based on the XForm fields
* received from the XMPP server.
*
* Parameters:
* (XMLElement) stanza - The IQ stanza received from the XMPP server.
*/
2017-04-04 17:26:06 +02:00
this.model.set('registration_form_rendered', true);
var $form = this.$('form'),
2017-07-22 22:21:05 +02:00
$stanza = $(stanza);
var $fields = void 0,
$input = void 0;
$form.empty().append(tpl_registration_form({
'domain': this.domain,
'title': this.title,
'instructions': this.instructions
}));
if (this.form_type === 'xform') {
$fields = $stanza.find('field');
_.each($fields, function (field) {
2017-07-22 22:21:05 +02:00
$form.append(utils.xForm2webForm.bind(_this2, $(field), $stanza));
});
} else {
// Show fields
_.each(_.keys(this.fields), function (key) {
if (key === "username") {
$input = tpl_form_username({
2017-07-22 22:21:05 +02:00
domain: " @" + _this2.domain,
name: key,
type: "text",
label: key,
value: '',
required: 1
});
} else {
2017-07-22 22:21:05 +02:00
$form.append("<label>" + key + "</label>");
$input = $("<input placeholder=\"" + key + "\" name=\"" + key + "\"></input>");
if (key === 'password' || key === 'email') {
$input.attr('type', key);
}
}
$form.append($input);
2017-07-22 22:21:05 +02:00
});
// Show urls
_.each(this.urls, function (url) {
$form.append($('<a target="blank"></a>').attr('href', url).text(url));
2017-07-22 22:21:05 +02:00
});
}
if (this.fields) {
2017-07-22 22:21:05 +02:00
$form.append("<input type=\"submit\" class=\"pure-button button-primary\" value=\"" + __('Register') + "\"/>");
$form.on('submit', this.submitRegistrationForm.bind(this));
2017-07-22 22:21:05 +02:00
$form.append("<input type=\"button\" class=\"pure-button button-cancel\" value=\"" + __('Cancel') + "\"/>");
$form.find('input[type=button]').on('click', this.cancelRegistration.bind(this));
} else {
2017-07-22 22:21:05 +02:00
$form.append("<input type=\"button\" class=\"submit\" value=\"" + __('Return') + "\"/>");
$form.find('input[type=button]').on('click', this.cancelRegistration.bind(this));
}
2017-04-04 17:26:06 +02:00
if (_converse.registration_domain) {
$form.find('input[type=button]').hide();
}
},
2017-07-22 22:21:05 +02:00
reportErrors: function reportErrors(stanza) {
/* Report back to the user any error messages received from the
* XMPP server after attempted registration.
*
* Parameters:
* (XMLElement) stanza - The IQ stanza received from the
* XMPP server.
*/
2017-07-22 22:21:05 +02:00
var $form = this.$('form'),
$errmsgs = $(stanza).find('error text');
var $flash = $form.find('.form-errors');
if (!$flash.length) {
2017-07-22 22:21:05 +02:00
var flash = '<legend class="form-errors"></legend>';
if ($form.find('p.instructions').length) {
$form.find('p.instructions').append(flash);
} else {
$form.prepend(flash);
}
$flash = $form.find('.form-errors');
} else {
$flash.empty();
}
$errmsgs.each(function (idx, txt) {
$flash.append($('<p>').text($(txt).text()));
});
if (!$errmsgs.length) {
2017-07-22 22:21:05 +02:00
$flash.append($('<p>').text(__('The provider rejected your registration attempt. ' + 'Please check the values you entered for correctness.')));
}
$flash.show();
},
2017-07-22 22:21:05 +02:00
cancelRegistration: function cancelRegistration(ev) {
/* Handler, when the user cancels the registration form.
*/
2017-07-22 22:21:05 +02:00
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2017-02-03 13:51:07 +01:00
_converse.connection.reset();
2017-04-04 17:26:06 +02:00
this.model.set('registration_form_rendered', false);
this.render();
2017-04-04 17:26:06 +02:00
if (_converse.registration_domain) {
2017-07-22 22:21:05 +02:00
document.querySelector('button.button-cancel').onclick = _.bind(this.fetchRegistrationForm, this, _converse.registration_domain, __('Retry'));
2017-04-04 17:26:06 +02:00
}
},
2017-07-22 22:21:05 +02:00
submitRegistrationForm: function submitRegistrationForm(ev) {
/* Handler, when the user submits the registration form.
* Provides form error feedback or starts the registration
* process.
*
* Parameters:
* (Event) ev - the submit event.
*/
2017-07-22 22:21:05 +02:00
if (ev && ev.preventDefault) {
ev.preventDefault();
}
var has_empty_inputs = _.reduce(this.el.querySelectorAll('input.required'), function (result, input) {
if (input.value === '') {
input.classList.add('error');
return result + 1;
}
return result;
}, 0);
if (has_empty_inputs) {
return;
}
var $inputs = $(ev.target).find(':input:not([type=button]):not([type=submit])'),
2017-07-22 22:21:05 +02:00
iq = $iq({ type: "set" }).c("query", { xmlns: Strophe.NS.REGISTER });
if (this.form_type === 'xform') {
2017-07-22 22:21:05 +02:00
iq.c("x", { xmlns: Strophe.NS.XFORM, type: 'submit' });
$inputs.each(function () {
iq.cnode(utils.webForm2xForm(this)).up();
});
} else {
$inputs.each(function () {
var $input = $(this);
iq.c($input.attr('name'), {}, $input.val());
});
}
2017-04-04 17:26:06 +02:00
this.model.set('registration_form_rendered', false);
2017-02-03 13:51:07 +01:00
_converse.connection._addSysHandler(this._onRegisterIQ.bind(this), null, "iq", null, null);
_converse.connection.send(iq);
this.setFields(iq.tree());
},
2017-07-22 22:21:05 +02:00
setFields: function setFields(stanza) {
/* Stores the values that will be sent to the XMPP server
* during attempted registration.
*
* Parameters:
* (XMLElement) stanza - the IQ stanza that will be sent to the XMPP server.
*/
2017-07-22 22:21:05 +02:00
var $query = $(stanza).find('query');
if ($query.length > 0) {
2017-07-22 22:21:05 +02:00
var $xform = $query.find("x[xmlns=\"" + Strophe.NS.XFORM + "\"]");
if ($xform.length > 0) {
this._setFieldsFromXForm($xform);
} else {
this._setFieldsFromLegacy($query);
}
}
},
2017-07-22 22:21:05 +02:00
_setFieldsFromLegacy: function _setFieldsFromLegacy($query) {
var _this3 = this;
$query.children().each(function (idx, field) {
var $field = $(field);
if (field.tagName.toLowerCase() === 'instructions') {
2017-07-22 22:21:05 +02:00
_this3.instructions = Strophe.getText(field);
return;
} else if (field.tagName.toLowerCase() === 'x') {
if ($field.attr('xmlns') === 'jabber:x:oob') {
$field.find('url').each(function (idx, url) {
2017-07-22 22:21:05 +02:00
_this3.urls.push($(url).text());
});
}
return;
}
2017-07-22 22:21:05 +02:00
_this3.fields[field.tagName.toLowerCase()] = Strophe.getText(field);
});
this.form_type = 'legacy';
},
2017-07-22 22:21:05 +02:00
_setFieldsFromXForm: function _setFieldsFromXForm($xform) {
var _this4 = this;
this.title = $xform.find('title').text();
this.instructions = $xform.find('instructions').text();
$xform.find('field').each(function (idx, field) {
var _var = field.getAttribute('var');
if (_var) {
2017-07-22 22:21:05 +02:00
_this4.fields[_var.toLowerCase()] = $(field).children('value').text();
} else {
// TODO: other option seems to be type="fixed"
2017-07-05 11:59:55 +02:00
_converse.log("Found field we couldn't parse", Strophe.LogLevel.WARN);
}
2017-07-22 22:21:05 +02:00
});
this.form_type = 'xform';
},
2017-07-22 22:21:05 +02:00
_onRegisterIQ: function _onRegisterIQ(stanza) {
/* Callback method that gets called when a return IQ stanza
* is received from the XMPP server, after attempting to
* register a new user.
*
* Parameters:
* (XMLElement) stanza - The IQ stanza.
*/
var error = null,
query = stanza.getElementsByTagName("query");
if (query.length > 0) {
query = query[0];
}
if (stanza.getAttribute("type") === "error") {
2017-07-05 11:59:55 +02:00
_converse.log("Registration failed.", Strophe.LogLevel.ERROR);
error = stanza.getElementsByTagName("error");
if (error.length !== 1) {
2017-02-03 13:51:07 +01:00
_converse.connection._changeConnectStatus(Strophe.Status.REGIFAIL, "unknown");
return false;
}
error = error[0].firstChild.tagName.toLowerCase();
if (error === 'conflict') {
2017-02-03 13:51:07 +01:00
_converse.connection._changeConnectStatus(Strophe.Status.CONFLICT, error);
} else if (error === 'not-acceptable') {
2017-02-03 13:51:07 +01:00
_converse.connection._changeConnectStatus(Strophe.Status.NOTACCEPTABLE, error);
} else {
2017-02-03 13:51:07 +01:00
_converse.connection._changeConnectStatus(Strophe.Status.REGIFAIL, error);
}
this.reportErrors(stanza);
} else {
2017-02-03 13:51:07 +01:00
_converse.connection._changeConnectStatus(Strophe.Status.REGISTERED, null);
}
return false;
},
2017-07-22 22:21:05 +02:00
remove: function remove() {
this.$tabs.empty();
this.$el.parent().empty();
}
});
}
});
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-register.js.map;
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
2017-02-13 17:16:13 +01:00
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
// Licensed under the Mozilla Public License (MPLv2)
//
/*global define */
/* This is a Converse.js plugin which add support for application-level pings
* as specified in XEP-0199 XMPP Ping.
*/
(function (root, factory) {
define('converse-ping',["converse-core", "strophe.ping"], factory);
2017-07-22 22:21:05 +02:00
})(undefined, function (converse) {
"use strict";
// Strophe methods for building stanzas
2017-07-22 22:21:05 +02:00
var _converse$env = converse.env,
Strophe = _converse$env.Strophe,
_ = _converse$env._;
converse.plugins.add('converse-ping', {
initialize: function initialize() {
/* The initialize function gets called as soon as the plugin is
* loaded by converse.js's plugin machinery.
*/
2017-02-03 13:51:07 +01:00
var _converse = this._converse;
2017-07-22 22:21:05 +02:00
2017-07-05 11:59:55 +02:00
_converse.api.settings.update({
ping_interval: 180 //in seconds
});
2017-02-03 13:51:07 +01:00
_converse.ping = function (jid, success, error, timeout) {
// XXX: We could first check here if the server advertised that
// it supports PING.
// However, some servers don't advertise while still keeping the
// connection option due to pings.
//
2017-07-22 22:21:05 +02:00
// var feature = _converse.disco_entities[_converse.domain].features.findWhere({'var': Strophe.NS.PING});
2017-02-03 13:51:07 +01:00
_converse.lastStanzaDate = new Date();
if (_.isNil(jid)) {
2017-02-03 13:51:07 +01:00
jid = Strophe.getDomainFromJid(_converse.bare_jid);
}
2017-07-22 22:21:05 +02:00
if (_.isUndefined(timeout)) {
timeout = null;
}
if (_.isUndefined(success)) {
success = null;
}
if (_.isUndefined(error)) {
error = null;
}
2017-02-03 13:51:07 +01:00
if (_converse.connection) {
_converse.connection.ping.ping(jid, success, error, timeout);
return true;
}
return false;
};
2017-02-03 13:51:07 +01:00
_converse.pong = function (ping) {
_converse.lastStanzaDate = new Date();
_converse.connection.ping.pong(ping);
return true;
};
2017-02-03 13:51:07 +01:00
_converse.registerPongHandler = function () {
_converse.connection.disco.addFeature(Strophe.NS.PING);
_converse.connection.ping.addPingHandler(_converse.pong);
};
2017-02-03 13:51:07 +01:00
_converse.registerPingHandler = function () {
_converse.registerPongHandler();
if (_converse.ping_interval > 0) {
_converse.connection.addHandler(function () {
/* Handler on each stanza, saves the received date
* in order to ping only when needed.
*/
2017-02-03 13:51:07 +01:00
_converse.lastStanzaDate = new Date();
return true;
});
2017-02-03 13:51:07 +01:00
_converse.connection.addTimedHandler(1000, function () {
var now = new Date();
2017-02-03 13:51:07 +01:00
if (!_converse.lastStanzaDate) {
_converse.lastStanzaDate = now;
}
2017-07-22 22:21:05 +02:00
if ((now - _converse.lastStanzaDate) / 1000 > _converse.ping_interval) {
2017-02-03 13:51:07 +01:00
return _converse.ping();
}
return true;
});
}
};
2017-07-22 22:21:05 +02:00
var onConnected = function onConnected() {
// Wrapper so that we can spy on registerPingHandler in tests
2017-02-03 13:51:07 +01:00
_converse.registerPingHandler();
};
2017-02-03 13:51:07 +01:00
_converse.on('connected', onConnected);
_converse.on('reconnected', onConnected);
}
});
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-ping.js.map;
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
2017-02-13 17:16:13 +01:00
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
// Licensed under the Mozilla Public License (MPLv2)
//
/*global define */
(function (root, factory) {
define('converse-notification',["converse-core"], factory);
2017-07-22 22:21:05 +02:00
})(undefined, function (converse) {
"use strict";
2017-07-22 22:21:05 +02:00
var _converse$env = converse.env,
utils = _converse$env.utils,
Strophe = _converse$env.Strophe,
_ = _converse$env._;
2017-07-22 22:21:05 +02:00
converse.plugins.add('converse-notification', {
initialize: function initialize() {
/* The initialize function gets called as soon as the plugin is
* loaded by converse.js's plugin machinery.
*/
2017-02-03 13:51:07 +01:00
var _converse = this._converse;
// For translations
2017-07-22 22:21:05 +02:00
2017-02-03 13:51:07 +01:00
var __ = _converse.__;
var ___ = _converse.___;
2017-07-22 22:21:05 +02:00
2017-02-03 13:51:07 +01:00
_converse.supports_html5_notification = "Notification" in window;
2017-07-05 11:59:55 +02:00
_converse.api.settings.update({
2016-06-20 21:11:43 +02:00
notify_all_room_messages: false,
show_desktop_notifications: true,
show_chatstate_notifications: false,
chatstate_notification_blacklist: [],
// ^ a list of JIDs to ignore concerning chat state notifications
2017-03-05 10:45:18 +01:00
play_sounds: true,
sounds_path: '/sounds/',
2016-07-26 08:00:30 +02:00
notification_icon: '/logo/conversejs128.png'
});
_converse.isOnlyChatStateNotification = function (msg) {
return (
2017-07-22 22:21:05 +02:00
// See XEP-0085 Chat State Notification
_.isNull(msg.querySelector('body')) && (_.isNull(msg.querySelector(_converse.ACTIVE)) || _.isNull(msg.querySelector(_converse.COMPOSING)) || _.isNull(msg.querySelector(_converse.INACTIVE)) || _.isNull(msg.querySelector(_converse.PAUSED)) || _.isNull(msg.querySelector(_converse.GONE)))
);
};
_converse.shouldNotifyOfGroupMessage = function (message) {
/* Is this a group message worthy of notification?
*/
2017-07-22 22:21:05 +02:00
var notify_all = _converse.notify_all_room_messages;
var jid = message.getAttribute('from'),
resource = Strophe.getResourceFromJid(jid),
2016-06-20 21:11:43 +02:00
room_jid = Strophe.getBareJidFromJid(jid),
sender = resource && Strophe.unescapeNode(resource) || '';
if (sender === '' || message.querySelectorAll('delay').length > 0) {
return false;
}
2017-02-03 13:51:07 +01:00
var room = _converse.chatboxes.get(room_jid);
var body = message.querySelector('body');
if (_.isNull(body)) {
2016-06-20 21:11:43 +02:00
return false;
}
2017-07-22 22:21:05 +02:00
var mentioned = new RegExp("\\b" + room.get('nick') + "\\b").test(body.textContent);
notify_all = notify_all === true || _.isArray(notify_all) && _.includes(notify_all, room_jid);
if (sender === room.get('nick') || !notify_all && !mentioned) {
return false;
}
return true;
};
2017-02-03 13:51:07 +01:00
_converse.shouldNotifyOfMessage = function (message) {
/* Is this a message worthy of notification?
*/
2016-06-20 21:11:43 +02:00
if (utils.isOTRMessage(message)) {
return false;
}
var forwarded = message.querySelector('forwarded');
if (!_.isNull(forwarded)) {
return false;
} else if (message.getAttribute('type') === 'groupchat') {
return _converse.shouldNotifyOfGroupMessage(message);
} else if (utils.isHeadlineMessage(message)) {
// We want to show notifications for headline messages.
return true;
}
2017-07-22 22:21:05 +02:00
var is_me = Strophe.getBareJidFromJid(message.getAttribute('from')) === _converse.bare_jid;
return !_converse.isOnlyChatStateNotification(message) && !is_me;
};
_converse.playSoundNotification = function () {
/* Plays a sound to notify that a new message was recieved.
*/
// XXX Eventually this can be refactored to use Notification's sound
// feature, but no browser currently supports it.
// https://developer.mozilla.org/en-US/docs/Web/API/notification/sound
2017-07-22 22:21:05 +02:00
var audio = void 0;
2017-03-05 10:45:18 +01:00
if (_converse.play_sounds && !_.isUndefined(window.Audio)) {
2017-07-22 22:21:05 +02:00
audio = new Audio(_converse.sounds_path + "msg_received.ogg");
if (audio.canPlayType('/audio/ogg')) {
audio.play();
} else {
2017-07-22 22:21:05 +02:00
audio = new Audio(_converse.sounds_path + "msg_received.mp3");
audio.play();
}
}
};
2017-02-03 13:51:07 +01:00
_converse.areDesktopNotificationsEnabled = function (ignore_hidden) {
2017-07-22 22:21:05 +02:00
var enabled = _converse.supports_html5_notification && _converse.show_desktop_notifications && Notification.permission === "granted";
2016-06-20 21:11:43 +02:00
if (ignore_hidden) {
return enabled;
} else {
2017-02-03 13:51:07 +01:00
return enabled && _converse.windowState === 'hidden';
}
};
_converse.showMessageNotification = function (message) {
/* Shows an HTML5 Notification to indicate that a new chat
* message was received.
*/
2017-07-22 22:21:05 +02:00
var title = void 0,
roster_item = void 0;
var full_from_jid = message.getAttribute('from'),
2017-04-23 19:02:44 +02:00
from_jid = Strophe.getBareJidFromJid(full_from_jid);
if (message.getAttribute('type') === 'headline') {
if (!_.includes(from_jid, '@') || _converse.allow_non_roster_messaging) {
title = __(___("Notification from %1$s"), from_jid);
} else {
return;
}
} else if (!_.includes(from_jid, '@')) {
// XXX: workaround for Prosody which doesn't give type "headline"
title = __(___("Notification from %1$s"), from_jid);
} else if (message.getAttribute('type') === 'groupchat') {
2017-04-23 19:02:44 +02:00
title = __(___("%1$s says"), Strophe.getResourceFromJid(full_from_jid));
} else {
if (_.isUndefined(_converse.roster)) {
2017-07-22 22:21:05 +02:00
_converse.log("Could not send notification, because roster is undefined", Strophe.LogLevel.ERROR);
return;
}
roster_item = _converse.roster.get(from_jid);
if (!_.isUndefined(roster_item)) {
title = __(___("%1$s says"), roster_item.get('fullname'));
2016-05-03 17:37:10 +02:00
} else {
if (_converse.allow_non_roster_messaging) {
title = __(___("%1$s says"), from_jid);
} else {
2016-05-03 17:37:10 +02:00
return;
}
}
}
var n = new Notification(title, {
2017-07-22 22:21:05 +02:00
body: message.querySelector('body').textContent,
lang: _converse.locale,
icon: _converse.notification_icon
});
setTimeout(n.close.bind(n), 5000);
};
2017-02-03 13:51:07 +01:00
_converse.showChatStateNotification = function (contact) {
/* Creates an HTML5 Notification to inform of a change in a
* contact's chat state.
*/
2017-02-03 13:51:07 +01:00
if (_.includes(_converse.chatstate_notification_blacklist, contact.jid)) {
// Don't notify if the user is being ignored.
return;
}
2017-07-22 22:21:05 +02:00
var chat_state = contact.chat_status;
var message = null;
if (chat_state === 'offline') {
message = __('has gone offline');
} else if (chat_state === 'away') {
message = __('has gone away');
2017-07-22 22:21:05 +02:00
} else if (chat_state === 'dnd') {
message = __('is busy');
} else if (chat_state === 'online') {
message = __('has come online');
}
if (message === null) {
return;
}
var n = new Notification(contact.fullname, {
2017-07-22 22:21:05 +02:00
body: message,
lang: _converse.locale,
icon: _converse.notification_icon
});
setTimeout(n.close.bind(n), 5000);
};
2017-02-03 13:51:07 +01:00
_converse.showContactRequestNotification = function (contact) {
var n = new Notification(contact.fullname, {
2017-07-22 22:21:05 +02:00
body: __('wants to be your contact'),
lang: _converse.locale,
icon: _converse.notification_icon
});
setTimeout(n.close.bind(n), 5000);
};
2017-02-03 13:51:07 +01:00
_converse.showFeedbackNotification = function (data) {
2016-09-16 14:35:02 +02:00
if (data.klass === 'error' || data.klass === 'warn') {
var n = new Notification(data.subject, {
2017-07-22 22:21:05 +02:00
body: data.message,
lang: _converse.locale,
icon: _converse.notification_icon
});
2016-09-16 14:35:02 +02:00
setTimeout(n.close.bind(n), 5000);
}
};
2016-03-16 12:49:35 +01:00
2017-02-03 13:51:07 +01:00
_converse.handleChatStateNotification = function (contact) {
/* Event handler for on('contactStatusChanged').
* Will show an HTML5 notification to indicate that the chat
* status has changed.
*/
2017-07-22 22:21:05 +02:00
if (_converse.areDesktopNotificationsEnabled() && _converse.show_chatstate_notifications) {
2017-02-03 13:51:07 +01:00
_converse.showChatStateNotification(contact);
}
};
2017-06-23 20:25:33 +02:00
_converse.handleMessageNotification = function (data) {
/* Event handler for the on('message') event. Will call methods
* to play sounds and show HTML5 notifications.
*/
2017-06-23 20:25:33 +02:00
var message = data.stanza;
2017-02-03 13:51:07 +01:00
if (!_converse.shouldNotifyOfMessage(message)) {
return false;
}
_converse.playSoundNotification();
2017-02-03 13:51:07 +01:00
if (_converse.areDesktopNotificationsEnabled()) {
_converse.showMessageNotification(message);
}
};
2017-02-03 13:51:07 +01:00
_converse.handleContactRequestNotification = function (contact) {
if (_converse.areDesktopNotificationsEnabled(true)) {
_converse.showContactRequestNotification(contact);
}
};
2017-02-03 13:51:07 +01:00
_converse.handleFeedback = function (data) {
if (_converse.areDesktopNotificationsEnabled(true)) {
_converse.showFeedbackNotification(data);
}
};
2017-02-03 13:51:07 +01:00
_converse.requestPermission = function () {
2017-07-22 22:21:05 +02:00
if (_converse.supports_html5_notification && !_.includes(['denied', 'granted'], Notification.permission)) {
// Ask user to enable HTML5 notifications
Notification.requestPermission();
}
};
2017-02-03 13:51:07 +01:00
_converse.on('pluginsInitialized', function () {
2016-06-20 21:11:43 +02:00
// We only register event handlers after all plugins are
// registered, because other plugins might override some of our
// handlers.
2017-07-22 22:21:05 +02:00
_converse.on('contactRequest', _converse.handleContactRequestNotification);
_converse.on('contactStatusChanged', _converse.handleChatStateNotification);
_converse.on('message', _converse.handleMessageNotification);
2017-02-03 13:51:07 +01:00
_converse.on('feedback', _converse.handleFeedback);
_converse.on('connected', _converse.requestPermission);
2016-06-20 21:11:43 +02:00
});
}
});
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-notification.js.map;
2016-11-07 15:43:48 +01:00
define('tpl!chatbox_minimize', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape;
with (obj) {
__p += '<a class="chatbox-btn toggle-chatbox-button icon-minus" title="' +
2017-02-13 17:16:13 +01:00
__e(info_minimize) +
2016-11-07 15:43:48 +01:00
'"></a>\n';
}
return __p
};});
define('tpl!toggle_chats', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
__p +=
2017-02-13 17:16:13 +01:00
__e(Minimized) +
' <span id="minimized-count">(' +
2017-02-13 17:16:13 +01:00
__e(num_minimized) +
2017-06-23 20:25:33 +02:00
')</span>\n<span class="unread-message-count\n ';
if (!num_unread) { ;
2017-06-23 20:25:33 +02:00
__p += ' unread-message-count-hidden ';
} ;
__p += '\n href="#">' +
2017-02-13 17:16:13 +01:00
__e(num_unread) +
2016-11-07 15:43:48 +01:00
'</span>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
define('tpl!trimmed_chat', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
2017-02-13 17:16:13 +01:00
var __t, __p = '', __e = _.escape, __j = Array.prototype.join;
function print() { __p += __j.call(arguments, '') }
with (obj) {
2017-06-23 20:25:33 +02:00
__p += '<a class="chatbox-btn close-chatbox-button icon-close"></a>\n<a class="chat-head-message-count\n ';
if (!num_unread) { ;
2017-06-23 20:25:33 +02:00
__p += ' chat-head-message-count-hidden ';
} ;
2017-06-23 20:25:33 +02:00
__p += '"\n href="#">' +
2017-02-13 17:16:13 +01:00
__e(num_unread) +
'</a>\n<a href="#" class="restore-chat" title="' +
2017-02-13 17:16:13 +01:00
__e(tooltip) +
'">\n ' +
2017-02-13 17:16:13 +01:00
__e( title ) +
2016-11-07 15:43:48 +01:00
'\n</a>\n';
2016-11-07 15:43:48 +01:00
}
return __p
};});
2016-11-07 15:43:48 +01:00
define('tpl!chats_panel', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '';
with (obj) {
__p += '<a id="toggle-minimized-chats" href="#"></a>\n<div class="flyout minimized-chats-flyout"></div>\n';
2016-11-30 17:27:20 +01:00
}
return __p
};});
2016-11-30 17:27:20 +01:00
2017-07-22 22:21:05 +02:00
2016-03-07 18:54:07 +01:00
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
2017-02-13 17:16:13 +01:00
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
2016-03-07 18:54:07 +01:00
// Licensed under the Mozilla Public License (MPLv2)
//
2017-04-23 19:02:44 +02:00
/*global define, window */
2016-03-07 18:54:07 +01:00
(function (root, factory) {
2017-07-22 22:21:05 +02:00
define('converse-minimize',["jquery.noconflict", "converse-core", "tpl!chatbox_minimize", "tpl!toggle_chats", "tpl!trimmed_chat", "tpl!chats_panel", "converse-controlbox", "converse-chatview", "converse-muc"], factory);
})(undefined, function ($, converse, tpl_chatbox_minimize, tpl_toggle_chats, tpl_trimmed_chat, tpl_chats_panel) {
"use strict";
2017-07-22 22:21:05 +02:00
var _converse$env = converse.env,
_ = _converse$env._,
utils = _converse$env.utils,
Backbone = _converse$env.Backbone,
b64_sha1 = _converse$env.b64_sha1,
moment = _converse$env.moment;
2016-03-07 18:54:07 +01:00
2017-02-03 13:51:07 +01:00
converse.plugins.add('converse-minimize', {
overrides: {
// Overrides mentioned here will be picked up by converse.js's
// plugin architecture they will replace existing methods on the
// relevant objects or classes.
//
// New functions which don't exist yet can also be added.
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
initChatBoxes: function initChatBoxes() {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2016-11-30 17:27:20 +01:00
var result = this.__super__.initChatBoxes.apply(this, arguments);
2017-02-03 13:51:07 +01:00
_converse.minimized_chats = new _converse.MinimizedChats({
model: _converse.chatboxes
});
2016-11-30 17:27:20 +01:00
return result;
},
2017-07-22 22:21:05 +02:00
registerGlobalEventHandlers: function registerGlobalEventHandlers() {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
$(window).on("resize", _.debounce(function (ev) {
2017-02-03 13:51:07 +01:00
if (_converse.connection.connected) {
_converse.chatboxviews.trimChats();
2016-03-07 18:54:07 +01:00
}
}, 200));
2016-09-16 14:35:02 +02:00
return this.__super__.registerGlobalEventHandlers.apply(this, arguments);
},
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
ChatBox: {
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-09-16 14:35:02 +02:00
this.__super__.initialize.apply(this, arguments);
if (this.get('id') === 'controlbox') {
return;
}
this.save({
'minimized': this.get('minimized') || false,
2017-07-22 22:21:05 +02:00
'time_minimized': this.get('time_minimized') || moment()
});
},
2017-07-22 22:21:05 +02:00
maximize: function maximize() {
2017-06-23 20:25:33 +02:00
utils.safeSave(this, {
'minimized': false,
'time_opened': moment().valueOf()
});
},
2017-07-22 22:21:05 +02:00
minimize: function minimize() {
2017-06-23 20:25:33 +02:00
utils.safeSave(this, {
'minimized': true,
'time_minimized': moment().format()
});
2017-07-22 22:21:05 +02:00
}
},
2016-03-07 18:54:07 +01:00
ChatBoxView: {
events: {
2017-07-22 22:21:05 +02:00
'click .toggle-chatbox-button': 'minimize'
},
2016-03-07 18:54:07 +01:00
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
this.model.on('change:minimized', this.onMinimizedChanged, this);
2016-09-16 14:35:02 +02:00
return this.__super__.initialize.apply(this, arguments);
},
2017-07-22 22:21:05 +02:00
_show: function _show() {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
if (!this.model.get('minimized')) {
2017-04-23 19:02:44 +02:00
this.__super__._show.apply(this, arguments);
2017-02-03 13:51:07 +01:00
_converse.chatboxviews.trimChats(this);
2017-04-23 19:02:44 +02:00
} else {
this.minimize();
}
},
2017-07-22 22:21:05 +02:00
isNewMessageHidden: function isNewMessageHidden() {
return this.model.get('minimized') || this.__super__.isNewMessageHidden.apply(this, arguments);
2017-06-23 20:25:33 +02:00
},
2017-07-22 22:21:05 +02:00
shouldShowOnTextMessage: function shouldShowOnTextMessage() {
return !this.model.get('minimized') && this.__super__.shouldShowOnTextMessage.apply(this, arguments);
},
2017-07-22 22:21:05 +02:00
setChatBoxHeight: function setChatBoxHeight(height) {
if (!this.model.get('minimized')) {
2016-09-16 14:35:02 +02:00
return this.__super__.setChatBoxHeight.apply(this, arguments);
}
},
2017-07-22 22:21:05 +02:00
setChatBoxWidth: function setChatBoxWidth(width) {
if (!this.model.get('minimized')) {
2016-09-16 14:35:02 +02:00
return this.__super__.setChatBoxWidth.apply(this, arguments);
}
},
2017-07-22 22:21:05 +02:00
onMinimizedChanged: function onMinimizedChanged(item) {
if (item.get('minimized')) {
this.minimize();
} else {
this.maximize();
}
},
2017-07-22 22:21:05 +02:00
maximize: function maximize() {
// Restores a minimized chat box
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2017-02-03 13:51:07 +01:00
this.$el.insertAfter(_converse.chatboxviews.get("controlbox").$el);
2017-06-23 20:25:33 +02:00
if (!this.model.isScrolledUp()) {
this.model.clearUnreadMsgCounter();
}
2016-11-30 17:27:20 +01:00
this.show();
2017-02-03 13:51:07 +01:00
_converse.emit('chatBoxMaximized', this);
return this;
},
2017-07-22 22:21:05 +02:00
minimize: function minimize(ev) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
if (ev && ev.preventDefault) {
ev.preventDefault();
}
// save the scroll position to restore it on maximize
2017-06-23 20:25:33 +02:00
if (this.model.collection && this.model.collection.browserStorage) {
2017-07-22 22:21:05 +02:00
this.model.save({ 'scroll': this.$content.scrollTop() });
2017-06-23 20:25:33 +02:00
} else {
2017-07-22 22:21:05 +02:00
this.model.set({ 'scroll': this.$content.scrollTop() });
2017-06-23 20:25:33 +02:00
}
2017-02-03 13:51:07 +01:00
this.setChatState(_converse.INACTIVE).model.minimize();
2016-11-30 17:27:20 +01:00
this.hide();
2017-02-03 13:51:07 +01:00
_converse.emit('chatBoxMinimized', this);
2017-07-22 22:21:05 +02:00
}
},
ChatRoomView: {
events: {
2017-07-22 22:21:05 +02:00
'click .toggle-chatbox-button': 'minimize'
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
this.model.on('change:minimized', function (item) {
if (item.get('minimized')) {
this.hide();
} else {
this.maximize();
}
}, this);
2016-09-16 14:35:02 +02:00
var result = this.__super__.initialize.apply(this, arguments);
if (this.model.get('minimized')) {
this.hide();
}
return result;
2016-12-13 20:46:07 +01:00
},
2017-07-22 22:21:05 +02:00
generateHeadingHTML: function generateHeadingHTML() {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse,
__ = _converse.__;
2017-07-22 22:21:05 +02:00
2016-12-13 20:46:07 +01:00
var html = this.__super__.generateHeadingHTML.apply(this, arguments);
var div = document.createElement('div');
div.innerHTML = html;
2017-07-22 22:21:05 +02:00
var el = tpl_chatbox_minimize({ info_minimize: __('Minimize this chat box') });
2016-12-13 20:46:07 +01:00
var button = div.querySelector('.close-chatbox-button');
button.insertAdjacentHTML('afterend', el);
return div.innerHTML;
}
},
ChatBoxes: {
2017-07-22 22:21:05 +02:00
chatBoxMayBeShown: function chatBoxMayBeShown(chatbox) {
return this.__super__.chatBoxMayBeShown.apply(this, arguments) && !chatbox.get('minimized');
}
},
2016-03-16 12:49:35 +01:00
ChatBoxViews: {
2017-07-22 22:21:05 +02:00
showChat: function showChat(attrs) {
/* Find the chat box and show it. If it doesn't exist, create it.
*/
2016-09-16 14:35:02 +02:00
var chatbox = this.__super__.showChat.apply(this, arguments);
2016-11-30 17:27:20 +01:00
var maximize = _.isUndefined(attrs.maximize) ? true : attrs.maximize;
if (chatbox.get('minimized') && maximize) {
chatbox.maximize();
}
return chatbox;
},
2017-07-22 22:21:05 +02:00
getChatBoxWidth: function getChatBoxWidth(view) {
if (!view.model.get('minimized') && view.$el.is(':visible')) {
return view.$el.outerWidth(true);
}
return 0;
},
2017-07-22 22:21:05 +02:00
getShownChats: function getShownChats() {
return this.filter(function (view) {
2016-06-20 21:11:43 +02:00
return (
2017-07-22 22:21:05 +02:00
// The controlbox can take a while to close,
// so we need to check its state. That's why we checked
// the 'closed' state.
!view.model.get('minimized') && !view.model.get('closed') && view.$el.is(':visible')
2016-06-20 21:11:43 +02:00
);
});
},
2017-07-22 22:21:05 +02:00
trimChats: function trimChats(newchat) {
var _this = this;
/* This method is called when a newly created chat box will
* be shown.
*
* It checks whether there is enough space on the page to show
* another chat box. Otherwise it minimizes the oldest chat box
* to create space.
*/
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
var shown_chats = this.getShownChats();
2017-02-03 13:51:07 +01:00
if (_converse.no_trimming || shown_chats.length <= 1) {
return;
}
if (this.getChatBoxWidth(shown_chats[0]) === $('body').outerWidth(true)) {
// If the chats shown are the same width as the body,
// then we're in responsive mode and the chats are
// fullscreen. In this case we don't trim.
return;
}
2017-07-22 22:21:05 +02:00
var $minimized = _converse.minimized_chats.$el,
minimized_width = _.includes(this.model.pluck('minimized'), true) ? $minimized.outerWidth(true) : 0,
new_id = newchat ? newchat.model.get('id') : null;
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
var boxes_width = _.reduce(this.xget(new_id), function (memo, view) {
return memo + _this.getChatBoxWidth(view);
}, newchat ? newchat.$el.outerWidth(true) : 0);
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
if (minimized_width + boxes_width > $('body').outerWidth(true)) {
var oldest_chat = this.getOldestMaximizedChat([new_id]);
if (oldest_chat) {
// We hide the chat immediately, because waiting
// for the event to fire (and letting the
// ChatBoxView hide it then) causes race
// conditions.
2017-07-22 22:21:05 +02:00
var view = this.get(oldest_chat.get('id'));
if (view) {
2016-11-30 17:27:20 +01:00
view.hide();
}
oldest_chat.minimize();
}
}
},
2017-07-22 22:21:05 +02:00
getOldestMaximizedChat: function getOldestMaximizedChat(exclude_ids) {
// Get oldest view (if its id is not excluded)
exclude_ids.push('controlbox');
var i = 0;
var model = this.model.sort().at(i);
2017-07-22 22:21:05 +02:00
while (_.includes(exclude_ids, model.get('id')) || model.get('minimized') === true) {
i++;
model = this.model.at(i);
if (!model) {
return null;
}
}
return model;
}
}
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
/* The initialize function gets called as soon as the plugin is
2017-02-03 13:51:07 +01:00
* loaded by Converse.js's plugin machinery.
*/
2017-02-03 13:51:07 +01:00
var _converse = this._converse,
__ = _converse.__;
// Add new HTML templates.
_converse.templates.chatbox_minimize = tpl_chatbox_minimize;
_converse.templates.toggle_chats = tpl_toggle_chats;
_converse.templates.trimmed_chat = tpl_trimmed_chat;
_converse.templates.chats_panel = tpl_chats_panel;
2017-07-05 11:59:55 +02:00
_converse.api.settings.update({
2017-07-22 22:21:05 +02:00
no_trimming: false // Set to true for phantomjs tests (where browser apparently has no width)
});
2017-02-03 13:51:07 +01:00
_converse.MinimizedChatBoxView = Backbone.View.extend({
tagName: 'div',
className: 'chat-head',
events: {
'click .close-chatbox-button': 'close',
'click .restore-chat': 'restore'
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2017-06-23 20:25:33 +02:00
this.model.on('change:num_unread', this.render, this);
},
2017-07-22 22:21:05 +02:00
render: function render() {
var data = _.extend(this.model.toJSON(), { 'tooltip': __('Click to restore this chat') });
if (this.model.get('type') === 'chatroom') {
data.title = this.model.get('name');
this.$el.addClass('chat-head-chatroom');
} else {
data.title = this.model.get('fullname');
this.$el.addClass('chat-head-chatbox');
}
return this.$el.html(tpl_trimmed_chat(data));
},
2017-07-22 22:21:05 +02:00
close: function close(ev) {
if (ev && ev.preventDefault) {
ev.preventDefault();
}
this.remove();
2017-02-03 13:51:07 +01:00
var view = _converse.chatboxviews.get(this.model.get('id'));
if (view) {
// This will call model.destroy(), removing it from the
// collection and will also emit 'chatBoxClosed'
view.close();
} else {
this.model.destroy();
2017-02-03 13:51:07 +01:00
_converse.emit('chatBoxClosed', this);
}
return this;
},
2017-07-22 22:21:05 +02:00
restore: _.debounce(function (ev) {
2017-07-22 22:21:05 +02:00
if (ev && ev.preventDefault) {
ev.preventDefault();
}
2017-06-23 20:25:33 +02:00
this.model.off('change:num_unread', null, this);
this.remove();
this.model.maximize();
2017-07-22 22:21:05 +02:00
}, 200, { 'leading': true })
});
2017-02-03 13:51:07 +01:00
_converse.MinimizedChats = Backbone.Overview.extend({
2016-11-30 17:27:20 +01:00
tagName: 'div',
id: "minimized-chats",
className: 'hidden',
events: {
"click #toggle-minimized-chats": "toggle"
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-11-30 17:27:20 +01:00
this.render();
this.initToggle();
this.model.on("add", this.onChanged, this);
this.model.on("destroy", this.removeChat, this);
this.model.on("change:minimized", this.onChanged, this);
this.model.on('change:num_unread', this.updateUnreadMessagesCounter, this);
},
2017-07-22 22:21:05 +02:00
tearDown: function tearDown() {
this.model.off("add", this.onChanged);
this.model.off("destroy", this.removeChat);
this.model.off("change:minimized", this.onChanged);
this.model.off('change:num_unread', this.updateUnreadMessagesCounter);
return this;
},
2017-07-22 22:21:05 +02:00
initToggle: function initToggle() {
2017-02-03 13:51:07 +01:00
this.toggleview = new _converse.MinimizedChatsToggleView({
model: new _converse.MinimizedChatsToggle()
});
2017-07-22 22:21:05 +02:00
var id = b64_sha1("converse.minchatstoggle" + _converse.bare_jid);
this.toggleview.model.id = id; // Appears to be necessary for backbone.browserStorage
2017-02-03 13:51:07 +01:00
this.toggleview.model.browserStorage = new Backbone.BrowserStorage[_converse.storage](id);
this.toggleview.model.fetch();
},
2017-07-22 22:21:05 +02:00
render: function render() {
2016-11-30 17:27:20 +01:00
if (!this.el.parentElement) {
this.el.innerHTML = tpl_chats_panel();
2017-02-03 13:51:07 +01:00
_converse.chatboxviews.el.appendChild(this.el);
2016-11-30 17:27:20 +01:00
}
if (this.keys().length === 0) {
2016-11-30 17:27:20 +01:00
this.el.classList.add('hidden');
2017-02-03 13:51:07 +01:00
_converse.chatboxviews.trimChats.bind(_converse.chatboxviews);
2016-11-30 17:27:20 +01:00
} else if (this.keys().length > 0 && !this.$el.is(':visible')) {
this.el.classList.remove('hidden');
2017-02-03 13:51:07 +01:00
_converse.chatboxviews.trimChats();
}
return this.$el;
},
2017-07-22 22:21:05 +02:00
toggle: function toggle(ev) {
if (ev && ev.preventDefault) {
ev.preventDefault();
}
this.toggleview.model.save({ 'collapsed': !this.toggleview.model.get('collapsed') });
this.$('.minimized-chats-flyout').toggle();
},
2017-07-22 22:21:05 +02:00
onChanged: function onChanged(item) {
if (item.get('id') === 'controlbox') {
// The ControlBox has it's own minimize toggle
2016-03-16 12:49:35 +01:00
return;
}
if (item.get('minimized')) {
this.addChat(item);
} else if (this.get(item.get('id'))) {
this.removeChat(item);
}
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
addChat: function addChat(item) {
var existing = this.get(item.get('id'));
if (existing && existing.$el.parent().length !== 0) {
return;
}
2017-07-22 22:21:05 +02:00
var view = new _converse.MinimizedChatBoxView({ model: item });
this.$('.minimized-chats-flyout').append(view.render());
this.add(item.get('id'), view);
2017-07-22 22:21:05 +02:00
this.toggleview.model.set({ 'num_minimized': this.keys().length });
this.render();
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
removeChat: function removeChat(item) {
this.remove(item.get('id'));
2017-07-22 22:21:05 +02:00
this.toggleview.model.set({ 'num_minimized': this.keys().length });
this.render();
2016-03-16 12:49:35 +01:00
},
2017-07-22 22:21:05 +02:00
updateUnreadMessagesCounter: function updateUnreadMessagesCounter() {
var ls = this.model.pluck('num_unread');
var count = 0,
i = void 0;
for (i = 0; i < ls.length; i++) {
count += ls[i];
}
this.toggleview.model.save({ 'num_unread': count });
this.render();
}
});
2017-02-03 13:51:07 +01:00
_converse.MinimizedChatsToggle = Backbone.Model.extend({
2017-06-23 20:25:33 +02:00
defaults: {
'collapsed': false,
'num_minimized': 0,
2017-07-22 22:21:05 +02:00
'num_unread': 0
}
});
2017-02-03 13:51:07 +01:00
_converse.MinimizedChatsToggleView = Backbone.View.extend({
el: '#toggle-minimized-chats',
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
this.model.on('change:num_minimized', this.render, this);
this.model.on('change:num_unread', this.render, this);
this.$flyout = this.$el.siblings('.minimized-chats-flyout');
},
2017-07-22 22:21:05 +02:00
render: function render() {
this.$el.html(tpl_toggle_chats(_.extend(this.model.toJSON(), {
'Minimized': __('Minimized')
})));
if (this.model.get('collapsed')) {
this.$flyout.hide();
} else {
this.$flyout.show();
2016-03-16 12:49:35 +01:00
}
return this.$el;
}
});
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
var renderMinimizeButton = function renderMinimizeButton(view) {
// Inserts a "minimize" button in the chatview's header
var $el = view.$el.find('.toggle-chatbox-button');
2017-07-22 22:21:05 +02:00
var $new_el = tpl_chatbox_minimize({ info_minimize: __('Minimize this chat box') });
if ($el.length) {
$el.replaceWith($new_el);
} else {
view.$el.find('.close-chatbox-button').after($new_el);
}
};
2017-02-03 13:51:07 +01:00
_converse.on('chatBoxOpened', renderMinimizeButton);
2017-02-03 13:51:07 +01:00
_converse.on('controlBoxOpened', function (chatbox) {
// Wrapped in anon method because at scan time, chatboxviews
// attr not set yet.
2017-02-03 13:51:07 +01:00
if (_converse.connection.connected) {
_converse.chatboxviews.trimChats(chatbox);
}
});
2017-02-01 12:05:32 +01:00
2017-07-22 22:21:05 +02:00
var logOut = function logOut() {
2017-02-03 13:51:07 +01:00
_converse.minimized_chats.remove();
2017-02-01 12:05:32 +01:00
};
2017-02-03 13:51:07 +01:00
_converse.on('logout', logOut);
}
});
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-minimize.js.map;
2016-12-13 20:46:07 +01:00
define('tpl!dragresize', ['lodash'], function(_) {return function(obj) {
obj || (obj = {});
var __t, __p = '';
with (obj) {
__p += '<div class="dragresize dragresize-top"></div>\n<div class="dragresize dragresize-topleft"></div>\n<div class="dragresize dragresize-left"></div>\n';
2016-12-13 20:46:07 +01:00
}
return __p
};});
2016-12-13 20:46:07 +01:00
2017-07-22 22:21:05 +02:00
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
2017-02-13 17:16:13 +01:00
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
// Licensed under the Mozilla Public License (MPLv2)
//
/*global define, window */
2016-03-16 12:49:35 +01:00
(function (root, factory) {
2017-07-22 22:21:05 +02:00
define('converse-dragresize',["jquery.noconflict", "converse-core", "tpl!dragresize", "converse-chatview", "converse-muc", // XXX: would like to remove this
"converse-controlbox"], factory);
})(undefined, function ($, converse, tpl_dragresize) {
"use strict";
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
var _ = converse.env._;
function renderDragResizeHandles(_converse, view) {
2017-06-23 20:25:33 +02:00
var flyout = view.el.querySelector('.box-flyout');
var div = document.createElement('div');
div.innerHTML = tpl_dragresize();
2017-07-22 22:21:05 +02:00
flyout.insertBefore(div, flyout.firstChild);
2017-06-23 20:25:33 +02:00
}
2017-02-03 13:51:07 +01:00
converse.plugins.add('converse-dragresize', {
2017-04-04 17:26:06 +02:00
/* Optional dependencies are other plugins which might be
* overridden or relied upon, and therefore need to be loaded before
* this plugin. They are called "optional" because they might not be
* available, in which case any overrides applicable to them will be
* ignored.
*
* It's possible however to make optional dependencies non-optional.
* If the setting "strict_plugin_dependencies" is set to true,
* an error will be raised if the plugin is not found.
*
* NB: These plugins need to have already been loaded via require.js.
*/
optional_dependencies: ["converse-headline"],
2016-03-16 12:49:35 +01:00
overrides: {
// Overrides mentioned here will be picked up by converse.js's
// plugin architecture they will replace existing methods on the
// relevant objects or classes.
//
// New functions which don't exist yet can also be added.
2016-06-20 21:11:43 +02:00
2017-07-22 22:21:05 +02:00
registerGlobalEventHandlers: function registerGlobalEventHandlers() {
2017-02-03 13:51:07 +01:00
var that = this;
2017-07-22 22:21:05 +02:00
$(document).on('mousemove', function (ev) {
2017-07-22 22:21:05 +02:00
if (!that.resizing || !that.allow_dragresize) {
return true;
}
ev.preventDefault();
2017-02-03 13:51:07 +01:00
that.resizing.chatbox.resizeChatBox(ev);
});
2016-03-16 12:49:35 +01:00
$(document).on('mouseup', function (ev) {
2017-07-22 22:21:05 +02:00
if (!that.resizing || !that.allow_dragresize) {
return true;
}
ev.preventDefault();
2017-07-22 22:21:05 +02:00
var height = that.applyDragResistance(that.resizing.chatbox.height, that.resizing.chatbox.model.get('default_height'));
var width = that.applyDragResistance(that.resizing.chatbox.width, that.resizing.chatbox.model.get('default_width'));
2017-02-03 13:51:07 +01:00
if (that.connection.connected) {
2017-07-22 22:21:05 +02:00
that.resizing.chatbox.model.save({ 'height': height });
that.resizing.chatbox.model.save({ 'width': width });
} else {
2017-07-22 22:21:05 +02:00
that.resizing.chatbox.model.set({ 'height': height });
that.resizing.chatbox.model.set({ 'width': width });
}
2017-02-03 13:51:07 +01:00
that.resizing = null;
});
2016-03-16 12:49:35 +01:00
2016-09-16 14:35:02 +02:00
return this.__super__.registerGlobalEventHandlers.apply(this, arguments);
},
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
ChatBox: {
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2016-09-16 14:35:02 +02:00
var result = this.__super__.initialize.apply(this, arguments),
2017-07-22 22:21:05 +02:00
height = this.get('height'),
width = this.get('width'),
save = this.get('id') === 'controlbox' ? this.set.bind(this) : this.save.bind(this);
save({
2017-02-03 13:51:07 +01:00
'height': _converse.applyDragResistance(height, this.get('default_height')),
2017-07-22 22:21:05 +02:00
'width': _converse.applyDragResistance(width, this.get('default_width'))
});
return result;
}
},
2016-03-16 12:49:35 +01:00
ChatBoxView: {
events: {
'mousedown .dragresize-top': 'onStartVerticalResize',
'mousedown .dragresize-left': 'onStartHorizontalResize',
'mousedown .dragresize-topleft': 'onStartDiagonalResize'
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
$(window).on('resize', _.debounce(this.setDimensions.bind(this), 100));
2016-09-16 14:35:02 +02:00
this.__super__.initialize.apply(this, arguments);
},
2017-07-22 22:21:05 +02:00
render: function render() {
2016-09-16 14:35:02 +02:00
var result = this.__super__.render.apply(this, arguments);
2017-06-23 20:25:33 +02:00
renderDragResizeHandles(this.__super__._converse, this);
this.setWidth();
return result;
},
2017-07-22 22:21:05 +02:00
setWidth: function setWidth() {
// If a custom width is applied (due to drag-resizing),
// then we need to set the width of the .chatbox element as well.
if (this.model.get('width')) {
this.$el.css('width', this.model.get('width'));
}
},
2017-07-22 22:21:05 +02:00
_show: function _show() {
this.initDragResize().setDimensions();
2016-09-16 14:35:02 +02:00
this.__super__._show.apply(this, arguments);
},
2017-07-22 22:21:05 +02:00
initDragResize: function initDragResize() {
/* Determine and store the default box size.
* We need this information for the drag-resizing feature.
*/
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
var $flyout = this.$el.find('.box-flyout');
if (_.isUndefined(this.model.get('height'))) {
var height = $flyout.height();
var width = $flyout.width();
this.model.set('height', height);
this.model.set('default_height', height);
this.model.set('width', width);
this.model.set('default_width', width);
}
var min_width = $flyout.css('min-width');
var min_height = $flyout.css('min-height');
2017-07-22 22:21:05 +02:00
this.model.set('min_width', min_width.endsWith('px') ? Number(min_width.replace(/px$/, '')) : 0);
this.model.set('min_height', min_height.endsWith('px') ? Number(min_height.replace(/px$/, '')) : 0);
// Initialize last known mouse position
this.prev_pageY = 0;
this.prev_pageX = 0;
2017-02-03 13:51:07 +01:00
if (_converse.connection.connected) {
this.height = this.model.get('height');
this.width = this.model.get('width');
}
return this;
},
2017-07-22 22:21:05 +02:00
setDimensions: function setDimensions() {
// Make sure the chat box has the right height and width.
this.adjustToViewport();
this.setChatBoxHeight(this.model.get('height'));
this.setChatBoxWidth(this.model.get('width'));
},
2017-07-22 22:21:05 +02:00
setChatBoxHeight: function setChatBoxHeight(height) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
if (height) {
2017-07-22 22:21:05 +02:00
height = _converse.applyDragResistance(height, this.model.get('default_height')) + 'px';
} else {
height = "";
}
this.$el.children('.box-flyout')[0].style.height = height;
},
2017-07-22 22:21:05 +02:00
setChatBoxWidth: function setChatBoxWidth(width) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
if (width) {
2017-07-22 22:21:05 +02:00
width = _converse.applyDragResistance(width, this.model.get('default_width')) + 'px';
} else {
width = "";
}
this.$el[0].style.width = width;
this.$el.children('.box-flyout')[0].style.width = width;
},
2017-07-22 22:21:05 +02:00
adjustToViewport: function adjustToViewport() {
/* Event handler called when viewport gets resized. We remove
* custom width/height from chat boxes.
*/
var viewport_width = Math.max(document.documentElement.clientWidth, window.innerWidth || 0);
var viewport_height = Math.max(document.documentElement.clientHeight, window.innerHeight || 0);
if (viewport_width <= 480) {
this.model.set('height', undefined);
this.model.set('width', undefined);
} else if (viewport_width <= this.model.get('width')) {
this.model.set('width', undefined);
} else if (viewport_height <= this.model.get('height')) {
this.model.set('height', undefined);
}
},
2017-07-22 22:21:05 +02:00
onStartVerticalResize: function onStartVerticalResize(ev) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
if (!_converse.allow_dragresize) {
return true;
}
// Record element attributes for mouseMove().
this.height = this.$el.children('.box-flyout').height();
2017-02-03 13:51:07 +01:00
_converse.resizing = {
'chatbox': this,
'direction': 'top'
};
this.prev_pageY = ev.pageY;
},
2017-07-22 22:21:05 +02:00
onStartHorizontalResize: function onStartHorizontalResize(ev) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
if (!_converse.allow_dragresize) {
return true;
}
this.width = this.$el.children('.box-flyout').width();
2017-02-03 13:51:07 +01:00
_converse.resizing = {
'chatbox': this,
'direction': 'left'
};
this.prev_pageX = ev.pageX;
},
2017-07-22 22:21:05 +02:00
onStartDiagonalResize: function onStartDiagonalResize(ev) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
this.onStartHorizontalResize(ev);
this.onStartVerticalResize(ev);
2017-02-03 13:51:07 +01:00
_converse.resizing.direction = 'topleft';
},
2017-07-22 22:21:05 +02:00
resizeChatBox: function resizeChatBox(ev) {
var diff = void 0;
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
2017-02-03 13:51:07 +01:00
if (_converse.resizing.direction.indexOf('top') === 0) {
diff = ev.pageY - this.prev_pageY;
if (diff) {
2017-07-22 22:21:05 +02:00
this.height = this.height - diff > (this.model.get('min_height') || 0) ? this.height - diff : this.model.get('min_height');
this.prev_pageY = ev.pageY;
this.setChatBoxHeight(this.height);
}
}
2017-02-03 13:51:07 +01:00
if (_.includes(_converse.resizing.direction, 'left')) {
diff = this.prev_pageX - ev.pageX;
if (diff) {
2017-07-22 22:21:05 +02:00
this.width = this.width + diff > (this.model.get('min_width') || 0) ? this.width + diff : this.model.get('min_width');
this.prev_pageX = ev.pageX;
this.setChatBoxWidth(this.width);
}
}
}
},
2017-02-03 13:51:07 +01:00
HeadlinesBoxView: {
events: {
'mousedown .dragresize-top': 'onStartVerticalResize',
'mousedown .dragresize-left': 'onStartHorizontalResize',
'mousedown .dragresize-topleft': 'onStartDiagonalResize'
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2017-02-03 13:51:07 +01:00
$(window).on('resize', _.debounce(this.setDimensions.bind(this), 100));
return this.__super__.initialize.apply(this, arguments);
},
2017-07-22 22:21:05 +02:00
render: function render() {
2017-06-23 20:25:33 +02:00
var result = this.__super__.render.apply(this, arguments);
renderDragResizeHandles(this.__super__._converse, this);
this.setWidth();
return result;
2017-02-03 13:51:07 +01:00
}
},
ControlBoxView: {
events: {
'mousedown .dragresize-top': 'onStartVerticalResize',
'mousedown .dragresize-left': 'onStartHorizontalResize',
'mousedown .dragresize-topleft': 'onStartDiagonalResize'
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
$(window).on('resize', _.debounce(this.setDimensions.bind(this), 100));
2016-09-16 14:35:02 +02:00
this.__super__.initialize.apply(this, arguments);
},
2017-07-22 22:21:05 +02:00
render: function render() {
2017-06-23 20:25:33 +02:00
var result = this.__super__.render.apply(this, arguments);
renderDragResizeHandles(this.__super__._converse, this);
this.setWidth();
return result;
},
2017-07-22 22:21:05 +02:00
renderLoginPanel: function renderLoginPanel() {
2016-09-16 14:35:02 +02:00
var result = this.__super__.renderLoginPanel.apply(this, arguments);
this.initDragResize().setDimensions();
return result;
},
2017-07-22 22:21:05 +02:00
renderContactsPanel: function renderContactsPanel() {
2016-09-16 14:35:02 +02:00
var result = this.__super__.renderContactsPanel.apply(this, arguments);
this.initDragResize().setDimensions();
return result;
}
},
ChatRoomView: {
events: {
'mousedown .dragresize-top': 'onStartVerticalResize',
'mousedown .dragresize-left': 'onStartHorizontalResize',
'mousedown .dragresize-topleft': 'onStartDiagonalResize'
},
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
$(window).on('resize', _.debounce(this.setDimensions.bind(this), 100));
2016-09-16 14:35:02 +02:00
this.__super__.initialize.apply(this, arguments);
},
2017-07-22 22:21:05 +02:00
render: function render() {
2016-09-16 14:35:02 +02:00
var result = this.__super__.render.apply(this, arguments);
2017-06-23 20:25:33 +02:00
renderDragResizeHandles(this.__super__._converse, this);
this.setWidth();
return result;
}
}
},
2016-03-16 12:49:35 +01:00
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
/* The initialize function gets called as soon as the plugin is
* loaded by converse.js's plugin machinery.
*/
2017-02-03 13:51:07 +01:00
var _converse = this._converse;
2017-07-22 22:21:05 +02:00
2017-07-05 11:59:55 +02:00
_converse.api.settings.update({
2017-07-22 22:21:05 +02:00
allow_dragresize: true
});
2017-02-03 13:51:07 +01:00
_converse.applyDragResistance = function (value, default_value) {
/* This method applies some resistance around the
* default_value. If value is close enough to
* default_value, then default_value is returned instead.
*/
if (_.isUndefined(value)) {
return undefined;
} else if (_.isUndefined(default_value)) {
return value;
}
var resistance = 10;
2017-07-22 22:21:05 +02:00
if (value !== default_value && Math.abs(value - default_value) < resistance) {
return default_value;
2016-03-16 12:49:35 +01:00
}
return value;
};
2016-03-16 12:49:35 +01:00
}
});
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-dragresize.js.map;
2016-03-16 12:49:35 +01:00
// Converse.js (A browser based XMPP chat client)
// http://conversejs.org
//
2017-02-13 17:16:13 +01:00
// Copyright (c) 2012-2017, Jan-Carel Brand <jc@opkode.com>
2016-03-16 12:49:35 +01:00
// Licensed under the Mozilla Public License (MPLv2)
//
2017-02-03 13:51:07 +01:00
/*global define */
2016-03-16 12:49:35 +01:00
(function (root, factory) {
2017-07-22 22:21:05 +02:00
define('converse-headline',["converse-core", "tpl!chatbox", "converse-chatview"], factory);
})(undefined, function (converse, tpl_chatbox) {
2016-03-16 12:49:35 +01:00
"use strict";
2017-07-22 22:21:05 +02:00
var _converse$env = converse.env,
_ = _converse$env._,
utils = _converse$env.utils;
2017-02-03 13:51:07 +01:00
converse.plugins.add('converse-headline', {
2016-03-16 12:49:35 +01:00
overrides: {
// Overrides mentioned here will be picked up by converse.js's
// plugin architecture they will replace existing methods on the
// relevant objects or classes.
//
// New functions which don't exist yet can also be added.
2016-06-20 21:11:43 +02:00
ChatBoxViews: {
2017-07-22 22:21:05 +02:00
onChatBoxAdded: function onChatBoxAdded(item) {
2017-02-03 13:51:07 +01:00
var _converse = this.__super__._converse;
2017-07-22 22:21:05 +02:00
var view = this.get(item.get('id'));
if (!view && item.get('type') === 'headline') {
2017-07-22 22:21:05 +02:00
view = new _converse.HeadlinesBoxView({ model: item });
this.add(item.get('id'), view);
return view;
} else {
2016-09-16 14:35:02 +02:00
return this.__super__.onChatBoxAdded.apply(this, arguments);
}
}
}
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
2016-03-16 12:49:35 +01:00
/* The initialize function gets called as soon as the plugin is
* loaded by converse.js's plugin machinery.
*/
2017-02-03 13:51:07 +01:00
var _converse = this._converse,
__ = _converse.__;
2017-07-22 22:21:05 +02:00
2017-02-03 13:51:07 +01:00
_converse.HeadlinesBoxView = _converse.ChatBoxView.extend({
className: 'chatbox headlines',
events: {
'click .close-chatbox-button': 'close',
'click .toggle-chatbox-button': 'minimize',
2017-02-03 13:51:07 +01:00
'keypress textarea.chat-textarea': 'keyPressed'
},
2017-07-22 22:21:05 +02:00
initialize: function initialize() {
this.disable_mam = true; // Don't do MAM queries for this box
this.model.messages.on('add', this.onMessageAdded, this);
this.model.on('show', this.show, this);
this.model.on('destroy', this.hide, this);
this.model.on('change:minimized', this.onMinimizedChanged, this);
2016-07-26 08:00:30 +02:00
this.render().fetchMessages().insertIntoDOM().hide();
2017-02-03 13:51:07 +01:00
_converse.emit('chatBoxInitialized', this);
},
2017-07-22 22:21:05 +02:00
render: function render() {
this.$el.attr('id', this.model.get('box_id')).html(tpl_chatbox(_.extend(this.model.toJSON(), {
show_toolbar: _converse.show_toolbar,
show_textarea: false,
show_send_button: _converse.show_send_button,
title: this.model.get('fullname'),
unread_msgs: __('You have unread messages'),
info_close: __('Close this box'),
label_personal_message: ''
})));
this.$content = this.$el.find('.chat-content');
2017-02-03 13:51:07 +01:00
_converse.emit('chatBoxOpened', this);
2016-11-07 15:43:48 +01:00
utils.refreshWebkit();
return this;
}
});
2017-07-22 22:21:05 +02:00
function onHeadlineMessage(message) {
/* Handler method for all incoming messages of type "headline". */
2017-02-03 13:51:07 +01:00
var from_jid = message.getAttribute('from');
if (utils.isHeadlineMessage(message)) {
if (_.includes(from_jid, '@') && !_converse.allow_non_roster_messaging) {
return;
}
2017-06-23 20:25:33 +02:00
var chatbox = _converse.chatboxes.create({
2017-02-03 13:51:07 +01:00
'id': from_jid,
'jid': from_jid,
2017-07-22 22:21:05 +02:00
'fullname': from_jid,
2017-02-03 13:51:07 +01:00
'type': 'headline'
2017-06-23 20:25:33 +02:00
});
chatbox.createMessage(message, undefined, message);
2017-07-22 22:21:05 +02:00
_converse.emit('message', { 'chatbox': chatbox, 'stanza': message });
2017-02-03 13:51:07 +01:00
}
return true;
2017-07-22 22:21:05 +02:00
}
2017-02-03 13:51:07 +01:00
2017-07-22 22:21:05 +02:00
function registerHeadlineHandler() {
_converse.connection.addHandler(onHeadlineMessage, null, 'message');
}
2017-02-03 13:51:07 +01:00
_converse.on('connected', registerHeadlineHandler);
_converse.on('reconnected', registerHeadlineHandler);
2016-03-07 18:54:07 +01:00
}
});
2017-07-22 22:21:05 +02:00
});
//# sourceMappingURL=converse-headline.js.map;
/*global define */
2016-03-07 18:54:07 +01:00
if (typeof define !== 'undefined') {
// The section below determines which plugins will be included in a build
define('converse',[
"converse-core",
2017-02-13 17:16:13 +01:00
// PLEASE NOTE: By default all translations are included.
// You can modify the file src/locales.js to include only those
// translations that you care about.
2016-03-07 18:54:07 +01:00
/* START: Removable components
2017-02-01 12:05:32 +01:00
* --------------------
* Any of the following components may be removed if they're not needed.
*/
2016-03-16 12:49:35 +01:00
"converse-chatview", // Renders standalone chat boxes for single user chat
2016-06-20 21:11:43 +02:00
"converse-controlbox", // The control box
2016-11-07 15:43:48 +01:00
"converse-bookmarks", // XEP-0048 Bookmarks
2017-06-23 20:25:33 +02:00
"converse-roomslist", // Show currently open chat rooms
2016-03-16 12:49:35 +01:00
"converse-mam", // XEP-0313 Message Archive Management
2016-03-07 18:54:07 +01:00
"converse-muc", // XEP-0045 Multi-user chat
2016-03-16 12:49:35 +01:00
"converse-vcard", // XEP-0054 VCard-temp
2016-03-07 18:54:07 +01:00
"converse-otr", // Off-the-record encryption for one-on-one messages
"converse-register", // XEP-0077 In-band registration
"converse-ping", // XEP-0199 XMPP Ping
"converse-notification",// HTML5 Notifications
2016-03-16 12:49:35 +01:00
"converse-minimize", // Allows chat boxes to be minimized
"converse-dragresize", // Allows chat boxes to be resized by dragging them
2016-03-16 12:49:35 +01:00
"converse-headline", // Support for headline messages
2016-03-07 18:54:07 +01:00
/* END: Removable components */
2017-03-05 10:45:18 +01:00
], function (converse) {
2017-02-03 13:51:07 +01:00
return converse;
2016-03-07 18:54:07 +01:00
});
}
;
2017-03-05 10:45:18 +01:00
define('jquery', [], function () { return jQuery; });
2017-04-23 19:02:44 +02:00
define('jquery.noconflict', [], function () { return jQuery; });
2017-03-05 10:45:18 +01:00
define('jquery.browser', [], function () { return jQuery; });
define('awesomplete', [], function () { return jQuery; });
define('lodash', [], function () { return _; });
define('lodash.converter', [], function () { return fp; });
2017-04-23 19:02:44 +02:00
define('lodash.noconflict', [], function () { return _; });
2017-03-05 10:45:18 +01:00
define('moment_with_locales', [], function () { return moment; });
define('strophe', [], function () {
return {
'Strophe': Strophe,
'$build': $build,
'$iq': $iq,
'$msg': $msg,
'$pres': $pres,
'SHA1': SHA1,
'MD5': MD5,
'b64_hmac_sha1': SHA1.b64_hmac_sha1,
'b64_sha1': SHA1.b64_sha1,
'str_hmac_sha1': SHA1.str_hmac_sha1,
'str_sha1': SHA1.str_sha1
};
});
var strophePlugin = function () { return Strophe; };
var emptyFunction = function () { };
define('strophe.disco', ['strophe'], strophePlugin);
define('strophe.ping', ['strophe'], strophePlugin);
define('strophe.rsm', ['strophe'], strophePlugin);
define('strophe.vcard', ['strophe'], strophePlugin);
2017-04-23 19:02:44 +02:00
define('backbone', [], function () { return Backbone; });
define('backbone.noconflict', [], function () { return Backbone; });
2017-03-05 10:45:18 +01:00
define('backbone.browserStorage', ['backbone'], emptyFunction);
define('backbone.overview', ['backbone'], emptyFunction);
define('otr', [], function () { return { 'DSA': DSA, 'OTR': OTR };});
define("locales", [], emptyFunction);
return require('converse');
}));