/******/ (() => { // webpackBootstrap /******/ var __webpack_modules__ = ({ /***/ 8926: /***/ ((module) => { function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { Promise.resolve(value).then(_next, _throw); } } function _asyncToGenerator(fn) { return function () { var self = this, args = arguments; return new Promise(function (resolve, reject) { var gen = fn.apply(self, args); function _next(value) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value); } function _throw(err) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err); } _next(undefined); }); }; } module.exports = _asyncToGenerator, module.exports.__esModule = true, module.exports["default"] = module.exports; /***/ }), /***/ 9713: /***/ ((module) => { function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } module.exports = _defineProperty, module.exports.__esModule = true, module.exports["default"] = module.exports; /***/ }), /***/ 5318: /***/ ((module) => { function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; } module.exports = _interopRequireDefault, module.exports.__esModule = true, module.exports["default"] = module.exports; /***/ }), /***/ 1553: /***/ ((module) => { /** * Copyright (c) 2014-present, Facebook, Inc. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. */ var runtime = (function (exports) { "use strict"; var Op = Object.prototype; var hasOwn = Op.hasOwnProperty; var undefined; // More compressible than void 0. var $Symbol = typeof Symbol === "function" ? Symbol : {}; var iteratorSymbol = $Symbol.iterator || "@@iterator"; var asyncIteratorSymbol = $Symbol.asyncIterator || "@@asyncIterator"; var toStringTagSymbol = $Symbol.toStringTag || "@@toStringTag"; function define(obj, key, value) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); return obj[key]; } try { // IE 8 has a broken Object.defineProperty that only works on DOM objects. define({}, ""); } catch (err) { define = function(obj, key, value) { return obj[key] = value; }; } function wrap(innerFn, outerFn, self, tryLocsList) { // If outerFn provided and outerFn.prototype is a Generator, then outerFn.prototype instanceof Generator. var protoGenerator = outerFn && outerFn.prototype instanceof Generator ? outerFn : Generator; var generator = Object.create(protoGenerator.prototype); var context = new Context(tryLocsList || []); // The ._invoke method unifies the implementations of the .next, // .throw, and .return methods. generator._invoke = makeInvokeMethod(innerFn, self, context); return generator; } exports.wrap = wrap; // Try/catch helper to minimize deoptimizations. Returns a completion // record like context.tryEntries[i].completion. This interface could // have been (and was previously) designed to take a closure to be // invoked without arguments, but in all the cases we care about we // already have an existing method we want to call, so there's no need // to create a new function object. We can even get away with assuming // the method takes exactly one argument, since that happens to be true // in every case, so we don't have to touch the arguments object. The // only additional allocation required is the completion record, which // has a stable shape and so hopefully should be cheap to allocate. function tryCatch(fn, obj, arg) { try { return { type: "normal", arg: fn.call(obj, arg) }; } catch (err) { return { type: "throw", arg: err }; } } var GenStateSuspendedStart = "suspendedStart"; var GenStateSuspendedYield = "suspendedYield"; var GenStateExecuting = "executing"; var GenStateCompleted = "completed"; // Returning this object from the innerFn has the same effect as // breaking out of the dispatch switch statement. var ContinueSentinel = {}; // Dummy constructor functions that we use as the .constructor and // .constructor.prototype properties for functions that return Generator // objects. For full spec compliance, you may wish to configure your // minifier not to mangle the names of these two functions. function Generator() {} function GeneratorFunction() {} function GeneratorFunctionPrototype() {} // This is a polyfill for %IteratorPrototype% for environments that // don't natively support it. var IteratorPrototype = {}; define(IteratorPrototype, iteratorSymbol, function () { return this; }); var getProto = Object.getPrototypeOf; var NativeIteratorPrototype = getProto && getProto(getProto(values([]))); if (NativeIteratorPrototype && NativeIteratorPrototype !== Op && hasOwn.call(NativeIteratorPrototype, iteratorSymbol)) { // This environment has a native %IteratorPrototype%; use it instead // of the polyfill. IteratorPrototype = NativeIteratorPrototype; } var Gp = GeneratorFunctionPrototype.prototype = Generator.prototype = Object.create(IteratorPrototype); GeneratorFunction.prototype = GeneratorFunctionPrototype; define(Gp, "constructor", GeneratorFunctionPrototype); define(GeneratorFunctionPrototype, "constructor", GeneratorFunction); GeneratorFunction.displayName = define( GeneratorFunctionPrototype, toStringTagSymbol, "GeneratorFunction" ); // Helper for defining the .next, .throw, and .return methods of the // Iterator interface in terms of a single ._invoke method. function defineIteratorMethods(prototype) { ["next", "throw", "return"].forEach(function(method) { define(prototype, method, function(arg) { return this._invoke(method, arg); }); }); } exports.isGeneratorFunction = function(genFun) { var ctor = typeof genFun === "function" && genFun.constructor; return ctor ? ctor === GeneratorFunction || // For the native GeneratorFunction constructor, the best we can // do is to check its .name property. (ctor.displayName || ctor.name) === "GeneratorFunction" : false; }; exports.mark = function(genFun) { if (Object.setPrototypeOf) { Object.setPrototypeOf(genFun, GeneratorFunctionPrototype); } else { genFun.__proto__ = GeneratorFunctionPrototype; define(genFun, toStringTagSymbol, "GeneratorFunction"); } genFun.prototype = Object.create(Gp); return genFun; }; // Within the body of any async function, `await x` is transformed to // `yield regeneratorRuntime.awrap(x)`, so that the runtime can test // `hasOwn.call(value, "__await")` to determine if the yielded value is // meant to be awaited. exports.awrap = function(arg) { return { __await: arg }; }; function AsyncIterator(generator, PromiseImpl) { function invoke(method, arg, resolve, reject) { var record = tryCatch(generator[method], generator, arg); if (record.type === "throw") { reject(record.arg); } else { var result = record.arg; var value = result.value; if (value && typeof value === "object" && hasOwn.call(value, "__await")) { return PromiseImpl.resolve(value.__await).then(function(value) { invoke("next", value, resolve, reject); }, function(err) { invoke("throw", err, resolve, reject); }); } return PromiseImpl.resolve(value).then(function(unwrapped) { // When a yielded Promise is resolved, its final value becomes // the .value of the Promise<{value,done}> result for the // current iteration. result.value = unwrapped; resolve(result); }, function(error) { // If a rejected Promise was yielded, throw the rejection back // into the async generator function so it can be handled there. return invoke("throw", error, resolve, reject); }); } } var previousPromise; function enqueue(method, arg) { function callInvokeWithMethodAndArg() { return new PromiseImpl(function(resolve, reject) { invoke(method, arg, resolve, reject); }); } return previousPromise = // If enqueue has been called before, then we want to wait until // all previous Promises have been resolved before calling invoke, // so that results are always delivered in the correct order. If // enqueue has not been called before, then it is important to // call invoke immediately, without waiting on a callback to fire, // so that the async generator function has the opportunity to do // any necessary setup in a predictable way. This predictability // is why the Promise constructor synchronously invokes its // executor callback, and why async functions synchronously // execute code before the first await. Since we implement simple // async functions in terms of async generators, it is especially // important to get this right, even though it requires care. previousPromise ? previousPromise.then( callInvokeWithMethodAndArg, // Avoid propagating failures to Promises returned by later // invocations of the iterator. callInvokeWithMethodAndArg ) : callInvokeWithMethodAndArg(); } // Define the unified helper method that is used to implement .next, // .throw, and .return (see defineIteratorMethods). this._invoke = enqueue; } defineIteratorMethods(AsyncIterator.prototype); define(AsyncIterator.prototype, asyncIteratorSymbol, function () { return this; }); exports.AsyncIterator = AsyncIterator; // Note that simple async functions are implemented on top of // AsyncIterator objects; they just return a Promise for the value of // the final result produced by the iterator. exports.async = function(innerFn, outerFn, self, tryLocsList, PromiseImpl) { if (PromiseImpl === void 0) PromiseImpl = Promise; var iter = new AsyncIterator( wrap(innerFn, outerFn, self, tryLocsList), PromiseImpl ); return exports.isGeneratorFunction(outerFn) ? iter // If outerFn is a generator, return the full iterator. : iter.next().then(function(result) { return result.done ? result.value : iter.next(); }); }; function makeInvokeMethod(innerFn, self, context) { var state = GenStateSuspendedStart; return function invoke(method, arg) { if (state === GenStateExecuting) { throw new Error("Generator is already running"); } if (state === GenStateCompleted) { if (method === "throw") { throw arg; } // Be forgiving, per 25.3.3.3.3 of the spec: // https://people.mozilla.org/~jorendorff/es6-draft.html#sec-generatorresume return doneResult(); } context.method = method; context.arg = arg; while (true) { var delegate = context.delegate; if (delegate) { var delegateResult = maybeInvokeDelegate(delegate, context); if (delegateResult) { if (delegateResult === ContinueSentinel) continue; return delegateResult; } } if (context.method === "next") { // Setting context._sent for legacy support of Babel's // function.sent implementation. context.sent = context._sent = context.arg; } else if (context.method === "throw") { if (state === GenStateSuspendedStart) { state = GenStateCompleted; throw context.arg; } context.dispatchException(context.arg); } else if (context.method === "return") { context.abrupt("return", context.arg); } state = GenStateExecuting; var record = tryCatch(innerFn, self, context); if (record.type === "normal") { // If an exception is thrown from innerFn, we leave state === // GenStateExecuting and loop back for another invocation. state = context.done ? GenStateCompleted : GenStateSuspendedYield; if (record.arg === ContinueSentinel) { continue; } return { value: record.arg, done: context.done }; } else if (record.type === "throw") { state = GenStateCompleted; // Dispatch the exception by looping back around to the // context.dispatchException(context.arg) call above. context.method = "throw"; context.arg = record.arg; } } }; } // Call delegate.iterator[context.method](context.arg) and handle the // result, either by returning a { value, done } result from the // delegate iterator, or by modifying context.method and context.arg, // setting context.delegate to null, and returning the ContinueSentinel. function maybeInvokeDelegate(delegate, context) { var method = delegate.iterator[context.method]; if (method === undefined) { // A .throw or .return when the delegate iterator has no .throw // method always terminates the yield* loop. context.delegate = null; if (context.method === "throw") { // Note: ["return"] must be used for ES3 parsing compatibility. if (delegate.iterator["return"]) { // If the delegate iterator has a return method, give it a // chance to clean up. context.method = "return"; context.arg = undefined; maybeInvokeDelegate(delegate, context); if (context.method === "throw") { // If maybeInvokeDelegate(context) changed context.method from // "return" to "throw", let that override the TypeError below. return ContinueSentinel; } } context.method = "throw"; context.arg = new TypeError( "The iterator does not provide a 'throw' method"); } return ContinueSentinel; } var record = tryCatch(method, delegate.iterator, context.arg); if (record.type === "throw") { context.method = "throw"; context.arg = record.arg; context.delegate = null; return ContinueSentinel; } var info = record.arg; if (! info) { context.method = "throw"; context.arg = new TypeError("iterator result is not an object"); context.delegate = null; return ContinueSentinel; } if (info.done) { // Assign the result of the finished delegate to the temporary // variable specified by delegate.resultName (see delegateYield). context[delegate.resultName] = info.value; // Resume execution at the desired location (see delegateYield). context.next = delegate.nextLoc; // If context.method was "throw" but the delegate handled the // exception, let the outer generator proceed normally. If // context.method was "next", forget context.arg since it has been // "consumed" by the delegate iterator. If context.method was // "return", allow the original .return call to continue in the // outer generator. if (context.method !== "return") { context.method = "next"; context.arg = undefined; } } else { // Re-yield the result returned by the delegate method. return info; } // The delegate iterator is finished, so forget it and continue with // the outer generator. context.delegate = null; return ContinueSentinel; } // Define Generator.prototype.{next,throw,return} in terms of the // unified ._invoke helper method. defineIteratorMethods(Gp); define(Gp, toStringTagSymbol, "Generator"); // A Generator should always return itself as the iterator object when the // @@iterator function is called on it. Some browsers' implementations of the // iterator prototype chain incorrectly implement this, causing the Generator // object to not be returned from this call. This ensures that doesn't happen. // See https://github.com/facebook/regenerator/issues/274 for more details. define(Gp, iteratorSymbol, function() { return this; }); define(Gp, "toString", function() { return "[object Generator]"; }); function pushTryEntry(locs) { var entry = { tryLoc: locs[0] }; if (1 in locs) { entry.catchLoc = locs[1]; } if (2 in locs) { entry.finallyLoc = locs[2]; entry.afterLoc = locs[3]; } this.tryEntries.push(entry); } function resetTryEntry(entry) { var record = entry.completion || {}; record.type = "normal"; delete record.arg; entry.completion = record; } function Context(tryLocsList) { // The root entry object (effectively a try statement without a catch // or a finally block) gives us a place to store values thrown from // locations where there is no enclosing try statement. this.tryEntries = [{ tryLoc: "root" }]; tryLocsList.forEach(pushTryEntry, this); this.reset(true); } exports.keys = function(object) { var keys = []; for (var key in object) { keys.push(key); } keys.reverse(); // Rather than returning an object with a next method, we keep // things simple and return the next function itself. return function next() { while (keys.length) { var key = keys.pop(); if (key in object) { next.value = key; next.done = false; return next; } } // To avoid creating an additional object, we just hang the .value // and .done properties off the next function object itself. This // also ensures that the minifier will not anonymize the function. next.done = true; return next; }; }; function values(iterable) { if (iterable) { var iteratorMethod = iterable[iteratorSymbol]; if (iteratorMethod) { return iteratorMethod.call(iterable); } if (typeof iterable.next === "function") { return iterable; } if (!isNaN(iterable.length)) { var i = -1, next = function next() { while (++i < iterable.length) { if (hasOwn.call(iterable, i)) { next.value = iterable[i]; next.done = false; return next; } } next.value = undefined; next.done = true; return next; }; return next.next = next; } } // Return an iterator with no values. return { next: doneResult }; } exports.values = values; function doneResult() { return { value: undefined, done: true }; } Context.prototype = { constructor: Context, reset: function(skipTempReset) { this.prev = 0; this.next = 0; // Resetting context._sent for legacy support of Babel's // function.sent implementation. this.sent = this._sent = undefined; this.done = false; this.delegate = null; this.method = "next"; this.arg = undefined; this.tryEntries.forEach(resetTryEntry); if (!skipTempReset) { for (var name in this) { // Not sure about the optimal order of these conditions: if (name.charAt(0) === "t" && hasOwn.call(this, name) && !isNaN(+name.slice(1))) { this[name] = undefined; } } } }, stop: function() { this.done = true; var rootEntry = this.tryEntries[0]; var rootRecord = rootEntry.completion; if (rootRecord.type === "throw") { throw rootRecord.arg; } return this.rval; }, dispatchException: function(exception) { if (this.done) { throw exception; } var context = this; function handle(loc, caught) { record.type = "throw"; record.arg = exception; context.next = loc; if (caught) { // If the dispatched exception was caught by a catch block, // then let that catch block handle the exception normally. context.method = "next"; context.arg = undefined; } return !! caught; } for (var i = this.tryEntries.length - 1; i >= 0; --i) { var entry = this.tryEntries[i]; var record = entry.completion; if (entry.tryLoc === "root") { // Exception thrown outside of any try block that could handle // it, so set the completion value of the entire function to // throw the exception. return handle("end"); } if (entry.tryLoc <= this.prev) { var hasCatch = hasOwn.call(entry, "catchLoc"); var hasFinally = hasOwn.call(entry, "finallyLoc"); if (hasCatch && hasFinally) { if (this.prev < entry.catchLoc) { return handle(entry.catchLoc, true); } else if (this.prev < entry.finallyLoc) { return handle(entry.finallyLoc); } } else if (hasCatch) { if (this.prev < entry.catchLoc) { return handle(entry.catchLoc, true); } } else if (hasFinally) { if (this.prev < entry.finallyLoc) { return handle(entry.finallyLoc); } } else { throw new Error("try statement without catch or finally"); } } } }, abrupt: function(type, arg) { for (var i = this.tryEntries.length - 1; i >= 0; --i) { var entry = this.tryEntries[i]; if (entry.tryLoc <= this.prev && hasOwn.call(entry, "finallyLoc") && this.prev < entry.finallyLoc) { var finallyEntry = entry; break; } } if (finallyEntry && (type === "break" || type === "continue") && finallyEntry.tryLoc <= arg && arg <= finallyEntry.finallyLoc) { // Ignore the finally entry if control is not jumping to a // location outside the try/catch block. finallyEntry = null; } var record = finallyEntry ? finallyEntry.completion : {}; record.type = type; record.arg = arg; if (finallyEntry) { this.method = "next"; this.next = finallyEntry.finallyLoc; return ContinueSentinel; } return this.complete(record); }, complete: function(record, afterLoc) { if (record.type === "throw") { throw record.arg; } if (record.type === "break" || record.type === "continue") { this.next = record.arg; } else if (record.type === "return") { this.rval = this.arg = record.arg; this.method = "return"; this.next = "end"; } else if (record.type === "normal" && afterLoc) { this.next = afterLoc; } return ContinueSentinel; }, finish: function(finallyLoc) { for (var i = this.tryEntries.length - 1; i >= 0; --i) { var entry = this.tryEntries[i]; if (entry.finallyLoc === finallyLoc) { this.complete(entry.completion, entry.afterLoc); resetTryEntry(entry); return ContinueSentinel; } } }, "catch": function(tryLoc) { for (var i = this.tryEntries.length - 1; i >= 0; --i) { var entry = this.tryEntries[i]; if (entry.tryLoc === tryLoc) { var record = entry.completion; if (record.type === "throw") { var thrown = record.arg; resetTryEntry(entry); } return thrown; } } // The context.catch method must only be called with a location // argument that corresponds to a known catch block. throw new Error("illegal catch attempt"); }, delegateYield: function(iterable, resultName, nextLoc) { this.delegate = { iterator: values(iterable), resultName: resultName, nextLoc: nextLoc }; if (this.method === "next") { // Deliberately forget the last sent value so that we don't // accidentally pass it on to the delegate. this.arg = undefined; } return ContinueSentinel; } }; // Regardless of whether this script is executing as a CommonJS module // or not, return the runtime object so that we can declare the variable // regeneratorRuntime in the outer scope, which allows this module to be // injected easily by `bin/regenerator --include-runtime script.js`. return exports; }( // If this script is executing as a CommonJS module, use module.exports // as the regeneratorRuntime namespace. Otherwise create a new empty // object. Either way, the resulting object will be used to initialize // the regeneratorRuntime variable at the top of this file. true ? module.exports : 0 )); try { regeneratorRuntime = runtime; } catch (accidentalStrictMode) { // This module should not be running in strict mode, so the above // assignment should always work unless something is misconfigured. Just // in case runtime.js accidentally runs in strict mode, in modern engines // we can explicitly access globalThis. In older engines we can escape // strict mode using a global Function call. This could conceivably fail // if a Content Security Policy forbids using Function, but in that case // the proper solution is to fix the accidental strict mode problem. If // you've misconfigured your bundler to force strict mode and applied a // CSP to forbid Function, and you're not willing to fix either of those // problems, please detail your unique predicament in a GitHub issue. if (typeof globalThis === "object") { globalThis.regeneratorRuntime = runtime; } else { Function("r", "regeneratorRuntime = r")(runtime); } } /***/ }), /***/ 7757: /***/ ((module, __unused_webpack_exports, __webpack_require__) => { module.exports = __webpack_require__(1553); /***/ }), /***/ 9494: /***/ ((module, __unused_webpack_exports, __webpack_require__) => { "use strict"; const atob = __webpack_require__(7672); const btoa = __webpack_require__(4817); module.exports = { atob, btoa }; /***/ }), /***/ 7672: /***/ ((module) => { "use strict"; /** * Implementation of atob() according to the HTML and Infra specs, except that * instead of throwing INVALID_CHARACTER_ERR we return null. */ function atob(data) { // Web IDL requires DOMStrings to just be converted using ECMAScript // ToString, which in our case amounts to using a template literal. data = `${data}`; // "Remove all ASCII whitespace from data." data = data.replace(/[ \t\n\f\r]/g, ""); // "If data's length divides by 4 leaving no remainder, then: if data ends // with one or two U+003D (=) code points, then remove them from data." if (data.length % 4 === 0) { data = data.replace(/==?$/, ""); } // "If data's length divides by 4 leaving a remainder of 1, then return // failure." // // "If data contains a code point that is not one of // // U+002B (+) // U+002F (/) // ASCII alphanumeric // // then return failure." if (data.length % 4 === 1 || /[^+/0-9A-Za-z]/.test(data)) { return null; } // "Let output be an empty byte sequence." let output = ""; // "Let buffer be an empty buffer that can have bits appended to it." // // We append bits via left-shift and or. accumulatedBits is used to track // when we've gotten to 24 bits. let buffer = 0; let accumulatedBits = 0; // "Let position be a position variable for data, initially pointing at the // start of data." // // "While position does not point past the end of data:" for (let i = 0; i < data.length; i++) { // "Find the code point pointed to by position in the second column of // Table 1: The Base 64 Alphabet of RFC 4648. Let n be the number given in // the first cell of the same row. // // "Append to buffer the six bits corresponding to n, most significant bit // first." // // atobLookup() implements the table from RFC 4648. buffer <<= 6; buffer |= atobLookup(data[i]); accumulatedBits += 6; // "If buffer has accumulated 24 bits, interpret them as three 8-bit // big-endian numbers. Append three bytes with values equal to those // numbers to output, in the same order, and then empty buffer." if (accumulatedBits === 24) { output += String.fromCharCode((buffer & 0xff0000) >> 16); output += String.fromCharCode((buffer & 0xff00) >> 8); output += String.fromCharCode(buffer & 0xff); buffer = accumulatedBits = 0; } // "Advance position by 1." } // "If buffer is not empty, it contains either 12 or 18 bits. If it contains // 12 bits, then discard the last four and interpret the remaining eight as // an 8-bit big-endian number. If it contains 18 bits, then discard the last // two and interpret the remaining 16 as two 8-bit big-endian numbers. Append // the one or two bytes with values equal to those one or two numbers to // output, in the same order." if (accumulatedBits === 12) { buffer >>= 4; output += String.fromCharCode(buffer); } else if (accumulatedBits === 18) { buffer >>= 2; output += String.fromCharCode((buffer & 0xff00) >> 8); output += String.fromCharCode(buffer & 0xff); } // "Return output." return output; } /** * A lookup table for atob(), which converts an ASCII character to the * corresponding six-bit number. */ const keystr = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; function atobLookup(chr) { const index = keystr.indexOf(chr); // Throw exception if character is not in the lookup string; should not be hit in tests return index < 0 ? undefined : index; } module.exports = atob; /***/ }), /***/ 4817: /***/ ((module) => { "use strict"; /** * btoa() as defined by the HTML and Infra specs, which mostly just references * RFC 4648. */ function btoa(s) { let i; // String conversion as required by Web IDL. s = `${s}`; // "The btoa() method must throw an "InvalidCharacterError" DOMException if // data contains any character whose code point is greater than U+00FF." for (i = 0; i < s.length; i++) { if (s.charCodeAt(i) > 255) { return null; } } let out = ""; for (i = 0; i < s.length; i += 3) { const groupsOfSix = [undefined, undefined, undefined, undefined]; groupsOfSix[0] = s.charCodeAt(i) >> 2; groupsOfSix[1] = (s.charCodeAt(i) & 0x03) << 4; if (s.length > i + 1) { groupsOfSix[1] |= s.charCodeAt(i + 1) >> 4; groupsOfSix[2] = (s.charCodeAt(i + 1) & 0x0f) << 2; } if (s.length > i + 2) { groupsOfSix[2] |= s.charCodeAt(i + 2) >> 6; groupsOfSix[3] = s.charCodeAt(i + 2) & 0x3f; } for (let j = 0; j < groupsOfSix.length; j++) { if (typeof groupsOfSix[j] === "undefined") { out += "="; } else { out += btoaLookup(groupsOfSix[j]); } } } return out; } /** * Lookup table for btoa(), which converts a six-bit number into the * corresponding ASCII character. */ const keystr = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; function btoaLookup(index) { if (index >= 0 && index < 64) { return keystr[index]; } // Throw INVALID_CHARACTER_ERR exception here -- won't be hit in the tests. return undefined; } module.exports = btoa; /***/ }), /***/ 4223: /***/ ((module, exports, __webpack_require__) => { var __WEBPACK_AMD_DEFINE_RESULT__;/* global window, exports, define */ !function () { 'use strict'; var re = { not_string: /[^s]/, not_bool: /[^t]/, not_type: /[^T]/, not_primitive: /[^v]/, number: /[diefg]/, numeric_arg: /[bcdiefguxX]/, json: /[j]/, not_json: /[^j]/, text: /^[^\x25]+/, modulo: /^\x25{2}/, placeholder: /^\x25(?:([1-9]\d*)\$|\(([^)]+)\))?(\+)?(0|'[^$])?(-)?(\d+)?(?:\.(\d+))?([b-gijostTuvxX])/, key: /^([a-z_][a-z_\d]*)/i, key_access: /^\.([a-z_][a-z_\d]*)/i, index_access: /^\[(\d+)\]/, sign: /^[+-]/ }; function sprintf(key) { // `arguments` is not an array, but should be fine for this call return sprintf_format(sprintf_parse(key), arguments); } function vsprintf(fmt, argv) { return sprintf.apply(null, [fmt].concat(argv || [])); } function sprintf_format(parse_tree, argv) { var cursor = 1, tree_length = parse_tree.length, arg, output = '', i, k, ph, pad, pad_character, pad_length, is_positive, sign; for (i = 0; i < tree_length; i++) { if (typeof parse_tree[i] === 'string') { output += parse_tree[i]; } else if (typeof parse_tree[i] === 'object') { ph = parse_tree[i]; // convenience purposes only if (ph.keys) { // keyword argument arg = argv[cursor]; for (k = 0; k < ph.keys.length; k++) { if (arg == undefined) { throw new Error(sprintf('[sprintf] Cannot access property "%s" of undefined value "%s"', ph.keys[k], ph.keys[k - 1])); } arg = arg[ph.keys[k]]; } } else if (ph.param_no) { // positional argument (explicit) arg = argv[ph.param_no]; } else { // positional argument (implicit) arg = argv[cursor++]; } if (re.not_type.test(ph.type) && re.not_primitive.test(ph.type) && arg instanceof Function) { arg = arg(); } if (re.numeric_arg.test(ph.type) && typeof arg !== 'number' && isNaN(arg)) { throw new TypeError(sprintf('[sprintf] expecting number but found %T', arg)); } if (re.number.test(ph.type)) { is_positive = arg >= 0; } switch (ph.type) { case 'b': arg = parseInt(arg, 10).toString(2); break; case 'c': arg = String.fromCharCode(parseInt(arg, 10)); break; case 'd': case 'i': arg = parseInt(arg, 10); break; case 'j': arg = JSON.stringify(arg, null, ph.width ? parseInt(ph.width) : 0); break; case 'e': arg = ph.precision ? parseFloat(arg).toExponential(ph.precision) : parseFloat(arg).toExponential(); break; case 'f': arg = ph.precision ? parseFloat(arg).toFixed(ph.precision) : parseFloat(arg); break; case 'g': arg = ph.precision ? String(Number(arg.toPrecision(ph.precision))) : parseFloat(arg); break; case 'o': arg = (parseInt(arg, 10) >>> 0).toString(8); break; case 's': arg = String(arg); arg = ph.precision ? arg.substring(0, ph.precision) : arg; break; case 't': arg = String(!!arg); arg = ph.precision ? arg.substring(0, ph.precision) : arg; break; case 'T': arg = Object.prototype.toString.call(arg).slice(8, -1).toLowerCase(); arg = ph.precision ? arg.substring(0, ph.precision) : arg; break; case 'u': arg = parseInt(arg, 10) >>> 0; break; case 'v': arg = arg.valueOf(); arg = ph.precision ? arg.substring(0, ph.precision) : arg; break; case 'x': arg = (parseInt(arg, 10) >>> 0).toString(16); break; case 'X': arg = (parseInt(arg, 10) >>> 0).toString(16).toUpperCase(); break; } if (re.json.test(ph.type)) { output += arg; } else { if (re.number.test(ph.type) && (!is_positive || ph.sign)) { sign = is_positive ? '+' : '-'; arg = arg.toString().replace(re.sign, ''); } else { sign = ''; } pad_character = ph.pad_char ? ph.pad_char === '0' ? '0' : ph.pad_char.charAt(1) : ' '; pad_length = ph.width - (sign + arg).length; pad = ph.width ? pad_length > 0 ? pad_character.repeat(pad_length) : '' : ''; output += ph.align ? sign + arg + pad : pad_character === '0' ? sign + pad + arg : pad + sign + arg; } } } return output; } var sprintf_cache = Object.create(null); function sprintf_parse(fmt) { if (sprintf_cache[fmt]) { return sprintf_cache[fmt]; } var _fmt = fmt, match, parse_tree = [], arg_names = 0; while (_fmt) { if ((match = re.text.exec(_fmt)) !== null) { parse_tree.push(match[0]); } else if ((match = re.modulo.exec(_fmt)) !== null) { parse_tree.push('%'); } else if ((match = re.placeholder.exec(_fmt)) !== null) { if (match[2]) { arg_names |= 1; var field_list = [], replacement_field = match[2], field_match = []; if ((field_match = re.key.exec(replacement_field)) !== null) { field_list.push(field_match[1]); while ((replacement_field = replacement_field.substring(field_match[0].length)) !== '') { if ((field_match = re.key_access.exec(replacement_field)) !== null) { field_list.push(field_match[1]); } else if ((field_match = re.index_access.exec(replacement_field)) !== null) { field_list.push(field_match[1]); } else { throw new SyntaxError('[sprintf] failed to parse named argument key'); } } } else { throw new SyntaxError('[sprintf] failed to parse named argument key'); } match[2] = field_list; } else { arg_names |= 2; } if (arg_names === 3) { throw new Error('[sprintf] mixing positional and named placeholders is not (yet) supported'); } parse_tree.push({ placeholder: match[0], param_no: match[1], keys: match[2], sign: match[3], pad_char: match[4], align: match[5], width: match[6], precision: match[7], type: match[8] }); } else { throw new SyntaxError('[sprintf] unexpected placeholder'); } _fmt = _fmt.substring(match[0].length); } return sprintf_cache[fmt] = parse_tree; } /** * export to either browser or node.js */ /* eslint-disable quote-props */ if (true) { exports.sprintf = sprintf; exports.vsprintf = vsprintf; } if (typeof window !== 'undefined') { window['sprintf'] = sprintf; window['vsprintf'] = vsprintf; if (true) { !(__WEBPACK_AMD_DEFINE_RESULT__ = (function () { return { 'sprintf': sprintf, 'vsprintf': vsprintf }; }).call(exports, __webpack_require__, exports, module), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__)); } } /* eslint-enable quote-props */ }(); // eslint-disable-line /***/ }), /***/ 8677: /***/ (function(module, exports, __webpack_require__) { var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_RESULT__;/*! * URI.js - Mutating URLs * IPv6 Support * * Version: 1.19.11 * * Author: Rodney Rehm * Web: http://medialize.github.io/URI.js/ * * Licensed under * MIT License http://www.opensource.org/licenses/mit-license * */ (function (root, factory) { 'use strict'; // https://github.com/umdjs/umd/blob/master/returnExports.js if ( true && module.exports) { // Node module.exports = factory(); } else if (true) { // AMD. Register as an anonymous module. !(__WEBPACK_AMD_DEFINE_FACTORY__ = (factory), __WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ? (__WEBPACK_AMD_DEFINE_FACTORY__.call(exports, __webpack_require__, exports, module)) : __WEBPACK_AMD_DEFINE_FACTORY__), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__)); } else {} })(this, function (root) { 'use strict'; /* var _in = "fe80:0000:0000:0000:0204:61ff:fe9d:f156"; var _out = IPv6.best(_in); var _expected = "fe80::204:61ff:fe9d:f156"; console.log(_in, _out, _expected, _out === _expected); */ // save current IPv6 variable, if any var _IPv6 = root && root.IPv6; function bestPresentation(address) { // based on: // Javascript to test an IPv6 address for proper format, and to // present the "best text representation" according to IETF Draft RFC at // http://tools.ietf.org/html/draft-ietf-6man-text-addr-representation-04 // 8 Feb 2010 Rich Brown, Dartware, LLC // Please feel free to use this code as long as you provide a link to // http://www.intermapper.com // http://intermapper.com/support/tools/IPV6-Validator.aspx // http://download.dartware.com/thirdparty/ipv6validator.js var _address = address.toLowerCase(); var segments = _address.split(':'); var length = segments.length; var total = 8; // trim colons (:: or ::a:b:c… or …a:b:c::) if (segments[0] === '' && segments[1] === '' && segments[2] === '') { // must have been :: // remove first two items segments.shift(); segments.shift(); } else if (segments[0] === '' && segments[1] === '') { // must have been ::xxxx // remove the first item segments.shift(); } else if (segments[length - 1] === '' && segments[length - 2] === '') { // must have been xxxx:: segments.pop(); } length = segments.length; // adjust total segments for IPv4 trailer if (segments[length - 1].indexOf('.') !== -1) { // found a "." which means IPv4 total = 7; } // fill empty segments them with "0000" var pos; for (pos = 0; pos < length; pos++) { if (segments[pos] === '') { break; } } if (pos < total) { segments.splice(pos, 1, '0000'); while (segments.length < total) { segments.splice(pos, 0, '0000'); } } // strip leading zeros var _segments; for (var i = 0; i < total; i++) { _segments = segments[i].split(''); for (var j = 0; j < 3; j++) { if (_segments[0] === '0' && _segments.length > 1) { _segments.splice(0, 1); } else { break; } } segments[i] = _segments.join(''); } // find longest sequence of zeroes and coalesce them into one segment var best = -1; var _best = 0; var _current = 0; var current = -1; var inzeroes = false; // i; already declared for (i = 0; i < total; i++) { if (inzeroes) { if (segments[i] === '0') { _current += 1; } else { inzeroes = false; if (_current > _best) { best = current; _best = _current; } } } else { if (segments[i] === '0') { inzeroes = true; current = i; _current = 1; } } } if (_current > _best) { best = current; _best = _current; } if (_best > 1) { segments.splice(best, _best, ''); } length = segments.length; // assemble remaining segments var result = ''; if (segments[0] === '') { result = ':'; } for (i = 0; i < length; i++) { result += segments[i]; if (i === length - 1) { break; } result += ':'; } if (segments[length - 1] === '') { result += ':'; } return result; } function noConflict() { /*jshint validthis: true */ if (root.IPv6 === this) { root.IPv6 = _IPv6; } return this; } return { best: bestPresentation, noConflict: noConflict }; }); /***/ }), /***/ 9827: /***/ (function(module, exports, __webpack_require__) { var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_RESULT__;/*! * URI.js - Mutating URLs * Second Level Domain (SLD) Support * * Version: 1.19.11 * * Author: Rodney Rehm * Web: http://medialize.github.io/URI.js/ * * Licensed under * MIT License http://www.opensource.org/licenses/mit-license * */ (function (root, factory) { 'use strict'; // https://github.com/umdjs/umd/blob/master/returnExports.js if ( true && module.exports) { // Node module.exports = factory(); } else if (true) { // AMD. Register as an anonymous module. !(__WEBPACK_AMD_DEFINE_FACTORY__ = (factory), __WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ? (__WEBPACK_AMD_DEFINE_FACTORY__.call(exports, __webpack_require__, exports, module)) : __WEBPACK_AMD_DEFINE_FACTORY__), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__)); } else {} })(this, function (root) { 'use strict'; // save current SecondLevelDomains variable, if any var _SecondLevelDomains = root && root.SecondLevelDomains; var SLD = { // list of known Second Level Domains // converted list of SLDs from https://github.com/gavingmiller/second-level-domains // ---- // publicsuffix.org is more current and actually used by a couple of browsers internally. // downside is it also contains domains like "dyndns.org" - which is fine for the security // issues browser have to deal with (SOP for cookies, etc) - but is way overboard for URI.js // ---- list: { 'ac': ' com gov mil net org ', 'ae': ' ac co gov mil name net org pro sch ', 'af': ' com edu gov net org ', 'al': ' com edu gov mil net org ', 'ao': ' co ed gv it og pb ', 'ar': ' com edu gob gov int mil net org tur ', 'at': ' ac co gv or ', 'au': ' asn com csiro edu gov id net org ', 'ba': ' co com edu gov mil net org rs unbi unmo unsa untz unze ', 'bb': ' biz co com edu gov info net org store tv ', 'bh': ' biz cc com edu gov info net org ', 'bn': ' com edu gov net org ', 'bo': ' com edu gob gov int mil net org tv ', 'br': ' adm adv agr am arq art ato b bio blog bmd cim cng cnt com coop ecn edu eng esp etc eti far flog fm fnd fot fst g12 ggf gov imb ind inf jor jus lel mat med mil mus net nom not ntr odo org ppg pro psc psi qsl rec slg srv tmp trd tur tv vet vlog wiki zlg ', 'bs': ' com edu gov net org ', 'bz': ' du et om ov rg ', 'ca': ' ab bc mb nb nf nl ns nt nu on pe qc sk yk ', 'ck': ' biz co edu gen gov info net org ', 'cn': ' ac ah bj com cq edu fj gd gov gs gx gz ha hb he hi hl hn jl js jx ln mil net nm nx org qh sc sd sh sn sx tj tw xj xz yn zj ', 'co': ' com edu gov mil net nom org ', 'cr': ' ac c co ed fi go or sa ', 'cy': ' ac biz com ekloges gov ltd name net org parliament press pro tm ', 'do': ' art com edu gob gov mil net org sld web ', 'dz': ' art asso com edu gov net org pol ', 'ec': ' com edu fin gov info med mil net org pro ', 'eg': ' com edu eun gov mil name net org sci ', 'er': ' com edu gov ind mil net org rochest w ', 'es': ' com edu gob nom org ', 'et': ' biz com edu gov info name net org ', 'fj': ' ac biz com info mil name net org pro ', 'fk': ' ac co gov net nom org ', 'fr': ' asso com f gouv nom prd presse tm ', 'gg': ' co net org ', 'gh': ' com edu gov mil org ', 'gn': ' ac com gov net org ', 'gr': ' com edu gov mil net org ', 'gt': ' com edu gob ind mil net org ', 'gu': ' com edu gov net org ', 'hk': ' com edu gov idv net org ', 'hu': ' 2000 agrar bolt casino city co erotica erotika film forum games hotel info ingatlan jogasz konyvelo lakas media news org priv reklam sex shop sport suli szex tm tozsde utazas video ', 'id': ' ac co go mil net or sch web ', 'il': ' ac co gov idf k12 muni net org ', 'in': ' ac co edu ernet firm gen gov i ind mil net nic org res ', 'iq': ' com edu gov i mil net org ', 'ir': ' ac co dnssec gov i id net org sch ', 'it': ' edu gov ', 'je': ' co net org ', 'jo': ' com edu gov mil name net org sch ', 'jp': ' ac ad co ed go gr lg ne or ', 'ke': ' ac co go info me mobi ne or sc ', 'kh': ' com edu gov mil net org per ', 'ki': ' biz com de edu gov info mob net org tel ', 'km': ' asso com coop edu gouv k medecin mil nom notaires pharmaciens presse tm veterinaire ', 'kn': ' edu gov net org ', 'kr': ' ac busan chungbuk chungnam co daegu daejeon es gangwon go gwangju gyeongbuk gyeonggi gyeongnam hs incheon jeju jeonbuk jeonnam k kg mil ms ne or pe re sc seoul ulsan ', 'kw': ' com edu gov net org ', 'ky': ' com edu gov net org ', 'kz': ' com edu gov mil net org ', 'lb': ' com edu gov net org ', 'lk': ' assn com edu gov grp hotel int ltd net ngo org sch soc web ', 'lr': ' com edu gov net org ', 'lv': ' asn com conf edu gov id mil net org ', 'ly': ' com edu gov id med net org plc sch ', 'ma': ' ac co gov m net org press ', 'mc': ' asso tm ', 'me': ' ac co edu gov its net org priv ', 'mg': ' com edu gov mil nom org prd tm ', 'mk': ' com edu gov inf name net org pro ', 'ml': ' com edu gov net org presse ', 'mn': ' edu gov org ', 'mo': ' com edu gov net org ', 'mt': ' com edu gov net org ', 'mv': ' aero biz com coop edu gov info int mil museum name net org pro ', 'mw': ' ac co com coop edu gov int museum net org ', 'mx': ' com edu gob net org ', 'my': ' com edu gov mil name net org sch ', 'nf': ' arts com firm info net other per rec store web ', 'ng': ' biz com edu gov mil mobi name net org sch ', 'ni': ' ac co com edu gob mil net nom org ', 'np': ' com edu gov mil net org ', 'nr': ' biz com edu gov info net org ', 'om': ' ac biz co com edu gov med mil museum net org pro sch ', 'pe': ' com edu gob mil net nom org sld ', 'ph': ' com edu gov i mil net ngo org ', 'pk': ' biz com edu fam gob gok gon gop gos gov net org web ', 'pl': ' art bialystok biz com edu gda gdansk gorzow gov info katowice krakow lodz lublin mil net ngo olsztyn org poznan pwr radom slupsk szczecin torun warszawa waw wroc wroclaw zgora ', 'pr': ' ac biz com edu est gov info isla name net org pro prof ', 'ps': ' com edu gov net org plo sec ', 'pw': ' belau co ed go ne or ', 'ro': ' arts com firm info nom nt org rec store tm www ', 'rs': ' ac co edu gov in org ', 'sb': ' com edu gov net org ', 'sc': ' com edu gov net org ', 'sh': ' co com edu gov net nom org ', 'sl': ' com edu gov net org ', 'st': ' co com consulado edu embaixada gov mil net org principe saotome store ', 'sv': ' com edu gob org red ', 'sz': ' ac co org ', 'tr': ' av bbs bel biz com dr edu gen gov info k12 name net org pol tel tsk tv web ', 'tt': ' aero biz cat co com coop edu gov info int jobs mil mobi museum name net org pro tel travel ', 'tw': ' club com ebiz edu game gov idv mil net org ', 'mu': ' ac co com gov net or org ', 'mz': ' ac co edu gov org ', 'na': ' co com ', 'nz': ' ac co cri geek gen govt health iwi maori mil net org parliament school ', 'pa': ' abo ac com edu gob ing med net nom org sld ', 'pt': ' com edu gov int net nome org publ ', 'py': ' com edu gov mil net org ', 'qa': ' com edu gov mil net org ', 're': ' asso com nom ', 'ru': ' ac adygeya altai amur arkhangelsk astrakhan bashkiria belgorod bir bryansk buryatia cbg chel chelyabinsk chita chukotka chuvashia com dagestan e-burg edu gov grozny int irkutsk ivanovo izhevsk jar joshkar-ola kalmykia kaluga kamchatka karelia kazan kchr kemerovo khabarovsk khakassia khv kirov koenig komi kostroma kranoyarsk kuban kurgan kursk lipetsk magadan mari mari-el marine mil mordovia mosreg msk murmansk nalchik net nnov nov novosibirsk nsk omsk orenburg org oryol penza perm pp pskov ptz rnd ryazan sakhalin samara saratov simbirsk smolensk spb stavropol stv surgut tambov tatarstan tom tomsk tsaritsyn tsk tula tuva tver tyumen udm udmurtia ulan-ude vladikavkaz vladimir vladivostok volgograd vologda voronezh vrn vyatka yakutia yamal yekaterinburg yuzhno-sakhalinsk ', 'rw': ' ac co com edu gouv gov int mil net ', 'sa': ' com edu gov med net org pub sch ', 'sd': ' com edu gov info med net org tv ', 'se': ' a ac b bd c d e f g h i k l m n o org p parti pp press r s t tm u w x y z ', 'sg': ' com edu gov idn net org per ', 'sn': ' art com edu gouv org perso univ ', 'sy': ' com edu gov mil net news org ', 'th': ' ac co go in mi net or ', 'tj': ' ac biz co com edu go gov info int mil name net nic org test web ', 'tn': ' agrinet com defense edunet ens fin gov ind info intl mincom nat net org perso rnrt rns rnu tourism ', 'tz': ' ac co go ne or ', 'ua': ' biz cherkassy chernigov chernovtsy ck cn co com crimea cv dn dnepropetrovsk donetsk dp edu gov if in ivano-frankivsk kh kharkov kherson khmelnitskiy kiev kirovograd km kr ks kv lg lugansk lutsk lviv me mk net nikolaev od odessa org pl poltava pp rovno rv sebastopol sumy te ternopil uzhgorod vinnica vn zaporizhzhe zhitomir zp zt ', 'ug': ' ac co go ne or org sc ', 'uk': ' ac bl british-library co cym gov govt icnet jet lea ltd me mil mod national-library-scotland nel net nhs nic nls org orgn parliament plc police sch scot soc ', 'us': ' dni fed isa kids nsn ', 'uy': ' com edu gub mil net org ', 've': ' co com edu gob info mil net org web ', 'vi': ' co com k12 net org ', 'vn': ' ac biz com edu gov health info int name net org pro ', 'ye': ' co com gov ltd me net org plc ', 'yu': ' ac co edu gov org ', 'za': ' ac agric alt bourse city co cybernet db edu gov grondar iaccess imt inca landesign law mil net ngo nis nom olivetti org pix school tm web ', 'zm': ' ac co com edu gov net org sch ', // https://en.wikipedia.org/wiki/CentralNic#Second-level_domains 'com': 'ar br cn de eu gb gr hu jpn kr no qc ru sa se uk us uy za ', 'net': 'gb jp se uk ', 'org': 'ae', 'de': 'com ' }, // gorhill 2013-10-25: Using indexOf() instead Regexp(). Significant boost // in both performance and memory footprint. No initialization required. // http://jsperf.com/uri-js-sld-regex-vs-binary-search/4 // Following methods use lastIndexOf() rather than array.split() in order // to avoid any memory allocations. has: function (domain) { var tldOffset = domain.lastIndexOf('.'); if (tldOffset <= 0 || tldOffset >= domain.length - 1) { return false; } var sldOffset = domain.lastIndexOf('.', tldOffset - 1); if (sldOffset <= 0 || sldOffset >= tldOffset - 1) { return false; } var sldList = SLD.list[domain.slice(tldOffset + 1)]; if (!sldList) { return false; } return sldList.indexOf(' ' + domain.slice(sldOffset + 1, tldOffset) + ' ') >= 0; }, is: function (domain) { var tldOffset = domain.lastIndexOf('.'); if (tldOffset <= 0 || tldOffset >= domain.length - 1) { return false; } var sldOffset = domain.lastIndexOf('.', tldOffset - 1); if (sldOffset >= 0) { return false; } var sldList = SLD.list[domain.slice(tldOffset + 1)]; if (!sldList) { return false; } return sldList.indexOf(' ' + domain.slice(0, tldOffset) + ' ') >= 0; }, get: function (domain) { var tldOffset = domain.lastIndexOf('.'); if (tldOffset <= 0 || tldOffset >= domain.length - 1) { return null; } var sldOffset = domain.lastIndexOf('.', tldOffset - 1); if (sldOffset <= 0 || sldOffset >= tldOffset - 1) { return null; } var sldList = SLD.list[domain.slice(tldOffset + 1)]; if (!sldList) { return null; } if (sldList.indexOf(' ' + domain.slice(sldOffset + 1, tldOffset) + ' ') < 0) { return null; } return domain.slice(sldOffset + 1); }, noConflict: function () { if (root.SecondLevelDomains === this) { root.SecondLevelDomains = _SecondLevelDomains; } return this; } }; return SLD; }); /***/ }), /***/ 5215: /***/ (function(module, exports, __webpack_require__) { var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_DEFINE_RESULT__;/*! * URI.js - Mutating URLs * * Version: 1.19.11 * * Author: Rodney Rehm * Web: http://medialize.github.io/URI.js/ * * Licensed under * MIT License http://www.opensource.org/licenses/mit-license * */ (function (root, factory) { 'use strict'; // https://github.com/umdjs/umd/blob/master/returnExports.js if ( true && module.exports) { // Node module.exports = factory(__webpack_require__(7819), __webpack_require__(8677), __webpack_require__(9827)); } else if (true) { // AMD. Register as an anonymous module. !(__WEBPACK_AMD_DEFINE_ARRAY__ = [__webpack_require__(7819), __webpack_require__(8677), __webpack_require__(9827)], __WEBPACK_AMD_DEFINE_FACTORY__ = (factory), __WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ? (__WEBPACK_AMD_DEFINE_FACTORY__.apply(exports, __WEBPACK_AMD_DEFINE_ARRAY__)) : __WEBPACK_AMD_DEFINE_FACTORY__), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__)); } else {} })(this, function (punycode, IPv6, SLD, root) { 'use strict'; /*global location, escape, unescape */ // FIXME: v2.0.0 renamce non-camelCase properties to uppercase /*jshint camelcase: false */ // save current URI variable, if any var _URI = root && root.URI; function URI(url, base) { var _urlSupplied = arguments.length >= 1; var _baseSupplied = arguments.length >= 2; // Allow instantiation without the 'new' keyword if (!(this instanceof URI)) { if (_urlSupplied) { if (_baseSupplied) { return new URI(url, base); } return new URI(url); } return new URI(); } if (url === undefined) { if (_urlSupplied) { throw new TypeError('undefined is not a valid argument for URI'); } if (typeof location !== 'undefined') { url = location.href + ''; } else { url = ''; } } if (url === null) { if (_urlSupplied) { throw new TypeError('null is not a valid argument for URI'); } } this.href(url); // resolve to base according to http://dvcs.w3.org/hg/url/raw-file/tip/Overview.html#constructor if (base !== undefined) { return this.absoluteTo(base); } return this; } function isInteger(value) { return /^[0-9]+$/.test(value); } URI.version = '1.19.11'; var p = URI.prototype; var hasOwn = Object.prototype.hasOwnProperty; function escapeRegEx(string) { // https://github.com/medialize/URI.js/commit/85ac21783c11f8ccab06106dba9735a31a86924d#commitcomment-821963 return string.replace(/([.*+?^=!:${}()|[\]\/\\])/g, '\\$1'); } function getType(value) { // IE8 doesn't return [Object Undefined] but [Object Object] for undefined value if (value === undefined) { return 'Undefined'; } return String(Object.prototype.toString.call(value)).slice(8, -1); } function isArray(obj) { return getType(obj) === 'Array'; } function filterArrayValues(data, value) { var lookup = {}; var i, length; if (getType(value) === 'RegExp') { lookup = null; } else if (isArray(value)) { for (i = 0, length = value.length; i < length; i++) { lookup[value[i]] = true; } } else { lookup[value] = true; } for (i = 0, length = data.length; i < length; i++) { /*jshint laxbreak: true */ var _match = lookup && lookup[data[i]] !== undefined || !lookup && value.test(data[i]); /*jshint laxbreak: false */ if (_match) { data.splice(i, 1); length--; i--; } } return data; } function arrayContains(list, value) { var i, length; // value may be string, number, array, regexp if (isArray(value)) { // Note: this can be optimized to O(n) (instead of current O(m * n)) for (i = 0, length = value.length; i < length; i++) { if (!arrayContains(list, value[i])) { return false; } } return true; } var _type = getType(value); for (i = 0, length = list.length; i < length; i++) { if (_type === 'RegExp') { if (typeof list[i] === 'string' && list[i].match(value)) { return true; } } else if (list[i] === value) { return true; } } return false; } function arraysEqual(one, two) { if (!isArray(one) || !isArray(two)) { return false; } // arrays can't be equal if they have different amount of content if (one.length !== two.length) { return false; } one.sort(); two.sort(); for (var i = 0, l = one.length; i < l; i++) { if (one[i] !== two[i]) { return false; } } return true; } function trimSlashes(text) { var trim_expression = /^\/+|\/+$/g; return text.replace(trim_expression, ''); } URI._parts = function () { return { protocol: null, username: null, password: null, hostname: null, urn: null, port: null, path: null, query: null, fragment: null, // state preventInvalidHostname: URI.preventInvalidHostname, duplicateQueryParameters: URI.duplicateQueryParameters, escapeQuerySpace: URI.escapeQuerySpace }; }; // state: throw on invalid hostname // see https://github.com/medialize/URI.js/pull/345 // and https://github.com/medialize/URI.js/issues/354 URI.preventInvalidHostname = false; // state: allow duplicate query parameters (a=1&a=1) URI.duplicateQueryParameters = false; // state: replaces + with %20 (space in query strings) URI.escapeQuerySpace = true; // static properties URI.protocol_expression = /^[a-z][a-z0-9.+-]*$/i; URI.idn_expression = /[^a-z0-9\._-]/i; URI.punycode_expression = /(xn--)/i; // well, 333.444.555.666 matches, but it sure ain't no IPv4 - do we care? URI.ip4_expression = /^\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}$/; // credits to Rich Brown // source: http://forums.intermapper.com/viewtopic.php?p=1096#1096 // specification: http://www.ietf.org/rfc/rfc4291.txt URI.ip6_expression = /^\s*((([0-9A-Fa-f]{1,4}:){7}([0-9A-Fa-f]{1,4}|:))|(([0-9A-Fa-f]{1,4}:){6}(:[0-9A-Fa-f]{1,4}|((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3})|:))|(([0-9A-Fa-f]{1,4}:){5}(((:[0-9A-Fa-f]{1,4}){1,2})|:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3})|:))|(([0-9A-Fa-f]{1,4}:){4}(((:[0-9A-Fa-f]{1,4}){1,3})|((:[0-9A-Fa-f]{1,4})?:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){3}(((:[0-9A-Fa-f]{1,4}){1,4})|((:[0-9A-Fa-f]{1,4}){0,2}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){2}(((:[0-9A-Fa-f]{1,4}){1,5})|((:[0-9A-Fa-f]{1,4}){0,3}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){1}(((:[0-9A-Fa-f]{1,4}){1,6})|((:[0-9A-Fa-f]{1,4}){0,4}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(:(((:[0-9A-Fa-f]{1,4}){1,7})|((:[0-9A-Fa-f]{1,4}){0,5}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:)))(%.+)?\s*$/; // expression used is "gruber revised" (@gruber v2) determined to be the // best solution in a regex-golf we did a couple of ages ago at // * http://mathiasbynens.be/demo/url-regex // * http://rodneyrehm.de/t/url-regex.html URI.find_uri_expression = /\b((?:[a-z][\w-]+:(?:\/{1,3}|[a-z0-9%])|www\d{0,3}[.]|[a-z0-9.\-]+[.][a-z]{2,4}\/)(?:[^\s()<>]+|\(([^\s()<>]+|(\([^\s()<>]+\)))*\))+(?:\(([^\s()<>]+|(\([^\s()<>]+\)))*\)|[^\s`!()\[\]{};:'".,<>?«»“”‘’]))/ig; URI.findUri = { // valid "scheme://" or "www." start: /\b(?:([a-z][a-z0-9.+-]*:\/\/)|www\.)/gi, // everything up to the next whitespace end: /[\s\r\n]|$/, // trim trailing punctuation captured by end RegExp trim: /[`!()\[\]{};:'".,<>?«»“”„‘’]+$/, // balanced parens inclusion (), [], {}, <> parens: /(\([^\)]*\)|\[[^\]]*\]|\{[^}]*\}|<[^>]*>)/g }; URI.leading_whitespace_expression = /^[\x00-\x20\u00a0\u1680\u2000-\u200a\u2028\u2029\u202f\u205f\u3000\ufeff]+/; // https://infra.spec.whatwg.org/#ascii-tab-or-newline URI.ascii_tab_whitespace = /[\u0009\u000A\u000D]+/g; // http://www.iana.org/assignments/uri-schemes.html // http://en.wikipedia.org/wiki/List_of_TCP_and_UDP_port_numbers#Well-known_ports URI.defaultPorts = { http: '80', https: '443', ftp: '21', gopher: '70', ws: '80', wss: '443' }; // list of protocols which always require a hostname URI.hostProtocols = ['http', 'https']; // allowed hostname characters according to RFC 3986 // ALPHA DIGIT "-" "." "_" "~" "!" "$" "&" "'" "(" ")" "*" "+" "," ";" "=" %encoded // I've never seen a (non-IDN) hostname other than: ALPHA DIGIT . - _ URI.invalid_hostname_characters = /[^a-zA-Z0-9\.\-:_]/; // map DOM Elements to their URI attribute URI.domAttributes = { 'a': 'href', 'blockquote': 'cite', 'link': 'href', 'base': 'href', 'script': 'src', 'form': 'action', 'img': 'src', 'area': 'href', 'iframe': 'src', 'embed': 'src', 'source': 'src', 'track': 'src', 'input': 'src', // but only if type="image" 'audio': 'src', 'video': 'src' }; URI.getDomAttribute = function (node) { if (!node || !node.nodeName) { return undefined; } var nodeName = node.nodeName.toLowerCase(); // should only expose src for type="image" if (nodeName === 'input' && node.type !== 'image') { return undefined; } return URI.domAttributes[nodeName]; }; function escapeForDumbFirefox36(value) { // https://github.com/medialize/URI.js/issues/91 return escape(value); } // encoding / decoding according to RFC3986 function strictEncodeURIComponent(string) { // see https://developer.mozilla.org/en-US/docs/JavaScript/Reference/Global_Objects/encodeURIComponent return encodeURIComponent(string).replace(/[!'()*]/g, escapeForDumbFirefox36).replace(/\*/g, '%2A'); } URI.encode = strictEncodeURIComponent; URI.decode = decodeURIComponent; URI.iso8859 = function () { URI.encode = escape; URI.decode = unescape; }; URI.unicode = function () { URI.encode = strictEncodeURIComponent; URI.decode = decodeURIComponent; }; URI.characters = { pathname: { encode: { // RFC3986 2.1: For consistency, URI producers and normalizers should // use uppercase hexadecimal digits for all percent-encodings. expression: /%(24|26|2B|2C|3B|3D|3A|40)/ig, map: { // -._~!'()* '%24': '$', '%26': '&', '%2B': '+', '%2C': ',', '%3B': ';', '%3D': '=', '%3A': ':', '%40': '@' } }, decode: { expression: /[\/\?#]/g, map: { '/': '%2F', '?': '%3F', '#': '%23' } } }, reserved: { encode: { // RFC3986 2.1: For consistency, URI producers and normalizers should // use uppercase hexadecimal digits for all percent-encodings. expression: /%(21|23|24|26|27|28|29|2A|2B|2C|2F|3A|3B|3D|3F|40|5B|5D)/ig, map: { // gen-delims '%3A': ':', '%2F': '/', '%3F': '?', '%23': '#', '%5B': '[', '%5D': ']', '%40': '@', // sub-delims '%21': '!', '%24': '$', '%26': '&', '%27': '\'', '%28': '(', '%29': ')', '%2A': '*', '%2B': '+', '%2C': ',', '%3B': ';', '%3D': '=' } } }, urnpath: { // The characters under `encode` are the characters called out by RFC 2141 as being acceptable // for usage in a URN. RFC2141 also calls out "-", ".", and "_" as acceptable characters, but // these aren't encoded by encodeURIComponent, so we don't have to call them out here. Also // note that the colon character is not featured in the encoding map; this is because URI.js // gives the colons in URNs semantic meaning as the delimiters of path segements, and so it // should not appear unencoded in a segment itself. // See also the note above about RFC3986 and capitalalized hex digits. encode: { expression: /%(21|24|27|28|29|2A|2B|2C|3B|3D|40)/ig, map: { '%21': '!', '%24': '$', '%27': '\'', '%28': '(', '%29': ')', '%2A': '*', '%2B': '+', '%2C': ',', '%3B': ';', '%3D': '=', '%40': '@' } }, // These characters are the characters called out by RFC2141 as "reserved" characters that // should never appear in a URN, plus the colon character (see note above). decode: { expression: /[\/\?#:]/g, map: { '/': '%2F', '?': '%3F', '#': '%23', ':': '%3A' } } } }; URI.encodeQuery = function (string, escapeQuerySpace) { var escaped = URI.encode(string + ''); if (escapeQuerySpace === undefined) { escapeQuerySpace = URI.escapeQuerySpace; } return escapeQuerySpace ? escaped.replace(/%20/g, '+') : escaped; }; URI.decodeQuery = function (string, escapeQuerySpace) { string += ''; if (escapeQuerySpace === undefined) { escapeQuerySpace = URI.escapeQuerySpace; } try { return URI.decode(escapeQuerySpace ? string.replace(/\+/g, '%20') : string); } catch (e) { // we're not going to mess with weird encodings, // give up and return the undecoded original string // see https://github.com/medialize/URI.js/issues/87 // see https://github.com/medialize/URI.js/issues/92 return string; } }; // generate encode/decode path functions var _parts = { 'encode': 'encode', 'decode': 'decode' }; var _part; var generateAccessor = function (_group, _part) { return function (string) { try { return URI[_part](string + '').replace(URI.characters[_group][_part].expression, function (c) { return URI.characters[_group][_part].map[c]; }); } catch (e) { // we're not going to mess with weird encodings, // give up and return the undecoded original string // see https://github.com/medialize/URI.js/issues/87 // see https://github.com/medialize/URI.js/issues/92 return string; } }; }; for (_part in _parts) { URI[_part + 'PathSegment'] = generateAccessor('pathname', _parts[_part]); URI[_part + 'UrnPathSegment'] = generateAccessor('urnpath', _parts[_part]); } var generateSegmentedPathFunction = function (_sep, _codingFuncName, _innerCodingFuncName) { return function (string) { // Why pass in names of functions, rather than the function objects themselves? The // definitions of some functions (but in particular, URI.decode) will occasionally change due // to URI.js having ISO8859 and Unicode modes. Passing in the name and getting it will ensure // that the functions we use here are "fresh". var actualCodingFunc; if (!_innerCodingFuncName) { actualCodingFunc = URI[_codingFuncName]; } else { actualCodingFunc = function (string) { return URI[_codingFuncName](URI[_innerCodingFuncName](string)); }; } var segments = (string + '').split(_sep); for (var i = 0, length = segments.length; i < length; i++) { segments[i] = actualCodingFunc(segments[i]); } return segments.join(_sep); }; }; // This takes place outside the above loop because we don't want, e.g., encodeUrnPath functions. URI.decodePath = generateSegmentedPathFunction('/', 'decodePathSegment'); URI.decodeUrnPath = generateSegmentedPathFunction(':', 'decodeUrnPathSegment'); URI.recodePath = generateSegmentedPathFunction('/', 'encodePathSegment', 'decode'); URI.recodeUrnPath = generateSegmentedPathFunction(':', 'encodeUrnPathSegment', 'decode'); URI.encodeReserved = generateAccessor('reserved', 'encode'); URI.parse = function (string, parts) { var pos; if (!parts) { parts = { preventInvalidHostname: URI.preventInvalidHostname }; } string = string.replace(URI.leading_whitespace_expression, ''); // https://infra.spec.whatwg.org/#ascii-tab-or-newline string = string.replace(URI.ascii_tab_whitespace, ''); // [protocol"://"[username[":"password]"@"]hostname[":"port]"/"?][path]["?"querystring]["#"fragment] // extract fragment pos = string.indexOf('#'); if (pos > -1) { // escaping? parts.fragment = string.substring(pos + 1) || null; string = string.substring(0, pos); } // extract query pos = string.indexOf('?'); if (pos > -1) { // escaping? parts.query = string.substring(pos + 1) || null; string = string.substring(0, pos); } // slashes and backslashes have lost all meaning for the web protocols (https, http, wss, ws) string = string.replace(/^(https?|ftp|wss?)?:+[/\\]*/i, '$1://'); // slashes and backslashes have lost all meaning for scheme relative URLs string = string.replace(/^[/\\]{2,}/i, '//'); // extract protocol if (string.substring(0, 2) === '//') { // relative-scheme parts.protocol = null; string = string.substring(2); // extract "user:pass@host:port" string = URI.parseAuthority(string, parts); } else { pos = string.indexOf(':'); if (pos > -1) { parts.protocol = string.substring(0, pos) || null; if (parts.protocol && !parts.protocol.match(URI.protocol_expression)) { // : may be within the path parts.protocol = undefined; } else if (string.substring(pos + 1, pos + 3).replace(/\\/g, '/') === '//') { string = string.substring(pos + 3); // extract "user:pass@host:port" string = URI.parseAuthority(string, parts); } else { string = string.substring(pos + 1); parts.urn = true; } } } // what's left must be the path parts.path = string; // and we're done return parts; }; URI.parseHost = function (string, parts) { if (!string) { string = ''; } // Copy chrome, IE, opera backslash-handling behavior. // Back slashes before the query string get converted to forward slashes // See: https://github.com/joyent/node/blob/386fd24f49b0e9d1a8a076592a404168faeecc34/lib/url.js#L115-L124 // See: https://code.google.com/p/chromium/issues/detail?id=25916 // https://github.com/medialize/URI.js/pull/233 string = string.replace(/\\/g, '/'); // extract host:port var pos = string.indexOf('/'); var bracketPos; var t; if (pos === -1) { pos = string.length; } if (string.charAt(0) === '[') { // IPv6 host - http://tools.ietf.org/html/draft-ietf-6man-text-addr-representation-04#section-6 // I claim most client software breaks on IPv6 anyways. To simplify things, URI only accepts // IPv6+port in the format [2001:db8::1]:80 (for the time being) bracketPos = string.indexOf(']'); parts.hostname = string.substring(1, bracketPos) || null; parts.port = string.substring(bracketPos + 2, pos) || null; if (parts.port === '/') { parts.port = null; } } else { var firstColon = string.indexOf(':'); var firstSlash = string.indexOf('/'); var nextColon = string.indexOf(':', firstColon + 1); if (nextColon !== -1 && (firstSlash === -1 || nextColon < firstSlash)) { // IPv6 host contains multiple colons - but no port // this notation is actually not allowed by RFC 3986, but we're a liberal parser parts.hostname = string.substring(0, pos) || null; parts.port = null; } else { t = string.substring(0, pos).split(':'); parts.hostname = t[0] || null; parts.port = t[1] || null; } } if (parts.hostname && string.substring(pos).charAt(0) !== '/') { pos++; string = '/' + string; } if (parts.preventInvalidHostname) { URI.ensureValidHostname(parts.hostname, parts.protocol); } if (parts.port) { URI.ensureValidPort(parts.port); } return string.substring(pos) || '/'; }; URI.parseAuthority = function (string, parts) { string = URI.parseUserinfo(string, parts); return URI.parseHost(string, parts); }; URI.parseUserinfo = function (string, parts) { // extract username:password var _string = string; var firstBackSlash = string.indexOf('\\'); if (firstBackSlash !== -1) { string = string.replace(/\\/g, '/'); } var firstSlash = string.indexOf('/'); var pos = string.lastIndexOf('@', firstSlash > -1 ? firstSlash : string.length - 1); var t; // authority@ must come before /path or \path if (pos > -1 && (firstSlash === -1 || pos < firstSlash)) { t = string.substring(0, pos).split(':'); parts.username = t[0] ? URI.decode(t[0]) : null; t.shift(); parts.password = t[0] ? URI.decode(t.join(':')) : null; string = _string.substring(pos + 1); } else { parts.username = null; parts.password = null; } return string; }; URI.parseQuery = function (string, escapeQuerySpace) { if (!string) { return {}; } // throw out the funky business - "?"[name"="value"&"]+ string = string.replace(/&+/g, '&').replace(/^\?*&*|&+$/g, ''); if (!string) { return {}; } var items = {}; var splits = string.split('&'); var length = splits.length; var v, name, value; for (var i = 0; i < length; i++) { v = splits[i].split('='); name = URI.decodeQuery(v.shift(), escapeQuerySpace); // no "=" is null according to http://dvcs.w3.org/hg/url/raw-file/tip/Overview.html#collect-url-parameters value = v.length ? URI.decodeQuery(v.join('='), escapeQuerySpace) : null; if (name === '__proto__') { // ignore attempt at exploiting JavaScript internals continue; } else if (hasOwn.call(items, name)) { if (typeof items[name] === 'string' || items[name] === null) { items[name] = [items[name]]; } items[name].push(value); } else { items[name] = value; } } return items; }; URI.build = function (parts) { var t = ''; var requireAbsolutePath = false; if (parts.protocol) { t += parts.protocol + ':'; } if (!parts.urn && (t || parts.hostname)) { t += '//'; requireAbsolutePath = true; } t += URI.buildAuthority(parts) || ''; if (typeof parts.path === 'string') { if (parts.path.charAt(0) !== '/' && requireAbsolutePath) { t += '/'; } t += parts.path; } if (typeof parts.query === 'string' && parts.query) { t += '?' + parts.query; } if (typeof parts.fragment === 'string' && parts.fragment) { t += '#' + parts.fragment; } return t; }; URI.buildHost = function (parts) { var t = ''; if (!parts.hostname) { return ''; } else if (URI.ip6_expression.test(parts.hostname)) { t += '[' + parts.hostname + ']'; } else { t += parts.hostname; } if (parts.port) { t += ':' + parts.port; } return t; }; URI.buildAuthority = function (parts) { return URI.buildUserinfo(parts) + URI.buildHost(parts); }; URI.buildUserinfo = function (parts) { var t = ''; if (parts.username) { t += URI.encode(parts.username); } if (parts.password) { t += ':' + URI.encode(parts.password); } if (t) { t += '@'; } return t; }; URI.buildQuery = function (data, duplicateQueryParameters, escapeQuerySpace) { // according to http://tools.ietf.org/html/rfc3986 or http://labs.apache.org/webarch/uri/rfc/rfc3986.html // being »-._~!$&'()*+,;=:@/?« %HEX and alnum are allowed // the RFC explicitly states ?/foo being a valid use case, no mention of parameter syntax! // URI.js treats the query string as being application/x-www-form-urlencoded // see http://www.w3.org/TR/REC-html40/interact/forms.html#form-content-type var t = ''; var unique, key, i, length; for (key in data) { if (key === '__proto__') { // ignore attempt at exploiting JavaScript internals continue; } else if (hasOwn.call(data, key)) { if (isArray(data[key])) { unique = {}; for (i = 0, length = data[key].length; i < length; i++) { if (data[key][i] !== undefined && unique[data[key][i] + ''] === undefined) { t += '&' + URI.buildQueryParameter(key, data[key][i], escapeQuerySpace); if (duplicateQueryParameters !== true) { unique[data[key][i] + ''] = true; } } } } else if (data[key] !== undefined) { t += '&' + URI.buildQueryParameter(key, data[key], escapeQuerySpace); } } } return t.substring(1); }; URI.buildQueryParameter = function (name, value, escapeQuerySpace) { // http://www.w3.org/TR/REC-html40/interact/forms.html#form-content-type -- application/x-www-form-urlencoded // don't append "=" for null values, according to http://dvcs.w3.org/hg/url/raw-file/tip/Overview.html#url-parameter-serialization return URI.encodeQuery(name, escapeQuerySpace) + (value !== null ? '=' + URI.encodeQuery(value, escapeQuerySpace) : ''); }; URI.addQuery = function (data, name, value) { if (typeof name === 'object') { for (var key in name) { if (hasOwn.call(name, key)) { URI.addQuery(data, key, name[key]); } } } else if (typeof name === 'string') { if (data[name] === undefined) { data[name] = value; return; } else if (typeof data[name] === 'string') { data[name] = [data[name]]; } if (!isArray(value)) { value = [value]; } data[name] = (data[name] || []).concat(value); } else { throw new TypeError('URI.addQuery() accepts an object, string as the name parameter'); } }; URI.setQuery = function (data, name, value) { if (typeof name === 'object') { for (var key in name) { if (hasOwn.call(name, key)) { URI.setQuery(data, key, name[key]); } } } else if (typeof name === 'string') { data[name] = value === undefined ? null : value; } else { throw new TypeError('URI.setQuery() accepts an object, string as the name parameter'); } }; URI.removeQuery = function (data, name, value) { var i, length, key; if (isArray(name)) { for (i = 0, length = name.length; i < length; i++) { data[name[i]] = undefined; } } else if (getType(name) === 'RegExp') { for (key in data) { if (name.test(key)) { data[key] = undefined; } } } else if (typeof name === 'object') { for (key in name) { if (hasOwn.call(name, key)) { URI.removeQuery(data, key, name[key]); } } } else if (typeof name === 'string') { if (value !== undefined) { if (getType(value) === 'RegExp') { if (!isArray(data[name]) && value.test(data[name])) { data[name] = undefined; } else { data[name] = filterArrayValues(data[name], value); } } else if (data[name] === String(value) && (!isArray(value) || value.length === 1)) { data[name] = undefined; } else if (isArray(data[name])) { data[name] = filterArrayValues(data[name], value); } } else { data[name] = undefined; } } else { throw new TypeError('URI.removeQuery() accepts an object, string, RegExp as the first parameter'); } }; URI.hasQuery = function (data, name, value, withinArray) { switch (getType(name)) { case 'String': // Nothing to do here break; case 'RegExp': for (var key in data) { if (hasOwn.call(data, key)) { if (name.test(key) && (value === undefined || URI.hasQuery(data, key, value))) { return true; } } } return false; case 'Object': for (var _key in name) { if (hasOwn.call(name, _key)) { if (!URI.hasQuery(data, _key, name[_key])) { return false; } } } return true; default: throw new TypeError('URI.hasQuery() accepts a string, regular expression or object as the name parameter'); } switch (getType(value)) { case 'Undefined': // true if exists (but may be empty) return name in data; // data[name] !== undefined; case 'Boolean': // true if exists and non-empty var _booly = Boolean(isArray(data[name]) ? data[name].length : data[name]); return value === _booly; case 'Function': // allow complex comparison return !!value(data[name], name, data); case 'Array': if (!isArray(data[name])) { return false; } var op = withinArray ? arrayContains : arraysEqual; return op(data[name], value); case 'RegExp': if (!isArray(data[name])) { return Boolean(data[name] && data[name].match(value)); } if (!withinArray) { return false; } return arrayContains(data[name], value); case 'Number': value = String(value); /* falls through */ case 'String': if (!isArray(data[name])) { return data[name] === value; } if (!withinArray) { return false; } return arrayContains(data[name], value); default: throw new TypeError('URI.hasQuery() accepts undefined, boolean, string, number, RegExp, Function as the value parameter'); } }; URI.joinPaths = function () { var input = []; var segments = []; var nonEmptySegments = 0; for (var i = 0; i < arguments.length; i++) { var url = new URI(arguments[i]); input.push(url); var _segments = url.segment(); for (var s = 0; s < _segments.length; s++) { if (typeof _segments[s] === 'string') { segments.push(_segments[s]); } if (_segments[s]) { nonEmptySegments++; } } } if (!segments.length || !nonEmptySegments) { return new URI(''); } var uri = new URI('').segment(segments); if (input[0].path() === '' || input[0].path().slice(0, 1) === '/') { uri.path('/' + uri.path()); } return uri.normalize(); }; URI.commonPath = function (one, two) { var length = Math.min(one.length, two.length); var pos; // find first non-matching character for (pos = 0; pos < length; pos++) { if (one.charAt(pos) !== two.charAt(pos)) { pos--; break; } } if (pos < 1) { return one.charAt(0) === two.charAt(0) && one.charAt(0) === '/' ? '/' : ''; } // revert to last / if (one.charAt(pos) !== '/' || two.charAt(pos) !== '/') { pos = one.substring(0, pos).lastIndexOf('/'); } return one.substring(0, pos + 1); }; URI.withinString = function (string, callback, options) { options || (options = {}); var _start = options.start || URI.findUri.start; var _end = options.end || URI.findUri.end; var _trim = options.trim || URI.findUri.trim; var _parens = options.parens || URI.findUri.parens; var _attributeOpen = /[a-z0-9-]=["']?$/i; _start.lastIndex = 0; while (true) { var match = _start.exec(string); if (!match) { break; } var start = match.index; if (options.ignoreHtml) { // attribut(e=["']?$) var attributeOpen = string.slice(Math.max(start - 3, 0), start); if (attributeOpen && _attributeOpen.test(attributeOpen)) { continue; } } var end = start + string.slice(start).search(_end); var slice = string.slice(start, end); // make sure we include well balanced parens var parensEnd = -1; while (true) { var parensMatch = _parens.exec(slice); if (!parensMatch) { break; } var parensMatchEnd = parensMatch.index + parensMatch[0].length; parensEnd = Math.max(parensEnd, parensMatchEnd); } if (parensEnd > -1) { slice = slice.slice(0, parensEnd) + slice.slice(parensEnd).replace(_trim, ''); } else { slice = slice.replace(_trim, ''); } if (slice.length <= match[0].length) { // the extract only contains the starting marker of a URI, // e.g. "www" or "http://" continue; } if (options.ignore && options.ignore.test(slice)) { continue; } end = start + slice.length; var result = callback(slice, start, end, string); if (result === undefined) { _start.lastIndex = end; continue; } result = String(result); string = string.slice(0, start) + result + string.slice(end); _start.lastIndex = start + result.length; } _start.lastIndex = 0; return string; }; URI.ensureValidHostname = function (v, protocol) { // Theoretically URIs allow percent-encoding in Hostnames (according to RFC 3986) // they are not part of DNS and therefore ignored by URI.js var hasHostname = !!v; // not null and not an empty string var hasProtocol = !!protocol; var rejectEmptyHostname = false; if (hasProtocol) { rejectEmptyHostname = arrayContains(URI.hostProtocols, protocol); } if (rejectEmptyHostname && !hasHostname) { throw new TypeError('Hostname cannot be empty, if protocol is ' + protocol); } else if (v && v.match(URI.invalid_hostname_characters)) { // test punycode if (!punycode) { throw new TypeError('Hostname "' + v + '" contains characters other than [A-Z0-9.-:_] and Punycode.js is not available'); } if (punycode.toASCII(v).match(URI.invalid_hostname_characters)) { throw new TypeError('Hostname "' + v + '" contains characters other than [A-Z0-9.-:_]'); } } }; URI.ensureValidPort = function (v) { if (!v) { return; } var port = Number(v); if (isInteger(port) && port > 0 && port < 65536) { return; } throw new TypeError('Port "' + v + '" is not a valid port'); }; // noConflict URI.noConflict = function (removeAll) { if (removeAll) { var unconflicted = { URI: this.noConflict() }; if (root.URITemplate && typeof root.URITemplate.noConflict === 'function') { unconflicted.URITemplate = root.URITemplate.noConflict(); } if (root.IPv6 && typeof root.IPv6.noConflict === 'function') { unconflicted.IPv6 = root.IPv6.noConflict(); } if (root.SecondLevelDomains && typeof root.SecondLevelDomains.noConflict === 'function') { unconflicted.SecondLevelDomains = root.SecondLevelDomains.noConflict(); } return unconflicted; } else if (root.URI === this) { root.URI = _URI; } return this; }; p.build = function (deferBuild) { if (deferBuild === true) { this._deferred_build = true; } else if (deferBuild === undefined || this._deferred_build) { this._string = URI.build(this._parts); this._deferred_build = false; } return this; }; p.clone = function () { return new URI(this); }; p.valueOf = p.toString = function () { return this.build(false)._string; }; function generateSimpleAccessor(_part) { return function (v, build) { if (v === undefined) { return this._parts[_part] || ''; } else { this._parts[_part] = v || null; this.build(!build); return this; } }; } function generatePrefixAccessor(_part, _key) { return function (v, build) { if (v === undefined) { return this._parts[_part] || ''; } else { if (v !== null) { v = v + ''; if (v.charAt(0) === _key) { v = v.substring(1); } } this._parts[_part] = v; this.build(!build); return this; } }; } p.protocol = generateSimpleAccessor('protocol'); p.username = generateSimpleAccessor('username'); p.password = generateSimpleAccessor('password'); p.hostname = generateSimpleAccessor('hostname'); p.port = generateSimpleAccessor('port'); p.query = generatePrefixAccessor('query', '?'); p.fragment = generatePrefixAccessor('fragment', '#'); p.search = function (v, build) { var t = this.query(v, build); return typeof t === 'string' && t.length ? '?' + t : t; }; p.hash = function (v, build) { var t = this.fragment(v, build); return typeof t === 'string' && t.length ? '#' + t : t; }; p.pathname = function (v, build) { if (v === undefined || v === true) { var res = this._parts.path || (this._parts.hostname ? '/' : ''); return v ? (this._parts.urn ? URI.decodeUrnPath : URI.decodePath)(res) : res; } else { if (this._parts.urn) { this._parts.path = v ? URI.recodeUrnPath(v) : ''; } else { this._parts.path = v ? URI.recodePath(v) : '/'; } this.build(!build); return this; } }; p.path = p.pathname; p.href = function (href, build) { var key; if (href === undefined) { return this.toString(); } this._string = ''; this._parts = URI._parts(); var _URI = href instanceof URI; var _object = typeof href === 'object' && (href.hostname || href.path || href.pathname); if (href.nodeName) { var attribute = URI.getDomAttribute(href); href = href[attribute] || ''; _object = false; } // window.location is reported to be an object, but it's not the sort // of object we're looking for: // * location.protocol ends with a colon // * location.query != object.search // * location.hash != object.fragment // simply serializing the unknown object should do the trick // (for location, not for everything...) if (!_URI && _object && href.pathname !== undefined) { href = href.toString(); } if (typeof href === 'string' || href instanceof String) { this._parts = URI.parse(String(href), this._parts); } else if (_URI || _object) { var src = _URI ? href._parts : href; for (key in src) { if (key === 'query') { continue; } if (hasOwn.call(this._parts, key)) { this._parts[key] = src[key]; } } if (src.query) { this.query(src.query, false); } } else { throw new TypeError('invalid input'); } this.build(!build); return this; }; // identification accessors p.is = function (what) { var ip = false; var ip4 = false; var ip6 = false; var name = false; var sld = false; var idn = false; var punycode = false; var relative = !this._parts.urn; if (this._parts.hostname) { relative = false; ip4 = URI.ip4_expression.test(this._parts.hostname); ip6 = URI.ip6_expression.test(this._parts.hostname); ip = ip4 || ip6; name = !ip; sld = name && SLD && SLD.has(this._parts.hostname); idn = name && URI.idn_expression.test(this._parts.hostname); punycode = name && URI.punycode_expression.test(this._parts.hostname); } switch (what.toLowerCase()) { case 'relative': return relative; case 'absolute': return !relative; // hostname identification case 'domain': case 'name': return name; case 'sld': return sld; case 'ip': return ip; case 'ip4': case 'ipv4': case 'inet4': return ip4; case 'ip6': case 'ipv6': case 'inet6': return ip6; case 'idn': return idn; case 'url': return !this._parts.urn; case 'urn': return !!this._parts.urn; case 'punycode': return punycode; } return null; }; // component specific input validation var _protocol = p.protocol; var _port = p.port; var _hostname = p.hostname; p.protocol = function (v, build) { if (v) { // accept trailing :// v = v.replace(/:(\/\/)?$/, ''); if (!v.match(URI.protocol_expression)) { throw new TypeError('Protocol "' + v + '" contains characters other than [A-Z0-9.+-] or doesn\'t start with [A-Z]'); } } return _protocol.call(this, v, build); }; p.scheme = p.protocol; p.port = function (v, build) { if (this._parts.urn) { return v === undefined ? '' : this; } if (v !== undefined) { if (v === 0) { v = null; } if (v) { v += ''; if (v.charAt(0) === ':') { v = v.substring(1); } URI.ensureValidPort(v); } } return _port.call(this, v, build); }; p.hostname = function (v, build) { if (this._parts.urn) { return v === undefined ? '' : this; } if (v !== undefined) { var x = { preventInvalidHostname: this._parts.preventInvalidHostname }; var res = URI.parseHost(v, x); if (res !== '/') { throw new TypeError('Hostname "' + v + '" contains characters other than [A-Z0-9.-]'); } v = x.hostname; if (this._parts.preventInvalidHostname) { URI.ensureValidHostname(v, this._parts.protocol); } } return _hostname.call(this, v, build); }; // compound accessors p.origin = function (v, build) { if (this._parts.urn) { return v === undefined ? '' : this; } if (v === undefined) { var protocol = this.protocol(); var authority = this.authority(); if (!authority) { return ''; } return (protocol ? protocol + '://' : '') + this.authority(); } else { var origin = URI(v); this.protocol(origin.protocol()).authority(origin.authority()).build(!build); return this; } }; p.host = function (v, build) { if (this._parts.urn) { return v === undefined ? '' : this; } if (v === undefined) { return this._parts.hostname ? URI.buildHost(this._parts) : ''; } else { var res = URI.parseHost(v, this._parts); if (res !== '/') { throw new TypeError('Hostname "' + v + '" contains characters other than [A-Z0-9.-]'); } this.build(!build); return this; } }; p.authority = function (v, build) { if (this._parts.urn) { return v === undefined ? '' : this; } if (v === undefined) { return this._parts.hostname ? URI.buildAuthority(this._parts) : ''; } else { var res = URI.parseAuthority(v, this._parts); if (res !== '/') { throw new TypeError('Hostname "' + v + '" contains characters other than [A-Z0-9.-]'); } this.build(!build); return this; } }; p.userinfo = function (v, build) { if (this._parts.urn) { return v === undefined ? '' : this; } if (v === undefined) { var t = URI.buildUserinfo(this._parts); return t ? t.substring(0, t.length - 1) : t; } else { if (v[v.length - 1] !== '@') { v += '@'; } URI.parseUserinfo(v, this._parts); this.build(!build); return this; } }; p.resource = function (v, build) { var parts; if (v === undefined) { return this.path() + this.search() + this.hash(); } parts = URI.parse(v); this._parts.path = parts.path; this._parts.query = parts.query; this._parts.fragment = parts.fragment; this.build(!build); return this; }; // fraction accessors p.subdomain = function (v, build) { if (this._parts.urn) { return v === undefined ? '' : this; } // convenience, return "www" from "www.example.org" if (v === undefined) { if (!this._parts.hostname || this.is('IP')) { return ''; } // grab domain and add another segment var end = this._parts.hostname.length - this.domain().length - 1; return this._parts.hostname.substring(0, end) || ''; } else { var e = this._parts.hostname.length - this.domain().length; var sub = this._parts.hostname.substring(0, e); var replace = new RegExp('^' + escapeRegEx(sub)); if (v && v.charAt(v.length - 1) !== '.') { v += '.'; } if (v.indexOf(':') !== -1) { throw new TypeError('Domains cannot contain colons'); } if (v) { URI.ensureValidHostname(v, this._parts.protocol); } this._parts.hostname = this._parts.hostname.replace(replace, v); this.build(!build); return this; } }; p.domain = function (v, build) { if (this._parts.urn) { return v === undefined ? '' : this; } if (typeof v === 'boolean') { build = v; v = undefined; } // convenience, return "example.org" from "www.example.org" if (v === undefined) { if (!this._parts.hostname || this.is('IP')) { return ''; } // if hostname consists of 1 or 2 segments, it must be the domain var t = this._parts.hostname.match(/\./g); if (t && t.length < 2) { return this._parts.hostname; } // grab tld and add another segment var end = this._parts.hostname.length - this.tld(build).length - 1; end = this._parts.hostname.lastIndexOf('.', end - 1) + 1; return this._parts.hostname.substring(end) || ''; } else { if (!v) { throw new TypeError('cannot set domain empty'); } if (v.indexOf(':') !== -1) { throw new TypeError('Domains cannot contain colons'); } URI.ensureValidHostname(v, this._parts.protocol); if (!this._parts.hostname || this.is('IP')) { this._parts.hostname = v; } else { var replace = new RegExp(escapeRegEx(this.domain()) + '$'); this._parts.hostname = this._parts.hostname.replace(replace, v); } this.build(!build); return this; } }; p.tld = function (v, build) { if (this._parts.urn) { return v === undefined ? '' : this; } if (typeof v === 'boolean') { build = v; v = undefined; } // return "org" from "www.example.org" if (v === undefined) { if (!this._parts.hostname || this.is('IP')) { return ''; } var pos = this._parts.hostname.lastIndexOf('.'); var tld = this._parts.hostname.substring(pos + 1); if (build !== true && SLD && SLD.list[tld.toLowerCase()]) { return SLD.get(this._parts.hostname) || tld; } return tld; } else { var replace; if (!v) { throw new TypeError('cannot set TLD empty'); } else if (v.match(/[^a-zA-Z0-9-]/)) { if (SLD && SLD.is(v)) { replace = new RegExp(escapeRegEx(this.tld()) + '$'); this._parts.hostname = this._parts.hostname.replace(replace, v); } else { throw new TypeError('TLD "' + v + '" contains characters other than [A-Z0-9]'); } } else if (!this._parts.hostname || this.is('IP')) { throw new ReferenceError('cannot set TLD on non-domain host'); } else { replace = new RegExp(escapeRegEx(this.tld()) + '$'); this._parts.hostname = this._parts.hostname.replace(replace, v); } this.build(!build); return this; } }; p.directory = function (v, build) { if (this._parts.urn) { return v === undefined ? '' : this; } if (v === undefined || v === true) { if (!this._parts.path && !this._parts.hostname) { return ''; } if (this._parts.path === '/') { return '/'; } var end = this._parts.path.length - this.filename().length - 1; var res = this._parts.path.substring(0, end) || (this._parts.hostname ? '/' : ''); return v ? URI.decodePath(res) : res; } else { var e = this._parts.path.length - this.filename().length; var directory = this._parts.path.substring(0, e); var replace = new RegExp('^' + escapeRegEx(directory)); // fully qualifier directories begin with a slash if (!this.is('relative')) { if (!v) { v = '/'; } if (v.charAt(0) !== '/') { v = '/' + v; } } // directories always end with a slash if (v && v.charAt(v.length - 1) !== '/') { v += '/'; } v = URI.recodePath(v); this._parts.path = this._parts.path.replace(replace, v); this.build(!build); return this; } }; p.filename = function (v, build) { if (this._parts.urn) { return v === undefined ? '' : this; } if (typeof v !== 'string') { if (!this._parts.path || this._parts.path === '/') { return ''; } var pos = this._parts.path.lastIndexOf('/'); var res = this._parts.path.substring(pos + 1); return v ? URI.decodePathSegment(res) : res; } else { var mutatedDirectory = false; if (v.charAt(0) === '/') { v = v.substring(1); } if (v.match(/\.?\//)) { mutatedDirectory = true; } var replace = new RegExp(escapeRegEx(this.filename()) + '$'); v = URI.recodePath(v); this._parts.path = this._parts.path.replace(replace, v); if (mutatedDirectory) { this.normalizePath(build); } else { this.build(!build); } return this; } }; p.suffix = function (v, build) { if (this._parts.urn) { return v === undefined ? '' : this; } if (v === undefined || v === true) { if (!this._parts.path || this._parts.path === '/') { return ''; } var filename = this.filename(); var pos = filename.lastIndexOf('.'); var s, res; if (pos === -1) { return ''; } // suffix may only contain alnum characters (yup, I made this up.) s = filename.substring(pos + 1); res = /^[a-z0-9%]+$/i.test(s) ? s : ''; return v ? URI.decodePathSegment(res) : res; } else { if (v.charAt(0) === '.') { v = v.substring(1); } var suffix = this.suffix(); var replace; if (!suffix) { if (!v) { return this; } this._parts.path += '.' + URI.recodePath(v); } else if (!v) { replace = new RegExp(escapeRegEx('.' + suffix) + '$'); } else { replace = new RegExp(escapeRegEx(suffix) + '$'); } if (replace) { v = URI.recodePath(v); this._parts.path = this._parts.path.replace(replace, v); } this.build(!build); return this; } }; p.segment = function (segment, v, build) { var separator = this._parts.urn ? ':' : '/'; var path = this.path(); var absolute = path.substring(0, 1) === '/'; var segments = path.split(separator); if (segment !== undefined && typeof segment !== 'number') { build = v; v = segment; segment = undefined; } if (segment !== undefined && typeof segment !== 'number') { throw new Error('Bad segment "' + segment + '", must be 0-based integer'); } if (absolute) { segments.shift(); } if (segment < 0) { // allow negative indexes to address from the end segment = Math.max(segments.length + segment, 0); } if (v === undefined) { /*jshint laxbreak: true */ return segment === undefined ? segments : segments[segment]; /*jshint laxbreak: false */ } else if (segment === null || segments[segment] === undefined) { if (isArray(v)) { segments = []; // collapse empty elements within array for (var i = 0, l = v.length; i < l; i++) { if (!v[i].length && (!segments.length || !segments[segments.length - 1].length)) { continue; } if (segments.length && !segments[segments.length - 1].length) { segments.pop(); } segments.push(trimSlashes(v[i])); } } else if (v || typeof v === 'string') { v = trimSlashes(v); if (segments[segments.length - 1] === '') { // empty trailing elements have to be overwritten // to prevent results such as /foo//bar segments[segments.length - 1] = v; } else { segments.push(v); } } } else { if (v) { segments[segment] = trimSlashes(v); } else { segments.splice(segment, 1); } } if (absolute) { segments.unshift(''); } return this.path(segments.join(separator), build); }; p.segmentCoded = function (segment, v, build) { var segments, i, l; if (typeof segment !== 'number') { build = v; v = segment; segment = undefined; } if (v === undefined) { segments = this.segment(segment, v, build); if (!isArray(segments)) { segments = segments !== undefined ? URI.decode(segments) : undefined; } else { for (i = 0, l = segments.length; i < l; i++) { segments[i] = URI.decode(segments[i]); } } return segments; } if (!isArray(v)) { v = typeof v === 'string' || v instanceof String ? URI.encode(v) : v; } else { for (i = 0, l = v.length; i < l; i++) { v[i] = URI.encode(v[i]); } } return this.segment(segment, v, build); }; // mutating query string var q = p.query; p.query = function (v, build) { if (v === true) { return URI.parseQuery(this._parts.query, this._parts.escapeQuerySpace); } else if (typeof v === 'function') { var data = URI.parseQuery(this._parts.query, this._parts.escapeQuerySpace); var result = v.call(this, data); this._parts.query = URI.buildQuery(result || data, this._parts.duplicateQueryParameters, this._parts.escapeQuerySpace); this.build(!build); return this; } else if (v !== undefined && typeof v !== 'string') { this._parts.query = URI.buildQuery(v, this._parts.duplicateQueryParameters, this._parts.escapeQuerySpace); this.build(!build); return this; } else { return q.call(this, v, build); } }; p.setQuery = function (name, value, build) { var data = URI.parseQuery(this._parts.query, this._parts.escapeQuerySpace); if (typeof name === 'string' || name instanceof String) { data[name] = value !== undefined ? value : null; } else if (typeof name === 'object') { for (var key in name) { if (hasOwn.call(name, key)) { data[key] = name[key]; } } } else { throw new TypeError('URI.addQuery() accepts an object, string as the name parameter'); } this._parts.query = URI.buildQuery(data, this._parts.duplicateQueryParameters, this._parts.escapeQuerySpace); if (typeof name !== 'string') { build = value; } this.build(!build); return this; }; p.addQuery = function (name, value, build) { var data = URI.parseQuery(this._parts.query, this._parts.escapeQuerySpace); URI.addQuery(data, name, value === undefined ? null : value); this._parts.query = URI.buildQuery(data, this._parts.duplicateQueryParameters, this._parts.escapeQuerySpace); if (typeof name !== 'string') { build = value; } this.build(!build); return this; }; p.removeQuery = function (name, value, build) { var data = URI.parseQuery(this._parts.query, this._parts.escapeQuerySpace); URI.removeQuery(data, name, value); this._parts.query = URI.buildQuery(data, this._parts.duplicateQueryParameters, this._parts.escapeQuerySpace); if (typeof name !== 'string') { build = value; } this.build(!build); return this; }; p.hasQuery = function (name, value, withinArray) { var data = URI.parseQuery(this._parts.query, this._parts.escapeQuerySpace); return URI.hasQuery(data, name, value, withinArray); }; p.setSearch = p.setQuery; p.addSearch = p.addQuery; p.removeSearch = p.removeQuery; p.hasSearch = p.hasQuery; // sanitizing URLs p.normalize = function () { if (this._parts.urn) { return this.normalizeProtocol(false).normalizePath(false).normalizeQuery(false).normalizeFragment(false).build(); } return this.normalizeProtocol(false).normalizeHostname(false).normalizePort(false).normalizePath(false).normalizeQuery(false).normalizeFragment(false).build(); }; p.normalizeProtocol = function (build) { if (typeof this._parts.protocol === 'string') { this._parts.protocol = this._parts.protocol.toLowerCase(); this.build(!build); } return this; }; p.normalizeHostname = function (build) { if (this._parts.hostname) { if (this.is('IDN') && punycode) { this._parts.hostname = punycode.toASCII(this._parts.hostname); } else if (this.is('IPv6') && IPv6) { this._parts.hostname = IPv6.best(this._parts.hostname); } this._parts.hostname = this._parts.hostname.toLowerCase(); this.build(!build); } return this; }; p.normalizePort = function (build) { // remove port of it's the protocol's default if (typeof this._parts.protocol === 'string' && this._parts.port === URI.defaultPorts[this._parts.protocol]) { this._parts.port = null; this.build(!build); } return this; }; p.normalizePath = function (build) { var _path = this._parts.path; if (!_path) { return this; } if (this._parts.urn) { this._parts.path = URI.recodeUrnPath(this._parts.path); this.build(!build); return this; } if (this._parts.path === '/') { return this; } _path = URI.recodePath(_path); var _was_relative; var _leadingParents = ''; var _parent, _pos; // handle relative paths if (_path.charAt(0) !== '/') { _was_relative = true; _path = '/' + _path; } // handle relative files (as opposed to directories) if (_path.slice(-3) === '/..' || _path.slice(-2) === '/.') { _path += '/'; } // resolve simples _path = _path.replace(/(\/(\.\/)+)|(\/\.$)/g, '/').replace(/\/{2,}/g, '/'); // remember leading parents if (_was_relative) { _leadingParents = _path.substring(1).match(/^(\.\.\/)+/) || ''; if (_leadingParents) { _leadingParents = _leadingParents[0]; } } // resolve parents while (true) { _parent = _path.search(/\/\.\.(\/|$)/); if (_parent === -1) { // no more ../ to resolve break; } else if (_parent === 0) { // top level cannot be relative, skip it _path = _path.substring(3); continue; } _pos = _path.substring(0, _parent).lastIndexOf('/'); if (_pos === -1) { _pos = _parent; } _path = _path.substring(0, _pos) + _path.substring(_parent + 3); } // revert to relative if (_was_relative && this.is('relative')) { _path = _leadingParents + _path.substring(1); } this._parts.path = _path; this.build(!build); return this; }; p.normalizePathname = p.normalizePath; p.normalizeQuery = function (build) { if (typeof this._parts.query === 'string') { if (!this._parts.query.length) { this._parts.query = null; } else { this.query(URI.parseQuery(this._parts.query, this._parts.escapeQuerySpace)); } this.build(!build); } return this; }; p.normalizeFragment = function (build) { if (!this._parts.fragment) { this._parts.fragment = null; this.build(!build); } return this; }; p.normalizeSearch = p.normalizeQuery; p.normalizeHash = p.normalizeFragment; p.iso8859 = function () { // expect unicode input, iso8859 output var e = URI.encode; var d = URI.decode; URI.encode = escape; URI.decode = decodeURIComponent; try { this.normalize(); } finally { URI.encode = e; URI.decode = d; } return this; }; p.unicode = function () { // expect iso8859 input, unicode output var e = URI.encode; var d = URI.decode; URI.encode = strictEncodeURIComponent; URI.decode = unescape; try { this.normalize(); } finally { URI.encode = e; URI.decode = d; } return this; }; p.readable = function () { var uri = this.clone(); // removing username, password, because they shouldn't be displayed according to RFC 3986 uri.username('').password('').normalize(); var t = ''; if (uri._parts.protocol) { t += uri._parts.protocol + '://'; } if (uri._parts.hostname) { if (uri.is('punycode') && punycode) { t += punycode.toUnicode(uri._parts.hostname); if (uri._parts.port) { t += ':' + uri._parts.port; } } else { t += uri.host(); } } if (uri._parts.hostname && uri._parts.path && uri._parts.path.charAt(0) !== '/') { t += '/'; } t += uri.path(true); if (uri._parts.query) { var q = ''; for (var i = 0, qp = uri._parts.query.split('&'), l = qp.length; i < l; i++) { var kv = (qp[i] || '').split('='); q += '&' + URI.decodeQuery(kv[0], this._parts.escapeQuerySpace).replace(/&/g, '%26'); if (kv[1] !== undefined) { q += '=' + URI.decodeQuery(kv[1], this._parts.escapeQuerySpace).replace(/&/g, '%26'); } } t += '?' + q.substring(1); } t += URI.decodeQuery(uri.hash(), true); return t; }; // resolving relative and absolute URLs p.absoluteTo = function (base) { var resolved = this.clone(); var properties = ['protocol', 'username', 'password', 'hostname', 'port']; var basedir, i, p; if (this._parts.urn) { throw new Error('URNs do not have any generally defined hierarchical components'); } if (!(base instanceof URI)) { base = new URI(base); } if (resolved._parts.protocol) { // Directly returns even if this._parts.hostname is empty. return resolved; } else { resolved._parts.protocol = base._parts.protocol; } if (this._parts.hostname) { return resolved; } for (i = 0; p = properties[i]; i++) { resolved._parts[p] = base._parts[p]; } if (!resolved._parts.path) { resolved._parts.path = base._parts.path; if (!resolved._parts.query) { resolved._parts.query = base._parts.query; } } else { if (resolved._parts.path.substring(-2) === '..') { resolved._parts.path += '/'; } if (resolved.path().charAt(0) !== '/') { basedir = base.directory(); basedir = basedir ? basedir : base.path().indexOf('/') === 0 ? '/' : ''; resolved._parts.path = (basedir ? basedir + '/' : '') + resolved._parts.path; resolved.normalizePath(); } } resolved.build(); return resolved; }; p.relativeTo = function (base) { var relative = this.clone().normalize(); var relativeParts, baseParts, common, relativePath, basePath; if (relative._parts.urn) { throw new Error('URNs do not have any generally defined hierarchical components'); } base = new URI(base).normalize(); relativeParts = relative._parts; baseParts = base._parts; relativePath = relative.path(); basePath = base.path(); if (relativePath.charAt(0) !== '/') { throw new Error('URI is already relative'); } if (basePath.charAt(0) !== '/') { throw new Error('Cannot calculate a URI relative to another relative URI'); } if (relativeParts.protocol === baseParts.protocol) { relativeParts.protocol = null; } if (relativeParts.username !== baseParts.username || relativeParts.password !== baseParts.password) { return relative.build(); } if (relativeParts.protocol !== null || relativeParts.username !== null || relativeParts.password !== null) { return relative.build(); } if (relativeParts.hostname === baseParts.hostname && relativeParts.port === baseParts.port) { relativeParts.hostname = null; relativeParts.port = null; } else { return relative.build(); } if (relativePath === basePath) { relativeParts.path = ''; return relative.build(); } // determine common sub path common = URI.commonPath(relativePath, basePath); // If the paths have nothing in common, return a relative URL with the absolute path. if (!common) { return relative.build(); } var parents = baseParts.path.substring(common.length).replace(/[^\/]*$/, '').replace(/.*?\//g, '../'); relativeParts.path = parents + relativeParts.path.substring(common.length) || './'; return relative.build(); }; // comparing URIs p.equals = function (uri) { var one = this.clone(); var two = new URI(uri); var one_map = {}; var two_map = {}; var checked = {}; var one_query, two_query, key; one.normalize(); two.normalize(); // exact match if (one.toString() === two.toString()) { return true; } // extract query string one_query = one.query(); two_query = two.query(); one.query(''); two.query(''); // definitely not equal if not even non-query parts match if (one.toString() !== two.toString()) { return false; } // query parameters have the same length, even if they're permuted if (one_query.length !== two_query.length) { return false; } one_map = URI.parseQuery(one_query, this._parts.escapeQuerySpace); two_map = URI.parseQuery(two_query, this._parts.escapeQuerySpace); for (key in one_map) { if (hasOwn.call(one_map, key)) { if (!isArray(one_map[key])) { if (one_map[key] !== two_map[key]) { return false; } } else if (!arraysEqual(one_map[key], two_map[key])) { return false; } checked[key] = true; } } for (key in two_map) { if (hasOwn.call(two_map, key)) { if (!checked[key]) { // two contains a parameter not present in one return false; } } } return true; }; // state p.preventInvalidHostname = function (v) { this._parts.preventInvalidHostname = !!v; return this; }; p.duplicateQueryParameters = function (v) { this._parts.duplicateQueryParameters = !!v; return this; }; p.escapeQuerySpace = function (v) { this._parts.escapeQuerySpace = !!v; return this; }; return URI; }); /***/ }), /***/ 7819: /***/ (function(module, exports, __webpack_require__) { /* module decorator */ module = __webpack_require__.nmd(module); var __WEBPACK_AMD_DEFINE_RESULT__;/*! https://mths.be/punycode v1.4.0 by @mathias */ ; (function (root) { /** Detect free variables */ var freeExports = true && exports && !exports.nodeType && exports; var freeModule = true && module && !module.nodeType && module; var freeGlobal = typeof __webpack_require__.g == 'object' && __webpack_require__.g; if (freeGlobal.global === freeGlobal || freeGlobal.window === freeGlobal || freeGlobal.self === freeGlobal) { root = freeGlobal; } /** * The `punycode` object. * @name punycode * @type Object */ var punycode, /** Highest positive signed 32-bit float value */ maxInt = 2147483647, // aka. 0x7FFFFFFF or 2^31-1 /** Bootstring parameters */ base = 36, tMin = 1, tMax = 26, skew = 38, damp = 700, initialBias = 72, initialN = 128, // 0x80 delimiter = '-', // '\x2D' /** Regular expressions */ regexPunycode = /^xn--/, regexNonASCII = /[^\x20-\x7E]/, // unprintable ASCII chars + non-ASCII chars regexSeparators = /[\x2E\u3002\uFF0E\uFF61]/g, // RFC 3490 separators /** Error messages */ errors = { 'overflow': 'Overflow: input needs wider integers to process', 'not-basic': 'Illegal input >= 0x80 (not a basic code point)', 'invalid-input': 'Invalid input' }, /** Convenience shortcuts */ baseMinusTMin = base - tMin, floor = Math.floor, stringFromCharCode = String.fromCharCode, /** Temporary variable */ key; /*--------------------------------------------------------------------------*/ /** * A generic error utility function. * @private * @param {String} type The error type. * @returns {Error} Throws a `RangeError` with the applicable error message. */ function error(type) { throw new RangeError(errors[type]); } /** * A generic `Array#map` utility function. * @private * @param {Array} array The array to iterate over. * @param {Function} callback The function that gets called for every array * item. * @returns {Array} A new array of values returned by the callback function. */ function map(array, fn) { var length = array.length; var result = []; while (length--) { result[length] = fn(array[length]); } return result; } /** * A simple `Array#map`-like wrapper to work with domain name strings or email * addresses. * @private * @param {String} domain The domain name or email address. * @param {Function} callback The function that gets called for every * character. * @returns {Array} A new string of characters returned by the callback * function. */ function mapDomain(string, fn) { var parts = string.split('@'); var result = ''; if (parts.length > 1) { // In email addresses, only the domain name should be punycoded. Leave // the local part (i.e. everything up to `@`) intact. result = parts[0] + '@'; string = parts[1]; } // Avoid `split(regex)` for IE8 compatibility. See #17. string = string.replace(regexSeparators, '\x2E'); var labels = string.split('.'); var encoded = map(labels, fn).join('.'); return result + encoded; } /** * Creates an array containing the numeric code points of each Unicode * character in the string. While JavaScript uses UCS-2 internally, * this function will convert a pair of surrogate halves (each of which * UCS-2 exposes as separate characters) into a single code point, * matching UTF-16. * @see `punycode.ucs2.encode` * @see * @memberOf punycode.ucs2 * @name decode * @param {String} string The Unicode input string (UCS-2). * @returns {Array} The new array of code points. */ function ucs2decode(string) { var output = [], counter = 0, length = string.length, value, extra; while (counter < length) { value = string.charCodeAt(counter++); if (value >= 0xD800 && value <= 0xDBFF && counter < length) { // high surrogate, and there is a next character extra = string.charCodeAt(counter++); if ((extra & 0xFC00) == 0xDC00) { // low surrogate output.push(((value & 0x3FF) << 10) + (extra & 0x3FF) + 0x10000); } else { // unmatched surrogate; only append this code unit, in case the next // code unit is the high surrogate of a surrogate pair output.push(value); counter--; } } else { output.push(value); } } return output; } /** * Creates a string based on an array of numeric code points. * @see `punycode.ucs2.decode` * @memberOf punycode.ucs2 * @name encode * @param {Array} codePoints The array of numeric code points. * @returns {String} The new Unicode string (UCS-2). */ function ucs2encode(array) { return map(array, function (value) { var output = ''; if (value > 0xFFFF) { value -= 0x10000; output += stringFromCharCode(value >>> 10 & 0x3FF | 0xD800); value = 0xDC00 | value & 0x3FF; } output += stringFromCharCode(value); return output; }).join(''); } /** * Converts a basic code point into a digit/integer. * @see `digitToBasic()` * @private * @param {Number} codePoint The basic numeric code point value. * @returns {Number} The numeric value of a basic code point (for use in * representing integers) in the range `0` to `base - 1`, or `base` if * the code point does not represent a value. */ function basicToDigit(codePoint) { if (codePoint - 48 < 10) { return codePoint - 22; } if (codePoint - 65 < 26) { return codePoint - 65; } if (codePoint - 97 < 26) { return codePoint - 97; } return base; } /** * Converts a digit/integer into a basic code point. * @see `basicToDigit()` * @private * @param {Number} digit The numeric value of a basic code point. * @returns {Number} The basic code point whose value (when used for * representing integers) is `digit`, which needs to be in the range * `0` to `base - 1`. If `flag` is non-zero, the uppercase form is * used; else, the lowercase form is used. The behavior is undefined * if `flag` is non-zero and `digit` has no uppercase form. */ function digitToBasic(digit, flag) { // 0..25 map to ASCII a..z or A..Z // 26..35 map to ASCII 0..9 return digit + 22 + 75 * (digit < 26) - ((flag != 0) << 5); } /** * Bias adaptation function as per section 3.4 of RFC 3492. * https://tools.ietf.org/html/rfc3492#section-3.4 * @private */ function adapt(delta, numPoints, firstTime) { var k = 0; delta = firstTime ? floor(delta / damp) : delta >> 1; delta += floor(delta / numPoints); for (; delta > baseMinusTMin * tMax >> 1; k += base) { delta = floor(delta / baseMinusTMin); } return floor(k + (baseMinusTMin + 1) * delta / (delta + skew)); } /** * Converts a Punycode string of ASCII-only symbols to a string of Unicode * symbols. * @memberOf punycode * @param {String} input The Punycode string of ASCII-only symbols. * @returns {String} The resulting string of Unicode symbols. */ function decode(input) { // Don't use UCS-2 var output = [], inputLength = input.length, out, i = 0, n = initialN, bias = initialBias, basic, j, index, oldi, w, k, digit, t, /** Cached calculation results */ baseMinusT; // Handle the basic code points: let `basic` be the number of input code // points before the last delimiter, or `0` if there is none, then copy // the first basic code points to the output. basic = input.lastIndexOf(delimiter); if (basic < 0) { basic = 0; } for (j = 0; j < basic; ++j) { // if it's not a basic code point if (input.charCodeAt(j) >= 0x80) { error('not-basic'); } output.push(input.charCodeAt(j)); } // Main decoding loop: start just after the last delimiter if any basic code // points were copied; start at the beginning otherwise. for (index = basic > 0 ? basic + 1 : 0; index < inputLength;) { // `index` is the index of the next character to be consumed. // Decode a generalized variable-length integer into `delta`, // which gets added to `i`. The overflow checking is easier // if we increase `i` as we go, then subtract off its starting // value at the end to obtain `delta`. for (oldi = i, w = 1, k = base;; k += base) { if (index >= inputLength) { error('invalid-input'); } digit = basicToDigit(input.charCodeAt(index++)); if (digit >= base || digit > floor((maxInt - i) / w)) { error('overflow'); } i += digit * w; t = k <= bias ? tMin : k >= bias + tMax ? tMax : k - bias; if (digit < t) { break; } baseMinusT = base - t; if (w > floor(maxInt / baseMinusT)) { error('overflow'); } w *= baseMinusT; } out = output.length + 1; bias = adapt(i - oldi, out, oldi == 0); // `i` was supposed to wrap around from `out` to `0`, // incrementing `n` each time, so we'll fix that now: if (floor(i / out) > maxInt - n) { error('overflow'); } n += floor(i / out); i %= out; // Insert `n` at position `i` of the output output.splice(i++, 0, n); } return ucs2encode(output); } /** * Converts a string of Unicode symbols (e.g. a domain name label) to a * Punycode string of ASCII-only symbols. * @memberOf punycode * @param {String} input The string of Unicode symbols. * @returns {String} The resulting Punycode string of ASCII-only symbols. */ function encode(input) { var n, delta, handledCPCount, basicLength, bias, j, m, q, k, t, currentValue, output = [], /** `inputLength` will hold the number of code points in `input`. */ inputLength, /** Cached calculation results */ handledCPCountPlusOne, baseMinusT, qMinusT; // Convert the input in UCS-2 to Unicode input = ucs2decode(input); // Cache the length inputLength = input.length; // Initialize the state n = initialN; delta = 0; bias = initialBias; // Handle the basic code points for (j = 0; j < inputLength; ++j) { currentValue = input[j]; if (currentValue < 0x80) { output.push(stringFromCharCode(currentValue)); } } handledCPCount = basicLength = output.length; // `handledCPCount` is the number of code points that have been handled; // `basicLength` is the number of basic code points. // Finish the basic string - if it is not empty - with a delimiter if (basicLength) { output.push(delimiter); } // Main encoding loop: while (handledCPCount < inputLength) { // All non-basic code points < n have been handled already. Find the next // larger one: for (m = maxInt, j = 0; j < inputLength; ++j) { currentValue = input[j]; if (currentValue >= n && currentValue < m) { m = currentValue; } } // Increase `delta` enough to advance the decoder's state to , // but guard against overflow handledCPCountPlusOne = handledCPCount + 1; if (m - n > floor((maxInt - delta) / handledCPCountPlusOne)) { error('overflow'); } delta += (m - n) * handledCPCountPlusOne; n = m; for (j = 0; j < inputLength; ++j) { currentValue = input[j]; if (currentValue < n && ++delta > maxInt) { error('overflow'); } if (currentValue == n) { // Represent delta as a generalized variable-length integer for (q = delta, k = base;; k += base) { t = k <= bias ? tMin : k >= bias + tMax ? tMax : k - bias; if (q < t) { break; } qMinusT = q - t; baseMinusT = base - t; output.push(stringFromCharCode(digitToBasic(t + qMinusT % baseMinusT, 0))); q = floor(qMinusT / baseMinusT); } output.push(stringFromCharCode(digitToBasic(q, 0))); bias = adapt(delta, handledCPCountPlusOne, handledCPCount == basicLength); delta = 0; ++handledCPCount; } } ++delta; ++n; } return output.join(''); } /** * Converts a Punycode string representing a domain name or an email address * to Unicode. Only the Punycoded parts of the input will be converted, i.e. * it doesn't matter if you call it on a string that has already been * converted to Unicode. * @memberOf punycode * @param {String} input The Punycoded domain name or email address to * convert to Unicode. * @returns {String} The Unicode representation of the given Punycode * string. */ function toUnicode(input) { return mapDomain(input, function (string) { return regexPunycode.test(string) ? decode(string.slice(4).toLowerCase()) : string; }); } /** * Converts a Unicode string representing a domain name or an email address to * Punycode. Only the non-ASCII parts of the domain name will be converted, * i.e. it doesn't matter if you call it with a domain that's already in * ASCII. * @memberOf punycode * @param {String} input The domain name or email address to convert, as a * Unicode string. * @returns {String} The Punycode representation of the given domain name or * email address. */ function toASCII(input) { return mapDomain(input, function (string) { return regexNonASCII.test(string) ? 'xn--' + encode(string) : string; }); } /*--------------------------------------------------------------------------*/ /** Define the public API */ punycode = { /** * A string representing the current Punycode.js version number. * @memberOf punycode * @type String */ 'version': '1.3.2', /** * An object of methods to convert from JavaScript's internal character * representation (UCS-2) to Unicode code points, and back. * @see * @memberOf punycode * @type Object */ 'ucs2': { 'decode': ucs2decode, 'encode': ucs2encode }, 'decode': decode, 'encode': encode, 'toASCII': toASCII, 'toUnicode': toUnicode }; /** Expose `punycode` */ // Some AMD build optimizers, like r.js, check for specific condition patterns // like the following: if (true) { !(__WEBPACK_AMD_DEFINE_RESULT__ = (function () { return punycode; }).call(exports, __webpack_require__, exports, module), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__)); } else {} })(this); /***/ }), /***/ 517: /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { "use strict"; // ESM COMPAT FLAG __webpack_require__.r(__webpack_exports__); // EXPORTS __webpack_require__.d(__webpack_exports__, { "default": () => (/* binding */ src_converse) }); // EXTERNAL MODULE: ./node_modules/urijs/src/URI.js var URI = __webpack_require__(5215); var URI_default = /*#__PURE__*/__webpack_require__.n(URI); // EXTERNAL MODULE: ./node_modules/sprintf-js/src/sprintf.js var sprintf = __webpack_require__(4223); ;// CONCATENATED MODULE: ./src/headless/shared/i18n.js /** * @namespace i18n */ /* harmony default export */ const i18n = ({ initialize() {}, /** * Overridable string wrapper method which can be used to provide i18n * support. * * The default implementation in @converse/headless simply calls sprintf * with the passed in arguments. * * If you install the full version of Converse, then this method gets * overwritten in src/i18n/index.js to return a translated string. * @method __ * @private * @memberOf i18n * @param { String } str */ __() { return (0,sprintf.sprintf)(...arguments); } }); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isObjectLike.js /** * Checks if `value` is object-like. A value is object-like if it's not `null` * and has a `typeof` result of "object". * * @static * @memberOf _ * @since 4.0.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is object-like, else `false`. * @example * * _.isObjectLike({}); * // => true * * _.isObjectLike([1, 2, 3]); * // => true * * _.isObjectLike(_.noop); * // => false * * _.isObjectLike(null); * // => false */ function isObjectLike(value) { return value != null && typeof value == 'object'; } /* harmony default export */ const lodash_es_isObjectLike = (isObjectLike); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_freeGlobal.js /** Detect free variable `global` from Node.js. */ var freeGlobal = typeof global == 'object' && global && global.Object === Object && global; /* harmony default export */ const _freeGlobal = (freeGlobal); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_root.js /** Detect free variable `self`. */ var freeSelf = typeof self == 'object' && self && self.Object === Object && self; /** Used as a reference to the global object. */ var root = _freeGlobal || freeSelf || Function('return this')(); /* harmony default export */ const _root = (root); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_Symbol.js /** Built-in value references. */ var _Symbol_Symbol = _root.Symbol; /* harmony default export */ const _Symbol = (_Symbol_Symbol); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_getRawTag.js /** Used for built-in method references. */ var objectProto = Object.prototype; /** Used to check objects for own properties. */ var _getRawTag_hasOwnProperty = objectProto.hasOwnProperty; /** * Used to resolve the * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring) * of values. */ var nativeObjectToString = objectProto.toString; /** Built-in value references. */ var symToStringTag = _Symbol ? _Symbol.toStringTag : undefined; /** * A specialized version of `baseGetTag` which ignores `Symbol.toStringTag` values. * * @private * @param {*} value The value to query. * @returns {string} Returns the raw `toStringTag`. */ function getRawTag(value) { var isOwn = _getRawTag_hasOwnProperty.call(value, symToStringTag), tag = value[symToStringTag]; try { value[symToStringTag] = undefined; var unmasked = true; } catch (e) {} var result = nativeObjectToString.call(value); if (unmasked) { if (isOwn) { value[symToStringTag] = tag; } else { delete value[symToStringTag]; } } return result; } /* harmony default export */ const _getRawTag = (getRawTag); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_objectToString.js /** Used for built-in method references. */ var _objectToString_objectProto = Object.prototype; /** * Used to resolve the * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring) * of values. */ var _objectToString_nativeObjectToString = _objectToString_objectProto.toString; /** * Converts `value` to a string using `Object.prototype.toString`. * * @private * @param {*} value The value to convert. * @returns {string} Returns the converted string. */ function objectToString(value) { return _objectToString_nativeObjectToString.call(value); } /* harmony default export */ const _objectToString = (objectToString); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseGetTag.js /** `Object#toString` result references. */ var nullTag = '[object Null]', undefinedTag = '[object Undefined]'; /** Built-in value references. */ var _baseGetTag_symToStringTag = _Symbol ? _Symbol.toStringTag : undefined; /** * The base implementation of `getTag` without fallbacks for buggy environments. * * @private * @param {*} value The value to query. * @returns {string} Returns the `toStringTag`. */ function baseGetTag(value) { if (value == null) { return value === undefined ? undefinedTag : nullTag; } return (_baseGetTag_symToStringTag && _baseGetTag_symToStringTag in Object(value)) ? _getRawTag(value) : _objectToString(value); } /* harmony default export */ const _baseGetTag = (baseGetTag); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_overArg.js /** * Creates a unary function that invokes `func` with its argument transformed. * * @private * @param {Function} func The function to wrap. * @param {Function} transform The argument transform. * @returns {Function} Returns the new function. */ function overArg(func, transform) { return function(arg) { return func(transform(arg)); }; } /* harmony default export */ const _overArg = (overArg); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_getPrototype.js /** Built-in value references. */ var getPrototype = _overArg(Object.getPrototypeOf, Object); /* harmony default export */ const _getPrototype = (getPrototype); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isPlainObject.js /** `Object#toString` result references. */ var objectTag = '[object Object]'; /** Used for built-in method references. */ var funcProto = Function.prototype, isPlainObject_objectProto = Object.prototype; /** Used to resolve the decompiled source of functions. */ var funcToString = funcProto.toString; /** Used to check objects for own properties. */ var isPlainObject_hasOwnProperty = isPlainObject_objectProto.hasOwnProperty; /** Used to infer the `Object` constructor. */ var objectCtorString = funcToString.call(Object); /** * Checks if `value` is a plain object, that is, an object created by the * `Object` constructor or one with a `[[Prototype]]` of `null`. * * @static * @memberOf _ * @since 0.8.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a plain object, else `false`. * @example * * function Foo() { * this.a = 1; * } * * _.isPlainObject(new Foo); * // => false * * _.isPlainObject([1, 2, 3]); * // => false * * _.isPlainObject({ 'x': 0, 'y': 0 }); * // => true * * _.isPlainObject(Object.create(null)); * // => true */ function isPlainObject(value) { if (!lodash_es_isObjectLike(value) || _baseGetTag(value) != objectTag) { return false; } var proto = _getPrototype(value); if (proto === null) { return true; } var Ctor = isPlainObject_hasOwnProperty.call(proto, 'constructor') && proto.constructor; return typeof Ctor == 'function' && Ctor instanceof Ctor && funcToString.call(Ctor) == objectCtorString; } /* harmony default export */ const lodash_es_isPlainObject = (isPlainObject); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isElement.js /** * Checks if `value` is likely a DOM element. * * @static * @memberOf _ * @since 0.1.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a DOM element, else `false`. * @example * * _.isElement(document.body); * // => true * * _.isElement(''); * // => false */ function isElement(value) { return lodash_es_isObjectLike(value) && value.nodeType === 1 && !lodash_es_isPlainObject(value); } /* harmony default export */ const lodash_es_isElement = (isElement); ;// CONCATENATED MODULE: ./src/headless/log.js var _console, _console2, _console3, _console4; const LEVELS = { 'debug': 0, 'info': 1, 'warn': 2, 'error': 3, 'fatal': 4 }; const logger = Object.assign({ 'debug': (_console = console) !== null && _console !== void 0 && _console.log ? console.log.bind(console) : function noop() {}, 'error': (_console2 = console) !== null && _console2 !== void 0 && _console2.log ? console.log.bind(console) : function noop() {}, 'info': (_console3 = console) !== null && _console3 !== void 0 && _console3.log ? console.log.bind(console) : function noop() {}, 'warn': (_console4 = console) !== null && _console4 !== void 0 && _console4.log ? console.log.bind(console) : function noop() {} }, console); /** * The log namespace * @namespace log */ const log = { /** * The the log-level, which determines how verbose the logging is. * @method log#setLogLevel * @param { integer } level - The loglevel which allows for filtering of log messages */ setLogLevel(level) { if (!['debug', 'info', 'warn', 'error', 'fatal'].includes(level)) { throw new Error(`Invalid loglevel: ${level}`); } this.loglevel = level; }, /** * Logs messages to the browser's developer console. * Available loglevels are 0 for 'debug', 1 for 'info', 2 for 'warn', * 3 for 'error' and 4 for 'fatal'. * When using the 'error' or 'warn' loglevels, a full stacktrace will be * logged as well. * @method log#log * @param { string | Error } message - The message to be logged * @param { integer } level - The loglevel which allows for filtering of log messages */ log(message, level) { let style = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : ''; if (LEVELS[level] < LEVELS[this.loglevel]) { return; } if (level === 'error' || level === 'fatal') { style = style || 'color: maroon'; } else if (level === 'debug') { style = style || 'color: green'; } if (message instanceof Error) { message = message.stack; } else if (lodash_es_isElement(message)) { message = message.outerHTML; } const prefix = style ? '%c' : ''; if (level === 'error') { logger.error(`${prefix} ERROR: ${message}`, style); } else if (level === 'warn') { logger.warn(`${prefix} ${new Date().toISOString()} WARNING: ${message}`, style); } else if (level === 'fatal') { logger.error(`${prefix} FATAL: ${message}`, style); } else if (level === 'debug') { logger.debug(`${prefix} ${new Date().toISOString()} DEBUG: ${message}`, style); } else { logger.info(`${prefix} ${new Date().toISOString()} INFO: ${message}`, style); } }, debug(message, style) { this.log(message, 'debug', style); }, error(message, style) { this.log(message, 'error', style); }, info(message, style) { this.log(message, 'info', style); }, warn(message, style) { this.log(message, 'warn', style); }, fatal(message, style) { this.log(message, 'fatal', style); } }; /* harmony default export */ const headless_log = (log); ;// CONCATENATED MODULE: ./src/strophe-shims.js const WebSocket = window.WebSocket; const strophe_shims_DOMParser = window.DOMParser; function getDummyXMLDOMDocument() { return document.implementation.createDocument('jabber:client', 'strophe', null); } ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/md5.js /* * A JavaScript implementation of the RSA Data Security, Inc. MD5 Message * Digest Algorithm, as defined in RFC 1321. * Version 2.1 Copyright (C) Paul Johnston 1999 - 2002. * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet * Distributed under the BSD License * See http://pajhome.org.uk/crypt/md5 for more info. */ /* * Everything that isn't used by Strophe has been stripped here! */ /* * Add integers, wrapping at 2^32. This uses 16-bit operations internally * to work around bugs in some JS interpreters. */ const safe_add = function (x, y) { const lsw = (x & 0xFFFF) + (y & 0xFFFF); const msw = (x >> 16) + (y >> 16) + (lsw >> 16); return msw << 16 | lsw & 0xFFFF; }; /* * Bitwise rotate a 32-bit number to the left. */ const bit_rol = function (num, cnt) { return num << cnt | num >>> 32 - cnt; }; /* * Convert a string to an array of little-endian words */ const str2binl = function (str) { if (typeof str !== "string") { throw new Error("str2binl was passed a non-string"); } const bin = []; for (let i = 0; i < str.length * 8; i += 8) { bin[i >> 5] |= (str.charCodeAt(i / 8) & 255) << i % 32; } return bin; }; /* * Convert an array of little-endian words to a string */ const binl2str = function (bin) { let str = ""; for (let i = 0; i < bin.length * 32; i += 8) { str += String.fromCharCode(bin[i >> 5] >>> i % 32 & 255); } return str; }; /* * Convert an array of little-endian words to a hex string. */ const binl2hex = function (binarray) { const hex_tab = "0123456789abcdef"; let str = ""; for (let i = 0; i < binarray.length * 4; i++) { str += hex_tab.charAt(binarray[i >> 2] >> i % 4 * 8 + 4 & 0xF) + hex_tab.charAt(binarray[i >> 2] >> i % 4 * 8 & 0xF); } return str; }; /* * These functions implement the four basic operations the algorithm uses. */ const md5_cmn = function (q, a, b, x, s, t) { return safe_add(bit_rol(safe_add(safe_add(a, q), safe_add(x, t)), s), b); }; const md5_ff = function (a, b, c, d, x, s, t) { return md5_cmn(b & c | ~b & d, a, b, x, s, t); }; const md5_gg = function (a, b, c, d, x, s, t) { return md5_cmn(b & d | c & ~d, a, b, x, s, t); }; const md5_hh = function (a, b, c, d, x, s, t) { return md5_cmn(b ^ c ^ d, a, b, x, s, t); }; const md5_ii = function (a, b, c, d, x, s, t) { return md5_cmn(c ^ (b | ~d), a, b, x, s, t); }; /* * Calculate the MD5 of an array of little-endian words, and a bit length */ const core_md5 = function (x, len) { /* append padding */ x[len >> 5] |= 0x80 << len % 32; x[(len + 64 >>> 9 << 4) + 14] = len; let a = 1732584193; let b = -271733879; let c = -1732584194; let d = 271733878; let olda, oldb, oldc, oldd; for (let i = 0; i < x.length; i += 16) { olda = a; oldb = b; oldc = c; oldd = d; a = md5_ff(a, b, c, d, x[i + 0], 7, -680876936); d = md5_ff(d, a, b, c, x[i + 1], 12, -389564586); c = md5_ff(c, d, a, b, x[i + 2], 17, 606105819); b = md5_ff(b, c, d, a, x[i + 3], 22, -1044525330); a = md5_ff(a, b, c, d, x[i + 4], 7, -176418897); d = md5_ff(d, a, b, c, x[i + 5], 12, 1200080426); c = md5_ff(c, d, a, b, x[i + 6], 17, -1473231341); b = md5_ff(b, c, d, a, x[i + 7], 22, -45705983); a = md5_ff(a, b, c, d, x[i + 8], 7, 1770035416); d = md5_ff(d, a, b, c, x[i + 9], 12, -1958414417); c = md5_ff(c, d, a, b, x[i + 10], 17, -42063); b = md5_ff(b, c, d, a, x[i + 11], 22, -1990404162); a = md5_ff(a, b, c, d, x[i + 12], 7, 1804603682); d = md5_ff(d, a, b, c, x[i + 13], 12, -40341101); c = md5_ff(c, d, a, b, x[i + 14], 17, -1502002290); b = md5_ff(b, c, d, a, x[i + 15], 22, 1236535329); a = md5_gg(a, b, c, d, x[i + 1], 5, -165796510); d = md5_gg(d, a, b, c, x[i + 6], 9, -1069501632); c = md5_gg(c, d, a, b, x[i + 11], 14, 643717713); b = md5_gg(b, c, d, a, x[i + 0], 20, -373897302); a = md5_gg(a, b, c, d, x[i + 5], 5, -701558691); d = md5_gg(d, a, b, c, x[i + 10], 9, 38016083); c = md5_gg(c, d, a, b, x[i + 15], 14, -660478335); b = md5_gg(b, c, d, a, x[i + 4], 20, -405537848); a = md5_gg(a, b, c, d, x[i + 9], 5, 568446438); d = md5_gg(d, a, b, c, x[i + 14], 9, -1019803690); c = md5_gg(c, d, a, b, x[i + 3], 14, -187363961); b = md5_gg(b, c, d, a, x[i + 8], 20, 1163531501); a = md5_gg(a, b, c, d, x[i + 13], 5, -1444681467); d = md5_gg(d, a, b, c, x[i + 2], 9, -51403784); c = md5_gg(c, d, a, b, x[i + 7], 14, 1735328473); b = md5_gg(b, c, d, a, x[i + 12], 20, -1926607734); a = md5_hh(a, b, c, d, x[i + 5], 4, -378558); d = md5_hh(d, a, b, c, x[i + 8], 11, -2022574463); c = md5_hh(c, d, a, b, x[i + 11], 16, 1839030562); b = md5_hh(b, c, d, a, x[i + 14], 23, -35309556); a = md5_hh(a, b, c, d, x[i + 1], 4, -1530992060); d = md5_hh(d, a, b, c, x[i + 4], 11, 1272893353); c = md5_hh(c, d, a, b, x[i + 7], 16, -155497632); b = md5_hh(b, c, d, a, x[i + 10], 23, -1094730640); a = md5_hh(a, b, c, d, x[i + 13], 4, 681279174); d = md5_hh(d, a, b, c, x[i + 0], 11, -358537222); c = md5_hh(c, d, a, b, x[i + 3], 16, -722521979); b = md5_hh(b, c, d, a, x[i + 6], 23, 76029189); a = md5_hh(a, b, c, d, x[i + 9], 4, -640364487); d = md5_hh(d, a, b, c, x[i + 12], 11, -421815835); c = md5_hh(c, d, a, b, x[i + 15], 16, 530742520); b = md5_hh(b, c, d, a, x[i + 2], 23, -995338651); a = md5_ii(a, b, c, d, x[i + 0], 6, -198630844); d = md5_ii(d, a, b, c, x[i + 7], 10, 1126891415); c = md5_ii(c, d, a, b, x[i + 14], 15, -1416354905); b = md5_ii(b, c, d, a, x[i + 5], 21, -57434055); a = md5_ii(a, b, c, d, x[i + 12], 6, 1700485571); d = md5_ii(d, a, b, c, x[i + 3], 10, -1894986606); c = md5_ii(c, d, a, b, x[i + 10], 15, -1051523); b = md5_ii(b, c, d, a, x[i + 1], 21, -2054922799); a = md5_ii(a, b, c, d, x[i + 8], 6, 1873313359); d = md5_ii(d, a, b, c, x[i + 15], 10, -30611744); c = md5_ii(c, d, a, b, x[i + 6], 15, -1560198380); b = md5_ii(b, c, d, a, x[i + 13], 21, 1309151649); a = md5_ii(a, b, c, d, x[i + 4], 6, -145523070); d = md5_ii(d, a, b, c, x[i + 11], 10, -1120210379); c = md5_ii(c, d, a, b, x[i + 2], 15, 718787259); b = md5_ii(b, c, d, a, x[i + 9], 21, -343485551); a = safe_add(a, olda); b = safe_add(b, oldb); c = safe_add(c, oldc); d = safe_add(d, oldd); } return [a, b, c, d]; }; /* * These are the functions you'll usually want to call. * They take string arguments and return either hex or base-64 encoded * strings. */ const MD5 = { hexdigest: function (s) { return binl2hex(core_md5(str2binl(s), s.length * 8)); }, hash: function (s) { return binl2str(core_md5(str2binl(s), s.length * 8)); } }; ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/sasl.js /** Class: Strophe.SASLMechanism * * Encapsulates an SASL authentication mechanism. * * User code may override the priority for each mechanism or disable it completely. * See for information about changing priority and for informatian on * how to disable a mechanism. * * By default, all mechanisms are enabled and the priorities are * * SCRAM-SHA-1 - 60 * PLAIN - 50 * OAUTHBEARER - 40 * X-OAUTH2 - 30 * ANONYMOUS - 20 * EXTERNAL - 10 * * See: Strophe.Connection.addSupportedSASLMechanisms */ class SASLMechanism { /** * PrivateConstructor: Strophe.SASLMechanism * SASL auth mechanism abstraction. * * Parameters: * (String) name - SASL Mechanism name. * (Boolean) isClientFirst - If client should send response first without challenge. * (Number) priority - Priority. * * Returns: * A new Strophe.SASLMechanism object. */ constructor(name, isClientFirst, priority) { /** PrivateVariable: mechname * Mechanism name. */ this.mechname = name; /** PrivateVariable: isClientFirst * If client sends response without initial server challenge. */ this.isClientFirst = isClientFirst; /** Variable: priority * Determines which is chosen for authentication (Higher is better). * Users may override this to prioritize mechanisms differently. * * Example: (This will cause Strophe to choose the mechanism that the server sent first) * * > Strophe.SASLPlain.priority = Strophe.SASLSHA1.priority; * * See for a list of available mechanisms. * */ this.priority = priority; } /** * Function: test * Checks if mechanism able to run. * To disable a mechanism, make this return false; * * To disable plain authentication run * > Strophe.SASLPlain.test = function() { * > return false; * > } * * See for a list of available mechanisms. * * Parameters: * (Strophe.Connection) connection - Target Connection. * * Returns: * (Boolean) If mechanism was able to run. */ test() { // eslint-disable-line class-methods-use-this return true; } /** PrivateFunction: onStart * Called before starting mechanism on some connection. * * Parameters: * (Strophe.Connection) connection - Target Connection. */ onStart(connection) { this._connection = connection; } /** PrivateFunction: onChallenge * Called by protocol implementation on incoming challenge. * * By deafult, if the client is expected to send data first (isClientFirst === true), * this method is called with `challenge` as null on the first call, * unless `clientChallenge` is overridden in the relevant subclass. * * Parameters: * (Strophe.Connection) connection - Target Connection. * (String) challenge - current challenge to handle. * * Returns: * (String) Mechanism response. */ onChallenge(connection, challenge) { // eslint-disable-line throw new Error("You should implement challenge handling!"); } /** PrivateFunction: clientChallenge * Called by the protocol implementation if the client is expected to send * data first in the authentication exchange (i.e. isClientFirst === true). * * Parameters: * (Strophe.Connection) connection - Target Connection. * * Returns: * (String) Mechanism response. */ clientChallenge(connection) { if (!this.isClientFirst) { throw new Error("clientChallenge should not be called if isClientFirst is false!"); } return this.onChallenge(connection); } /** PrivateFunction: onFailure * Protocol informs mechanism implementation about SASL failure. */ onFailure() { this._connection = null; } /** PrivateFunction: onSuccess * Protocol informs mechanism implementation about SASL success. */ onSuccess() { this._connection = null; } } ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/sasl-anon.js // Building SASL callbacks class SASLAnonymous extends SASLMechanism { /** PrivateConstructor: SASLAnonymous * SASL ANONYMOUS authentication. */ constructor() { let mechname = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 'ANONYMOUS'; let isClientFirst = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false; let priority = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 20; super(mechname, isClientFirst, priority); } test(connection) { // eslint-disable-line class-methods-use-this return connection.authcid === null; } } ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/sasl-external.js class SASLExternal extends SASLMechanism { /** PrivateConstructor: SASLExternal * SASL EXTERNAL authentication. * * The EXTERNAL mechanism allows a client to request the server to use * credentials established by means external to the mechanism to * authenticate the client. The external means may be, for instance, * TLS services. */ constructor() { let mechname = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 'EXTERNAL'; let isClientFirst = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : true; let priority = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 10; super(mechname, isClientFirst, priority); } onChallenge(connection) { // eslint-disable-line class-methods-use-this /** According to XEP-178, an authzid SHOULD NOT be presented when the * authcid contained or implied in the client certificate is the JID (i.e. * authzid) with which the user wants to log in as. * * To NOT send the authzid, the user should therefore set the authcid equal * to the JID when instantiating a new Strophe.Connection object. */ return connection.authcid === connection.authzid ? '' : connection.authzid; } } ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/utils.js const utils = { utf16to8: function (str) { var i, c; var out = ""; var len = str.length; for (i = 0; i < len; i++) { c = str.charCodeAt(i); if (c >= 0x0000 && c <= 0x007F) { out += str.charAt(i); } else if (c > 0x07FF) { out += String.fromCharCode(0xE0 | c >> 12 & 0x0F); out += String.fromCharCode(0x80 | c >> 6 & 0x3F); out += String.fromCharCode(0x80 | c >> 0 & 0x3F); } else { out += String.fromCharCode(0xC0 | c >> 6 & 0x1F); out += String.fromCharCode(0x80 | c >> 0 & 0x3F); } } return out; }, addCookies: function (cookies) { /* Parameters: * (Object) cookies - either a map of cookie names * to string values or to maps of cookie values. * * For example: * { "myCookie": "1234" } * * or: * { "myCookie": { * "value": "1234", * "domain": ".example.org", * "path": "/", * "expires": expirationDate * } * } * * These values get passed to Strophe.Connection via * options.cookies */ cookies = cookies || {}; for (const cookieName in cookies) { if (Object.prototype.hasOwnProperty.call(cookies, cookieName)) { let expires = ''; let domain = ''; let path = ''; const cookieObj = cookies[cookieName]; const isObj = typeof cookieObj === "object"; const cookieValue = escape(unescape(isObj ? cookieObj.value : cookieObj)); if (isObj) { expires = cookieObj.expires ? ";expires=" + cookieObj.expires : ''; domain = cookieObj.domain ? ";domain=" + cookieObj.domain : ''; path = cookieObj.path ? ";path=" + cookieObj.path : ''; } document.cookie = cookieName + '=' + cookieValue + expires + domain + path; } } } }; ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/sasl-oauthbearer.js class SASLOAuthBearer extends SASLMechanism { /** PrivateConstructor: SASLOAuthBearer * SASL OAuth Bearer authentication. */ constructor() { let mechname = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 'OAUTHBEARER'; let isClientFirst = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : true; let priority = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 40; super(mechname, isClientFirst, priority); } test(connection) { // eslint-disable-line class-methods-use-this return connection.pass !== null; } onChallenge(connection) { // eslint-disable-line class-methods-use-this let auth_str = 'n,'; if (connection.authcid !== null) { auth_str = auth_str + 'a=' + connection.authzid; } auth_str = auth_str + ','; auth_str = auth_str + "\u0001"; auth_str = auth_str + 'auth=Bearer '; auth_str = auth_str + connection.pass; auth_str = auth_str + "\u0001"; auth_str = auth_str + "\u0001"; return utils.utf16to8(auth_str); } } ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/sasl-plain.js class SASLPlain extends SASLMechanism { /** PrivateConstructor: SASLPlain * SASL PLAIN authentication. */ constructor() { let mechname = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 'PLAIN'; let isClientFirst = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : true; let priority = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 50; super(mechname, isClientFirst, priority); } test(connection) { // eslint-disable-line class-methods-use-this return connection.authcid !== null; } onChallenge(connection) { // eslint-disable-line class-methods-use-this const { authcid, authzid, domain, pass } = connection; if (!domain) { throw new Error("SASLPlain onChallenge: domain is not defined!"); } // Only include authzid if it differs from authcid. // See: https://tools.ietf.org/html/rfc6120#section-6.3.8 let auth_str = authzid !== `${authcid}@${domain}` ? authzid : ''; auth_str = auth_str + "\u0000"; auth_str = auth_str + authcid; auth_str = auth_str + "\u0000"; auth_str = auth_str + pass; return utils.utf16to8(auth_str); } } ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/sha1.js /* * A JavaScript implementation of the Secure Hash Algorithm, SHA-1, as defined * in FIPS PUB 180-1 * Version 2.1a Copyright Paul Johnston 2000 - 2002. * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet * Distributed under the BSD License * See http://pajhome.org.uk/crypt/md5 for details. */ /* global define */ /* Some functions and variables have been stripped for use with Strophe */ /* * Calculate the SHA-1 of an array of big-endian words, and a bit length */ function core_sha1(x, len) { /* append padding */ x[len >> 5] |= 0x80 << 24 - len % 32; x[(len + 64 >> 9 << 4) + 15] = len; var w = new Array(80); var a = 1732584193; var b = -271733879; var c = -1732584194; var d = 271733878; var e = -1009589776; var i, j, t, olda, oldb, oldc, oldd, olde; for (i = 0; i < x.length; i += 16) { olda = a; oldb = b; oldc = c; oldd = d; olde = e; for (j = 0; j < 80; j++) { if (j < 16) { w[j] = x[i + j]; } else { w[j] = rol(w[j - 3] ^ w[j - 8] ^ w[j - 14] ^ w[j - 16], 1); } t = sha1_safe_add(sha1_safe_add(rol(a, 5), sha1_ft(j, b, c, d)), sha1_safe_add(sha1_safe_add(e, w[j]), sha1_kt(j))); e = d; d = c; c = rol(b, 30); b = a; a = t; } a = sha1_safe_add(a, olda); b = sha1_safe_add(b, oldb); c = sha1_safe_add(c, oldc); d = sha1_safe_add(d, oldd); e = sha1_safe_add(e, olde); } return [a, b, c, d, e]; } /* * Perform the appropriate triplet combination function for the current * iteration */ function sha1_ft(t, b, c, d) { if (t < 20) { return b & c | ~b & d; } if (t < 40) { return b ^ c ^ d; } if (t < 60) { return b & c | b & d | c & d; } return b ^ c ^ d; } /* * Determine the appropriate additive constant for the current iteration */ function sha1_kt(t) { return t < 20 ? 1518500249 : t < 40 ? 1859775393 : t < 60 ? -1894007588 : -899497514; } /* * Calculate the HMAC-SHA1 of a key and some data */ function core_hmac_sha1(key, data) { var bkey = str2binb(key); if (bkey.length > 16) { bkey = core_sha1(bkey, key.length * 8); } var ipad = new Array(16), opad = new Array(16); for (var i = 0; i < 16; i++) { ipad[i] = bkey[i] ^ 0x36363636; opad[i] = bkey[i] ^ 0x5C5C5C5C; } var hash = core_sha1(ipad.concat(str2binb(data)), 512 + data.length * 8); return core_sha1(opad.concat(hash), 512 + 160); } /* * Add integers, wrapping at 2^32. This uses 16-bit operations internally * to work around bugs in some JS interpreters. */ function sha1_safe_add(x, y) { var lsw = (x & 0xFFFF) + (y & 0xFFFF); var msw = (x >> 16) + (y >> 16) + (lsw >> 16); return msw << 16 | lsw & 0xFFFF; } /* * Bitwise rotate a 32-bit number to the left. */ function rol(num, cnt) { return num << cnt | num >>> 32 - cnt; } /* * Convert an 8-bit or 16-bit string to an array of big-endian words * In 8-bit function, characters >255 have their hi-byte silently ignored. */ function str2binb(str) { var bin = []; var mask = 255; for (var i = 0; i < str.length * 8; i += 8) { bin[i >> 5] |= (str.charCodeAt(i / 8) & mask) << 24 - i % 32; } return bin; } /* * Convert an array of big-endian words to a base-64 string */ function binb2b64(binarray) { var tab = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; var str = ""; var triplet, j; for (var i = 0; i < binarray.length * 4; i += 3) { triplet = (binarray[i >> 2] >> 8 * (3 - i % 4) & 0xFF) << 16 | (binarray[i + 1 >> 2] >> 8 * (3 - (i + 1) % 4) & 0xFF) << 8 | binarray[i + 2 >> 2] >> 8 * (3 - (i + 2) % 4) & 0xFF; for (j = 0; j < 4; j++) { if (i * 8 + j * 6 > binarray.length * 32) { str += "="; } else { str += tab.charAt(triplet >> 6 * (3 - j) & 0x3F); } } } return str; } /* * Convert an array of big-endian words to a string */ function binb2str(bin) { var str = ""; var mask = 255; for (var i = 0; i < bin.length * 32; i += 8) { str += String.fromCharCode(bin[i >> 5] >>> 24 - i % 32 & mask); } return str; } /* * These are the functions you'll usually want to call * They take string arguments and return either hex or base-64 encoded strings */ const SHA1 = { b64_hmac_sha1: function (key, data) { return binb2b64(core_hmac_sha1(key, data)); }, b64_sha1: function (s) { return binb2b64(core_sha1(str2binb(s), s.length * 8)); }, binb2str: binb2str, core_hmac_sha1: core_hmac_sha1, str_hmac_sha1: function (key, data) { return binb2str(core_hmac_sha1(key, data)); }, str_sha1: function (s) { return binb2str(core_sha1(str2binb(s), s.length * 8)); } }; ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/sasl-sha1.js class SASLSHA1 extends SASLMechanism { /** PrivateConstructor: SASLSHA1 * SASL SCRAM SHA 1 authentication. */ constructor() { let mechname = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 'SCRAM-SHA-1'; let isClientFirst = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : true; let priority = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 60; super(mechname, isClientFirst, priority); } test(connection) { // eslint-disable-line class-methods-use-this return connection.authcid !== null; } onChallenge(connection, challenge) { // eslint-disable-line class-methods-use-this let nonce, salt, iter, Hi, U, U_old, i, k; let responseText = "c=biws,"; let authMessage = `${connection._sasl_data["client-first-message-bare"]},${challenge},`; const cnonce = connection._sasl_data.cnonce; const attribMatch = /([a-z]+)=([^,]+)(,|$)/; while (challenge.match(attribMatch)) { const matches = challenge.match(attribMatch); challenge = challenge.replace(matches[0], ""); switch (matches[1]) { case "r": nonce = matches[2]; break; case "s": salt = matches[2]; break; case "i": iter = matches[2]; break; } } if (nonce.slice(0, cnonce.length) !== cnonce) { connection._sasl_data = {}; return connection._sasl_failure_cb(); } responseText += "r=" + nonce; authMessage += responseText; salt = atob(salt); salt += "\x00\x00\x00\x01"; const pass = utils.utf16to8(connection.pass); Hi = U_old = SHA1.core_hmac_sha1(pass, salt); for (i = 1; i < iter; i++) { U = SHA1.core_hmac_sha1(pass, SHA1.binb2str(U_old)); for (k = 0; k < 5; k++) { Hi[k] ^= U[k]; } U_old = U; } Hi = SHA1.binb2str(Hi); const clientKey = SHA1.core_hmac_sha1(Hi, "Client Key"); const serverKey = SHA1.str_hmac_sha1(Hi, "Server Key"); const clientSignature = SHA1.core_hmac_sha1(SHA1.str_sha1(SHA1.binb2str(clientKey)), authMessage); connection._sasl_data["server-signature"] = SHA1.b64_hmac_sha1(serverKey, authMessage); for (k = 0; k < 5; k++) { clientKey[k] ^= clientSignature[k]; } responseText += ",p=" + btoa(SHA1.binb2str(clientKey)); return responseText; } clientChallenge(connection, test_cnonce) { // eslint-disable-line class-methods-use-this const cnonce = test_cnonce || MD5.hexdigest("" + Math.random() * 1234567890); let auth_str = "n=" + utils.utf16to8(connection.authcid); auth_str += ",r="; auth_str += cnonce; connection._sasl_data.cnonce = cnonce; connection._sasl_data["client-first-message-bare"] = auth_str; auth_str = "n,," + auth_str; return auth_str; } } ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/sasl-xoauth2.js class SASLXOAuth2 extends SASLMechanism { /** PrivateConstructor: SASLXOAuth2 * SASL X-OAuth2 authentication. */ constructor() { let mechname = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 'X-OAUTH2'; let isClientFirst = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : true; let priority = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 30; super(mechname, isClientFirst, priority); } test(connection) { // eslint-disable-line class-methods-use-this return connection.pass !== null; } onChallenge(connection) { // eslint-disable-line class-methods-use-this let auth_str = '\u0000'; if (connection.authcid !== null) { auth_str = auth_str + connection.authzid; } auth_str = auth_str + "\u0000"; auth_str = auth_str + connection.pass; return utils.utf16to8(auth_str); } } // EXTERNAL MODULE: ./node_modules/abab/index.js var abab = __webpack_require__(9494); ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/core.js /* This program is distributed under the terms of the MIT license. Please see the LICENSE file for details. Copyright 2006-2018, OGG, LLC */ /*global define, document, sessionStorage, setTimeout, clearTimeout, ActiveXObject, DOMParser, btoa, atob */ /** Function: $build * Create a Strophe.Builder. * This is an alias for 'new Strophe.Builder(name, attrs)'. * * Parameters: * (String) name - The root element name. * (Object) attrs - The attributes for the root element in object notation. * * Returns: * A new Strophe.Builder object. */ function $build(name, attrs) { return new Strophe.Builder(name, attrs); } /** Function: $msg * Create a Strophe.Builder with a element as the root. * * Parameters: * (Object) attrs - The element attributes in object notation. * * Returns: * A new Strophe.Builder object. */ function $msg(attrs) { return new Strophe.Builder("message", attrs); } /** Function: $iq * Create a Strophe.Builder with an element as the root. * * Parameters: * (Object) attrs - The element attributes in object notation. * * Returns: * A new Strophe.Builder object. */ function $iq(attrs) { return new Strophe.Builder("iq", attrs); } /** Function: $pres * Create a Strophe.Builder with a element as the root. * * Parameters: * (Object) attrs - The element attributes in object notation. * * Returns: * A new Strophe.Builder object. */ function $pres(attrs) { return new Strophe.Builder("presence", attrs); } /** Class: Strophe * An object container for all Strophe library functions. * * This class is just a container for all the objects and constants * used in the library. It is not meant to be instantiated, but to * provide a namespace for library objects, constants, and functions. */ const Strophe = { /** Constant: VERSION */ VERSION: "1.5.0", /** Constants: XMPP Namespace Constants * Common namespace constants from the XMPP RFCs and XEPs. * * NS.HTTPBIND - HTTP BIND namespace from XEP 124. * NS.BOSH - BOSH namespace from XEP 206. * NS.CLIENT - Main XMPP client namespace. * NS.AUTH - Legacy authentication namespace. * NS.ROSTER - Roster operations namespace. * NS.PROFILE - Profile namespace. * NS.DISCO_INFO - Service discovery info namespace from XEP 30. * NS.DISCO_ITEMS - Service discovery items namespace from XEP 30. * NS.MUC - Multi-User Chat namespace from XEP 45. * NS.SASL - XMPP SASL namespace from RFC 3920. * NS.STREAM - XMPP Streams namespace from RFC 3920. * NS.BIND - XMPP Binding namespace from RFC 3920 and RFC 6120. * NS.SESSION - XMPP Session namespace from RFC 3920. * NS.XHTML_IM - XHTML-IM namespace from XEP 71. * NS.XHTML - XHTML body namespace from XEP 71. */ NS: { HTTPBIND: "http://jabber.org/protocol/httpbind", BOSH: "urn:xmpp:xbosh", CLIENT: "jabber:client", AUTH: "jabber:iq:auth", ROSTER: "jabber:iq:roster", PROFILE: "jabber:iq:profile", DISCO_INFO: "http://jabber.org/protocol/disco#info", DISCO_ITEMS: "http://jabber.org/protocol/disco#items", MUC: "http://jabber.org/protocol/muc", SASL: "urn:ietf:params:xml:ns:xmpp-sasl", STREAM: "http://etherx.jabber.org/streams", FRAMING: "urn:ietf:params:xml:ns:xmpp-framing", BIND: "urn:ietf:params:xml:ns:xmpp-bind", SESSION: "urn:ietf:params:xml:ns:xmpp-session", VERSION: "jabber:iq:version", STANZAS: "urn:ietf:params:xml:ns:xmpp-stanzas", XHTML_IM: "http://jabber.org/protocol/xhtml-im", XHTML: "http://www.w3.org/1999/xhtml" }, /** Constants: XHTML_IM Namespace * contains allowed tags, tag attributes, and css properties. * Used in the createHtml function to filter incoming html into the allowed XHTML-IM subset. * See http://xmpp.org/extensions/xep-0071.html#profile-summary for the list of recommended * allowed tags and their attributes. */ XHTML: { tags: ['a', 'blockquote', 'br', 'cite', 'em', 'img', 'li', 'ol', 'p', 'span', 'strong', 'ul', 'body'], attributes: { 'a': ['href'], 'blockquote': ['style'], 'br': [], 'cite': ['style'], 'em': [], 'img': ['src', 'alt', 'style', 'height', 'width'], 'li': ['style'], 'ol': ['style'], 'p': ['style'], 'span': ['style'], 'strong': [], 'ul': ['style'], 'body': [] }, css: ['background-color', 'color', 'font-family', 'font-size', 'font-style', 'font-weight', 'margin-left', 'margin-right', 'text-align', 'text-decoration'], /** Function: XHTML.validTag * * Utility method to determine whether a tag is allowed * in the XHTML_IM namespace. * * XHTML tag names are case sensitive and must be lower case. */ validTag(tag) { for (let i = 0; i < Strophe.XHTML.tags.length; i++) { if (tag === Strophe.XHTML.tags[i]) { return true; } } return false; }, /** Function: XHTML.validAttribute * * Utility method to determine whether an attribute is allowed * as recommended per XEP-0071 * * XHTML attribute names are case sensitive and must be lower case. */ validAttribute(tag, attribute) { if (typeof Strophe.XHTML.attributes[tag] !== 'undefined' && Strophe.XHTML.attributes[tag].length > 0) { for (let i = 0; i < Strophe.XHTML.attributes[tag].length; i++) { if (attribute === Strophe.XHTML.attributes[tag][i]) { return true; } } } return false; }, validCSS(style) { for (let i = 0; i < Strophe.XHTML.css.length; i++) { if (style === Strophe.XHTML.css[i]) { return true; } } return false; } }, /** Constants: Connection Status Constants * Connection status constants for use by the connection handler * callback. * * Status.ERROR - An error has occurred * Status.CONNECTING - The connection is currently being made * Status.CONNFAIL - The connection attempt failed * Status.AUTHENTICATING - The connection is authenticating * Status.AUTHFAIL - The authentication attempt failed * Status.CONNECTED - The connection has succeeded * Status.DISCONNECTED - The connection has been terminated * Status.DISCONNECTING - The connection is currently being terminated * Status.ATTACHED - The connection has been attached * Status.REDIRECT - The connection has been redirected * Status.CONNTIMEOUT - The connection has timed out */ Status: { ERROR: 0, CONNECTING: 1, CONNFAIL: 2, AUTHENTICATING: 3, AUTHFAIL: 4, CONNECTED: 5, DISCONNECTED: 6, DISCONNECTING: 7, ATTACHED: 8, REDIRECT: 9, CONNTIMEOUT: 10, BINDREQUIRED: 11, ATTACHFAIL: 12 }, ErrorCondition: { BAD_FORMAT: "bad-format", CONFLICT: "conflict", MISSING_JID_NODE: "x-strophe-bad-non-anon-jid", NO_AUTH_MECH: "no-auth-mech", UNKNOWN_REASON: "unknown" }, /** Constants: Log Level Constants * Logging level indicators. * * LogLevel.DEBUG - Debug output * LogLevel.INFO - Informational output * LogLevel.WARN - Warnings * LogLevel.ERROR - Errors * LogLevel.FATAL - Fatal errors */ LogLevel: { DEBUG: 0, INFO: 1, WARN: 2, ERROR: 3, FATAL: 4 }, /** PrivateConstants: DOM Element Type Constants * DOM element types. * * ElementType.NORMAL - Normal element. * ElementType.TEXT - Text data element. * ElementType.FRAGMENT - XHTML fragment element. */ ElementType: { NORMAL: 1, TEXT: 3, CDATA: 4, FRAGMENT: 11 }, /** PrivateConstants: Timeout Values * Timeout values for error states. These values are in seconds. * These should not be changed unless you know exactly what you are * doing. * * TIMEOUT - Timeout multiplier. A waiting request will be considered * failed after Math.floor(TIMEOUT * wait) seconds have elapsed. * This defaults to 1.1, and with default wait, 66 seconds. * SECONDARY_TIMEOUT - Secondary timeout multiplier. In cases where * Strophe can detect early failure, it will consider the request * failed if it doesn't return after * Math.floor(SECONDARY_TIMEOUT * wait) seconds have elapsed. * This defaults to 0.1, and with default wait, 6 seconds. */ TIMEOUT: 1.1, SECONDARY_TIMEOUT: 0.1, /** Function: addNamespace * This function is used to extend the current namespaces in * Strophe.NS. It takes a key and a value with the key being the * name of the new namespace, with its actual value. * For example: * Strophe.addNamespace('PUBSUB', "http://jabber.org/protocol/pubsub"); * * Parameters: * (String) name - The name under which the namespace will be * referenced under Strophe.NS * (String) value - The actual namespace. */ addNamespace(name, value) { Strophe.NS[name] = value; }, /** Function: forEachChild * Map a function over some or all child elements of a given element. * * This is a small convenience function for mapping a function over * some or all of the children of an element. If elemName is null, all * children will be passed to the function, otherwise only children * whose tag names match elemName will be passed. * * Parameters: * (XMLElement) elem - The element to operate on. * (String) elemName - The child element tag name filter. * (Function) func - The function to apply to each child. This * function should take a single argument, a DOM element. */ forEachChild(elem, elemName, func) { for (let i = 0; i < elem.childNodes.length; i++) { const childNode = elem.childNodes[i]; if (childNode.nodeType === Strophe.ElementType.NORMAL && (!elemName || this.isTagEqual(childNode, elemName))) { func(childNode); } } }, /** Function: isTagEqual * Compare an element's tag name with a string. * * This function is case sensitive. * * Parameters: * (XMLElement) el - A DOM element. * (String) name - The element name. * * Returns: * true if the element's tag name matches _el_, and false * otherwise. */ isTagEqual(el, name) { return el.tagName === name; }, /** PrivateVariable: _xmlGenerator * _Private_ variable that caches a DOM document to * generate elements. */ _xmlGenerator: null, /** Function: xmlGenerator * Get the DOM document to generate elements. * * Returns: * The currently used DOM document. */ xmlGenerator() { if (!Strophe._xmlGenerator) { Strophe._xmlGenerator = getDummyXMLDOMDocument(); } return Strophe._xmlGenerator; }, /** Function: xmlElement * Create an XML DOM element. * * This function creates an XML DOM element correctly across all * implementations. Note that these are not HTML DOM elements, which * aren't appropriate for XMPP stanzas. * * Parameters: * (String) name - The name for the element. * (Array|Object) attrs - An optional array or object containing * key/value pairs to use as element attributes. The object should * be in the format {'key': 'value'} or {key: 'value'}. The array * should have the format [['key1', 'value1'], ['key2', 'value2']]. * (String) text - The text child data for the element. * * Returns: * A new XML DOM element. */ xmlElement(name) { if (!name) { return null; } const node = Strophe.xmlGenerator().createElement(name); // FIXME: this should throw errors if args are the wrong type or // there are more than two optional args for (let a = 1; a < arguments.length; a++) { const arg = arguments[a]; if (!arg) { continue; } if (typeof arg === "string" || typeof arg === "number") { node.appendChild(Strophe.xmlTextNode(arg)); } else if (typeof arg === "object" && typeof arg.sort === "function") { for (let i = 0; i < arg.length; i++) { const attr = arg[i]; if (typeof attr === "object" && typeof attr.sort === "function" && attr[1] !== undefined && attr[1] !== null) { node.setAttribute(attr[0], attr[1]); } } } else if (typeof arg === "object") { for (const k in arg) { if (Object.prototype.hasOwnProperty.call(arg, k) && arg[k] !== undefined && arg[k] !== null) { node.setAttribute(k, arg[k]); } } } } return node; }, /* Function: xmlescape * Excapes invalid xml characters. * * Parameters: * (String) text - text to escape. * * Returns: * Escaped text. */ xmlescape(text) { text = text.replace(/\&/g, "&"); text = text.replace(//g, ">"); text = text.replace(/'/g, "'"); text = text.replace(/"/g, """); return text; }, /* Function: xmlunescape * Unexcapes invalid xml characters. * * Parameters: * (String) text - text to unescape. * * Returns: * Unescaped text. */ xmlunescape(text) { text = text.replace(/\&/g, "&"); text = text.replace(/</g, "<"); text = text.replace(/>/g, ">"); text = text.replace(/'/g, "'"); text = text.replace(/"/g, "\""); return text; }, /** Function: xmlTextNode * Creates an XML DOM text node. * * Provides a cross implementation version of document.createTextNode. * * Parameters: * (String) text - The content of the text node. * * Returns: * A new XML DOM text node. */ xmlTextNode(text) { return Strophe.xmlGenerator().createTextNode(text); }, /** Function: xmlHtmlNode * Creates an XML DOM html node. * * Parameters: * (String) html - The content of the html node. * * Returns: * A new XML DOM text node. */ xmlHtmlNode(html) { let node; //ensure text is escaped if (strophe_shims_DOMParser) { const parser = new strophe_shims_DOMParser(); node = parser.parseFromString(html, "text/xml"); } else { node = new ActiveXObject("Microsoft.XMLDOM"); node.async = "false"; node.loadXML(html); } return node; }, /** Function: getText * Get the concatenation of all text children of an element. * * Parameters: * (XMLElement) elem - A DOM element. * * Returns: * A String with the concatenated text of all text element children. */ getText(elem) { if (!elem) { return null; } let str = ""; if (elem.childNodes.length === 0 && elem.nodeType === Strophe.ElementType.TEXT) { str += elem.nodeValue; } for (let i = 0; i < elem.childNodes.length; i++) { if (elem.childNodes[i].nodeType === Strophe.ElementType.TEXT) { str += elem.childNodes[i].nodeValue; } } return Strophe.xmlescape(str); }, /** Function: copyElement * Copy an XML DOM element. * * This function copies a DOM element and all its descendants and returns * the new copy. * * Parameters: * (XMLElement) elem - A DOM element. * * Returns: * A new, copied DOM element tree. */ copyElement(elem) { let el; if (elem.nodeType === Strophe.ElementType.NORMAL) { el = Strophe.xmlElement(elem.tagName); for (let i = 0; i < elem.attributes.length; i++) { el.setAttribute(elem.attributes[i].nodeName, elem.attributes[i].value); } for (let i = 0; i < elem.childNodes.length; i++) { el.appendChild(Strophe.copyElement(elem.childNodes[i])); } } else if (elem.nodeType === Strophe.ElementType.TEXT) { el = Strophe.xmlGenerator().createTextNode(elem.nodeValue); } return el; }, /** Function: createHtml * Copy an HTML DOM element into an XML DOM. * * This function copies a DOM element and all its descendants and returns * the new copy. * * Parameters: * (HTMLElement) elem - A DOM element. * * Returns: * A new, copied DOM element tree. */ createHtml(elem) { let el; if (elem.nodeType === Strophe.ElementType.NORMAL) { const tag = elem.nodeName.toLowerCase(); // XHTML tags must be lower case. if (Strophe.XHTML.validTag(tag)) { try { el = Strophe.xmlElement(tag); for (let i = 0; i < Strophe.XHTML.attributes[tag].length; i++) { const attribute = Strophe.XHTML.attributes[tag][i]; let value = elem.getAttribute(attribute); if (typeof value === 'undefined' || value === null || value === '' || value === false || value === 0) { continue; } if (attribute === 'style' && typeof value === 'object' && typeof value.cssText !== 'undefined') { value = value.cssText; // we're dealing with IE, need to get CSS out } // filter out invalid css styles if (attribute === 'style') { const css = []; const cssAttrs = value.split(';'); for (let j = 0; j < cssAttrs.length; j++) { const attr = cssAttrs[j].split(':'); const cssName = attr[0].replace(/^\s*/, "").replace(/\s*$/, "").toLowerCase(); if (Strophe.XHTML.validCSS(cssName)) { const cssValue = attr[1].replace(/^\s*/, "").replace(/\s*$/, ""); css.push(cssName + ': ' + cssValue); } } if (css.length > 0) { value = css.join('; '); el.setAttribute(attribute, value); } } else { el.setAttribute(attribute, value); } } for (let i = 0; i < elem.childNodes.length; i++) { el.appendChild(Strophe.createHtml(elem.childNodes[i])); } } catch (e) { // invalid elements el = Strophe.xmlTextNode(''); } } else { el = Strophe.xmlGenerator().createDocumentFragment(); for (let i = 0; i < elem.childNodes.length; i++) { el.appendChild(Strophe.createHtml(elem.childNodes[i])); } } } else if (elem.nodeType === Strophe.ElementType.FRAGMENT) { el = Strophe.xmlGenerator().createDocumentFragment(); for (let i = 0; i < elem.childNodes.length; i++) { el.appendChild(Strophe.createHtml(elem.childNodes[i])); } } else if (elem.nodeType === Strophe.ElementType.TEXT) { el = Strophe.xmlTextNode(elem.nodeValue); } return el; }, /** Function: escapeNode * Escape the node part (also called local part) of a JID. * * Parameters: * (String) node - A node (or local part). * * Returns: * An escaped node (or local part). */ escapeNode(node) { if (typeof node !== "string") { return node; } return node.replace(/^\s+|\s+$/g, '').replace(/\\/g, "\\5c").replace(/ /g, "\\20").replace(/\"/g, "\\22").replace(/\&/g, "\\26").replace(/\'/g, "\\27").replace(/\//g, "\\2f").replace(/:/g, "\\3a").replace(//g, "\\3e").replace(/@/g, "\\40"); }, /** Function: unescapeNode * Unescape a node part (also called local part) of a JID. * * Parameters: * (String) node - A node (or local part). * * Returns: * An unescaped node (or local part). */ unescapeNode(node) { if (typeof node !== "string") { return node; } return node.replace(/\\20/g, " ").replace(/\\22/g, '"').replace(/\\26/g, "&").replace(/\\27/g, "'").replace(/\\2f/g, "/").replace(/\\3a/g, ":").replace(/\\3c/g, "<").replace(/\\3e/g, ">").replace(/\\40/g, "@").replace(/\\5c/g, "\\"); }, /** Function: getNodeFromJid * Get the node portion of a JID String. * * Parameters: * (String) jid - A JID. * * Returns: * A String containing the node. */ getNodeFromJid(jid) { if (jid.indexOf("@") < 0) { return null; } return jid.split("@")[0]; }, /** Function: getDomainFromJid * Get the domain portion of a JID String. * * Parameters: * (String) jid - A JID. * * Returns: * A String containing the domain. */ getDomainFromJid(jid) { const bare = Strophe.getBareJidFromJid(jid); if (bare.indexOf("@") < 0) { return bare; } else { const parts = bare.split("@"); parts.splice(0, 1); return parts.join('@'); } }, /** Function: getResourceFromJid * Get the resource portion of a JID String. * * Parameters: * (String) jid - A JID. * * Returns: * A String containing the resource. */ getResourceFromJid(jid) { if (!jid) { return null; } const s = jid.split("/"); if (s.length < 2) { return null; } s.splice(0, 1); return s.join('/'); }, /** Function: getBareJidFromJid * Get the bare JID from a JID String. * * Parameters: * (String) jid - A JID. * * Returns: * A String containing the bare JID. */ getBareJidFromJid(jid) { return jid ? jid.split("/")[0] : null; }, /** PrivateFunction: _handleError * _Private_ function that properly logs an error to the console */ _handleError(e) { if (typeof e.stack !== "undefined") { Strophe.fatal(e.stack); } if (e.sourceURL) { Strophe.fatal("error: " + this.handler + " " + e.sourceURL + ":" + e.line + " - " + e.name + ": " + e.message); } else if (e.fileName) { Strophe.fatal("error: " + this.handler + " " + e.fileName + ":" + e.lineNumber + " - " + e.name + ": " + e.message); } else { Strophe.fatal("error: " + e.message); } }, /** Function: log * User overrideable logging function. * * This function is called whenever the Strophe library calls any * of the logging functions. The default implementation of this * function logs only fatal errors. If client code wishes to handle the logging * messages, it should override this with * > Strophe.log = function (level, msg) { * > (user code here) * > }; * * Please note that data sent and received over the wire is logged * via Strophe.Connection.rawInput() and Strophe.Connection.rawOutput(). * * The different levels and their meanings are * * DEBUG - Messages useful for debugging purposes. * INFO - Informational messages. This is mostly information like * 'disconnect was called' or 'SASL auth succeeded'. * WARN - Warnings about potential problems. This is mostly used * to report transient connection errors like request timeouts. * ERROR - Some error occurred. * FATAL - A non-recoverable fatal error occurred. * * Parameters: * (Integer) level - The log level of the log message. This will * be one of the values in Strophe.LogLevel. * (String) msg - The log message. */ log(level, msg) { if (level === this.LogLevel.FATAL) { var _console; (_console = console) === null || _console === void 0 ? void 0 : _console.error(msg); } }, /** Function: debug * Log a message at the Strophe.LogLevel.DEBUG level. * * Parameters: * (String) msg - The log message. */ debug(msg) { this.log(this.LogLevel.DEBUG, msg); }, /** Function: info * Log a message at the Strophe.LogLevel.INFO level. * * Parameters: * (String) msg - The log message. */ info(msg) { this.log(this.LogLevel.INFO, msg); }, /** Function: warn * Log a message at the Strophe.LogLevel.WARN level. * * Parameters: * (String) msg - The log message. */ warn(msg) { this.log(this.LogLevel.WARN, msg); }, /** Function: error * Log a message at the Strophe.LogLevel.ERROR level. * * Parameters: * (String) msg - The log message. */ error(msg) { this.log(this.LogLevel.ERROR, msg); }, /** Function: fatal * Log a message at the Strophe.LogLevel.FATAL level. * * Parameters: * (String) msg - The log message. */ fatal(msg) { this.log(this.LogLevel.FATAL, msg); }, /** Function: serialize * Render a DOM element and all descendants to a String. * * Parameters: * (XMLElement) elem - A DOM element. * * Returns: * The serialized element tree as a String. */ serialize(elem) { if (!elem) { return null; } if (typeof elem.tree === "function") { elem = elem.tree(); } const names = [...Array(elem.attributes.length).keys()].map(i => elem.attributes[i].nodeName); names.sort(); let result = names.reduce((a, n) => `${a} ${n}="${Strophe.xmlescape(elem.attributes.getNamedItem(n).value)}"`, `<${elem.nodeName}`); if (elem.childNodes.length > 0) { result += ">"; for (let i = 0; i < elem.childNodes.length; i++) { const child = elem.childNodes[i]; switch (child.nodeType) { case Strophe.ElementType.NORMAL: // normal element, so recurse result += Strophe.serialize(child); break; case Strophe.ElementType.TEXT: // text element to escape values result += Strophe.xmlescape(child.nodeValue); break; case Strophe.ElementType.CDATA: // cdata section so don't escape values result += ""; } } result += ""; } else { result += "/>"; } return result; }, /** PrivateVariable: _requestId * _Private_ variable that keeps track of the request ids for * connections. */ _requestId: 0, /** PrivateVariable: Strophe.connectionPlugins * _Private_ variable Used to store plugin names that need * initialization on Strophe.Connection construction. */ _connectionPlugins: {}, /** Function: addConnectionPlugin * Extends the Strophe.Connection object with the given plugin. * * Parameters: * (String) name - The name of the extension. * (Object) ptype - The plugin's prototype. */ addConnectionPlugin(name, ptype) { Strophe._connectionPlugins[name] = ptype; } }; /** Class: Strophe.Builder * XML DOM builder. * * This object provides an interface similar to JQuery but for building * DOM elements easily and rapidly. All the functions except for toString() * and tree() return the object, so calls can be chained. Here's an * example using the $iq() builder helper. * > $iq({to: 'you', from: 'me', type: 'get', id: '1'}) * > .c('query', {xmlns: 'strophe:example'}) * > .c('example') * > .toString() * * The above generates this XML fragment * > * > * > * > * > * The corresponding DOM manipulations to get a similar fragment would be * a lot more tedious and probably involve several helper variables. * * Since adding children makes new operations operate on the child, up() * is provided to traverse up the tree. To add two children, do * > builder.c('child1', ...).up().c('child2', ...) * The next operation on the Builder will be relative to the second child. */ /** Constructor: Strophe.Builder * Create a Strophe.Builder object. * * The attributes should be passed in object notation. For example * > let b = new Builder('message', {to: 'you', from: 'me'}); * or * > let b = new Builder('messsage', {'xml:lang': 'en'}); * * Parameters: * (String) name - The name of the root element. * (Object) attrs - The attributes for the root element in object notation. * * Returns: * A new Strophe.Builder. */ Strophe.Builder = class Builder { constructor(name, attrs) { // Set correct namespace for jabber:client elements if (name === "presence" || name === "message" || name === "iq") { if (attrs && !attrs.xmlns) { attrs.xmlns = Strophe.NS.CLIENT; } else if (!attrs) { attrs = { xmlns: Strophe.NS.CLIENT }; } } // Holds the tree being built. this.nodeTree = Strophe.xmlElement(name, attrs); // Points to the current operation node. this.node = this.nodeTree; } /** Function: tree * Return the DOM tree. * * This function returns the current DOM tree as an element object. This * is suitable for passing to functions like Strophe.Connection.send(). * * Returns: * The DOM tree as a element object. */ tree() { return this.nodeTree; } /** Function: toString * Serialize the DOM tree to a String. * * This function returns a string serialization of the current DOM * tree. It is often used internally to pass data to a * Strophe.Request object. * * Returns: * The serialized DOM tree in a String. */ toString() { return Strophe.serialize(this.nodeTree); } /** Function: up * Make the current parent element the new current element. * * This function is often used after c() to traverse back up the tree. * For example, to add two children to the same element * > builder.c('child1', {}).up().c('child2', {}); * * Returns: * The Stophe.Builder object. */ up() { this.node = this.node.parentNode; return this; } /** Function: root * Make the root element the new current element. * * When at a deeply nested element in the tree, this function can be used * to jump back to the root of the tree, instead of having to repeatedly * call up(). * * Returns: * The Stophe.Builder object. */ root() { this.node = this.nodeTree; return this; } /** Function: attrs * Add or modify attributes of the current element. * * The attributes should be passed in object notation. This function * does not move the current element pointer. * * Parameters: * (Object) moreattrs - The attributes to add/modify in object notation. * * Returns: * The Strophe.Builder object. */ attrs(moreattrs) { for (const k in moreattrs) { if (Object.prototype.hasOwnProperty.call(moreattrs, k)) { if (moreattrs[k] === undefined) { this.node.removeAttribute(k); } else { this.node.setAttribute(k, moreattrs[k]); } } } return this; } /** Function: c * Add a child to the current element and make it the new current * element. * * This function moves the current element pointer to the child, * unless text is provided. If you need to add another child, it * is necessary to use up() to go back to the parent in the tree. * * Parameters: * (String) name - The name of the child. * (Object) attrs - The attributes of the child in object notation. * (String) text - The text to add to the child. * * Returns: * The Strophe.Builder object. */ c(name, attrs, text) { const child = Strophe.xmlElement(name, attrs, text); this.node.appendChild(child); if (typeof text !== "string" && typeof text !== "number") { this.node = child; } return this; } /** Function: cnode * Add a child to the current element and make it the new current * element. * * This function is the same as c() except that instead of using a * name and an attributes object to create the child it uses an * existing DOM element object. * * Parameters: * (XMLElement) elem - A DOM element. * * Returns: * The Strophe.Builder object. */ cnode(elem) { let impNode; const xmlGen = Strophe.xmlGenerator(); try { impNode = xmlGen.importNode !== undefined; } catch (e) { impNode = false; } const newElem = impNode ? xmlGen.importNode(elem, true) : Strophe.copyElement(elem); this.node.appendChild(newElem); this.node = newElem; return this; } /** Function: t * Add a child text element. * * This *does not* make the child the new current element since there * are no children of text elements. * * Parameters: * (String) text - The text data to append to the current element. * * Returns: * The Strophe.Builder object. */ t(text) { const child = Strophe.xmlTextNode(text); this.node.appendChild(child); return this; } /** Function: h * Replace current element contents with the HTML passed in. * * This *does not* make the child the new current element * * Parameters: * (String) html - The html to insert as contents of current element. * * Returns: * The Strophe.Builder object. */ h(html) { const fragment = Strophe.xmlGenerator().createElement('body'); // force the browser to try and fix any invalid HTML tags fragment.innerHTML = html; // copy cleaned html into an xml dom const xhtml = Strophe.createHtml(fragment); while (xhtml.childNodes.length > 0) { this.node.appendChild(xhtml.childNodes[0]); } return this; } }; /** PrivateClass: Strophe.Handler * _Private_ helper class for managing stanza handlers. * * A Strophe.Handler encapsulates a user provided callback function to be * executed when matching stanzas are received by the connection. * Handlers can be either one-off or persistant depending on their * return value. Returning true will cause a Handler to remain active, and * returning false will remove the Handler. * * Users will not use Strophe.Handler objects directly, but instead they * will use Strophe.Connection.addHandler() and * Strophe.Connection.deleteHandler(). */ /** PrivateConstructor: Strophe.Handler * Create and initialize a new Strophe.Handler. * * Parameters: * (Function) handler - A function to be executed when the handler is run. * (String) ns - The namespace to match. * (String) name - The element name to match. * (String) type - The element type to match. * (String) id - The element id attribute to match. * (String) from - The element from attribute to match. * (Object) options - Handler options * * Returns: * A new Strophe.Handler object. */ Strophe.Handler = function (handler, ns, name, type, id, from, options) { this.handler = handler; this.ns = ns; this.name = name; this.type = type; this.id = id; this.options = options || { 'matchBareFromJid': false, 'ignoreNamespaceFragment': false }; // BBB: Maintain backward compatibility with old `matchBare` option if (this.options.matchBare) { Strophe.warn('The "matchBare" option is deprecated, use "matchBareFromJid" instead.'); this.options.matchBareFromJid = this.options.matchBare; delete this.options.matchBare; } if (this.options.matchBareFromJid) { this.from = from ? Strophe.getBareJidFromJid(from) : null; } else { this.from = from; } // whether the handler is a user handler or a system handler this.user = true; }; Strophe.Handler.prototype = { /** PrivateFunction: getNamespace * Returns the XML namespace attribute on an element. * If `ignoreNamespaceFragment` was passed in for this handler, then the * URL fragment will be stripped. * * Parameters: * (XMLElement) elem - The XML element with the namespace. * * Returns: * The namespace, with optionally the fragment stripped. */ getNamespace(elem) { let elNamespace = elem.getAttribute("xmlns"); if (elNamespace && this.options.ignoreNamespaceFragment) { elNamespace = elNamespace.split('#')[0]; } return elNamespace; }, /** PrivateFunction: namespaceMatch * Tests if a stanza matches the namespace set for this Strophe.Handler. * * Parameters: * (XMLElement) elem - The XML element to test. * * Returns: * true if the stanza matches and false otherwise. */ namespaceMatch(elem) { let nsMatch = false; if (!this.ns) { return true; } else { Strophe.forEachChild(elem, null, elem => { if (this.getNamespace(elem) === this.ns) { nsMatch = true; } }); return nsMatch || this.getNamespace(elem) === this.ns; } }, /** PrivateFunction: isMatch * Tests if a stanza matches the Strophe.Handler. * * Parameters: * (XMLElement) elem - The XML element to test. * * Returns: * true if the stanza matches and false otherwise. */ isMatch(elem) { let from = elem.getAttribute('from'); if (this.options.matchBareFromJid) { from = Strophe.getBareJidFromJid(from); } const elem_type = elem.getAttribute("type"); if (this.namespaceMatch(elem) && (!this.name || Strophe.isTagEqual(elem, this.name)) && (!this.type || (Array.isArray(this.type) ? this.type.indexOf(elem_type) !== -1 : elem_type === this.type)) && (!this.id || elem.getAttribute("id") === this.id) && (!this.from || from === this.from)) { return true; } return false; }, /** PrivateFunction: run * Run the callback on a matching stanza. * * Parameters: * (XMLElement) elem - The DOM element that triggered the * Strophe.Handler. * * Returns: * A boolean indicating if the handler should remain active. */ run(elem) { let result = null; try { result = this.handler(elem); } catch (e) { Strophe._handleError(e); throw e; } return result; }, /** PrivateFunction: toString * Get a String representation of the Strophe.Handler object. * * Returns: * A String. */ toString() { return "{Handler: " + this.handler + "(" + this.name + "," + this.id + "," + this.ns + ")}"; } }; /** PrivateClass: Strophe.TimedHandler * _Private_ helper class for managing timed handlers. * * A Strophe.TimedHandler encapsulates a user provided callback that * should be called after a certain period of time or at regular * intervals. The return value of the callback determines whether the * Strophe.TimedHandler will continue to fire. * * Users will not use Strophe.TimedHandler objects directly, but instead * they will use Strophe.Connection.addTimedHandler() and * Strophe.Connection.deleteTimedHandler(). */ Strophe.TimedHandler = class TimedHandler { /** PrivateConstructor: Strophe.TimedHandler * Create and initialize a new Strophe.TimedHandler object. * * Parameters: * (Integer) period - The number of milliseconds to wait before the * handler is called. * (Function) handler - The callback to run when the handler fires. This * function should take no arguments. * * Returns: * A new Strophe.TimedHandler object. */ constructor(period, handler) { this.period = period; this.handler = handler; this.lastCalled = new Date().getTime(); this.user = true; } /** PrivateFunction: run * Run the callback for the Strophe.TimedHandler. * * Returns: * true if the Strophe.TimedHandler should be called again, and false * otherwise. */ run() { this.lastCalled = new Date().getTime(); return this.handler(); } /** PrivateFunction: reset * Reset the last called time for the Strophe.TimedHandler. */ reset() { this.lastCalled = new Date().getTime(); } /** PrivateFunction: toString * Get a string representation of the Strophe.TimedHandler object. * * Returns: * The string representation. */ toString() { return "{TimedHandler: " + this.handler + "(" + this.period + ")}"; } }; /** Class: Strophe.Connection * XMPP Connection manager. * * This class is the main part of Strophe. It manages a BOSH or websocket * connection to an XMPP server and dispatches events to the user callbacks * as data arrives. It supports SASL PLAIN, SASL SCRAM-SHA-1 * and legacy authentication. * * After creating a Strophe.Connection object, the user will typically * call connect() with a user supplied callback to handle connection level * events like authentication failure, disconnection, or connection * complete. * * The user will also have several event handlers defined by using * addHandler() and addTimedHandler(). These will allow the user code to * respond to interesting stanzas or do something periodically with the * connection. These handlers will be active once authentication is * finished. * * To send data to the connection, use send(). */ /** Constructor: Strophe.Connection * Create and initialize a Strophe.Connection object. * * The transport-protocol for this connection will be chosen automatically * based on the given service parameter. URLs starting with "ws://" or * "wss://" will use WebSockets, URLs starting with "http://", "https://" * or without a protocol will use BOSH. * * To make Strophe connect to the current host you can leave out the protocol * and host part and just pass the path, e.g. * * > let conn = new Strophe.Connection("/http-bind/"); * * Options common to both Websocket and BOSH: * ------------------------------------------ * * cookies: * * The *cookies* option allows you to pass in cookies to be added to the * document. These cookies will then be included in the BOSH XMLHttpRequest * or in the websocket connection. * * The passed in value must be a map of cookie names and string values. * * > { "myCookie": { * > "value": "1234", * > "domain": ".example.org", * > "path": "/", * > "expires": expirationDate * > } * > } * * Note that cookies can't be set in this way for other domains (i.e. cross-domain). * Those cookies need to be set under those domains, for example they can be * set server-side by making a XHR call to that domain to ask it to set any * necessary cookies. * * mechanisms: * * The *mechanisms* option allows you to specify the SASL mechanisms that this * instance of Strophe.Connection (and therefore your XMPP client) will * support. * * The value must be an array of objects with Strophe.SASLMechanism * prototypes. * * If nothing is specified, then the following mechanisms (and their * priorities) are registered: * * SCRAM-SHA-1 - 60 * PLAIN - 50 * OAUTHBEARER - 40 * X-OAUTH2 - 30 * ANONYMOUS - 20 * EXTERNAL - 10 * * explicitResourceBinding: * * If `explicitResourceBinding` is set to a truthy value, then the XMPP client * needs to explicitly call `Strophe.Connection.prototype.bind` once the XMPP * server has advertised the "urn:ietf:params:xml:ns:xmpp-bind" feature. * * Making this step explicit allows client authors to first finish other * stream related tasks, such as setting up an XEP-0198 Stream Management * session, before binding the JID resource for this session. * * WebSocket options: * ------------------ * * protocol: * * If you want to connect to the current host with a WebSocket connection you * can tell Strophe to use WebSockets through a "protocol" attribute in the * optional options parameter. Valid values are "ws" for WebSocket and "wss" * for Secure WebSocket. * So to connect to "wss://CURRENT_HOSTNAME/xmpp-websocket" you would call * * > let conn = new Strophe.Connection("/xmpp-websocket/", {protocol: "wss"}); * * Note that relative URLs _NOT_ starting with a "/" will also include the path * of the current site. * * Also because downgrading security is not permitted by browsers, when using * relative URLs both BOSH and WebSocket connections will use their secure * variants if the current connection to the site is also secure (https). * * worker: * * Set this option to URL from where the shared worker script should be loaded. * * To run the websocket connection inside a shared worker. * This allows you to share a single websocket-based connection between * multiple Strophe.Connection instances, for example one per browser tab. * * The script to use is the one in `src/shared-connection-worker.js`. * * BOSH options: * ------------- * * By adding "sync" to the options, you can control if requests will * be made synchronously or not. The default behaviour is asynchronous. * If you want to make requests synchronous, make "sync" evaluate to true. * > let conn = new Strophe.Connection("/http-bind/", {sync: true}); * * You can also toggle this on an already established connection. * > conn.options.sync = true; * * The *customHeaders* option can be used to provide custom HTTP headers to be * included in the XMLHttpRequests made. * * The *keepalive* option can be used to instruct Strophe to maintain the * current BOSH session across interruptions such as webpage reloads. * * It will do this by caching the sessions tokens in sessionStorage, and when * "restore" is called it will check whether there are cached tokens with * which it can resume an existing session. * * The *withCredentials* option should receive a Boolean value and is used to * indicate wether cookies should be included in ajax requests (by default * they're not). * Set this value to true if you are connecting to a BOSH service * and for some reason need to send cookies to it. * In order for this to work cross-domain, the server must also enable * credentials by setting the Access-Control-Allow-Credentials response header * to "true". For most usecases however this setting should be false (which * is the default). * Additionally, when using Access-Control-Allow-Credentials, the * Access-Control-Allow-Origin header can't be set to the wildcard "*", but * instead must be restricted to actual domains. * * The *contentType* option can be set to change the default Content-Type * of "text/xml; charset=utf-8", which can be useful to reduce the amount of * CORS preflight requests that are sent to the server. * * Parameters: * (String) service - The BOSH or WebSocket service URL. * (Object) options - A hash of configuration options * * Returns: * A new Strophe.Connection object. */ Strophe.Connection = class Connection { constructor(service, options) { // The service URL this.service = service; // Configuration options this.options = options || {}; this.setProtocol(); /* The connected JID. */ this.jid = ""; /* the JIDs domain */ this.domain = null; /* stream:features */ this.features = null; // SASL this._sasl_data = {}; this.do_bind = false; this.do_session = false; this.mechanisms = {}; // handler lists this.timedHandlers = []; this.handlers = []; this.removeTimeds = []; this.removeHandlers = []; this.addTimeds = []; this.addHandlers = []; this.protocolErrorHandlers = { 'HTTP': {}, 'websocket': {} }; this._idleTimeout = null; this._disconnectTimeout = null; this.authenticated = false; this.connected = false; this.disconnecting = false; this.do_authentication = true; this.paused = false; this.restored = false; this._data = []; this._uniqueId = 0; this._sasl_success_handler = null; this._sasl_failure_handler = null; this._sasl_challenge_handler = null; // Max retries before disconnecting this.maxRetries = 5; // Call onIdle callback every 1/10th of a second this._idleTimeout = setTimeout(() => this._onIdle(), 100); utils.addCookies(this.options.cookies); this.registerSASLMechanisms(this.options.mechanisms); // A client must always respond to incoming IQ "set" and "get" stanzas. // See https://datatracker.ietf.org/doc/html/rfc6120#section-8.2.3 // // This is a fallback handler which gets called when no other handler // was called for a received IQ "set" or "get". this.iqFallbackHandler = new Strophe.Handler(iq => this.send($iq({ type: 'error', id: iq.getAttribute('id') }).c('error', { 'type': 'cancel' }).c('service-unavailable', { 'xmlns': Strophe.NS.STANZAS })), null, 'iq', ['get', 'set']); // initialize plugins for (const k in Strophe._connectionPlugins) { if (Object.prototype.hasOwnProperty.call(Strophe._connectionPlugins, k)) { const F = function () {}; F.prototype = Strophe._connectionPlugins[k]; this[k] = new F(); this[k].init(this); } } } /** Function: setProtocol * Select protocal based on this.options or this.service */ setProtocol() { const proto = this.options.protocol || ""; if (this.options.worker) { this._proto = new Strophe.WorkerWebsocket(this); } else if (this.service.indexOf("ws:") === 0 || this.service.indexOf("wss:") === 0 || proto.indexOf("ws") === 0) { this._proto = new Strophe.Websocket(this); } else { this._proto = new Strophe.Bosh(this); } } /** Function: reset * Reset the connection. * * This function should be called after a connection is disconnected * before that connection is reused. */ reset() { this._proto._reset(); // SASL this.do_session = false; this.do_bind = false; // handler lists this.timedHandlers = []; this.handlers = []; this.removeTimeds = []; this.removeHandlers = []; this.addTimeds = []; this.addHandlers = []; this.authenticated = false; this.connected = false; this.disconnecting = false; this.restored = false; this._data = []; this._requests = []; this._uniqueId = 0; } /** Function: pause * Pause the request manager. * * This will prevent Strophe from sending any more requests to the * server. This is very useful for temporarily pausing * BOSH-Connections while a lot of send() calls are happening quickly. * This causes Strophe to send the data in a single request, saving * many request trips. */ pause() { this.paused = true; } /** Function: resume * Resume the request manager. * * This resumes after pause() has been called. */ resume() { this.paused = false; } /** Function: getUniqueId * Generate a unique ID for use in elements. * * All stanzas are required to have unique id attributes. This * function makes creating these easy. Each connection instance has * a counter which starts from zero, and the value of this counter * plus a colon followed by the suffix becomes the unique id. If no * suffix is supplied, the counter is used as the unique id. * * Suffixes are used to make debugging easier when reading the stream * data, and their use is recommended. The counter resets to 0 for * every new connection for the same reason. For connections to the * same server that authenticate the same way, all the ids should be * the same, which makes it easy to see changes. This is useful for * automated testing as well. * * Parameters: * (String) suffix - A optional suffix to append to the id. * * Returns: * A unique string to be used for the id attribute. */ getUniqueId(suffix) { // eslint-disable-line class-methods-use-this const uuid = 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) { const r = Math.random() * 16 | 0, v = c === 'x' ? r : r & 0x3 | 0x8; return v.toString(16); }); if (typeof suffix === "string" || typeof suffix === "number") { return uuid + ":" + suffix; } else { return uuid + ""; } } /** Function: addProtocolErrorHandler * Register a handler function for when a protocol (websocker or HTTP) * error occurs. * * NOTE: Currently only HTTP errors for BOSH requests are handled. * Patches that handle websocket errors would be very welcome. * * Parameters: * (String) protocol - 'HTTP' or 'websocket' * (Integer) status_code - Error status code (e.g 500, 400 or 404) * (Function) callback - Function that will fire on Http error * * Example: * function onError(err_code){ * //do stuff * } * * let conn = Strophe.connect('http://example.com/http-bind'); * conn.addProtocolErrorHandler('HTTP', 500, onError); * // Triggers HTTP 500 error and onError handler will be called * conn.connect('user_jid@incorrect_jabber_host', 'secret', onConnect); */ addProtocolErrorHandler(protocol, status_code, callback) { this.protocolErrorHandlers[protocol][status_code] = callback; } /** Function: connect * Starts the connection process. * * As the connection process proceeds, the user supplied callback will * be triggered multiple times with status updates. The callback * should take two arguments - the status code and the error condition. * * The status code will be one of the values in the Strophe.Status * constants. The error condition will be one of the conditions * defined in RFC 3920 or the condition 'strophe-parsererror'. * * The Parameters _wait_, _hold_ and _route_ are optional and only relevant * for BOSH connections. Please see XEP 124 for a more detailed explanation * of the optional parameters. * * Parameters: * (String) jid - The user's JID. This may be a bare JID, * or a full JID. If a node is not supplied, SASL OAUTHBEARER or * SASL ANONYMOUS authentication will be attempted (OAUTHBEARER will * process the provided password value as an access token). * (String) pass - The user's password. * (Function) callback - The connect callback function. * (Integer) wait - The optional HTTPBIND wait value. This is the * time the server will wait before returning an empty result for * a request. The default setting of 60 seconds is recommended. * (Integer) hold - The optional HTTPBIND hold value. This is the * number of connections the server will hold at one time. This * should almost always be set to 1 (the default). * (String) route - The optional route value. * (String) authcid - The optional alternative authentication identity * (username) if intending to impersonate another user. * When using the SASL-EXTERNAL authentication mechanism, for example * with client certificates, then the authcid value is used to * determine whether an authorization JID (authzid) should be sent to * the server. The authzid should NOT be sent to the server if the * authzid and authcid are the same. So to prevent it from being sent * (for example when the JID is already contained in the client * certificate), set authcid to that same JID. See XEP-178 for more * details. * (Integer) disconnection_timeout - The optional disconnection timeout * in milliseconds before _doDisconnect will be called. */ connect(jid, pass, callback, wait, hold, route, authcid) { let disconnection_timeout = arguments.length > 7 && arguments[7] !== undefined ? arguments[7] : 3000; this.jid = jid; /** Variable: authzid * Authorization identity. */ this.authzid = Strophe.getBareJidFromJid(this.jid); /** Variable: authcid * Authentication identity (User name). */ this.authcid = authcid || Strophe.getNodeFromJid(this.jid); /** Variable: pass * Authentication identity (User password). */ this.pass = pass; this.connect_callback = callback; this.disconnecting = false; this.connected = false; this.authenticated = false; this.restored = false; this.disconnection_timeout = disconnection_timeout; // parse jid for domain this.domain = Strophe.getDomainFromJid(this.jid); this._changeConnectStatus(Strophe.Status.CONNECTING, null); this._proto._connect(wait, hold, route); } /** Function: attach * Attach to an already created and authenticated BOSH session. * * This function is provided to allow Strophe to attach to BOSH * sessions which have been created externally, perhaps by a Web * application. This is often used to support auto-login type features * without putting user credentials into the page. * * Parameters: * (String) jid - The full JID that is bound by the session. * (String) sid - The SID of the BOSH session. * (String) rid - The current RID of the BOSH session. This RID * will be used by the next request. * (Function) callback The connect callback function. * (Integer) wait - The optional HTTPBIND wait value. This is the * time the server will wait before returning an empty result for * a request. The default setting of 60 seconds is recommended. * Other settings will require tweaks to the Strophe.TIMEOUT value. * (Integer) hold - The optional HTTPBIND hold value. This is the * number of connections the server will hold at one time. This * should almost always be set to 1 (the default). * (Integer) wind - The optional HTTBIND window value. This is the * allowed range of request ids that are valid. The default is 5. */ attach(jid, sid, rid, callback, wait, hold, wind) { if (this._proto._attach) { return this._proto._attach(jid, sid, rid, callback, wait, hold, wind); } else { const error = new Error('The "attach" method is not available for your connection protocol'); error.name = 'StropheSessionError'; throw error; } } /** Function: restore * Attempt to restore a cached BOSH session. * * This function is only useful in conjunction with providing the * "keepalive":true option when instantiating a new Strophe.Connection. * * When "keepalive" is set to true, Strophe will cache the BOSH tokens * RID (Request ID) and SID (Session ID) and then when this function is * called, it will attempt to restore the session from those cached * tokens. * * This function must therefore be called instead of connect or attach. * * For an example on how to use it, please see examples/restore.js * * Parameters: * (String) jid - The user's JID. This may be a bare JID or a full JID. * (Function) callback - The connect callback function. * (Integer) wait - The optional HTTPBIND wait value. This is the * time the server will wait before returning an empty result for * a request. The default setting of 60 seconds is recommended. * (Integer) hold - The optional HTTPBIND hold value. This is the * number of connections the server will hold at one time. This * should almost always be set to 1 (the default). * (Integer) wind - The optional HTTBIND window value. This is the * allowed range of request ids that are valid. The default is 5. */ restore(jid, callback, wait, hold, wind) { if (this._sessionCachingSupported()) { this._proto._restore(jid, callback, wait, hold, wind); } else { const error = new Error('The "restore" method can only be used with a BOSH connection.'); error.name = 'StropheSessionError'; throw error; } } /** PrivateFunction: _sessionCachingSupported * Checks whether sessionStorage and JSON are supported and whether we're * using BOSH. */ _sessionCachingSupported() { if (this._proto instanceof Strophe.Bosh) { if (!JSON) { return false; } try { sessionStorage.setItem('_strophe_', '_strophe_'); sessionStorage.removeItem('_strophe_'); } catch (e) { return false; } return true; } return false; } /** Function: xmlInput * User overrideable function that receives XML data coming into the * connection. * * The default function does nothing. User code can override this with * > Strophe.Connection.xmlInput = function (elem) { * > (user code) * > }; * * Due to limitations of current Browsers' XML-Parsers the opening and closing * tag for WebSocket-Connoctions will be passed as selfclosing here. * * BOSH-Connections will have all stanzas wrapped in a tag. See * if you want to strip this tag. * * Parameters: * (XMLElement) elem - The XML data received by the connection. */ xmlInput(elem) { // eslint-disable-line return; } /** Function: xmlOutput * User overrideable function that receives XML data sent to the * connection. * * The default function does nothing. User code can override this with * > Strophe.Connection.xmlOutput = function (elem) { * > (user code) * > }; * * Due to limitations of current Browsers' XML-Parsers the opening and closing * tag for WebSocket-Connoctions will be passed as selfclosing here. * * BOSH-Connections will have all stanzas wrapped in a tag. See * if you want to strip this tag. * * Parameters: * (XMLElement) elem - The XMLdata sent by the connection. */ xmlOutput(elem) { // eslint-disable-line return; } /** Function: rawInput * User overrideable function that receives raw data coming into the * connection. * * The default function does nothing. User code can override this with * > Strophe.Connection.rawInput = function (data) { * > (user code) * > }; * * Parameters: * (String) data - The data received by the connection. */ rawInput(data) { // eslint-disable-line return; } /** Function: rawOutput * User overrideable function that receives raw data sent to the * connection. * * The default function does nothing. User code can override this with * > Strophe.Connection.rawOutput = function (data) { * > (user code) * > }; * * Parameters: * (String) data - The data sent by the connection. */ rawOutput(data) { // eslint-disable-line return; } /** Function: nextValidRid * User overrideable function that receives the new valid rid. * * The default function does nothing. User code can override this with * > Strophe.Connection.nextValidRid = function (rid) { * > (user code) * > }; * * Parameters: * (Number) rid - The next valid rid */ nextValidRid(rid) { // eslint-disable-line return; } /** Function: send * Send a stanza. * * This function is called to push data onto the send queue to * go out over the wire. Whenever a request is sent to the BOSH * server, all pending data is sent and the queue is flushed. * * Parameters: * (XMLElement | * [XMLElement] | * Strophe.Builder) elem - The stanza to send. */ send(elem) { if (elem === null) { return; } if (typeof elem.sort === "function") { for (let i = 0; i < elem.length; i++) { this._queueData(elem[i]); } } else if (typeof elem.tree === "function") { this._queueData(elem.tree()); } else { this._queueData(elem); } this._proto._send(); } /** Function: flush * Immediately send any pending outgoing data. * * Normally send() queues outgoing data until the next idle period * (100ms), which optimizes network use in the common cases when * several send()s are called in succession. flush() can be used to * immediately send all pending data. */ flush() { // cancel the pending idle period and run the idle function // immediately clearTimeout(this._idleTimeout); this._onIdle(); } /** Function: sendPresence * Helper function to send presence stanzas. The main benefit is for * sending presence stanzas for which you expect a responding presence * stanza with the same id (for example when leaving a chat room). * * Parameters: * (XMLElement) elem - The stanza to send. * (Function) callback - The callback function for a successful request. * (Function) errback - The callback function for a failed or timed * out request. On timeout, the stanza will be null. * (Integer) timeout - The time specified in milliseconds for a * timeout to occur. * * Returns: * The id used to send the presence. */ sendPresence(elem, callback, errback, timeout) { let timeoutHandler = null; if (typeof elem.tree === "function") { elem = elem.tree(); } let id = elem.getAttribute('id'); if (!id) { // inject id if not found id = this.getUniqueId("sendPresence"); elem.setAttribute("id", id); } if (typeof callback === "function" || typeof errback === "function") { const handler = this.addHandler(stanza => { // remove timeout handler if there is one if (timeoutHandler) { this.deleteTimedHandler(timeoutHandler); } if (stanza.getAttribute('type') === 'error') { if (errback) { errback(stanza); } } else if (callback) { callback(stanza); } }, null, 'presence', null, id); // if timeout specified, set up a timeout handler. if (timeout) { timeoutHandler = this.addTimedHandler(timeout, () => { // get rid of normal handler this.deleteHandler(handler); // call errback on timeout with null stanza if (errback) { errback(null); } return false; }); } } this.send(elem); return id; } /** Function: sendIQ * Helper function to send IQ stanzas. * * Parameters: * (XMLElement) elem - The stanza to send. * (Function) callback - The callback function for a successful request. * (Function) errback - The callback function for a failed or timed * out request. On timeout, the stanza will be null. * (Integer) timeout - The time specified in milliseconds for a * timeout to occur. * * Returns: * The id used to send the IQ. */ sendIQ(elem, callback, errback, timeout) { let timeoutHandler = null; if (typeof elem.tree === "function") { elem = elem.tree(); } let id = elem.getAttribute('id'); if (!id) { // inject id if not found id = this.getUniqueId("sendIQ"); elem.setAttribute("id", id); } if (typeof callback === "function" || typeof errback === "function") { const handler = this.addHandler(stanza => { // remove timeout handler if there is one if (timeoutHandler) { this.deleteTimedHandler(timeoutHandler); } const iqtype = stanza.getAttribute('type'); if (iqtype === 'result') { if (callback) { callback(stanza); } } else if (iqtype === 'error') { if (errback) { errback(stanza); } } else { const error = new Error(`Got bad IQ type of ${iqtype}`); error.name = "StropheError"; throw error; } }, null, 'iq', ['error', 'result'], id); // if timeout specified, set up a timeout handler. if (timeout) { timeoutHandler = this.addTimedHandler(timeout, () => { // get rid of normal handler this.deleteHandler(handler); // call errback on timeout with null stanza if (errback) { errback(null); } return false; }); } } this.send(elem); return id; } /** PrivateFunction: _queueData * Queue outgoing data for later sending. Also ensures that the data * is a DOMElement. */ _queueData(element) { if (element === null || !element.tagName || !element.childNodes) { const error = new Error("Cannot queue non-DOMElement."); error.name = "StropheError"; throw error; } this._data.push(element); } /** PrivateFunction: _sendRestart * Send an xmpp:restart stanza. */ _sendRestart() { this._data.push("restart"); this._proto._sendRestart(); this._idleTimeout = setTimeout(() => this._onIdle(), 100); } /** Function: addTimedHandler * Add a timed handler to the connection. * * This function adds a timed handler. The provided handler will * be called every period milliseconds until it returns false, * the connection is terminated, or the handler is removed. Handlers * that wish to continue being invoked should return true. * * Because of method binding it is necessary to save the result of * this function if you wish to remove a handler with * deleteTimedHandler(). * * Note that user handlers are not active until authentication is * successful. * * Parameters: * (Integer) period - The period of the handler. * (Function) handler - The callback function. * * Returns: * A reference to the handler that can be used to remove it. */ addTimedHandler(period, handler) { const thand = new Strophe.TimedHandler(period, handler); this.addTimeds.push(thand); return thand; } /** Function: deleteTimedHandler * Delete a timed handler for a connection. * * This function removes a timed handler from the connection. The * handRef parameter is *not* the function passed to addTimedHandler(), * but is the reference returned from addTimedHandler(). * * Parameters: * (Strophe.TimedHandler) handRef - The handler reference. */ deleteTimedHandler(handRef) { // this must be done in the Idle loop so that we don't change // the handlers during iteration this.removeTimeds.push(handRef); } /** Function: addHandler * Add a stanza handler for the connection. * * This function adds a stanza handler to the connection. The * handler callback will be called for any stanza that matches * the parameters. Note that if multiple parameters are supplied, * they must all match for the handler to be invoked. * * The handler will receive the stanza that triggered it as its argument. * *The handler should return true if it is to be invoked again; * returning false will remove the handler after it returns.* * * As a convenience, the ns parameters applies to the top level element * and also any of its immediate children. This is primarily to make * matching /iq/query elements easy. * * Options * ~~~~~~~ * With the options argument, you can specify boolean flags that affect how * matches are being done. * * Currently two flags exist: * * - matchBareFromJid: * When set to true, the from parameter and the * from attribute on the stanza will be matched as bare JIDs instead * of full JIDs. To use this, pass {matchBareFromJid: true} as the * value of options. The default value for matchBareFromJid is false. * * - ignoreNamespaceFragment: * When set to true, a fragment specified on the stanza's namespace * URL will be ignored when it's matched with the one configured for * the handler. * * This means that if you register like this: * > connection.addHandler( * > handler, * > 'http://jabber.org/protocol/muc', * > null, null, null, null, * > {'ignoreNamespaceFragment': true} * > ); * * Then a stanza with XML namespace of * 'http://jabber.org/protocol/muc#user' will also be matched. If * 'ignoreNamespaceFragment' is false, then only stanzas with * 'http://jabber.org/protocol/muc' will be matched. * * Deleting the handler * ~~~~~~~~~~~~~~~~~~~~ * The return value should be saved if you wish to remove the handler * with deleteHandler(). * * Parameters: * (Function) handler - The user callback. * (String) ns - The namespace to match. * (String) name - The stanza name to match. * (String|Array) type - The stanza type (or types if an array) to match. * (String) id - The stanza id attribute to match. * (String) from - The stanza from attribute to match. * (String) options - The handler options * * Returns: * A reference to the handler that can be used to remove it. */ addHandler(handler, ns, name, type, id, from, options) { const hand = new Strophe.Handler(handler, ns, name, type, id, from, options); this.addHandlers.push(hand); return hand; } /** Function: deleteHandler * Delete a stanza handler for a connection. * * This function removes a stanza handler from the connection. The * handRef parameter is *not* the function passed to addHandler(), * but is the reference returned from addHandler(). * * Parameters: * (Strophe.Handler) handRef - The handler reference. */ deleteHandler(handRef) { // this must be done in the Idle loop so that we don't change // the handlers during iteration this.removeHandlers.push(handRef); // If a handler is being deleted while it is being added, // prevent it from getting added const i = this.addHandlers.indexOf(handRef); if (i >= 0) { this.addHandlers.splice(i, 1); } } /** Function: registerSASLMechanisms * * Register the SASL mechanisms which will be supported by this instance of * Strophe.Connection (i.e. which this XMPP client will support). * * Parameters: * (Array) mechanisms - Array of objects with Strophe.SASLMechanism prototypes * */ registerSASLMechanisms(mechanisms) { this.mechanisms = {}; mechanisms = mechanisms || [Strophe.SASLAnonymous, Strophe.SASLExternal, Strophe.SASLOAuthBearer, Strophe.SASLXOAuth2, Strophe.SASLPlain, Strophe.SASLSHA1]; mechanisms.forEach(m => this.registerSASLMechanism(m)); } /** Function: registerSASLMechanism * * Register a single SASL mechanism, to be supported by this client. * * Parameters: * (Object) mechanism - Object with a Strophe.SASLMechanism prototype * */ registerSASLMechanism(Mechanism) { const mechanism = new Mechanism(); this.mechanisms[mechanism.mechname] = mechanism; } /** Function: disconnect * Start the graceful disconnection process. * * This function starts the disconnection process. This process starts * by sending unavailable presence and sending BOSH body of type * terminate. A timeout handler makes sure that disconnection happens * even if the BOSH server does not respond. * If the Connection object isn't connected, at least tries to abort all pending requests * so the connection object won't generate successful requests (which were already opened). * * The user supplied connection callback will be notified of the * progress as this process happens. * * Parameters: * (String) reason - The reason the disconnect is occuring. */ disconnect(reason) { this._changeConnectStatus(Strophe.Status.DISCONNECTING, reason); if (reason) { Strophe.warn("Disconnect was called because: " + reason); } else { Strophe.info("Disconnect was called"); } if (this.connected) { let pres = false; this.disconnecting = true; if (this.authenticated) { pres = $pres({ 'xmlns': Strophe.NS.CLIENT, 'type': 'unavailable' }); } // setup timeout handler this._disconnectTimeout = this._addSysTimedHandler(this.disconnection_timeout, this._onDisconnectTimeout.bind(this)); this._proto._disconnect(pres); } else { Strophe.warn("Disconnect was called before Strophe connected to the server"); this._proto._abortAllRequests(); this._doDisconnect(); } } /** PrivateFunction: _changeConnectStatus * _Private_ helper function that makes sure plugins and the user's * callback are notified of connection status changes. * * Parameters: * (Integer) status - the new connection status, one of the values * in Strophe.Status * (String) condition - the error condition or null * (XMLElement) elem - The triggering stanza. */ _changeConnectStatus(status, condition, elem) { // notify all plugins listening for status changes for (const k in Strophe._connectionPlugins) { if (Object.prototype.hasOwnProperty.call(Strophe._connectionPlugins, k)) { const plugin = this[k]; if (plugin.statusChanged) { try { plugin.statusChanged(status, condition); } catch (err) { Strophe.error(`${k} plugin caused an exception changing status: ${err}`); } } } } // notify the user's callback if (this.connect_callback) { try { this.connect_callback(status, condition, elem); } catch (e) { Strophe._handleError(e); Strophe.error(`User connection callback caused an exception: ${e}`); } } } /** PrivateFunction: _doDisconnect * _Private_ function to disconnect. * * This is the last piece of the disconnection logic. This resets the * connection and alerts the user's connection callback. */ _doDisconnect(condition) { if (typeof this._idleTimeout === "number") { clearTimeout(this._idleTimeout); } // Cancel Disconnect Timeout if (this._disconnectTimeout !== null) { this.deleteTimedHandler(this._disconnectTimeout); this._disconnectTimeout = null; } Strophe.debug("_doDisconnect was called"); this._proto._doDisconnect(); this.authenticated = false; this.disconnecting = false; this.restored = false; // delete handlers this.handlers = []; this.timedHandlers = []; this.removeTimeds = []; this.removeHandlers = []; this.addTimeds = []; this.addHandlers = []; // tell the parent we disconnected this._changeConnectStatus(Strophe.Status.DISCONNECTED, condition); this.connected = false; } /** PrivateFunction: _dataRecv * _Private_ handler to processes incoming data from the the connection. * * Except for _connect_cb handling the initial connection request, * this function handles the incoming data for all requests. This * function also fires stanza handlers that match each incoming * stanza. * * Parameters: * (Strophe.Request) req - The request that has data ready. * (string) req - The stanza a raw string (optiona). */ _dataRecv(req, raw) { const elem = this._proto._reqToData(req); if (elem === null) { return; } if (this.xmlInput !== Strophe.Connection.prototype.xmlInput) { if (elem.nodeName === this._proto.strip && elem.childNodes.length) { this.xmlInput(elem.childNodes[0]); } else { this.xmlInput(elem); } } if (this.rawInput !== Strophe.Connection.prototype.rawInput) { if (raw) { this.rawInput(raw); } else { this.rawInput(Strophe.serialize(elem)); } } // remove handlers scheduled for deletion while (this.removeHandlers.length > 0) { const hand = this.removeHandlers.pop(); const i = this.handlers.indexOf(hand); if (i >= 0) { this.handlers.splice(i, 1); } } // add handlers scheduled for addition while (this.addHandlers.length > 0) { this.handlers.push(this.addHandlers.pop()); } // handle graceful disconnect if (this.disconnecting && this._proto._emptyQueue()) { this._doDisconnect(); return; } const type = elem.getAttribute("type"); if (type !== null && type === "terminate") { // Don't process stanzas that come in after disconnect if (this.disconnecting) { return; } // an error occurred let cond = elem.getAttribute("condition"); const conflict = elem.getElementsByTagName("conflict"); if (cond !== null) { if (cond === "remote-stream-error" && conflict.length > 0) { cond = "conflict"; } this._changeConnectStatus(Strophe.Status.CONNFAIL, cond); } else { this._changeConnectStatus(Strophe.Status.CONNFAIL, Strophe.ErrorCondition.UNKOWN_REASON); } this._doDisconnect(cond); return; } // send each incoming stanza through the handler chain Strophe.forEachChild(elem, null, child => { const matches = []; this.handlers = this.handlers.reduce((handlers, handler) => { try { if (handler.isMatch(child) && (this.authenticated || !handler.user)) { if (handler.run(child)) { handlers.push(handler); } matches.push(handler); } else { handlers.push(handler); } } catch (e) { // if the handler throws an exception, we consider it as false Strophe.warn('Removing Strophe handlers due to uncaught exception: ' + e.message); } return handlers; }, []); // If no handler was fired for an incoming IQ with type="set", // then we return an IQ error stanza with service-unavailable. if (!matches.length && this.iqFallbackHandler.isMatch(child)) { this.iqFallbackHandler.run(child); } }); } /** PrivateFunction: _connect_cb * _Private_ handler for initial connection request. * * This handler is used to process the initial connection request * response from the BOSH server. It is used to set up authentication * handlers and start the authentication process. * * SASL authentication will be attempted if available, otherwise * the code will fall back to legacy authentication. * * Parameters: * (Strophe.Request) req - The current request. * (Function) _callback - low level (xmpp) connect callback function. * Useful for plugins with their own xmpp connect callback (when they * want to do something special). */ _connect_cb(req, _callback, raw) { Strophe.debug("_connect_cb was called"); this.connected = true; let bodyWrap; try { bodyWrap = this._proto._reqToData(req); } catch (e) { if (e.name !== Strophe.ErrorCondition.BAD_FORMAT) { throw e; } this._changeConnectStatus(Strophe.Status.CONNFAIL, Strophe.ErrorCondition.BAD_FORMAT); this._doDisconnect(Strophe.ErrorCondition.BAD_FORMAT); } if (!bodyWrap) { return; } if (this.xmlInput !== Strophe.Connection.prototype.xmlInput) { if (bodyWrap.nodeName === this._proto.strip && bodyWrap.childNodes.length) { this.xmlInput(bodyWrap.childNodes[0]); } else { this.xmlInput(bodyWrap); } } if (this.rawInput !== Strophe.Connection.prototype.rawInput) { if (raw) { this.rawInput(raw); } else { this.rawInput(Strophe.serialize(bodyWrap)); } } const conncheck = this._proto._connect_cb(bodyWrap); if (conncheck === Strophe.Status.CONNFAIL) { return; } // Check for the stream:features tag let hasFeatures; if (bodyWrap.getElementsByTagNameNS) { hasFeatures = bodyWrap.getElementsByTagNameNS(Strophe.NS.STREAM, "features").length > 0; } else { hasFeatures = bodyWrap.getElementsByTagName("stream:features").length > 0 || bodyWrap.getElementsByTagName("features").length > 0; } if (!hasFeatures) { this._proto._no_auth_received(_callback); return; } const matched = Array.from(bodyWrap.getElementsByTagName("mechanism")).map(m => this.mechanisms[m.textContent]).filter(m => m); if (matched.length === 0) { if (bodyWrap.getElementsByTagName("auth").length === 0) { // There are no matching SASL mechanisms and also no legacy // auth available. this._proto._no_auth_received(_callback); return; } } if (this.do_authentication !== false) { this.authenticate(matched); } } /** Function: sortMechanismsByPriority * * Sorts an array of objects with prototype SASLMechanism according to * their priorities. * * Parameters: * (Array) mechanisms - Array of SASL mechanisms. * */ sortMechanismsByPriority(mechanisms) { // eslint-disable-line class-methods-use-this // Sorting mechanisms according to priority. for (let i = 0; i < mechanisms.length - 1; ++i) { let higher = i; for (let j = i + 1; j < mechanisms.length; ++j) { if (mechanisms[j].priority > mechanisms[higher].priority) { higher = j; } } if (higher !== i) { const swap = mechanisms[i]; mechanisms[i] = mechanisms[higher]; mechanisms[higher] = swap; } } return mechanisms; } /** Function: authenticate * Set up authentication * * Continues the initial connection request by setting up authentication * handlers and starting the authentication process. * * SASL authentication will be attempted if available, otherwise * the code will fall back to legacy authentication. * * Parameters: * (Array) matched - Array of SASL mechanisms supported. * */ authenticate(matched) { if (!this._attemptSASLAuth(matched)) { this._attemptLegacyAuth(); } } /** PrivateFunction: _attemptSASLAuth * * Iterate through an array of SASL mechanisms and attempt authentication * with the highest priority (enabled) mechanism. * * Parameters: * (Array) mechanisms - Array of SASL mechanisms. * * Returns: * (Boolean) mechanism_found - true or false, depending on whether a * valid SASL mechanism was found with which authentication could be * started. */ _attemptSASLAuth(mechanisms) { mechanisms = this.sortMechanismsByPriority(mechanisms || []); let mechanism_found = false; for (let i = 0; i < mechanisms.length; ++i) { if (!mechanisms[i].test(this)) { continue; } this._sasl_success_handler = this._addSysHandler(this._sasl_success_cb.bind(this), null, "success", null, null); this._sasl_failure_handler = this._addSysHandler(this._sasl_failure_cb.bind(this), null, "failure", null, null); this._sasl_challenge_handler = this._addSysHandler(this._sasl_challenge_cb.bind(this), null, "challenge", null, null); this._sasl_mechanism = mechanisms[i]; this._sasl_mechanism.onStart(this); const request_auth_exchange = $build("auth", { 'xmlns': Strophe.NS.SASL, 'mechanism': this._sasl_mechanism.mechname }); if (this._sasl_mechanism.isClientFirst) { const response = this._sasl_mechanism.clientChallenge(this); request_auth_exchange.t((0,abab.btoa)(response)); } this.send(request_auth_exchange.tree()); mechanism_found = true; break; } return mechanism_found; } /** PrivateFunction: _sasl_challenge_cb * _Private_ handler for the SASL challenge * */ _sasl_challenge_cb(elem) { const challenge = (0,abab.atob)(Strophe.getText(elem)); const response = this._sasl_mechanism.onChallenge(this, challenge); const stanza = $build('response', { 'xmlns': Strophe.NS.SASL }); if (response !== "") { stanza.t((0,abab.btoa)(response)); } this.send(stanza.tree()); return true; } /** PrivateFunction: _attemptLegacyAuth * * Attempt legacy (i.e. non-SASL) authentication. */ _attemptLegacyAuth() { if (Strophe.getNodeFromJid(this.jid) === null) { // we don't have a node, which is required for non-anonymous // client connections this._changeConnectStatus(Strophe.Status.CONNFAIL, Strophe.ErrorCondition.MISSING_JID_NODE); this.disconnect(Strophe.ErrorCondition.MISSING_JID_NODE); } else { // Fall back to legacy authentication this._changeConnectStatus(Strophe.Status.AUTHENTICATING, null); this._addSysHandler(this._onLegacyAuthIQResult.bind(this), null, null, null, "_auth_1"); this.send($iq({ 'type': "get", 'to': this.domain, 'id': "_auth_1" }).c("query", { xmlns: Strophe.NS.AUTH }).c("username", {}).t(Strophe.getNodeFromJid(this.jid)).tree()); } } /** PrivateFunction: _onLegacyAuthIQResult * _Private_ handler for legacy authentication. * * This handler is called in response to the initial * for legacy authentication. It builds an authentication and * sends it, creating a handler (calling back to _auth2_cb()) to * handle the result * * Parameters: * (XMLElement) elem - The stanza that triggered the callback. * * Returns: * false to remove the handler. */ _onLegacyAuthIQResult(elem) { // eslint-disable-line no-unused-vars // build plaintext auth iq const iq = $iq({ type: "set", id: "_auth_2" }).c('query', { xmlns: Strophe.NS.AUTH }).c('username', {}).t(Strophe.getNodeFromJid(this.jid)).up().c('password').t(this.pass); if (!Strophe.getResourceFromJid(this.jid)) { // since the user has not supplied a resource, we pick // a default one here. unlike other auth methods, the server // cannot do this for us. this.jid = Strophe.getBareJidFromJid(this.jid) + '/strophe'; } iq.up().c('resource', {}).t(Strophe.getResourceFromJid(this.jid)); this._addSysHandler(this._auth2_cb.bind(this), null, null, null, "_auth_2"); this.send(iq.tree()); return false; } /** PrivateFunction: _sasl_success_cb * _Private_ handler for succesful SASL authentication. * * Parameters: * (XMLElement) elem - The matching stanza. * * Returns: * false to remove the handler. */ _sasl_success_cb(elem) { if (this._sasl_data["server-signature"]) { let serverSignature; const success = (0,abab.atob)(Strophe.getText(elem)); const attribMatch = /([a-z]+)=([^,]+)(,|$)/; const matches = success.match(attribMatch); if (matches[1] === "v") { serverSignature = matches[2]; } if (serverSignature !== this._sasl_data["server-signature"]) { // remove old handlers this.deleteHandler(this._sasl_failure_handler); this._sasl_failure_handler = null; if (this._sasl_challenge_handler) { this.deleteHandler(this._sasl_challenge_handler); this._sasl_challenge_handler = null; } this._sasl_data = {}; return this._sasl_failure_cb(null); } } Strophe.info("SASL authentication succeeded."); if (this._sasl_mechanism) { this._sasl_mechanism.onSuccess(); } // remove old handlers this.deleteHandler(this._sasl_failure_handler); this._sasl_failure_handler = null; if (this._sasl_challenge_handler) { this.deleteHandler(this._sasl_challenge_handler); this._sasl_challenge_handler = null; } const streamfeature_handlers = []; const wrapper = (handlers, elem) => { while (handlers.length) { this.deleteHandler(handlers.pop()); } this._onStreamFeaturesAfterSASL(elem); return false; }; streamfeature_handlers.push(this._addSysHandler(elem => wrapper(streamfeature_handlers, elem), null, "stream:features", null, null)); streamfeature_handlers.push(this._addSysHandler(elem => wrapper(streamfeature_handlers, elem), Strophe.NS.STREAM, "features", null, null)); // we must send an xmpp:restart now this._sendRestart(); return false; } /** PrivateFunction: _onStreamFeaturesAfterSASL * Parameters: * (XMLElement) elem - The matching stanza. * * Returns: * false to remove the handler. */ _onStreamFeaturesAfterSASL(elem) { // save stream:features for future usage this.features = elem; for (let i = 0; i < elem.childNodes.length; i++) { const child = elem.childNodes[i]; if (child.nodeName === 'bind') { this.do_bind = true; } if (child.nodeName === 'session') { this.do_session = true; } } if (!this.do_bind) { this._changeConnectStatus(Strophe.Status.AUTHFAIL, null); return false; } else if (!this.options.explicitResourceBinding) { this.bind(); } else { this._changeConnectStatus(Strophe.Status.BINDREQUIRED, null); } return false; } /** Function: bind * * Sends an IQ to the XMPP server to bind a JID resource for this session. * * https://tools.ietf.org/html/rfc6120#section-7.5 * * If `explicitResourceBinding` was set to a truthy value in the options * passed to the Strophe.Connection constructor, then this function needs * to be called explicitly by the client author. * * Otherwise it'll be called automatically as soon as the XMPP server * advertises the "urn:ietf:params:xml:ns:xmpp-bind" stream feature. */ bind() { if (!this.do_bind) { Strophe.log(Strophe.LogLevel.INFO, `Strophe.Connection.prototype.bind called but "do_bind" is false`); return; } this._addSysHandler(this._onResourceBindResultIQ.bind(this), null, null, null, "_bind_auth_2"); const resource = Strophe.getResourceFromJid(this.jid); if (resource) { this.send($iq({ type: "set", id: "_bind_auth_2" }).c('bind', { xmlns: Strophe.NS.BIND }).c('resource', {}).t(resource).tree()); } else { this.send($iq({ type: "set", id: "_bind_auth_2" }).c('bind', { xmlns: Strophe.NS.BIND }).tree()); } } /** PrivateFunction: _onResourceBindIQ * _Private_ handler for binding result and session start. * * Parameters: * (XMLElement) elem - The matching stanza. * * Returns: * false to remove the handler. */ _onResourceBindResultIQ(elem) { if (elem.getAttribute("type") === "error") { Strophe.warn("Resource binding failed."); const conflict = elem.getElementsByTagName("conflict"); let condition; if (conflict.length > 0) { condition = Strophe.ErrorCondition.CONFLICT; } this._changeConnectStatus(Strophe.Status.AUTHFAIL, condition, elem); return false; } // TODO - need to grab errors const bind = elem.getElementsByTagName("bind"); if (bind.length > 0) { const jidNode = bind[0].getElementsByTagName("jid"); if (jidNode.length > 0) { this.authenticated = true; this.jid = Strophe.getText(jidNode[0]); if (this.do_session) { this._establishSession(); } else { this._changeConnectStatus(Strophe.Status.CONNECTED, null); } } } else { Strophe.warn("Resource binding failed."); this._changeConnectStatus(Strophe.Status.AUTHFAIL, null, elem); return false; } } /** PrivateFunction: _establishSession * Send IQ request to establish a session with the XMPP server. * * See https://xmpp.org/rfcs/rfc3921.html#session * * Note: The protocol for session establishment has been determined as * unnecessary and removed in RFC-6121. */ _establishSession() { if (!this.do_session) { throw new Error(`Strophe.Connection.prototype._establishSession ` + `called but apparently ${Strophe.NS.SESSION} wasn't advertised by the server`); } this._addSysHandler(this._onSessionResultIQ.bind(this), null, null, null, "_session_auth_2"); this.send($iq({ type: "set", id: "_session_auth_2" }).c('session', { xmlns: Strophe.NS.SESSION }).tree()); } /** PrivateFunction: _onSessionResultIQ * _Private_ handler for the server's IQ response to a client's session * request. * * This sets Connection.authenticated to true on success, which * starts the processing of user handlers. * * See https://xmpp.org/rfcs/rfc3921.html#session * * Note: The protocol for session establishment has been determined as * unnecessary and removed in RFC-6121. * * Parameters: * (XMLElement) elem - The matching stanza. * * Returns: * false to remove the handler. */ _onSessionResultIQ(elem) { if (elem.getAttribute("type") === "result") { this.authenticated = true; this._changeConnectStatus(Strophe.Status.CONNECTED, null); } else if (elem.getAttribute("type") === "error") { this.authenticated = false; Strophe.warn("Session creation failed."); this._changeConnectStatus(Strophe.Status.AUTHFAIL, null, elem); return false; } return false; } /** PrivateFunction: _sasl_failure_cb * _Private_ handler for SASL authentication failure. * * Parameters: * (XMLElement) elem - The matching stanza. * * Returns: * false to remove the handler. */ _sasl_failure_cb(elem) { // delete unneeded handlers if (this._sasl_success_handler) { this.deleteHandler(this._sasl_success_handler); this._sasl_success_handler = null; } if (this._sasl_challenge_handler) { this.deleteHandler(this._sasl_challenge_handler); this._sasl_challenge_handler = null; } if (this._sasl_mechanism) this._sasl_mechanism.onFailure(); this._changeConnectStatus(Strophe.Status.AUTHFAIL, null, elem); return false; } /** PrivateFunction: _auth2_cb * _Private_ handler to finish legacy authentication. * * This handler is called when the result from the jabber:iq:auth * stanza is returned. * * Parameters: * (XMLElement) elem - The stanza that triggered the callback. * * Returns: * false to remove the handler. */ _auth2_cb(elem) { if (elem.getAttribute("type") === "result") { this.authenticated = true; this._changeConnectStatus(Strophe.Status.CONNECTED, null); } else if (elem.getAttribute("type") === "error") { this._changeConnectStatus(Strophe.Status.AUTHFAIL, null, elem); this.disconnect('authentication failed'); } return false; } /** PrivateFunction: _addSysTimedHandler * _Private_ function to add a system level timed handler. * * This function is used to add a Strophe.TimedHandler for the * library code. System timed handlers are allowed to run before * authentication is complete. * * Parameters: * (Integer) period - The period of the handler. * (Function) handler - The callback function. */ _addSysTimedHandler(period, handler) { const thand = new Strophe.TimedHandler(period, handler); thand.user = false; this.addTimeds.push(thand); return thand; } /** PrivateFunction: _addSysHandler * _Private_ function to add a system level stanza handler. * * This function is used to add a Strophe.Handler for the * library code. System stanza handlers are allowed to run before * authentication is complete. * * Parameters: * (Function) handler - The callback function. * (String) ns - The namespace to match. * (String) name - The stanza name to match. * (String) type - The stanza type attribute to match. * (String) id - The stanza id attribute to match. */ _addSysHandler(handler, ns, name, type, id) { const hand = new Strophe.Handler(handler, ns, name, type, id); hand.user = false; this.addHandlers.push(hand); return hand; } /** PrivateFunction: _onDisconnectTimeout * _Private_ timeout handler for handling non-graceful disconnection. * * If the graceful disconnect process does not complete within the * time allotted, this handler finishes the disconnect anyway. * * Returns: * false to remove the handler. */ _onDisconnectTimeout() { Strophe.debug("_onDisconnectTimeout was called"); this._changeConnectStatus(Strophe.Status.CONNTIMEOUT, null); this._proto._onDisconnectTimeout(); // actually disconnect this._doDisconnect(); return false; } /** PrivateFunction: _onIdle * _Private_ handler to process events during idle cycle. * * This handler is called every 100ms to fire timed handlers that * are ready and keep poll requests going. */ _onIdle() { // add timed handlers scheduled for addition // NOTE: we add before remove in the case a timed handler is // added and then deleted before the next _onIdle() call. while (this.addTimeds.length > 0) { this.timedHandlers.push(this.addTimeds.pop()); } // remove timed handlers that have been scheduled for deletion while (this.removeTimeds.length > 0) { const thand = this.removeTimeds.pop(); const i = this.timedHandlers.indexOf(thand); if (i >= 0) { this.timedHandlers.splice(i, 1); } } // call ready timed handlers const now = new Date().getTime(); const newList = []; for (let i = 0; i < this.timedHandlers.length; i++) { const thand = this.timedHandlers[i]; if (this.authenticated || !thand.user) { const since = thand.lastCalled + thand.period; if (since - now <= 0) { if (thand.run()) { newList.push(thand); } } else { newList.push(thand); } } } this.timedHandlers = newList; clearTimeout(this._idleTimeout); this._proto._onIdle(); // reactivate the timer only if connected if (this.connected) { this._idleTimeout = setTimeout(() => this._onIdle(), 100); } } }; Strophe.SASLMechanism = SASLMechanism; /** Constants: SASL mechanisms * Available authentication mechanisms * * Strophe.SASLAnonymous - SASL ANONYMOUS authentication. * Strophe.SASLPlain - SASL PLAIN authentication. * Strophe.SASLSHA1 - SASL SCRAM-SHA-1 authentication * Strophe.SASLOAuthBearer - SASL OAuth Bearer authentication * Strophe.SASLExternal - SASL EXTERNAL authentication * Strophe.SASLXOAuth2 - SASL X-OAuth2 authentication */ Strophe.SASLAnonymous = SASLAnonymous; Strophe.SASLPlain = SASLPlain; Strophe.SASLSHA1 = SASLSHA1; Strophe.SASLOAuthBearer = SASLOAuthBearer; Strophe.SASLExternal = SASLExternal; Strophe.SASLXOAuth2 = SASLXOAuth2; /* harmony default export */ const core = ({ '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 }); ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/bosh.js /* This program is distributed under the terms of the MIT license. Please see the LICENSE file for details. Copyright 2006-2008, OGG, LLC */ /* global ActiveXObject */ /** PrivateClass: Strophe.Request * _Private_ helper class that provides a cross implementation abstraction * for a BOSH related XMLHttpRequest. * * The Strophe.Request class is used internally to encapsulate BOSH request * information. It is not meant to be used from user's code. */ Strophe.Request = class Request { /** PrivateConstructor: Strophe.Request * Create and initialize a new Strophe.Request object. * * Parameters: * (XMLElement) elem - The XML data to be sent in the request. * (Function) func - The function that will be called when the * XMLHttpRequest readyState changes. * (Integer) rid - The BOSH rid attribute associated with this request. * (Integer) sends - The number of times this same request has been sent. */ constructor(elem, func, rid, sends) { this.id = ++Strophe._requestId; this.xmlData = elem; this.data = Strophe.serialize(elem); // save original function in case we need to make a new request // from this one. this.origFunc = func; this.func = func; this.rid = rid; this.date = NaN; this.sends = sends || 0; this.abort = false; this.dead = null; this.age = function () { if (!this.date) { return 0; } const now = new Date(); return (now - this.date) / 1000; }; this.timeDead = function () { if (!this.dead) { return 0; } const now = new Date(); return (now - this.dead) / 1000; }; this.xhr = this._newXHR(); } /** PrivateFunction: getResponse * Get a response from the underlying XMLHttpRequest. * * This function attempts to get a response from the request and checks * for errors. * * Throws: * "parsererror" - A parser error occured. * "bad-format" - The entity has sent XML that cannot be processed. * * Returns: * The DOM element tree of the response. */ getResponse() { let node = null; if (this.xhr.responseXML && this.xhr.responseXML.documentElement) { node = this.xhr.responseXML.documentElement; if (node.tagName === "parsererror") { Strophe.error("invalid response received"); Strophe.error("responseText: " + this.xhr.responseText); Strophe.error("responseXML: " + Strophe.serialize(this.xhr.responseXML)); throw new Error("parsererror"); } } else if (this.xhr.responseText) { // In React Native, we may get responseText but no responseXML. We can try to parse it manually. Strophe.debug("Got responseText but no responseXML; attempting to parse it with DOMParser..."); node = new strophe_shims_DOMParser().parseFromString(this.xhr.responseText, 'application/xml').documentElement; if (!node) { throw new Error('Parsing produced null node'); } else if (node.querySelector('parsererror')) { Strophe.error("invalid response received: " + node.querySelector('parsererror').textContent); Strophe.error("responseText: " + this.xhr.responseText); const error = new Error(); error.name = Strophe.ErrorCondition.BAD_FORMAT; throw error; } } return node; } /** PrivateFunction: _newXHR * _Private_ helper function to create XMLHttpRequests. * * This function creates XMLHttpRequests across all implementations. * * Returns: * A new XMLHttpRequest. */ _newXHR() { let xhr = null; if (window.XMLHttpRequest) { xhr = new XMLHttpRequest(); if (xhr.overrideMimeType) { xhr.overrideMimeType("text/xml; charset=utf-8"); } } else if (window.ActiveXObject) { xhr = new ActiveXObject("Microsoft.XMLHTTP"); } // use Function.bind() to prepend ourselves as an argument xhr.onreadystatechange = this.func.bind(null, this); return xhr; } }; /** Class: Strophe.Bosh * _Private_ helper class that handles BOSH Connections * * The Strophe.Bosh class is used internally by Strophe.Connection * to encapsulate BOSH sessions. It is not meant to be used from user's code. */ /** File: bosh.js * A JavaScript library to enable BOSH in Strophejs. * * this library uses Bidirectional-streams Over Synchronous HTTP (BOSH) * to emulate a persistent, stateful, two-way connection to an XMPP server. * More information on BOSH can be found in XEP 124. */ /** PrivateConstructor: Strophe.Bosh * Create and initialize a Strophe.Bosh object. * * Parameters: * (Strophe.Connection) connection - The Strophe.Connection that will use BOSH. * * Returns: * A new Strophe.Bosh object. */ Strophe.Bosh = class Bosh { constructor(connection) { this._conn = connection; /* request id for body tags */ this.rid = Math.floor(Math.random() * 4294967295); /* The current session ID. */ this.sid = null; // default BOSH values this.hold = 1; this.wait = 60; this.window = 5; this.errors = 0; this.inactivity = null; this.lastResponseHeaders = null; this._requests = []; } /** PrivateFunction: _buildBody * _Private_ helper function to generate the wrapper for BOSH. * * Returns: * A Strophe.Builder with a element. */ _buildBody() { const bodyWrap = $build('body', { 'rid': this.rid++, 'xmlns': Strophe.NS.HTTPBIND }); if (this.sid !== null) { bodyWrap.attrs({ 'sid': this.sid }); } if (this._conn.options.keepalive && this._conn._sessionCachingSupported()) { this._cacheSession(); } return bodyWrap; } /** PrivateFunction: _reset * Reset the connection. * * This function is called by the reset function of the Strophe Connection */ _reset() { this.rid = Math.floor(Math.random() * 4294967295); this.sid = null; this.errors = 0; if (this._conn._sessionCachingSupported()) { window.sessionStorage.removeItem('strophe-bosh-session'); } this._conn.nextValidRid(this.rid); } /** PrivateFunction: _connect * _Private_ function that initializes the BOSH connection. * * Creates and sends the Request that initializes the BOSH connection. */ _connect(wait, hold, route) { this.wait = wait || this.wait; this.hold = hold || this.hold; this.errors = 0; const body = this._buildBody().attrs({ "to": this._conn.domain, "xml:lang": "en", "wait": this.wait, "hold": this.hold, "content": "text/xml; charset=utf-8", "ver": "1.6", "xmpp:version": "1.0", "xmlns:xmpp": Strophe.NS.BOSH }); if (route) { body.attrs({ 'route': route }); } const _connect_cb = this._conn._connect_cb; this._requests.push(new Strophe.Request(body.tree(), this._onRequestStateChange.bind(this, _connect_cb.bind(this._conn)), body.tree().getAttribute("rid"))); this._throttledRequestHandler(); } /** PrivateFunction: _attach * Attach to an already created and authenticated BOSH session. * * This function is provided to allow Strophe to attach to BOSH * sessions which have been created externally, perhaps by a Web * application. This is often used to support auto-login type features * without putting user credentials into the page. * * Parameters: * (String) jid - The full JID that is bound by the session. * (String) sid - The SID of the BOSH session. * (String) rid - The current RID of the BOSH session. This RID * will be used by the next request. * (Function) callback The connect callback function. * (Integer) wait - The optional HTTPBIND wait value. This is the * time the server will wait before returning an empty result for * a request. The default setting of 60 seconds is recommended. * Other settings will require tweaks to the Strophe.TIMEOUT value. * (Integer) hold - The optional HTTPBIND hold value. This is the * number of connections the server will hold at one time. This * should almost always be set to 1 (the default). * (Integer) wind - The optional HTTBIND window value. This is the * allowed range of request ids that are valid. The default is 5. */ _attach(jid, sid, rid, callback, wait, hold, wind) { this._conn.jid = jid; this.sid = sid; this.rid = rid; this._conn.connect_callback = callback; this._conn.domain = Strophe.getDomainFromJid(this._conn.jid); this._conn.authenticated = true; this._conn.connected = true; this.wait = wait || this.wait; this.hold = hold || this.hold; this.window = wind || this.window; this._conn._changeConnectStatus(Strophe.Status.ATTACHED, null); } /** PrivateFunction: _restore * Attempt to restore a cached BOSH session * * Parameters: * (String) jid - The full JID that is bound by the session. * This parameter is optional but recommended, specifically in cases * where prebinded BOSH sessions are used where it's important to know * that the right session is being restored. * (Function) callback The connect callback function. * (Integer) wait - The optional HTTPBIND wait value. This is the * time the server will wait before returning an empty result for * a request. The default setting of 60 seconds is recommended. * Other settings will require tweaks to the Strophe.TIMEOUT value. * (Integer) hold - The optional HTTPBIND hold value. This is the * number of connections the server will hold at one time. This * should almost always be set to 1 (the default). * (Integer) wind - The optional HTTBIND window value. This is the * allowed range of request ids that are valid. The default is 5. */ _restore(jid, callback, wait, hold, wind) { const session = JSON.parse(window.sessionStorage.getItem('strophe-bosh-session')); if (typeof session !== "undefined" && session !== null && session.rid && session.sid && session.jid && (typeof jid === "undefined" || jid === null || Strophe.getBareJidFromJid(session.jid) === Strophe.getBareJidFromJid(jid) || // If authcid is null, then it's an anonymous login, so // we compare only the domains: Strophe.getNodeFromJid(jid) === null && Strophe.getDomainFromJid(session.jid) === jid)) { this._conn.restored = true; this._attach(session.jid, session.sid, session.rid, callback, wait, hold, wind); } else { const error = new Error("_restore: no restoreable session."); error.name = "StropheSessionError"; throw error; } } /** PrivateFunction: _cacheSession * _Private_ handler for the beforeunload event. * * This handler is used to process the Bosh-part of the initial request. * Parameters: * (Strophe.Request) bodyWrap - The received stanza. */ _cacheSession() { if (this._conn.authenticated) { if (this._conn.jid && this.rid && this.sid) { window.sessionStorage.setItem('strophe-bosh-session', JSON.stringify({ 'jid': this._conn.jid, 'rid': this.rid, 'sid': this.sid })); } } else { window.sessionStorage.removeItem('strophe-bosh-session'); } } /** PrivateFunction: _connect_cb * _Private_ handler for initial connection request. * * This handler is used to process the Bosh-part of the initial request. * Parameters: * (Strophe.Request) bodyWrap - The received stanza. */ _connect_cb(bodyWrap) { const typ = bodyWrap.getAttribute("type"); if (typ !== null && typ === "terminate") { // an error occurred let cond = bodyWrap.getAttribute("condition"); Strophe.error("BOSH-Connection failed: " + cond); const conflict = bodyWrap.getElementsByTagName("conflict"); if (cond !== null) { if (cond === "remote-stream-error" && conflict.length > 0) { cond = "conflict"; } this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, cond); } else { this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "unknown"); } this._conn._doDisconnect(cond); return Strophe.Status.CONNFAIL; } // check to make sure we don't overwrite these if _connect_cb is // called multiple times in the case of missing stream:features if (!this.sid) { this.sid = bodyWrap.getAttribute("sid"); } const wind = bodyWrap.getAttribute('requests'); if (wind) { this.window = parseInt(wind, 10); } const hold = bodyWrap.getAttribute('hold'); if (hold) { this.hold = parseInt(hold, 10); } const wait = bodyWrap.getAttribute('wait'); if (wait) { this.wait = parseInt(wait, 10); } const inactivity = bodyWrap.getAttribute('inactivity'); if (inactivity) { this.inactivity = parseInt(inactivity, 10); } } /** PrivateFunction: _disconnect * _Private_ part of Connection.disconnect for Bosh * * Parameters: * (Request) pres - This stanza will be sent before disconnecting. */ _disconnect(pres) { this._sendTerminate(pres); } /** PrivateFunction: _doDisconnect * _Private_ function to disconnect. * * Resets the SID and RID. */ _doDisconnect() { this.sid = null; this.rid = Math.floor(Math.random() * 4294967295); if (this._conn._sessionCachingSupported()) { window.sessionStorage.removeItem('strophe-bosh-session'); } this._conn.nextValidRid(this.rid); } /** PrivateFunction: _emptyQueue * _Private_ function to check if the Request queue is empty. * * Returns: * True, if there are no Requests queued, False otherwise. */ _emptyQueue() { return this._requests.length === 0; } /** PrivateFunction: _callProtocolErrorHandlers * _Private_ function to call error handlers registered for HTTP errors. * * Parameters: * (Strophe.Request) req - The request that is changing readyState. */ _callProtocolErrorHandlers(req) { const reqStatus = Bosh._getRequestStatus(req); const err_callback = this._conn.protocolErrorHandlers.HTTP[reqStatus]; if (err_callback) { err_callback.call(this, reqStatus); } } /** PrivateFunction: _hitError * _Private_ function to handle the error count. * * Requests are resent automatically until their error count reaches * 5. Each time an error is encountered, this function is called to * increment the count and disconnect if the count is too high. * * Parameters: * (Integer) reqStatus - The request status. */ _hitError(reqStatus) { this.errors++; Strophe.warn("request errored, status: " + reqStatus + ", number of errors: " + this.errors); if (this.errors > 4) { this._conn._onDisconnectTimeout(); } } /** PrivateFunction: _no_auth_received * * Called on stream start/restart when no stream:features * has been received and sends a blank poll request. */ _no_auth_received(callback) { Strophe.warn("Server did not yet offer a supported authentication " + "mechanism. Sending a blank poll request."); if (callback) { callback = callback.bind(this._conn); } else { callback = this._conn._connect_cb.bind(this._conn); } const body = this._buildBody(); this._requests.push(new Strophe.Request(body.tree(), this._onRequestStateChange.bind(this, callback), body.tree().getAttribute("rid"))); this._throttledRequestHandler(); } /** PrivateFunction: _onDisconnectTimeout * _Private_ timeout handler for handling non-graceful disconnection. * * Cancels all remaining Requests and clears the queue. */ _onDisconnectTimeout() { this._abortAllRequests(); } /** PrivateFunction: _abortAllRequests * _Private_ helper function that makes sure all pending requests are aborted. */ _abortAllRequests() { while (this._requests.length > 0) { const req = this._requests.pop(); req.abort = true; req.xhr.abort(); req.xhr.onreadystatechange = function () {}; } } /** PrivateFunction: _onIdle * _Private_ handler called by Strophe.Connection._onIdle * * Sends all queued Requests or polls with empty Request if there are none. */ _onIdle() { const data = this._conn._data; // if no requests are in progress, poll if (this._conn.authenticated && this._requests.length === 0 && data.length === 0 && !this._conn.disconnecting) { Strophe.debug("no requests during idle cycle, sending blank request"); data.push(null); } if (this._conn.paused) { return; } if (this._requests.length < 2 && data.length > 0) { const body = this._buildBody(); for (let i = 0; i < data.length; i++) { if (data[i] !== null) { if (data[i] === "restart") { body.attrs({ "to": this._conn.domain, "xml:lang": "en", "xmpp:restart": "true", "xmlns:xmpp": Strophe.NS.BOSH }); } else { body.cnode(data[i]).up(); } } } delete this._conn._data; this._conn._data = []; this._requests.push(new Strophe.Request(body.tree(), this._onRequestStateChange.bind(this, this._conn._dataRecv.bind(this._conn)), body.tree().getAttribute("rid"))); this._throttledRequestHandler(); } if (this._requests.length > 0) { const time_elapsed = this._requests[0].age(); if (this._requests[0].dead !== null) { if (this._requests[0].timeDead() > Math.floor(Strophe.SECONDARY_TIMEOUT * this.wait)) { this._throttledRequestHandler(); } } if (time_elapsed > Math.floor(Strophe.TIMEOUT * this.wait)) { Strophe.warn("Request " + this._requests[0].id + " timed out, over " + Math.floor(Strophe.TIMEOUT * this.wait) + " seconds since last activity"); this._throttledRequestHandler(); } } } /** PrivateFunction: _getRequestStatus * * Returns the HTTP status code from a Strophe.Request * * Parameters: * (Strophe.Request) req - The Strophe.Request instance. * (Integer) def - The default value that should be returned if no * status value was found. */ static _getRequestStatus(req, def) { let reqStatus; if (req.xhr.readyState === 4) { try { reqStatus = req.xhr.status; } catch (e) { // ignore errors from undefined status attribute. Works // around a browser bug Strophe.error("Caught an error while retrieving a request's status, " + "reqStatus: " + reqStatus); } } if (typeof reqStatus === "undefined") { reqStatus = typeof def === 'number' ? def : 0; } return reqStatus; } /** PrivateFunction: _onRequestStateChange * _Private_ handler for Strophe.Request state changes. * * This function is called when the XMLHttpRequest readyState changes. * It contains a lot of error handling logic for the many ways that * requests can fail, and calls the request callback when requests * succeed. * * Parameters: * (Function) func - The handler for the request. * (Strophe.Request) req - The request that is changing readyState. */ _onRequestStateChange(func, req) { Strophe.debug("request id " + req.id + "." + req.sends + " state changed to " + req.xhr.readyState); if (req.abort) { req.abort = false; return; } if (req.xhr.readyState !== 4) { // The request is not yet complete return; } const reqStatus = Bosh._getRequestStatus(req); this.lastResponseHeaders = req.xhr.getAllResponseHeaders(); if (this._conn.disconnecting && reqStatus >= 400) { this._hitError(reqStatus); this._callProtocolErrorHandlers(req); return; } const reqIs0 = this._requests[0] === req; const reqIs1 = this._requests[1] === req; const valid_request = reqStatus > 0 && reqStatus < 500; const too_many_retries = req.sends > this._conn.maxRetries; if (valid_request || too_many_retries) { // remove from internal queue this._removeRequest(req); Strophe.debug("request id " + req.id + " should now be removed"); } if (reqStatus === 200) { // request succeeded // if request 1 finished, or request 0 finished and request // 1 is over Strophe.SECONDARY_TIMEOUT seconds old, we need to // restart the other - both will be in the first spot, as the // completed request has been removed from the queue already if (reqIs1 || reqIs0 && this._requests.length > 0 && this._requests[0].age() > Math.floor(Strophe.SECONDARY_TIMEOUT * this.wait)) { this._restartRequest(0); } this._conn.nextValidRid(Number(req.rid) + 1); Strophe.debug("request id " + req.id + "." + req.sends + " got 200"); func(req); // call handler this.errors = 0; } else if (reqStatus === 0 || reqStatus >= 400 && reqStatus < 600 || reqStatus >= 12000) { // request failed Strophe.error("request id " + req.id + "." + req.sends + " error " + reqStatus + " happened"); this._hitError(reqStatus); this._callProtocolErrorHandlers(req); if (reqStatus >= 400 && reqStatus < 500) { this._conn._changeConnectStatus(Strophe.Status.DISCONNECTING, null); this._conn._doDisconnect(); } } else { Strophe.error("request id " + req.id + "." + req.sends + " error " + reqStatus + " happened"); } if (!valid_request && !too_many_retries) { this._throttledRequestHandler(); } else if (too_many_retries && !this._conn.connected) { this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "giving-up"); } } /** PrivateFunction: _processRequest * _Private_ function to process a request in the queue. * * This function takes requests off the queue and sends them and * restarts dead requests. * * Parameters: * (Integer) i - The index of the request in the queue. */ _processRequest(i) { let req = this._requests[i]; const reqStatus = Bosh._getRequestStatus(req, -1); // make sure we limit the number of retries if (req.sends > this._conn.maxRetries) { this._conn._onDisconnectTimeout(); return; } const time_elapsed = req.age(); const primary_timeout = !isNaN(time_elapsed) && time_elapsed > Math.floor(Strophe.TIMEOUT * this.wait); const secondary_timeout = req.dead !== null && req.timeDead() > Math.floor(Strophe.SECONDARY_TIMEOUT * this.wait); const server_error = req.xhr.readyState === 4 && (reqStatus < 1 || reqStatus >= 500); if (primary_timeout || secondary_timeout || server_error) { if (secondary_timeout) { Strophe.error(`Request ${this._requests[i].id} timed out (secondary), restarting`); } req.abort = true; req.xhr.abort(); // setting to null fails on IE6, so set to empty function req.xhr.onreadystatechange = function () {}; this._requests[i] = new Strophe.Request(req.xmlData, req.origFunc, req.rid, req.sends); req = this._requests[i]; } if (req.xhr.readyState === 0) { Strophe.debug("request id " + req.id + "." + req.sends + " posting"); try { const content_type = this._conn.options.contentType || "text/xml; charset=utf-8"; req.xhr.open("POST", this._conn.service, this._conn.options.sync ? false : true); if (typeof req.xhr.setRequestHeader !== 'undefined') { // IE9 doesn't have setRequestHeader req.xhr.setRequestHeader("Content-Type", content_type); } if (this._conn.options.withCredentials) { req.xhr.withCredentials = true; } } catch (e2) { Strophe.error("XHR open failed: " + e2.toString()); if (!this._conn.connected) { this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "bad-service"); } this._conn.disconnect(); return; } // Fires the XHR request -- may be invoked immediately // or on a gradually expanding retry window for reconnects const sendFunc = () => { req.date = new Date(); if (this._conn.options.customHeaders) { const headers = this._conn.options.customHeaders; for (const header in headers) { if (Object.prototype.hasOwnProperty.call(headers, header)) { req.xhr.setRequestHeader(header, headers[header]); } } } req.xhr.send(req.data); }; // Implement progressive backoff for reconnects -- // First retry (send === 1) should also be instantaneous if (req.sends > 1) { // Using a cube of the retry number creates a nicely // expanding retry window const backoff = Math.min(Math.floor(Strophe.TIMEOUT * this.wait), Math.pow(req.sends, 3)) * 1000; setTimeout(function () { // XXX: setTimeout should be called only with function expressions (23974bc1) sendFunc(); }, backoff); } else { sendFunc(); } req.sends++; if (this._conn.xmlOutput !== Strophe.Connection.prototype.xmlOutput) { if (req.xmlData.nodeName === this.strip && req.xmlData.childNodes.length) { this._conn.xmlOutput(req.xmlData.childNodes[0]); } else { this._conn.xmlOutput(req.xmlData); } } if (this._conn.rawOutput !== Strophe.Connection.prototype.rawOutput) { this._conn.rawOutput(req.data); } } else { Strophe.debug("_processRequest: " + (i === 0 ? "first" : "second") + " request has readyState of " + req.xhr.readyState); } } /** PrivateFunction: _removeRequest * _Private_ function to remove a request from the queue. * * Parameters: * (Strophe.Request) req - The request to remove. */ _removeRequest(req) { Strophe.debug("removing request"); for (let i = this._requests.length - 1; i >= 0; i--) { if (req === this._requests[i]) { this._requests.splice(i, 1); } } // IE6 fails on setting to null, so set to empty function req.xhr.onreadystatechange = function () {}; this._throttledRequestHandler(); } /** PrivateFunction: _restartRequest * _Private_ function to restart a request that is presumed dead. * * Parameters: * (Integer) i - The index of the request in the queue. */ _restartRequest(i) { const req = this._requests[i]; if (req.dead === null) { req.dead = new Date(); } this._processRequest(i); } /** PrivateFunction: _reqToData * _Private_ function to get a stanza out of a request. * * Tries to extract a stanza out of a Request Object. * When this fails the current connection will be disconnected. * * Parameters: * (Object) req - The Request. * * Returns: * The stanza that was passed. */ _reqToData(req) { try { return req.getResponse(); } catch (e) { if (e.message !== "parsererror") { throw e; } this._conn.disconnect("strophe-parsererror"); } } /** PrivateFunction: _sendTerminate * _Private_ function to send initial disconnect sequence. * * This is the first step in a graceful disconnect. It sends * the BOSH server a terminate body and includes an unavailable * presence if authentication has completed. */ _sendTerminate(pres) { Strophe.debug("_sendTerminate was called"); const body = this._buildBody().attrs({ type: "terminate" }); if (pres) { body.cnode(pres.tree()); } const req = new Strophe.Request(body.tree(), this._onRequestStateChange.bind(this, this._conn._dataRecv.bind(this._conn)), body.tree().getAttribute("rid")); this._requests.push(req); this._throttledRequestHandler(); } /** PrivateFunction: _send * _Private_ part of the Connection.send function for BOSH * * Just triggers the RequestHandler to send the messages that are in the queue */ _send() { clearTimeout(this._conn._idleTimeout); this._throttledRequestHandler(); this._conn._idleTimeout = setTimeout(() => this._conn._onIdle(), 100); } /** PrivateFunction: _sendRestart * * Send an xmpp:restart stanza. */ _sendRestart() { this._throttledRequestHandler(); clearTimeout(this._conn._idleTimeout); } /** PrivateFunction: _throttledRequestHandler * _Private_ function to throttle requests to the connection window. * * This function makes sure we don't send requests so fast that the * request ids overflow the connection window in the case that one * request died. */ _throttledRequestHandler() { if (!this._requests) { Strophe.debug("_throttledRequestHandler called with " + "undefined requests"); } else { Strophe.debug("_throttledRequestHandler called with " + this._requests.length + " requests"); } if (!this._requests || this._requests.length === 0) { return; } if (this._requests.length > 0) { this._processRequest(0); } if (this._requests.length > 1 && Math.abs(this._requests[0].rid - this._requests[1].rid) < this.window) { this._processRequest(1); } } }; /** Variable: strip * * BOSH-Connections will have all stanzas wrapped in a tag when * passed to or . * To strip this tag, User code can set to "body": * * > Strophe.Bosh.prototype.strip = "body"; * * This will enable stripping of the body tag in both * and . */ Strophe.Bosh.prototype.strip = null; ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/websocket.js /* This program is distributed under the terms of the MIT license. Please see the LICENSE file for details. Copyright 2006-2008, OGG, LLC */ /* global window, clearTimeout, WebSocket, DOMParser */ /** Class: Strophe.WebSocket * _Private_ helper class that handles WebSocket Connections * * The Strophe.WebSocket class is used internally by Strophe.Connection * to encapsulate WebSocket sessions. It is not meant to be used from user's code. */ /** File: websocket.js * A JavaScript library to enable XMPP over Websocket in Strophejs. * * This file implements XMPP over WebSockets for Strophejs. * If a Connection is established with a Websocket url (ws://...) * Strophe will use WebSockets. * For more information on XMPP-over-WebSocket see RFC 7395: * http://tools.ietf.org/html/rfc7395 * * WebSocket support implemented by Andreas Guth (andreas.guth@rwth-aachen.de) */ Strophe.Websocket = class Websocket { /** PrivateConstructor: Strophe.Websocket * Create and initialize a Strophe.WebSocket object. * Currently only sets the connection Object. * * Parameters: * (Strophe.Connection) connection - The Strophe.Connection that will use WebSockets. * * Returns: * A new Strophe.WebSocket object. */ constructor(connection) { this._conn = connection; this.strip = "wrapper"; const service = connection.service; if (service.indexOf("ws:") !== 0 && service.indexOf("wss:") !== 0) { // If the service is not an absolute URL, assume it is a path and put the absolute // URL together from options, current URL and the path. let new_service = ""; if (connection.options.protocol === "ws" && window.location.protocol !== "https:") { new_service += "ws"; } else { new_service += "wss"; } new_service += "://" + window.location.host; if (service.indexOf("/") !== 0) { new_service += window.location.pathname + service; } else { new_service += service; } connection.service = new_service; } } /** PrivateFunction: _buildStream * _Private_ helper function to generate the start tag for WebSockets * * Returns: * A Strophe.Builder with a element. */ _buildStream() { return $build("open", { "xmlns": Strophe.NS.FRAMING, "to": this._conn.domain, "version": '1.0' }); } /** PrivateFunction: _checkStreamError * _Private_ checks a message for stream:error * * Parameters: * (Strophe.Request) bodyWrap - The received stanza. * connectstatus - The ConnectStatus that will be set on error. * Returns: * true if there was a streamerror, false otherwise. */ _checkStreamError(bodyWrap, connectstatus) { let errors; if (bodyWrap.getElementsByTagNameNS) { errors = bodyWrap.getElementsByTagNameNS(Strophe.NS.STREAM, "error"); } else { errors = bodyWrap.getElementsByTagName("stream:error"); } if (errors.length === 0) { return false; } const error = errors[0]; let condition = ""; let text = ""; const ns = "urn:ietf:params:xml:ns:xmpp-streams"; for (let i = 0; i < error.childNodes.length; i++) { const e = error.childNodes[i]; if (e.getAttribute("xmlns") !== ns) { break; } if (e.nodeName === "text") { text = e.textContent; } else { condition = e.nodeName; } } let errorString = "WebSocket stream error: "; if (condition) { errorString += condition; } else { errorString += "unknown"; } if (text) { errorString += " - " + text; } Strophe.error(errorString); // close the connection on stream_error this._conn._changeConnectStatus(connectstatus, condition); this._conn._doDisconnect(); return true; } /** PrivateFunction: _reset * Reset the connection. * * This function is called by the reset function of the Strophe Connection. * Is not needed by WebSockets. */ _reset() { // eslint-disable-line class-methods-use-this return; } /** PrivateFunction: _connect * _Private_ function called by Strophe.Connection.connect * * Creates a WebSocket for a connection and assigns Callbacks to it. * Does nothing if there already is a WebSocket. */ _connect() { // Ensure that there is no open WebSocket from a previous Connection. this._closeSocket(); this.socket = new WebSocket(this._conn.service, "xmpp"); this.socket.onopen = () => this._onOpen(); this.socket.onerror = e => this._onError(e); this.socket.onclose = e => this._onClose(e); // Gets replaced with this._onMessage once _onInitialMessage is called this.socket.onmessage = message => this._onInitialMessage(message); } /** PrivateFunction: _connect_cb * _Private_ function called by Strophe.Connection._connect_cb * * checks for stream:error * * Parameters: * (Strophe.Request) bodyWrap - The received stanza. */ _connect_cb(bodyWrap) { const error = this._checkStreamError(bodyWrap, Strophe.Status.CONNFAIL); if (error) { return Strophe.Status.CONNFAIL; } } /** PrivateFunction: _handleStreamStart * _Private_ function that checks the opening tag for errors. * * Disconnects if there is an error and returns false, true otherwise. * * Parameters: * (Node) message - Stanza containing the tag. */ _handleStreamStart(message) { let error = false; // Check for errors in the tag const ns = message.getAttribute("xmlns"); if (typeof ns !== "string") { error = "Missing xmlns in "; } else if (ns !== Strophe.NS.FRAMING) { error = "Wrong xmlns in : " + ns; } const ver = message.getAttribute("version"); if (typeof ver !== "string") { error = "Missing version in "; } else if (ver !== "1.0") { error = "Wrong version in : " + ver; } if (error) { this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, error); this._conn._doDisconnect(); return false; } return true; } /** PrivateFunction: _onInitialMessage * _Private_ function that handles the first connection messages. * * On receiving an opening stream tag this callback replaces itself with the real * message handler. On receiving a stream error the connection is terminated. */ _onInitialMessage(message) { if (message.data.indexOf("\s*)*/, ""); if (data === '') return; const streamStart = new strophe_shims_DOMParser().parseFromString(data, "text/xml").documentElement; this._conn.xmlInput(streamStart); this._conn.rawInput(message.data); //_handleStreamSteart will check for XML errors and disconnect on error if (this._handleStreamStart(streamStart)) { //_connect_cb will check for stream:error and disconnect on error this._connect_cb(streamStart); } } else if (message.data.indexOf("WSS, WS->ANY const isSecureRedirect = service.indexOf("wss:") >= 0 && see_uri.indexOf("wss:") >= 0 || service.indexOf("ws:") >= 0; if (isSecureRedirect) { this._conn._changeConnectStatus(Strophe.Status.REDIRECT, "Received see-other-uri, resetting connection"); this._conn.reset(); this._conn.service = see_uri; this._connect(); } } else { this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "Received closing stream"); this._conn._doDisconnect(); } } else { this._replaceMessageHandler(); const string = this._streamWrap(message.data); const elem = new strophe_shims_DOMParser().parseFromString(string, "text/xml").documentElement; this._conn._connect_cb(elem, null, message.data); } } /** PrivateFunction: _replaceMessageHandler * * Called by _onInitialMessage in order to replace itself with the general message handler. * This method is overridden by Strophe.WorkerWebsocket, which manages a * websocket connection via a service worker and doesn't have direct access * to the socket. */ _replaceMessageHandler() { this.socket.onmessage = m => this._onMessage(m); } /** PrivateFunction: _disconnect * _Private_ function called by Strophe.Connection.disconnect * * Disconnects and sends a last stanza if one is given * * Parameters: * (Request) pres - This stanza will be sent before disconnecting. */ _disconnect(pres) { if (this.socket && this.socket.readyState !== WebSocket.CLOSED) { if (pres) { this._conn.send(pres); } const close = $build("close", { "xmlns": Strophe.NS.FRAMING }); this._conn.xmlOutput(close.tree()); const closeString = Strophe.serialize(close); this._conn.rawOutput(closeString); try { this.socket.send(closeString); } catch (e) { Strophe.warn("Couldn't send tag."); } } setTimeout(() => this._conn._doDisconnect, 0); } /** PrivateFunction: _doDisconnect * _Private_ function to disconnect. * * Just closes the Socket for WebSockets */ _doDisconnect() { Strophe.debug("WebSockets _doDisconnect was called"); this._closeSocket(); } /** PrivateFunction _streamWrap * _Private_ helper function to wrap a stanza in a tag. * This is used so Strophe can process stanzas from WebSockets like BOSH */ _streamWrap(stanza) { // eslint-disable-line class-methods-use-this return "" + stanza + ''; } /** PrivateFunction: _closeSocket * _Private_ function to close the WebSocket. * * Closes the socket if it is still open and deletes it */ _closeSocket() { if (this.socket) { try { this.socket.onclose = null; this.socket.onerror = null; this.socket.onmessage = null; this.socket.close(); } catch (e) { Strophe.debug(e.message); } } this.socket = null; } /** PrivateFunction: _emptyQueue * _Private_ function to check if the message queue is empty. * * Returns: * True, because WebSocket messages are send immediately after queueing. */ _emptyQueue() { // eslint-disable-line class-methods-use-this return true; } /** PrivateFunction: _onClose * _Private_ function to handle websockets closing. */ _onClose(e) { if (this._conn.connected && !this._conn.disconnecting) { Strophe.error("Websocket closed unexpectedly"); this._conn._doDisconnect(); } else if (e && e.code === 1006 && !this._conn.connected && this.socket) { // in case the onError callback was not called (Safari 10 does not // call onerror when the initial connection fails) we need to // dispatch a CONNFAIL status update to be consistent with the // behavior on other browsers. Strophe.error("Websocket closed unexcectedly"); this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "The WebSocket connection could not be established or was disconnected."); this._conn._doDisconnect(); } else { Strophe.debug("Websocket closed"); } } /** PrivateFunction: _no_auth_received * * Called on stream start/restart when no stream:features * has been received. */ _no_auth_received(callback) { Strophe.error("Server did not offer a supported authentication mechanism"); this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, Strophe.ErrorCondition.NO_AUTH_MECH); if (callback) { callback.call(this._conn); } this._conn._doDisconnect(); } /** PrivateFunction: _onDisconnectTimeout * _Private_ timeout handler for handling non-graceful disconnection. * * This does nothing for WebSockets */ _onDisconnectTimeout() {} // eslint-disable-line class-methods-use-this /** PrivateFunction: _abortAllRequests * _Private_ helper function that makes sure all pending requests are aborted. */ _abortAllRequests() {} // eslint-disable-line class-methods-use-this /** PrivateFunction: _onError * _Private_ function to handle websockets errors. * * Parameters: * (Object) error - The websocket error. */ _onError(error) { Strophe.error("Websocket error " + JSON.stringify(error)); this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "The WebSocket connection could not be established or was disconnected."); this._disconnect(); } /** PrivateFunction: _onIdle * _Private_ function called by Strophe.Connection._onIdle * * sends all queued stanzas */ _onIdle() { const data = this._conn._data; if (data.length > 0 && !this._conn.paused) { for (let i = 0; i < data.length; i++) { if (data[i] !== null) { let stanza; if (data[i] === "restart") { stanza = this._buildStream().tree(); } else { stanza = data[i]; } const rawStanza = Strophe.serialize(stanza); this._conn.xmlOutput(stanza); this._conn.rawOutput(rawStanza); this.socket.send(rawStanza); } } this._conn._data = []; } } /** PrivateFunction: _onMessage * _Private_ function to handle websockets messages. * * This function parses each of the messages as if they are full documents. * [TODO : We may actually want to use a SAX Push parser]. * * Since all XMPP traffic starts with * * * The first stanza will always fail to be parsed. * * Additionally, the seconds stanza will always be with * the stream NS defined in the previous stanza, so we need to 'force' * the inclusion of the NS in this stanza. * * Parameters: * (string) message - The websocket message. */ _onMessage(message) { let elem; // check for closing stream const close = ''; if (message.data === close) { this._conn.rawInput(close); this._conn.xmlInput(message); if (!this._conn.disconnecting) { this._conn._doDisconnect(); } return; } else if (message.data.search(" tag before we close the connection return; } this._conn._dataRecv(elem, message.data); } /** PrivateFunction: _onOpen * _Private_ function to handle websockets connection setup. * * The opening stream tag is sent here. */ _onOpen() { Strophe.debug("Websocket open"); const start = this._buildStream(); this._conn.xmlOutput(start.tree()); const startString = Strophe.serialize(start); this._conn.rawOutput(startString); this.socket.send(startString); } /** PrivateFunction: _reqToData * _Private_ function to get a stanza out of a request. * * WebSockets don't use requests, so the passed argument is just returned. * * Parameters: * (Object) stanza - The stanza. * * Returns: * The stanza that was passed. */ _reqToData(stanza) { // eslint-disable-line class-methods-use-this return stanza; } /** PrivateFunction: _send * _Private_ part of the Connection.send function for WebSocket * * Just flushes the messages that are in the queue */ _send() { this._conn.flush(); } /** PrivateFunction: _sendRestart * * Send an xmpp:restart stanza. */ _sendRestart() { clearTimeout(this._conn._idleTimeout); this._conn._onIdle.bind(this._conn)(); } }; ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/worker-websocket.js /* This program is distributed under the terms of the MIT license. Please see the LICENSE file for details. Copyright 2020, JC Brand */ const lmap = {}; lmap['debug'] = Strophe.LogLevel.DEBUG; lmap['info'] = Strophe.LogLevel.INFO; lmap['warn'] = Strophe.LogLevel.WARN; lmap['error'] = Strophe.LogLevel.ERROR; lmap['fatal'] = Strophe.LogLevel.FATAL; /** Class: Strophe.WorkerWebsocket * _Private_ helper class that handles a websocket connection inside a shared worker. */ Strophe.WorkerWebsocket = class WorkerWebsocket extends Strophe.Websocket { /** PrivateConstructor: Strophe.WorkerWebsocket * Create and initialize a Strophe.WorkerWebsocket object. * * Parameters: * (Strophe.Connection) connection - The Strophe.Connection * * Returns: * A new Strophe.WorkerWebsocket object. */ constructor(connection) { super(connection); this._conn = connection; this.worker = new SharedWorker(this._conn.options.worker, 'Strophe XMPP Connection'); this.worker.onerror = e => { var _console; (_console = console) === null || _console === void 0 ? void 0 : _console.error(e); Strophe.log(Strophe.LogLevel.ERROR, `Shared Worker Error: ${e}`); }; } get socket() { return { 'send': str => this.worker.port.postMessage(['send', str]) }; } _connect() { this._messageHandler = m => this._onInitialMessage(m); this.worker.port.start(); this.worker.port.onmessage = ev => this._onWorkerMessage(ev); this.worker.port.postMessage(['_connect', this._conn.service, this._conn.jid]); } _attach(callback) { this._messageHandler = m => this._onMessage(m); this._conn.connect_callback = callback; this.worker.port.start(); this.worker.port.onmessage = ev => this._onWorkerMessage(ev); this.worker.port.postMessage(['_attach', this._conn.service]); } _attachCallback(status, jid) { if (status === Strophe.Status.ATTACHED) { this._conn.jid = jid; this._conn.authenticated = true; this._conn.connected = true; this._conn.restored = true; this._conn._changeConnectStatus(Strophe.Status.ATTACHED); } else if (status === Strophe.Status.ATTACHFAIL) { this._conn.authenticated = false; this._conn.connected = false; this._conn.restored = false; this._conn._changeConnectStatus(Strophe.Status.ATTACHFAIL); } } _disconnect(readyState, pres) { pres && this._conn.send(pres); const close = $build("close", { "xmlns": Strophe.NS.FRAMING }); this._conn.xmlOutput(close.tree()); const closeString = Strophe.serialize(close); this._conn.rawOutput(closeString); this.worker.port.postMessage(['send', closeString]); this._conn._doDisconnect(); } _onClose(e) { if (this._conn.connected && !this._conn.disconnecting) { Strophe.error("Websocket closed unexpectedly"); this._conn._doDisconnect(); } else if (e && e.code === 1006 && !this._conn.connected) { // in case the onError callback was not called (Safari 10 does not // call onerror when the initial connection fails) we need to // dispatch a CONNFAIL status update to be consistent with the // behavior on other browsers. Strophe.error("Websocket closed unexcectedly"); this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "The WebSocket connection could not be established or was disconnected."); this._conn._doDisconnect(); } else { Strophe.debug("Websocket closed"); } } _closeSocket() { this.worker.port.postMessage(['_closeSocket']); } /** PrivateFunction: _replaceMessageHandler * * Called by _onInitialMessage in order to replace itself with the general message handler. * This method is overridden by Strophe.WorkerWebsocket, which manages a * websocket connection via a service worker and doesn't have direct access * to the socket. */ _replaceMessageHandler() { this._messageHandler = m => this._onMessage(m); } /** PrivateFunction: _onWorkerMessage * _Private_ function that handles messages received from the service worker */ _onWorkerMessage(ev) { const { data } = ev; const method_name = data[0]; if (method_name === '_onMessage') { this._messageHandler(data[1]); } else if (method_name in this) { try { this[method_name].apply(this, ev.data.slice(1)); } catch (e) { Strophe.log(Strophe.LogLevel.ERROR, e); } } else if (method_name === 'log') { const level = data[1]; const msg = data[2]; Strophe.log(lmap[level], msg); } else { Strophe.log(Strophe.LogLevel.ERROR, `Found unhandled service worker message: ${data}`); } } }; ;// CONCATENATED MODULE: ./node_modules/strophe.js/src/strophe.js /*global global*/ __webpack_require__.g.$build = core.$build; __webpack_require__.g.$iq = core.$iq; __webpack_require__.g.$msg = core.$msg; __webpack_require__.g.$pres = core.$pres; __webpack_require__.g.Strophe = core.Strophe; const { b64_sha1 } = SHA1; ;// CONCATENATED MODULE: ./src/headless/shared/constants.js const BOSH_WAIT = 59; const CONNECTION_STATUS = {}; CONNECTION_STATUS[Strophe.Status.ATTACHED] = 'ATTACHED'; CONNECTION_STATUS[Strophe.Status.AUTHENTICATING] = 'AUTHENTICATING'; CONNECTION_STATUS[Strophe.Status.AUTHFAIL] = 'AUTHFAIL'; CONNECTION_STATUS[Strophe.Status.CONNECTED] = 'CONNECTED'; CONNECTION_STATUS[Strophe.Status.CONNECTING] = 'CONNECTING'; CONNECTION_STATUS[Strophe.Status.CONNFAIL] = 'CONNFAIL'; CONNECTION_STATUS[Strophe.Status.DISCONNECTED] = 'DISCONNECTED'; CONNECTION_STATUS[Strophe.Status.DISCONNECTING] = 'DISCONNECTING'; CONNECTION_STATUS[Strophe.Status.ERROR] = 'ERROR'; CONNECTION_STATUS[Strophe.Status.RECONNECTING] = 'RECONNECTING'; CONNECTION_STATUS[Strophe.Status.REDIRECT] = 'REDIRECT'; // Core plugins are whitelisted automatically // These are just the @converse/headless plugins, for the full converse, // the other plugins are whitelisted in src/consts.js const CORE_PLUGINS = ['converse-adhoc', 'converse-bookmarks', 'converse-bosh', 'converse-caps', 'converse-carbons', 'converse-chat', 'converse-chatboxes', 'converse-disco', 'converse-emoji', 'converse-headlines', 'converse-mam', 'converse-muc', 'converse-ping', 'converse-pubsub', 'converse-roster', 'converse-smacks', 'converse-status', 'converse-vcard']; const URL_PARSE_OPTIONS = { 'start': /(\b|_)(?:([a-z][a-z0-9.+-]*:\/\/)|xmpp:|mailto:|www\.)/gi }; const CHAT_STATES = ['active', 'composing', 'gone', 'inactive', 'paused']; const KEYCODES = { TAB: 9, ENTER: 13, SHIFT: 16, CTRL: 17, ALT: 18, ESCAPE: 27, LEFT_ARROW: 37, UP_ARROW: 38, RIGHT_ARROW: 39, DOWN_ARROW: 40, FORWARD_SLASH: 47, AT: 50, META: 91, META_RIGHT: 93 }; ;// CONCATENATED MODULE: ./node_modules/lodash-es/isObject.js /** * Checks if `value` is the * [language type](http://www.ecma-international.org/ecma-262/7.0/#sec-ecmascript-language-types) * of `Object`. (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`) * * @static * @memberOf _ * @since 0.1.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is an object, else `false`. * @example * * _.isObject({}); * // => true * * _.isObject([1, 2, 3]); * // => true * * _.isObject(_.noop); * // => true * * _.isObject(null); * // => false */ function isObject(value) { var type = typeof value; return value != null && (type == 'object' || type == 'function'); } /* harmony default export */ const lodash_es_isObject = (isObject); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isFunction.js /** `Object#toString` result references. */ var asyncTag = '[object AsyncFunction]', funcTag = '[object Function]', genTag = '[object GeneratorFunction]', proxyTag = '[object Proxy]'; /** * Checks if `value` is classified as a `Function` object. * * @static * @memberOf _ * @since 0.1.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a function, else `false`. * @example * * _.isFunction(_); * // => true * * _.isFunction(/abc/); * // => false */ function isFunction(value) { if (!lodash_es_isObject(value)) { return false; } // The use of `Object#toString` avoids issues with the `typeof` operator // in Safari 9 which returns 'object' for typed arrays and other constructors. var tag = _baseGetTag(value); return tag == funcTag || tag == genTag || tag == asyncTag || tag == proxyTag; } /* harmony default export */ const lodash_es_isFunction = (isFunction); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_coreJsData.js /** Used to detect overreaching core-js shims. */ var coreJsData = _root["__core-js_shared__"]; /* harmony default export */ const _coreJsData = (coreJsData); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_isMasked.js /** Used to detect methods masquerading as native. */ var maskSrcKey = (function() { var uid = /[^.]+$/.exec(_coreJsData && _coreJsData.keys && _coreJsData.keys.IE_PROTO || ''); return uid ? ('Symbol(src)_1.' + uid) : ''; }()); /** * Checks if `func` has its source masked. * * @private * @param {Function} func The function to check. * @returns {boolean} Returns `true` if `func` is masked, else `false`. */ function isMasked(func) { return !!maskSrcKey && (maskSrcKey in func); } /* harmony default export */ const _isMasked = (isMasked); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_toSource.js /** Used for built-in method references. */ var _toSource_funcProto = Function.prototype; /** Used to resolve the decompiled source of functions. */ var _toSource_funcToString = _toSource_funcProto.toString; /** * Converts `func` to its source code. * * @private * @param {Function} func The function to convert. * @returns {string} Returns the source code. */ function toSource(func) { if (func != null) { try { return _toSource_funcToString.call(func); } catch (e) {} try { return (func + ''); } catch (e) {} } return ''; } /* harmony default export */ const _toSource = (toSource); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseIsNative.js /** * Used to match `RegExp` * [syntax characters](http://ecma-international.org/ecma-262/7.0/#sec-patterns). */ var reRegExpChar = /[\\^$.*+?()[\]{}|]/g; /** Used to detect host constructors (Safari). */ var reIsHostCtor = /^\[object .+?Constructor\]$/; /** Used for built-in method references. */ var _baseIsNative_funcProto = Function.prototype, _baseIsNative_objectProto = Object.prototype; /** Used to resolve the decompiled source of functions. */ var _baseIsNative_funcToString = _baseIsNative_funcProto.toString; /** Used to check objects for own properties. */ var _baseIsNative_hasOwnProperty = _baseIsNative_objectProto.hasOwnProperty; /** Used to detect if a method is native. */ var reIsNative = RegExp('^' + _baseIsNative_funcToString.call(_baseIsNative_hasOwnProperty).replace(reRegExpChar, '\\$&') .replace(/hasOwnProperty|(function).*?(?=\\\()| for .+?(?=\\\])/g, '$1.*?') + '$' ); /** * The base implementation of `_.isNative` without bad shim checks. * * @private * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a native function, * else `false`. */ function baseIsNative(value) { if (!lodash_es_isObject(value) || _isMasked(value)) { return false; } var pattern = lodash_es_isFunction(value) ? reIsNative : reIsHostCtor; return pattern.test(_toSource(value)); } /* harmony default export */ const _baseIsNative = (baseIsNative); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_getValue.js /** * Gets the value at `key` of `object`. * * @private * @param {Object} [object] The object to query. * @param {string} key The key of the property to get. * @returns {*} Returns the property value. */ function getValue(object, key) { return object == null ? undefined : object[key]; } /* harmony default export */ const _getValue = (getValue); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_getNative.js /** * Gets the native function at `key` of `object`. * * @private * @param {Object} object The object to query. * @param {string} key The key of the method to get. * @returns {*} Returns the function if it's native, else `undefined`. */ function getNative(object, key) { var value = _getValue(object, key); return _baseIsNative(value) ? value : undefined; } /* harmony default export */ const _getNative = (getNative); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_defineProperty.js var defineProperty = (function() { try { var func = _getNative(Object, 'defineProperty'); func({}, '', {}); return func; } catch (e) {} }()); /* harmony default export */ const _defineProperty = (defineProperty); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseAssignValue.js /** * The base implementation of `assignValue` and `assignMergeValue` without * value checks. * * @private * @param {Object} object The object to modify. * @param {string} key The key of the property to assign. * @param {*} value The value to assign. */ function baseAssignValue(object, key, value) { if (key == '__proto__' && _defineProperty) { _defineProperty(object, key, { 'configurable': true, 'enumerable': true, 'value': value, 'writable': true }); } else { object[key] = value; } } /* harmony default export */ const _baseAssignValue = (baseAssignValue); ;// CONCATENATED MODULE: ./node_modules/lodash-es/eq.js /** * Performs a * [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero) * comparison between two values to determine if they are equivalent. * * @static * @memberOf _ * @since 4.0.0 * @category Lang * @param {*} value The value to compare. * @param {*} other The other value to compare. * @returns {boolean} Returns `true` if the values are equivalent, else `false`. * @example * * var object = { 'a': 1 }; * var other = { 'a': 1 }; * * _.eq(object, object); * // => true * * _.eq(object, other); * // => false * * _.eq('a', 'a'); * // => true * * _.eq('a', Object('a')); * // => false * * _.eq(NaN, NaN); * // => true */ function eq(value, other) { return value === other || (value !== value && other !== other); } /* harmony default export */ const lodash_es_eq = (eq); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_assignValue.js /** Used for built-in method references. */ var _assignValue_objectProto = Object.prototype; /** Used to check objects for own properties. */ var _assignValue_hasOwnProperty = _assignValue_objectProto.hasOwnProperty; /** * Assigns `value` to `key` of `object` if the existing value is not equivalent * using [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero) * for equality comparisons. * * @private * @param {Object} object The object to modify. * @param {string} key The key of the property to assign. * @param {*} value The value to assign. */ function assignValue(object, key, value) { var objValue = object[key]; if (!(_assignValue_hasOwnProperty.call(object, key) && lodash_es_eq(objValue, value)) || (value === undefined && !(key in object))) { _baseAssignValue(object, key, value); } } /* harmony default export */ const _assignValue = (assignValue); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_copyObject.js /** * Copies properties of `source` to `object`. * * @private * @param {Object} source The object to copy properties from. * @param {Array} props The property identifiers to copy. * @param {Object} [object={}] The object to copy properties to. * @param {Function} [customizer] The function to customize copied values. * @returns {Object} Returns `object`. */ function copyObject(source, props, object, customizer) { var isNew = !object; object || (object = {}); var index = -1, length = props.length; while (++index < length) { var key = props[index]; var newValue = customizer ? customizer(object[key], source[key], key, object, source) : undefined; if (newValue === undefined) { newValue = source[key]; } if (isNew) { _baseAssignValue(object, key, newValue); } else { _assignValue(object, key, newValue); } } return object; } /* harmony default export */ const _copyObject = (copyObject); ;// CONCATENATED MODULE: ./node_modules/lodash-es/identity.js /** * This method returns the first argument it receives. * * @static * @since 0.1.0 * @memberOf _ * @category Util * @param {*} value Any value. * @returns {*} Returns `value`. * @example * * var object = { 'a': 1 }; * * console.log(_.identity(object) === object); * // => true */ function identity(value) { return value; } /* harmony default export */ const lodash_es_identity = (identity); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_apply.js /** * A faster alternative to `Function#apply`, this function invokes `func` * with the `this` binding of `thisArg` and the arguments of `args`. * * @private * @param {Function} func The function to invoke. * @param {*} thisArg The `this` binding of `func`. * @param {Array} args The arguments to invoke `func` with. * @returns {*} Returns the result of `func`. */ function apply(func, thisArg, args) { switch (args.length) { case 0: return func.call(thisArg); case 1: return func.call(thisArg, args[0]); case 2: return func.call(thisArg, args[0], args[1]); case 3: return func.call(thisArg, args[0], args[1], args[2]); } return func.apply(thisArg, args); } /* harmony default export */ const _apply = (apply); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_overRest.js /* Built-in method references for those with the same name as other `lodash` methods. */ var nativeMax = Math.max; /** * A specialized version of `baseRest` which transforms the rest array. * * @private * @param {Function} func The function to apply a rest parameter to. * @param {number} [start=func.length-1] The start position of the rest parameter. * @param {Function} transform The rest array transform. * @returns {Function} Returns the new function. */ function overRest(func, start, transform) { start = nativeMax(start === undefined ? (func.length - 1) : start, 0); return function() { var args = arguments, index = -1, length = nativeMax(args.length - start, 0), array = Array(length); while (++index < length) { array[index] = args[start + index]; } index = -1; var otherArgs = Array(start + 1); while (++index < start) { otherArgs[index] = args[index]; } otherArgs[start] = transform(array); return _apply(func, this, otherArgs); }; } /* harmony default export */ const _overRest = (overRest); ;// CONCATENATED MODULE: ./node_modules/lodash-es/constant.js /** * Creates a function that returns `value`. * * @static * @memberOf _ * @since 2.4.0 * @category Util * @param {*} value The value to return from the new function. * @returns {Function} Returns the new constant function. * @example * * var objects = _.times(2, _.constant({ 'a': 1 })); * * console.log(objects); * // => [{ 'a': 1 }, { 'a': 1 }] * * console.log(objects[0] === objects[1]); * // => true */ function constant(value) { return function() { return value; }; } /* harmony default export */ const lodash_es_constant = (constant); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseSetToString.js /** * The base implementation of `setToString` without support for hot loop shorting. * * @private * @param {Function} func The function to modify. * @param {Function} string The `toString` result. * @returns {Function} Returns `func`. */ var baseSetToString = !_defineProperty ? lodash_es_identity : function(func, string) { return _defineProperty(func, 'toString', { 'configurable': true, 'enumerable': false, 'value': lodash_es_constant(string), 'writable': true }); }; /* harmony default export */ const _baseSetToString = (baseSetToString); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_shortOut.js /** Used to detect hot functions by number of calls within a span of milliseconds. */ var HOT_COUNT = 800, HOT_SPAN = 16; /* Built-in method references for those with the same name as other `lodash` methods. */ var nativeNow = Date.now; /** * Creates a function that'll short out and invoke `identity` instead * of `func` when it's called `HOT_COUNT` or more times in `HOT_SPAN` * milliseconds. * * @private * @param {Function} func The function to restrict. * @returns {Function} Returns the new shortable function. */ function shortOut(func) { var count = 0, lastCalled = 0; return function() { var stamp = nativeNow(), remaining = HOT_SPAN - (stamp - lastCalled); lastCalled = stamp; if (remaining > 0) { if (++count >= HOT_COUNT) { return arguments[0]; } } else { count = 0; } return func.apply(undefined, arguments); }; } /* harmony default export */ const _shortOut = (shortOut); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_setToString.js /** * Sets the `toString` method of `func` to return `string`. * * @private * @param {Function} func The function to modify. * @param {Function} string The `toString` result. * @returns {Function} Returns `func`. */ var setToString = _shortOut(_baseSetToString); /* harmony default export */ const _setToString = (setToString); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseRest.js /** * The base implementation of `_.rest` which doesn't validate or coerce arguments. * * @private * @param {Function} func The function to apply a rest parameter to. * @param {number} [start=func.length-1] The start position of the rest parameter. * @returns {Function} Returns the new function. */ function baseRest(func, start) { return _setToString(_overRest(func, start, lodash_es_identity), func + ''); } /* harmony default export */ const _baseRest = (baseRest); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isLength.js /** Used as references for various `Number` constants. */ var MAX_SAFE_INTEGER = 9007199254740991; /** * Checks if `value` is a valid array-like length. * * **Note:** This method is loosely based on * [`ToLength`](http://ecma-international.org/ecma-262/7.0/#sec-tolength). * * @static * @memberOf _ * @since 4.0.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a valid length, else `false`. * @example * * _.isLength(3); * // => true * * _.isLength(Number.MIN_VALUE); * // => false * * _.isLength(Infinity); * // => false * * _.isLength('3'); * // => false */ function isLength(value) { return typeof value == 'number' && value > -1 && value % 1 == 0 && value <= MAX_SAFE_INTEGER; } /* harmony default export */ const lodash_es_isLength = (isLength); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isArrayLike.js /** * Checks if `value` is array-like. A value is considered array-like if it's * not a function and has a `value.length` that's an integer greater than or * equal to `0` and less than or equal to `Number.MAX_SAFE_INTEGER`. * * @static * @memberOf _ * @since 4.0.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is array-like, else `false`. * @example * * _.isArrayLike([1, 2, 3]); * // => true * * _.isArrayLike(document.body.children); * // => true * * _.isArrayLike('abc'); * // => true * * _.isArrayLike(_.noop); * // => false */ function isArrayLike(value) { return value != null && lodash_es_isLength(value.length) && !lodash_es_isFunction(value); } /* harmony default export */ const lodash_es_isArrayLike = (isArrayLike); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_isIndex.js /** Used as references for various `Number` constants. */ var _isIndex_MAX_SAFE_INTEGER = 9007199254740991; /** Used to detect unsigned integer values. */ var reIsUint = /^(?:0|[1-9]\d*)$/; /** * Checks if `value` is a valid array-like index. * * @private * @param {*} value The value to check. * @param {number} [length=MAX_SAFE_INTEGER] The upper bounds of a valid index. * @returns {boolean} Returns `true` if `value` is a valid index, else `false`. */ function isIndex(value, length) { var type = typeof value; length = length == null ? _isIndex_MAX_SAFE_INTEGER : length; return !!length && (type == 'number' || (type != 'symbol' && reIsUint.test(value))) && (value > -1 && value % 1 == 0 && value < length); } /* harmony default export */ const _isIndex = (isIndex); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_isIterateeCall.js /** * Checks if the given arguments are from an iteratee call. * * @private * @param {*} value The potential iteratee value argument. * @param {*} index The potential iteratee index or key argument. * @param {*} object The potential iteratee object argument. * @returns {boolean} Returns `true` if the arguments are from an iteratee call, * else `false`. */ function isIterateeCall(value, index, object) { if (!lodash_es_isObject(object)) { return false; } var type = typeof index; if (type == 'number' ? (lodash_es_isArrayLike(object) && _isIndex(index, object.length)) : (type == 'string' && index in object) ) { return lodash_es_eq(object[index], value); } return false; } /* harmony default export */ const _isIterateeCall = (isIterateeCall); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_createAssigner.js /** * Creates a function like `_.assign`. * * @private * @param {Function} assigner The function to assign values. * @returns {Function} Returns the new assigner function. */ function createAssigner(assigner) { return _baseRest(function(object, sources) { var index = -1, length = sources.length, customizer = length > 1 ? sources[length - 1] : undefined, guard = length > 2 ? sources[2] : undefined; customizer = (assigner.length > 3 && typeof customizer == 'function') ? (length--, customizer) : undefined; if (guard && _isIterateeCall(sources[0], sources[1], guard)) { customizer = length < 3 ? undefined : customizer; length = 1; } object = Object(object); while (++index < length) { var source = sources[index]; if (source) { assigner(object, source, index, customizer); } } return object; }); } /* harmony default export */ const _createAssigner = (createAssigner); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseTimes.js /** * The base implementation of `_.times` without support for iteratee shorthands * or max array length checks. * * @private * @param {number} n The number of times to invoke `iteratee`. * @param {Function} iteratee The function invoked per iteration. * @returns {Array} Returns the array of results. */ function baseTimes(n, iteratee) { var index = -1, result = Array(n); while (++index < n) { result[index] = iteratee(index); } return result; } /* harmony default export */ const _baseTimes = (baseTimes); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseIsArguments.js /** `Object#toString` result references. */ var argsTag = '[object Arguments]'; /** * The base implementation of `_.isArguments`. * * @private * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is an `arguments` object, */ function baseIsArguments(value) { return lodash_es_isObjectLike(value) && _baseGetTag(value) == argsTag; } /* harmony default export */ const _baseIsArguments = (baseIsArguments); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isArguments.js /** Used for built-in method references. */ var isArguments_objectProto = Object.prototype; /** Used to check objects for own properties. */ var isArguments_hasOwnProperty = isArguments_objectProto.hasOwnProperty; /** Built-in value references. */ var propertyIsEnumerable = isArguments_objectProto.propertyIsEnumerable; /** * Checks if `value` is likely an `arguments` object. * * @static * @memberOf _ * @since 0.1.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is an `arguments` object, * else `false`. * @example * * _.isArguments(function() { return arguments; }()); * // => true * * _.isArguments([1, 2, 3]); * // => false */ var isArguments = _baseIsArguments(function() { return arguments; }()) ? _baseIsArguments : function(value) { return lodash_es_isObjectLike(value) && isArguments_hasOwnProperty.call(value, 'callee') && !propertyIsEnumerable.call(value, 'callee'); }; /* harmony default export */ const lodash_es_isArguments = (isArguments); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isArray.js /** * Checks if `value` is classified as an `Array` object. * * @static * @memberOf _ * @since 0.1.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is an array, else `false`. * @example * * _.isArray([1, 2, 3]); * // => true * * _.isArray(document.body.children); * // => false * * _.isArray('abc'); * // => false * * _.isArray(_.noop); * // => false */ var isArray = Array.isArray; /* harmony default export */ const lodash_es_isArray = (isArray); ;// CONCATENATED MODULE: ./node_modules/lodash-es/stubFalse.js /** * This method returns `false`. * * @static * @memberOf _ * @since 4.13.0 * @category Util * @returns {boolean} Returns `false`. * @example * * _.times(2, _.stubFalse); * // => [false, false] */ function stubFalse() { return false; } /* harmony default export */ const lodash_es_stubFalse = (stubFalse); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isBuffer.js /** Detect free variable `exports`. */ var freeExports = typeof exports == 'object' && exports && !exports.nodeType && exports; /** Detect free variable `module`. */ var freeModule = freeExports && typeof module == 'object' && module && !module.nodeType && module; /** Detect the popular CommonJS extension `module.exports`. */ var moduleExports = freeModule && freeModule.exports === freeExports; /** Built-in value references. */ var Buffer = moduleExports ? _root.Buffer : undefined; /* Built-in method references for those with the same name as other `lodash` methods. */ var nativeIsBuffer = Buffer ? Buffer.isBuffer : undefined; /** * Checks if `value` is a buffer. * * @static * @memberOf _ * @since 4.3.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a buffer, else `false`. * @example * * _.isBuffer(new Buffer(2)); * // => true * * _.isBuffer(new Uint8Array(2)); * // => false */ var isBuffer = nativeIsBuffer || lodash_es_stubFalse; /* harmony default export */ const lodash_es_isBuffer = (isBuffer); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseIsTypedArray.js /** `Object#toString` result references. */ var _baseIsTypedArray_argsTag = '[object Arguments]', arrayTag = '[object Array]', boolTag = '[object Boolean]', dateTag = '[object Date]', errorTag = '[object Error]', _baseIsTypedArray_funcTag = '[object Function]', mapTag = '[object Map]', numberTag = '[object Number]', _baseIsTypedArray_objectTag = '[object Object]', regexpTag = '[object RegExp]', setTag = '[object Set]', stringTag = '[object String]', weakMapTag = '[object WeakMap]'; var arrayBufferTag = '[object ArrayBuffer]', dataViewTag = '[object DataView]', float32Tag = '[object Float32Array]', float64Tag = '[object Float64Array]', int8Tag = '[object Int8Array]', int16Tag = '[object Int16Array]', int32Tag = '[object Int32Array]', uint8Tag = '[object Uint8Array]', uint8ClampedTag = '[object Uint8ClampedArray]', uint16Tag = '[object Uint16Array]', uint32Tag = '[object Uint32Array]'; /** Used to identify `toStringTag` values of typed arrays. */ var typedArrayTags = {}; typedArrayTags[float32Tag] = typedArrayTags[float64Tag] = typedArrayTags[int8Tag] = typedArrayTags[int16Tag] = typedArrayTags[int32Tag] = typedArrayTags[uint8Tag] = typedArrayTags[uint8ClampedTag] = typedArrayTags[uint16Tag] = typedArrayTags[uint32Tag] = true; typedArrayTags[_baseIsTypedArray_argsTag] = typedArrayTags[arrayTag] = typedArrayTags[arrayBufferTag] = typedArrayTags[boolTag] = typedArrayTags[dataViewTag] = typedArrayTags[dateTag] = typedArrayTags[errorTag] = typedArrayTags[_baseIsTypedArray_funcTag] = typedArrayTags[mapTag] = typedArrayTags[numberTag] = typedArrayTags[_baseIsTypedArray_objectTag] = typedArrayTags[regexpTag] = typedArrayTags[setTag] = typedArrayTags[stringTag] = typedArrayTags[weakMapTag] = false; /** * The base implementation of `_.isTypedArray` without Node.js optimizations. * * @private * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a typed array, else `false`. */ function baseIsTypedArray(value) { return lodash_es_isObjectLike(value) && lodash_es_isLength(value.length) && !!typedArrayTags[_baseGetTag(value)]; } /* harmony default export */ const _baseIsTypedArray = (baseIsTypedArray); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseUnary.js /** * The base implementation of `_.unary` without support for storing metadata. * * @private * @param {Function} func The function to cap arguments for. * @returns {Function} Returns the new capped function. */ function baseUnary(func) { return function(value) { return func(value); }; } /* harmony default export */ const _baseUnary = (baseUnary); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_nodeUtil.js /** Detect free variable `exports`. */ var _nodeUtil_freeExports = typeof exports == 'object' && exports && !exports.nodeType && exports; /** Detect free variable `module`. */ var _nodeUtil_freeModule = _nodeUtil_freeExports && typeof module == 'object' && module && !module.nodeType && module; /** Detect the popular CommonJS extension `module.exports`. */ var _nodeUtil_moduleExports = _nodeUtil_freeModule && _nodeUtil_freeModule.exports === _nodeUtil_freeExports; /** Detect free variable `process` from Node.js. */ var freeProcess = _nodeUtil_moduleExports && _freeGlobal.process; /** Used to access faster Node.js helpers. */ var nodeUtil = (function() { try { // Use `util.types` for Node.js 10+. var types = _nodeUtil_freeModule && _nodeUtil_freeModule.require && _nodeUtil_freeModule.require('util').types; if (types) { return types; } // Legacy `process.binding('util')` for Node.js < 10. return freeProcess && freeProcess.binding && freeProcess.binding('util'); } catch (e) {} }()); /* harmony default export */ const _nodeUtil = (nodeUtil); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isTypedArray.js /* Node.js helper references. */ var nodeIsTypedArray = _nodeUtil && _nodeUtil.isTypedArray; /** * Checks if `value` is classified as a typed array. * * @static * @memberOf _ * @since 3.0.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a typed array, else `false`. * @example * * _.isTypedArray(new Uint8Array); * // => true * * _.isTypedArray([]); * // => false */ var isTypedArray = nodeIsTypedArray ? _baseUnary(nodeIsTypedArray) : _baseIsTypedArray; /* harmony default export */ const lodash_es_isTypedArray = (isTypedArray); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_arrayLikeKeys.js /** Used for built-in method references. */ var _arrayLikeKeys_objectProto = Object.prototype; /** Used to check objects for own properties. */ var _arrayLikeKeys_hasOwnProperty = _arrayLikeKeys_objectProto.hasOwnProperty; /** * Creates an array of the enumerable property names of the array-like `value`. * * @private * @param {*} value The value to query. * @param {boolean} inherited Specify returning inherited property names. * @returns {Array} Returns the array of property names. */ function arrayLikeKeys(value, inherited) { var isArr = lodash_es_isArray(value), isArg = !isArr && lodash_es_isArguments(value), isBuff = !isArr && !isArg && lodash_es_isBuffer(value), isType = !isArr && !isArg && !isBuff && lodash_es_isTypedArray(value), skipIndexes = isArr || isArg || isBuff || isType, result = skipIndexes ? _baseTimes(value.length, String) : [], length = result.length; for (var key in value) { if ((inherited || _arrayLikeKeys_hasOwnProperty.call(value, key)) && !(skipIndexes && ( // Safari 9 has enumerable `arguments.length` in strict mode. key == 'length' || // Node.js 0.10 has enumerable non-index properties on buffers. (isBuff && (key == 'offset' || key == 'parent')) || // PhantomJS 2 has enumerable non-index properties on typed arrays. (isType && (key == 'buffer' || key == 'byteLength' || key == 'byteOffset')) || // Skip index properties. _isIndex(key, length) ))) { result.push(key); } } return result; } /* harmony default export */ const _arrayLikeKeys = (arrayLikeKeys); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_isPrototype.js /** Used for built-in method references. */ var _isPrototype_objectProto = Object.prototype; /** * Checks if `value` is likely a prototype object. * * @private * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a prototype, else `false`. */ function isPrototype(value) { var Ctor = value && value.constructor, proto = (typeof Ctor == 'function' && Ctor.prototype) || _isPrototype_objectProto; return value === proto; } /* harmony default export */ const _isPrototype = (isPrototype); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_nativeKeysIn.js /** * This function is like * [`Object.keys`](http://ecma-international.org/ecma-262/7.0/#sec-object.keys) * except that it includes inherited enumerable properties. * * @private * @param {Object} object The object to query. * @returns {Array} Returns the array of property names. */ function nativeKeysIn(object) { var result = []; if (object != null) { for (var key in Object(object)) { result.push(key); } } return result; } /* harmony default export */ const _nativeKeysIn = (nativeKeysIn); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseKeysIn.js /** Used for built-in method references. */ var _baseKeysIn_objectProto = Object.prototype; /** Used to check objects for own properties. */ var _baseKeysIn_hasOwnProperty = _baseKeysIn_objectProto.hasOwnProperty; /** * The base implementation of `_.keysIn` which doesn't treat sparse arrays as dense. * * @private * @param {Object} object The object to query. * @returns {Array} Returns the array of property names. */ function baseKeysIn(object) { if (!lodash_es_isObject(object)) { return _nativeKeysIn(object); } var isProto = _isPrototype(object), result = []; for (var key in object) { if (!(key == 'constructor' && (isProto || !_baseKeysIn_hasOwnProperty.call(object, key)))) { result.push(key); } } return result; } /* harmony default export */ const _baseKeysIn = (baseKeysIn); ;// CONCATENATED MODULE: ./node_modules/lodash-es/keysIn.js /** * Creates an array of the own and inherited enumerable property names of `object`. * * **Note:** Non-object values are coerced to objects. * * @static * @memberOf _ * @since 3.0.0 * @category Object * @param {Object} object The object to query. * @returns {Array} Returns the array of property names. * @example * * function Foo() { * this.a = 1; * this.b = 2; * } * * Foo.prototype.c = 3; * * _.keysIn(new Foo); * // => ['a', 'b', 'c'] (iteration order is not guaranteed) */ function keysIn(object) { return lodash_es_isArrayLike(object) ? _arrayLikeKeys(object, true) : _baseKeysIn(object); } /* harmony default export */ const lodash_es_keysIn = (keysIn); ;// CONCATENATED MODULE: ./node_modules/lodash-es/assignIn.js /** * This method is like `_.assign` except that it iterates over own and * inherited source properties. * * **Note:** This method mutates `object`. * * @static * @memberOf _ * @since 4.0.0 * @alias extend * @category Object * @param {Object} object The destination object. * @param {...Object} [sources] The source objects. * @returns {Object} Returns `object`. * @see _.assign * @example * * function Foo() { * this.a = 1; * } * * function Bar() { * this.c = 3; * } * * Foo.prototype.b = 2; * Bar.prototype.d = 4; * * _.assignIn({ 'a': 0 }, new Foo, new Bar); * // => { 'a': 1, 'b': 2, 'c': 3, 'd': 4 } */ var assignIn = _createAssigner(function(object, source) { _copyObject(source, lodash_es_keysIn(source), object); }); /* harmony default export */ const lodash_es_assignIn = (assignIn); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_arraySome.js /** * A specialized version of `_.some` for arrays without support for iteratee * shorthands. * * @private * @param {Array} [array] The array to iterate over. * @param {Function} predicate The function invoked per iteration. * @returns {boolean} Returns `true` if any element passes the predicate check, * else `false`. */ function arraySome(array, predicate) { var index = -1, length = array == null ? 0 : array.length; while (++index < length) { if (predicate(array[index], index, array)) { return true; } } return false; } /* harmony default export */ const _arraySome = (arraySome); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_listCacheClear.js /** * Removes all key-value entries from the list cache. * * @private * @name clear * @memberOf ListCache */ function listCacheClear() { this.__data__ = []; this.size = 0; } /* harmony default export */ const _listCacheClear = (listCacheClear); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_assocIndexOf.js /** * Gets the index at which the `key` is found in `array` of key-value pairs. * * @private * @param {Array} array The array to inspect. * @param {*} key The key to search for. * @returns {number} Returns the index of the matched value, else `-1`. */ function assocIndexOf(array, key) { var length = array.length; while (length--) { if (lodash_es_eq(array[length][0], key)) { return length; } } return -1; } /* harmony default export */ const _assocIndexOf = (assocIndexOf); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_listCacheDelete.js /** Used for built-in method references. */ var arrayProto = Array.prototype; /** Built-in value references. */ var splice = arrayProto.splice; /** * Removes `key` and its value from the list cache. * * @private * @name delete * @memberOf ListCache * @param {string} key The key of the value to remove. * @returns {boolean} Returns `true` if the entry was removed, else `false`. */ function listCacheDelete(key) { var data = this.__data__, index = _assocIndexOf(data, key); if (index < 0) { return false; } var lastIndex = data.length - 1; if (index == lastIndex) { data.pop(); } else { splice.call(data, index, 1); } --this.size; return true; } /* harmony default export */ const _listCacheDelete = (listCacheDelete); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_listCacheGet.js /** * Gets the list cache value for `key`. * * @private * @name get * @memberOf ListCache * @param {string} key The key of the value to get. * @returns {*} Returns the entry value. */ function listCacheGet(key) { var data = this.__data__, index = _assocIndexOf(data, key); return index < 0 ? undefined : data[index][1]; } /* harmony default export */ const _listCacheGet = (listCacheGet); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_listCacheHas.js /** * Checks if a list cache value for `key` exists. * * @private * @name has * @memberOf ListCache * @param {string} key The key of the entry to check. * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`. */ function listCacheHas(key) { return _assocIndexOf(this.__data__, key) > -1; } /* harmony default export */ const _listCacheHas = (listCacheHas); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_listCacheSet.js /** * Sets the list cache `key` to `value`. * * @private * @name set * @memberOf ListCache * @param {string} key The key of the value to set. * @param {*} value The value to set. * @returns {Object} Returns the list cache instance. */ function listCacheSet(key, value) { var data = this.__data__, index = _assocIndexOf(data, key); if (index < 0) { ++this.size; data.push([key, value]); } else { data[index][1] = value; } return this; } /* harmony default export */ const _listCacheSet = (listCacheSet); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_ListCache.js /** * Creates an list cache object. * * @private * @constructor * @param {Array} [entries] The key-value pairs to cache. */ function ListCache(entries) { var index = -1, length = entries == null ? 0 : entries.length; this.clear(); while (++index < length) { var entry = entries[index]; this.set(entry[0], entry[1]); } } // Add methods to `ListCache`. ListCache.prototype.clear = _listCacheClear; ListCache.prototype['delete'] = _listCacheDelete; ListCache.prototype.get = _listCacheGet; ListCache.prototype.has = _listCacheHas; ListCache.prototype.set = _listCacheSet; /* harmony default export */ const _ListCache = (ListCache); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_stackClear.js /** * Removes all key-value entries from the stack. * * @private * @name clear * @memberOf Stack */ function stackClear() { this.__data__ = new _ListCache; this.size = 0; } /* harmony default export */ const _stackClear = (stackClear); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_stackDelete.js /** * Removes `key` and its value from the stack. * * @private * @name delete * @memberOf Stack * @param {string} key The key of the value to remove. * @returns {boolean} Returns `true` if the entry was removed, else `false`. */ function stackDelete(key) { var data = this.__data__, result = data['delete'](key); this.size = data.size; return result; } /* harmony default export */ const _stackDelete = (stackDelete); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_stackGet.js /** * Gets the stack value for `key`. * * @private * @name get * @memberOf Stack * @param {string} key The key of the value to get. * @returns {*} Returns the entry value. */ function stackGet(key) { return this.__data__.get(key); } /* harmony default export */ const _stackGet = (stackGet); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_stackHas.js /** * Checks if a stack value for `key` exists. * * @private * @name has * @memberOf Stack * @param {string} key The key of the entry to check. * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`. */ function stackHas(key) { return this.__data__.has(key); } /* harmony default export */ const _stackHas = (stackHas); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_Map.js /* Built-in method references that are verified to be native. */ var _Map_Map = _getNative(_root, 'Map'); /* harmony default export */ const _Map = (_Map_Map); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_nativeCreate.js /* Built-in method references that are verified to be native. */ var nativeCreate = _getNative(Object, 'create'); /* harmony default export */ const _nativeCreate = (nativeCreate); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_hashClear.js /** * Removes all key-value entries from the hash. * * @private * @name clear * @memberOf Hash */ function hashClear() { this.__data__ = _nativeCreate ? _nativeCreate(null) : {}; this.size = 0; } /* harmony default export */ const _hashClear = (hashClear); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_hashDelete.js /** * Removes `key` and its value from the hash. * * @private * @name delete * @memberOf Hash * @param {Object} hash The hash to modify. * @param {string} key The key of the value to remove. * @returns {boolean} Returns `true` if the entry was removed, else `false`. */ function hashDelete(key) { var result = this.has(key) && delete this.__data__[key]; this.size -= result ? 1 : 0; return result; } /* harmony default export */ const _hashDelete = (hashDelete); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_hashGet.js /** Used to stand-in for `undefined` hash values. */ var HASH_UNDEFINED = '__lodash_hash_undefined__'; /** Used for built-in method references. */ var _hashGet_objectProto = Object.prototype; /** Used to check objects for own properties. */ var _hashGet_hasOwnProperty = _hashGet_objectProto.hasOwnProperty; /** * Gets the hash value for `key`. * * @private * @name get * @memberOf Hash * @param {string} key The key of the value to get. * @returns {*} Returns the entry value. */ function hashGet(key) { var data = this.__data__; if (_nativeCreate) { var result = data[key]; return result === HASH_UNDEFINED ? undefined : result; } return _hashGet_hasOwnProperty.call(data, key) ? data[key] : undefined; } /* harmony default export */ const _hashGet = (hashGet); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_hashHas.js /** Used for built-in method references. */ var _hashHas_objectProto = Object.prototype; /** Used to check objects for own properties. */ var _hashHas_hasOwnProperty = _hashHas_objectProto.hasOwnProperty; /** * Checks if a hash value for `key` exists. * * @private * @name has * @memberOf Hash * @param {string} key The key of the entry to check. * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`. */ function hashHas(key) { var data = this.__data__; return _nativeCreate ? (data[key] !== undefined) : _hashHas_hasOwnProperty.call(data, key); } /* harmony default export */ const _hashHas = (hashHas); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_hashSet.js /** Used to stand-in for `undefined` hash values. */ var _hashSet_HASH_UNDEFINED = '__lodash_hash_undefined__'; /** * Sets the hash `key` to `value`. * * @private * @name set * @memberOf Hash * @param {string} key The key of the value to set. * @param {*} value The value to set. * @returns {Object} Returns the hash instance. */ function hashSet(key, value) { var data = this.__data__; this.size += this.has(key) ? 0 : 1; data[key] = (_nativeCreate && value === undefined) ? _hashSet_HASH_UNDEFINED : value; return this; } /* harmony default export */ const _hashSet = (hashSet); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_Hash.js /** * Creates a hash object. * * @private * @constructor * @param {Array} [entries] The key-value pairs to cache. */ function Hash(entries) { var index = -1, length = entries == null ? 0 : entries.length; this.clear(); while (++index < length) { var entry = entries[index]; this.set(entry[0], entry[1]); } } // Add methods to `Hash`. Hash.prototype.clear = _hashClear; Hash.prototype['delete'] = _hashDelete; Hash.prototype.get = _hashGet; Hash.prototype.has = _hashHas; Hash.prototype.set = _hashSet; /* harmony default export */ const _Hash = (Hash); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_mapCacheClear.js /** * Removes all key-value entries from the map. * * @private * @name clear * @memberOf MapCache */ function mapCacheClear() { this.size = 0; this.__data__ = { 'hash': new _Hash, 'map': new (_Map || _ListCache), 'string': new _Hash }; } /* harmony default export */ const _mapCacheClear = (mapCacheClear); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_isKeyable.js /** * Checks if `value` is suitable for use as unique object key. * * @private * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is suitable, else `false`. */ function isKeyable(value) { var type = typeof value; return (type == 'string' || type == 'number' || type == 'symbol' || type == 'boolean') ? (value !== '__proto__') : (value === null); } /* harmony default export */ const _isKeyable = (isKeyable); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_getMapData.js /** * Gets the data for `map`. * * @private * @param {Object} map The map to query. * @param {string} key The reference key. * @returns {*} Returns the map data. */ function getMapData(map, key) { var data = map.__data__; return _isKeyable(key) ? data[typeof key == 'string' ? 'string' : 'hash'] : data.map; } /* harmony default export */ const _getMapData = (getMapData); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_mapCacheDelete.js /** * Removes `key` and its value from the map. * * @private * @name delete * @memberOf MapCache * @param {string} key The key of the value to remove. * @returns {boolean} Returns `true` if the entry was removed, else `false`. */ function mapCacheDelete(key) { var result = _getMapData(this, key)['delete'](key); this.size -= result ? 1 : 0; return result; } /* harmony default export */ const _mapCacheDelete = (mapCacheDelete); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_mapCacheGet.js /** * Gets the map value for `key`. * * @private * @name get * @memberOf MapCache * @param {string} key The key of the value to get. * @returns {*} Returns the entry value. */ function mapCacheGet(key) { return _getMapData(this, key).get(key); } /* harmony default export */ const _mapCacheGet = (mapCacheGet); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_mapCacheHas.js /** * Checks if a map value for `key` exists. * * @private * @name has * @memberOf MapCache * @param {string} key The key of the entry to check. * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`. */ function mapCacheHas(key) { return _getMapData(this, key).has(key); } /* harmony default export */ const _mapCacheHas = (mapCacheHas); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_mapCacheSet.js /** * Sets the map `key` to `value`. * * @private * @name set * @memberOf MapCache * @param {string} key The key of the value to set. * @param {*} value The value to set. * @returns {Object} Returns the map cache instance. */ function mapCacheSet(key, value) { var data = _getMapData(this, key), size = data.size; data.set(key, value); this.size += data.size == size ? 0 : 1; return this; } /* harmony default export */ const _mapCacheSet = (mapCacheSet); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_MapCache.js /** * Creates a map cache object to store key-value pairs. * * @private * @constructor * @param {Array} [entries] The key-value pairs to cache. */ function MapCache(entries) { var index = -1, length = entries == null ? 0 : entries.length; this.clear(); while (++index < length) { var entry = entries[index]; this.set(entry[0], entry[1]); } } // Add methods to `MapCache`. MapCache.prototype.clear = _mapCacheClear; MapCache.prototype['delete'] = _mapCacheDelete; MapCache.prototype.get = _mapCacheGet; MapCache.prototype.has = _mapCacheHas; MapCache.prototype.set = _mapCacheSet; /* harmony default export */ const _MapCache = (MapCache); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_stackSet.js /** Used as the size to enable large array optimizations. */ var LARGE_ARRAY_SIZE = 200; /** * Sets the stack `key` to `value`. * * @private * @name set * @memberOf Stack * @param {string} key The key of the value to set. * @param {*} value The value to set. * @returns {Object} Returns the stack cache instance. */ function stackSet(key, value) { var data = this.__data__; if (data instanceof _ListCache) { var pairs = data.__data__; if (!_Map || (pairs.length < LARGE_ARRAY_SIZE - 1)) { pairs.push([key, value]); this.size = ++data.size; return this; } data = this.__data__ = new _MapCache(pairs); } data.set(key, value); this.size = data.size; return this; } /* harmony default export */ const _stackSet = (stackSet); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_Stack.js /** * Creates a stack cache object to store key-value pairs. * * @private * @constructor * @param {Array} [entries] The key-value pairs to cache. */ function Stack(entries) { var data = this.__data__ = new _ListCache(entries); this.size = data.size; } // Add methods to `Stack`. Stack.prototype.clear = _stackClear; Stack.prototype['delete'] = _stackDelete; Stack.prototype.get = _stackGet; Stack.prototype.has = _stackHas; Stack.prototype.set = _stackSet; /* harmony default export */ const _Stack = (Stack); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_setCacheAdd.js /** Used to stand-in for `undefined` hash values. */ var _setCacheAdd_HASH_UNDEFINED = '__lodash_hash_undefined__'; /** * Adds `value` to the array cache. * * @private * @name add * @memberOf SetCache * @alias push * @param {*} value The value to cache. * @returns {Object} Returns the cache instance. */ function setCacheAdd(value) { this.__data__.set(value, _setCacheAdd_HASH_UNDEFINED); return this; } /* harmony default export */ const _setCacheAdd = (setCacheAdd); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_setCacheHas.js /** * Checks if `value` is in the array cache. * * @private * @name has * @memberOf SetCache * @param {*} value The value to search for. * @returns {number} Returns `true` if `value` is found, else `false`. */ function setCacheHas(value) { return this.__data__.has(value); } /* harmony default export */ const _setCacheHas = (setCacheHas); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_SetCache.js /** * * Creates an array cache object to store unique values. * * @private * @constructor * @param {Array} [values] The values to cache. */ function SetCache(values) { var index = -1, length = values == null ? 0 : values.length; this.__data__ = new _MapCache; while (++index < length) { this.add(values[index]); } } // Add methods to `SetCache`. SetCache.prototype.add = SetCache.prototype.push = _setCacheAdd; SetCache.prototype.has = _setCacheHas; /* harmony default export */ const _SetCache = (SetCache); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_cacheHas.js /** * Checks if a `cache` value for `key` exists. * * @private * @param {Object} cache The cache to query. * @param {string} key The key of the entry to check. * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`. */ function cacheHas(cache, key) { return cache.has(key); } /* harmony default export */ const _cacheHas = (cacheHas); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_equalArrays.js /** Used to compose bitmasks for value comparisons. */ var COMPARE_PARTIAL_FLAG = 1, COMPARE_UNORDERED_FLAG = 2; /** * A specialized version of `baseIsEqualDeep` for arrays with support for * partial deep comparisons. * * @private * @param {Array} array The array to compare. * @param {Array} other The other array to compare. * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details. * @param {Function} customizer The function to customize comparisons. * @param {Function} equalFunc The function to determine equivalents of values. * @param {Object} stack Tracks traversed `array` and `other` objects. * @returns {boolean} Returns `true` if the arrays are equivalent, else `false`. */ function equalArrays(array, other, bitmask, customizer, equalFunc, stack) { var isPartial = bitmask & COMPARE_PARTIAL_FLAG, arrLength = array.length, othLength = other.length; if (arrLength != othLength && !(isPartial && othLength > arrLength)) { return false; } // Check that cyclic values are equal. var arrStacked = stack.get(array); var othStacked = stack.get(other); if (arrStacked && othStacked) { return arrStacked == other && othStacked == array; } var index = -1, result = true, seen = (bitmask & COMPARE_UNORDERED_FLAG) ? new _SetCache : undefined; stack.set(array, other); stack.set(other, array); // Ignore non-index properties. while (++index < arrLength) { var arrValue = array[index], othValue = other[index]; if (customizer) { var compared = isPartial ? customizer(othValue, arrValue, index, other, array, stack) : customizer(arrValue, othValue, index, array, other, stack); } if (compared !== undefined) { if (compared) { continue; } result = false; break; } // Recursively compare arrays (susceptible to call stack limits). if (seen) { if (!_arraySome(other, function(othValue, othIndex) { if (!_cacheHas(seen, othIndex) && (arrValue === othValue || equalFunc(arrValue, othValue, bitmask, customizer, stack))) { return seen.push(othIndex); } })) { result = false; break; } } else if (!( arrValue === othValue || equalFunc(arrValue, othValue, bitmask, customizer, stack) )) { result = false; break; } } stack['delete'](array); stack['delete'](other); return result; } /* harmony default export */ const _equalArrays = (equalArrays); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_Uint8Array.js /** Built-in value references. */ var _Uint8Array_Uint8Array = _root.Uint8Array; /* harmony default export */ const _Uint8Array = (_Uint8Array_Uint8Array); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_mapToArray.js /** * Converts `map` to its key-value pairs. * * @private * @param {Object} map The map to convert. * @returns {Array} Returns the key-value pairs. */ function mapToArray(map) { var index = -1, result = Array(map.size); map.forEach(function(value, key) { result[++index] = [key, value]; }); return result; } /* harmony default export */ const _mapToArray = (mapToArray); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_setToArray.js /** * Converts `set` to an array of its values. * * @private * @param {Object} set The set to convert. * @returns {Array} Returns the values. */ function setToArray(set) { var index = -1, result = Array(set.size); set.forEach(function(value) { result[++index] = value; }); return result; } /* harmony default export */ const _setToArray = (setToArray); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_equalByTag.js /** Used to compose bitmasks for value comparisons. */ var _equalByTag_COMPARE_PARTIAL_FLAG = 1, _equalByTag_COMPARE_UNORDERED_FLAG = 2; /** `Object#toString` result references. */ var _equalByTag_boolTag = '[object Boolean]', _equalByTag_dateTag = '[object Date]', _equalByTag_errorTag = '[object Error]', _equalByTag_mapTag = '[object Map]', _equalByTag_numberTag = '[object Number]', _equalByTag_regexpTag = '[object RegExp]', _equalByTag_setTag = '[object Set]', _equalByTag_stringTag = '[object String]', symbolTag = '[object Symbol]'; var _equalByTag_arrayBufferTag = '[object ArrayBuffer]', _equalByTag_dataViewTag = '[object DataView]'; /** Used to convert symbols to primitives and strings. */ var symbolProto = _Symbol ? _Symbol.prototype : undefined, symbolValueOf = symbolProto ? symbolProto.valueOf : undefined; /** * A specialized version of `baseIsEqualDeep` for comparing objects of * the same `toStringTag`. * * **Note:** This function only supports comparing values with tags of * `Boolean`, `Date`, `Error`, `Number`, `RegExp`, or `String`. * * @private * @param {Object} object The object to compare. * @param {Object} other The other object to compare. * @param {string} tag The `toStringTag` of the objects to compare. * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details. * @param {Function} customizer The function to customize comparisons. * @param {Function} equalFunc The function to determine equivalents of values. * @param {Object} stack Tracks traversed `object` and `other` objects. * @returns {boolean} Returns `true` if the objects are equivalent, else `false`. */ function equalByTag(object, other, tag, bitmask, customizer, equalFunc, stack) { switch (tag) { case _equalByTag_dataViewTag: if ((object.byteLength != other.byteLength) || (object.byteOffset != other.byteOffset)) { return false; } object = object.buffer; other = other.buffer; case _equalByTag_arrayBufferTag: if ((object.byteLength != other.byteLength) || !equalFunc(new _Uint8Array(object), new _Uint8Array(other))) { return false; } return true; case _equalByTag_boolTag: case _equalByTag_dateTag: case _equalByTag_numberTag: // Coerce booleans to `1` or `0` and dates to milliseconds. // Invalid dates are coerced to `NaN`. return lodash_es_eq(+object, +other); case _equalByTag_errorTag: return object.name == other.name && object.message == other.message; case _equalByTag_regexpTag: case _equalByTag_stringTag: // Coerce regexes to strings and treat strings, primitives and objects, // as equal. See http://www.ecma-international.org/ecma-262/7.0/#sec-regexp.prototype.tostring // for more details. return object == (other + ''); case _equalByTag_mapTag: var convert = _mapToArray; case _equalByTag_setTag: var isPartial = bitmask & _equalByTag_COMPARE_PARTIAL_FLAG; convert || (convert = _setToArray); if (object.size != other.size && !isPartial) { return false; } // Assume cyclic values are equal. var stacked = stack.get(object); if (stacked) { return stacked == other; } bitmask |= _equalByTag_COMPARE_UNORDERED_FLAG; // Recursively compare objects (susceptible to call stack limits). stack.set(object, other); var result = _equalArrays(convert(object), convert(other), bitmask, customizer, equalFunc, stack); stack['delete'](object); return result; case symbolTag: if (symbolValueOf) { return symbolValueOf.call(object) == symbolValueOf.call(other); } } return false; } /* harmony default export */ const _equalByTag = (equalByTag); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_arrayPush.js /** * Appends the elements of `values` to `array`. * * @private * @param {Array} array The array to modify. * @param {Array} values The values to append. * @returns {Array} Returns `array`. */ function arrayPush(array, values) { var index = -1, length = values.length, offset = array.length; while (++index < length) { array[offset + index] = values[index]; } return array; } /* harmony default export */ const _arrayPush = (arrayPush); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseGetAllKeys.js /** * The base implementation of `getAllKeys` and `getAllKeysIn` which uses * `keysFunc` and `symbolsFunc` to get the enumerable property names and * symbols of `object`. * * @private * @param {Object} object The object to query. * @param {Function} keysFunc The function to get the keys of `object`. * @param {Function} symbolsFunc The function to get the symbols of `object`. * @returns {Array} Returns the array of property names and symbols. */ function baseGetAllKeys(object, keysFunc, symbolsFunc) { var result = keysFunc(object); return lodash_es_isArray(object) ? result : _arrayPush(result, symbolsFunc(object)); } /* harmony default export */ const _baseGetAllKeys = (baseGetAllKeys); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_arrayFilter.js /** * A specialized version of `_.filter` for arrays without support for * iteratee shorthands. * * @private * @param {Array} [array] The array to iterate over. * @param {Function} predicate The function invoked per iteration. * @returns {Array} Returns the new filtered array. */ function arrayFilter(array, predicate) { var index = -1, length = array == null ? 0 : array.length, resIndex = 0, result = []; while (++index < length) { var value = array[index]; if (predicate(value, index, array)) { result[resIndex++] = value; } } return result; } /* harmony default export */ const _arrayFilter = (arrayFilter); ;// CONCATENATED MODULE: ./node_modules/lodash-es/stubArray.js /** * This method returns a new empty array. * * @static * @memberOf _ * @since 4.13.0 * @category Util * @returns {Array} Returns the new empty array. * @example * * var arrays = _.times(2, _.stubArray); * * console.log(arrays); * // => [[], []] * * console.log(arrays[0] === arrays[1]); * // => false */ function stubArray() { return []; } /* harmony default export */ const lodash_es_stubArray = (stubArray); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_getSymbols.js /** Used for built-in method references. */ var _getSymbols_objectProto = Object.prototype; /** Built-in value references. */ var _getSymbols_propertyIsEnumerable = _getSymbols_objectProto.propertyIsEnumerable; /* Built-in method references for those with the same name as other `lodash` methods. */ var nativeGetSymbols = Object.getOwnPropertySymbols; /** * Creates an array of the own enumerable symbols of `object`. * * @private * @param {Object} object The object to query. * @returns {Array} Returns the array of symbols. */ var getSymbols = !nativeGetSymbols ? lodash_es_stubArray : function(object) { if (object == null) { return []; } object = Object(object); return _arrayFilter(nativeGetSymbols(object), function(symbol) { return _getSymbols_propertyIsEnumerable.call(object, symbol); }); }; /* harmony default export */ const _getSymbols = (getSymbols); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_nativeKeys.js /* Built-in method references for those with the same name as other `lodash` methods. */ var nativeKeys = _overArg(Object.keys, Object); /* harmony default export */ const _nativeKeys = (nativeKeys); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseKeys.js /** Used for built-in method references. */ var _baseKeys_objectProto = Object.prototype; /** Used to check objects for own properties. */ var _baseKeys_hasOwnProperty = _baseKeys_objectProto.hasOwnProperty; /** * The base implementation of `_.keys` which doesn't treat sparse arrays as dense. * * @private * @param {Object} object The object to query. * @returns {Array} Returns the array of property names. */ function baseKeys(object) { if (!_isPrototype(object)) { return _nativeKeys(object); } var result = []; for (var key in Object(object)) { if (_baseKeys_hasOwnProperty.call(object, key) && key != 'constructor') { result.push(key); } } return result; } /* harmony default export */ const _baseKeys = (baseKeys); ;// CONCATENATED MODULE: ./node_modules/lodash-es/keys.js /** * Creates an array of the own enumerable property names of `object`. * * **Note:** Non-object values are coerced to objects. See the * [ES spec](http://ecma-international.org/ecma-262/7.0/#sec-object.keys) * for more details. * * @static * @since 0.1.0 * @memberOf _ * @category Object * @param {Object} object The object to query. * @returns {Array} Returns the array of property names. * @example * * function Foo() { * this.a = 1; * this.b = 2; * } * * Foo.prototype.c = 3; * * _.keys(new Foo); * // => ['a', 'b'] (iteration order is not guaranteed) * * _.keys('hi'); * // => ['0', '1'] */ function keys(object) { return lodash_es_isArrayLike(object) ? _arrayLikeKeys(object) : _baseKeys(object); } /* harmony default export */ const lodash_es_keys = (keys); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_getAllKeys.js /** * Creates an array of own enumerable property names and symbols of `object`. * * @private * @param {Object} object The object to query. * @returns {Array} Returns the array of property names and symbols. */ function getAllKeys(object) { return _baseGetAllKeys(object, lodash_es_keys, _getSymbols); } /* harmony default export */ const _getAllKeys = (getAllKeys); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_equalObjects.js /** Used to compose bitmasks for value comparisons. */ var _equalObjects_COMPARE_PARTIAL_FLAG = 1; /** Used for built-in method references. */ var _equalObjects_objectProto = Object.prototype; /** Used to check objects for own properties. */ var _equalObjects_hasOwnProperty = _equalObjects_objectProto.hasOwnProperty; /** * A specialized version of `baseIsEqualDeep` for objects with support for * partial deep comparisons. * * @private * @param {Object} object The object to compare. * @param {Object} other The other object to compare. * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details. * @param {Function} customizer The function to customize comparisons. * @param {Function} equalFunc The function to determine equivalents of values. * @param {Object} stack Tracks traversed `object` and `other` objects. * @returns {boolean} Returns `true` if the objects are equivalent, else `false`. */ function equalObjects(object, other, bitmask, customizer, equalFunc, stack) { var isPartial = bitmask & _equalObjects_COMPARE_PARTIAL_FLAG, objProps = _getAllKeys(object), objLength = objProps.length, othProps = _getAllKeys(other), othLength = othProps.length; if (objLength != othLength && !isPartial) { return false; } var index = objLength; while (index--) { var key = objProps[index]; if (!(isPartial ? key in other : _equalObjects_hasOwnProperty.call(other, key))) { return false; } } // Check that cyclic values are equal. var objStacked = stack.get(object); var othStacked = stack.get(other); if (objStacked && othStacked) { return objStacked == other && othStacked == object; } var result = true; stack.set(object, other); stack.set(other, object); var skipCtor = isPartial; while (++index < objLength) { key = objProps[index]; var objValue = object[key], othValue = other[key]; if (customizer) { var compared = isPartial ? customizer(othValue, objValue, key, other, object, stack) : customizer(objValue, othValue, key, object, other, stack); } // Recursively compare objects (susceptible to call stack limits). if (!(compared === undefined ? (objValue === othValue || equalFunc(objValue, othValue, bitmask, customizer, stack)) : compared )) { result = false; break; } skipCtor || (skipCtor = key == 'constructor'); } if (result && !skipCtor) { var objCtor = object.constructor, othCtor = other.constructor; // Non `Object` object instances with different constructors are not equal. if (objCtor != othCtor && ('constructor' in object && 'constructor' in other) && !(typeof objCtor == 'function' && objCtor instanceof objCtor && typeof othCtor == 'function' && othCtor instanceof othCtor)) { result = false; } } stack['delete'](object); stack['delete'](other); return result; } /* harmony default export */ const _equalObjects = (equalObjects); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_DataView.js /* Built-in method references that are verified to be native. */ var DataView = _getNative(_root, 'DataView'); /* harmony default export */ const _DataView = (DataView); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_Promise.js /* Built-in method references that are verified to be native. */ var _Promise_Promise = _getNative(_root, 'Promise'); /* harmony default export */ const _Promise = (_Promise_Promise); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_Set.js /* Built-in method references that are verified to be native. */ var _Set_Set = _getNative(_root, 'Set'); /* harmony default export */ const _Set = (_Set_Set); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_WeakMap.js /* Built-in method references that are verified to be native. */ var _WeakMap_WeakMap = _getNative(_root, 'WeakMap'); /* harmony default export */ const _WeakMap = (_WeakMap_WeakMap); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_getTag.js /** `Object#toString` result references. */ var _getTag_mapTag = '[object Map]', _getTag_objectTag = '[object Object]', promiseTag = '[object Promise]', _getTag_setTag = '[object Set]', _getTag_weakMapTag = '[object WeakMap]'; var _getTag_dataViewTag = '[object DataView]'; /** Used to detect maps, sets, and weakmaps. */ var dataViewCtorString = _toSource(_DataView), mapCtorString = _toSource(_Map), promiseCtorString = _toSource(_Promise), setCtorString = _toSource(_Set), weakMapCtorString = _toSource(_WeakMap); /** * Gets the `toStringTag` of `value`. * * @private * @param {*} value The value to query. * @returns {string} Returns the `toStringTag`. */ var getTag = _baseGetTag; // Fallback for data views, maps, sets, and weak maps in IE 11 and promises in Node.js < 6. if ((_DataView && getTag(new _DataView(new ArrayBuffer(1))) != _getTag_dataViewTag) || (_Map && getTag(new _Map) != _getTag_mapTag) || (_Promise && getTag(_Promise.resolve()) != promiseTag) || (_Set && getTag(new _Set) != _getTag_setTag) || (_WeakMap && getTag(new _WeakMap) != _getTag_weakMapTag)) { getTag = function(value) { var result = _baseGetTag(value), Ctor = result == _getTag_objectTag ? value.constructor : undefined, ctorString = Ctor ? _toSource(Ctor) : ''; if (ctorString) { switch (ctorString) { case dataViewCtorString: return _getTag_dataViewTag; case mapCtorString: return _getTag_mapTag; case promiseCtorString: return promiseTag; case setCtorString: return _getTag_setTag; case weakMapCtorString: return _getTag_weakMapTag; } } return result; }; } /* harmony default export */ const _getTag = (getTag); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseIsEqualDeep.js /** Used to compose bitmasks for value comparisons. */ var _baseIsEqualDeep_COMPARE_PARTIAL_FLAG = 1; /** `Object#toString` result references. */ var _baseIsEqualDeep_argsTag = '[object Arguments]', _baseIsEqualDeep_arrayTag = '[object Array]', _baseIsEqualDeep_objectTag = '[object Object]'; /** Used for built-in method references. */ var _baseIsEqualDeep_objectProto = Object.prototype; /** Used to check objects for own properties. */ var _baseIsEqualDeep_hasOwnProperty = _baseIsEqualDeep_objectProto.hasOwnProperty; /** * A specialized version of `baseIsEqual` for arrays and objects which performs * deep comparisons and tracks traversed objects enabling objects with circular * references to be compared. * * @private * @param {Object} object The object to compare. * @param {Object} other The other object to compare. * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details. * @param {Function} customizer The function to customize comparisons. * @param {Function} equalFunc The function to determine equivalents of values. * @param {Object} [stack] Tracks traversed `object` and `other` objects. * @returns {boolean} Returns `true` if the objects are equivalent, else `false`. */ function baseIsEqualDeep(object, other, bitmask, customizer, equalFunc, stack) { var objIsArr = lodash_es_isArray(object), othIsArr = lodash_es_isArray(other), objTag = objIsArr ? _baseIsEqualDeep_arrayTag : _getTag(object), othTag = othIsArr ? _baseIsEqualDeep_arrayTag : _getTag(other); objTag = objTag == _baseIsEqualDeep_argsTag ? _baseIsEqualDeep_objectTag : objTag; othTag = othTag == _baseIsEqualDeep_argsTag ? _baseIsEqualDeep_objectTag : othTag; var objIsObj = objTag == _baseIsEqualDeep_objectTag, othIsObj = othTag == _baseIsEqualDeep_objectTag, isSameTag = objTag == othTag; if (isSameTag && lodash_es_isBuffer(object)) { if (!lodash_es_isBuffer(other)) { return false; } objIsArr = true; objIsObj = false; } if (isSameTag && !objIsObj) { stack || (stack = new _Stack); return (objIsArr || lodash_es_isTypedArray(object)) ? _equalArrays(object, other, bitmask, customizer, equalFunc, stack) : _equalByTag(object, other, objTag, bitmask, customizer, equalFunc, stack); } if (!(bitmask & _baseIsEqualDeep_COMPARE_PARTIAL_FLAG)) { var objIsWrapped = objIsObj && _baseIsEqualDeep_hasOwnProperty.call(object, '__wrapped__'), othIsWrapped = othIsObj && _baseIsEqualDeep_hasOwnProperty.call(other, '__wrapped__'); if (objIsWrapped || othIsWrapped) { var objUnwrapped = objIsWrapped ? object.value() : object, othUnwrapped = othIsWrapped ? other.value() : other; stack || (stack = new _Stack); return equalFunc(objUnwrapped, othUnwrapped, bitmask, customizer, stack); } } if (!isSameTag) { return false; } stack || (stack = new _Stack); return _equalObjects(object, other, bitmask, customizer, equalFunc, stack); } /* harmony default export */ const _baseIsEqualDeep = (baseIsEqualDeep); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseIsEqual.js /** * The base implementation of `_.isEqual` which supports partial comparisons * and tracks traversed objects. * * @private * @param {*} value The value to compare. * @param {*} other The other value to compare. * @param {boolean} bitmask The bitmask flags. * 1 - Unordered comparison * 2 - Partial comparison * @param {Function} [customizer] The function to customize comparisons. * @param {Object} [stack] Tracks traversed `value` and `other` objects. * @returns {boolean} Returns `true` if the values are equivalent, else `false`. */ function baseIsEqual(value, other, bitmask, customizer, stack) { if (value === other) { return true; } if (value == null || other == null || (!lodash_es_isObjectLike(value) && !lodash_es_isObjectLike(other))) { return value !== value && other !== other; } return _baseIsEqualDeep(value, other, bitmask, customizer, baseIsEqual, stack); } /* harmony default export */ const _baseIsEqual = (baseIsEqual); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseIsMatch.js /** Used to compose bitmasks for value comparisons. */ var _baseIsMatch_COMPARE_PARTIAL_FLAG = 1, _baseIsMatch_COMPARE_UNORDERED_FLAG = 2; /** * The base implementation of `_.isMatch` without support for iteratee shorthands. * * @private * @param {Object} object The object to inspect. * @param {Object} source The object of property values to match. * @param {Array} matchData The property names, values, and compare flags to match. * @param {Function} [customizer] The function to customize comparisons. * @returns {boolean} Returns `true` if `object` is a match, else `false`. */ function baseIsMatch(object, source, matchData, customizer) { var index = matchData.length, length = index, noCustomizer = !customizer; if (object == null) { return !length; } object = Object(object); while (index--) { var data = matchData[index]; if ((noCustomizer && data[2]) ? data[1] !== object[data[0]] : !(data[0] in object) ) { return false; } } while (++index < length) { data = matchData[index]; var key = data[0], objValue = object[key], srcValue = data[1]; if (noCustomizer && data[2]) { if (objValue === undefined && !(key in object)) { return false; } } else { var stack = new _Stack; if (customizer) { var result = customizer(objValue, srcValue, key, object, source, stack); } if (!(result === undefined ? _baseIsEqual(srcValue, objValue, _baseIsMatch_COMPARE_PARTIAL_FLAG | _baseIsMatch_COMPARE_UNORDERED_FLAG, customizer, stack) : result )) { return false; } } } return true; } /* harmony default export */ const _baseIsMatch = (baseIsMatch); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_isStrictComparable.js /** * Checks if `value` is suitable for strict equality comparisons, i.e. `===`. * * @private * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` if suitable for strict * equality comparisons, else `false`. */ function isStrictComparable(value) { return value === value && !lodash_es_isObject(value); } /* harmony default export */ const _isStrictComparable = (isStrictComparable); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_getMatchData.js /** * Gets the property names, values, and compare flags of `object`. * * @private * @param {Object} object The object to query. * @returns {Array} Returns the match data of `object`. */ function getMatchData(object) { var result = lodash_es_keys(object), length = result.length; while (length--) { var key = result[length], value = object[key]; result[length] = [key, value, _isStrictComparable(value)]; } return result; } /* harmony default export */ const _getMatchData = (getMatchData); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_matchesStrictComparable.js /** * A specialized version of `matchesProperty` for source values suitable * for strict equality comparisons, i.e. `===`. * * @private * @param {string} key The key of the property to get. * @param {*} srcValue The value to match. * @returns {Function} Returns the new spec function. */ function matchesStrictComparable(key, srcValue) { return function(object) { if (object == null) { return false; } return object[key] === srcValue && (srcValue !== undefined || (key in Object(object))); }; } /* harmony default export */ const _matchesStrictComparable = (matchesStrictComparable); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseMatches.js /** * The base implementation of `_.matches` which doesn't clone `source`. * * @private * @param {Object} source The object of property values to match. * @returns {Function} Returns the new spec function. */ function baseMatches(source) { var matchData = _getMatchData(source); if (matchData.length == 1 && matchData[0][2]) { return _matchesStrictComparable(matchData[0][0], matchData[0][1]); } return function(object) { return object === source || _baseIsMatch(object, source, matchData); }; } /* harmony default export */ const _baseMatches = (baseMatches); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isSymbol.js /** `Object#toString` result references. */ var isSymbol_symbolTag = '[object Symbol]'; /** * Checks if `value` is classified as a `Symbol` primitive or object. * * @static * @memberOf _ * @since 4.0.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a symbol, else `false`. * @example * * _.isSymbol(Symbol.iterator); * // => true * * _.isSymbol('abc'); * // => false */ function isSymbol(value) { return typeof value == 'symbol' || (lodash_es_isObjectLike(value) && _baseGetTag(value) == isSymbol_symbolTag); } /* harmony default export */ const lodash_es_isSymbol = (isSymbol); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_isKey.js /** Used to match property names within property paths. */ var reIsDeepProp = /\.|\[(?:[^[\]]*|(["'])(?:(?!\1)[^\\]|\\.)*?\1)\]/, reIsPlainProp = /^\w*$/; /** * Checks if `value` is a property name and not a property path. * * @private * @param {*} value The value to check. * @param {Object} [object] The object to query keys on. * @returns {boolean} Returns `true` if `value` is a property name, else `false`. */ function isKey(value, object) { if (lodash_es_isArray(value)) { return false; } var type = typeof value; if (type == 'number' || type == 'symbol' || type == 'boolean' || value == null || lodash_es_isSymbol(value)) { return true; } return reIsPlainProp.test(value) || !reIsDeepProp.test(value) || (object != null && value in Object(object)); } /* harmony default export */ const _isKey = (isKey); ;// CONCATENATED MODULE: ./node_modules/lodash-es/memoize.js /** Error message constants. */ var FUNC_ERROR_TEXT = 'Expected a function'; /** * Creates a function that memoizes the result of `func`. If `resolver` is * provided, it determines the cache key for storing the result based on the * arguments provided to the memoized function. By default, the first argument * provided to the memoized function is used as the map cache key. The `func` * is invoked with the `this` binding of the memoized function. * * **Note:** The cache is exposed as the `cache` property on the memoized * function. Its creation may be customized by replacing the `_.memoize.Cache` * constructor with one whose instances implement the * [`Map`](http://ecma-international.org/ecma-262/7.0/#sec-properties-of-the-map-prototype-object) * method interface of `clear`, `delete`, `get`, `has`, and `set`. * * @static * @memberOf _ * @since 0.1.0 * @category Function * @param {Function} func The function to have its output memoized. * @param {Function} [resolver] The function to resolve the cache key. * @returns {Function} Returns the new memoized function. * @example * * var object = { 'a': 1, 'b': 2 }; * var other = { 'c': 3, 'd': 4 }; * * var values = _.memoize(_.values); * values(object); * // => [1, 2] * * values(other); * // => [3, 4] * * object.a = 2; * values(object); * // => [1, 2] * * // Modify the result cache. * values.cache.set(object, ['a', 'b']); * values(object); * // => ['a', 'b'] * * // Replace `_.memoize.Cache`. * _.memoize.Cache = WeakMap; */ function memoize(func, resolver) { if (typeof func != 'function' || (resolver != null && typeof resolver != 'function')) { throw new TypeError(FUNC_ERROR_TEXT); } var memoized = function() { var args = arguments, key = resolver ? resolver.apply(this, args) : args[0], cache = memoized.cache; if (cache.has(key)) { return cache.get(key); } var result = func.apply(this, args); memoized.cache = cache.set(key, result) || cache; return result; }; memoized.cache = new (memoize.Cache || _MapCache); return memoized; } // Expose `MapCache`. memoize.Cache = _MapCache; /* harmony default export */ const lodash_es_memoize = (memoize); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_memoizeCapped.js /** Used as the maximum memoize cache size. */ var MAX_MEMOIZE_SIZE = 500; /** * A specialized version of `_.memoize` which clears the memoized function's * cache when it exceeds `MAX_MEMOIZE_SIZE`. * * @private * @param {Function} func The function to have its output memoized. * @returns {Function} Returns the new memoized function. */ function memoizeCapped(func) { var result = lodash_es_memoize(func, function(key) { if (cache.size === MAX_MEMOIZE_SIZE) { cache.clear(); } return key; }); var cache = result.cache; return result; } /* harmony default export */ const _memoizeCapped = (memoizeCapped); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_stringToPath.js /** Used to match property names within property paths. */ var rePropName = /[^.[\]]+|\[(?:(-?\d+(?:\.\d+)?)|(["'])((?:(?!\2)[^\\]|\\.)*?)\2)\]|(?=(?:\.|\[\])(?:\.|\[\]|$))/g; /** Used to match backslashes in property paths. */ var reEscapeChar = /\\(\\)?/g; /** * Converts `string` to a property path array. * * @private * @param {string} string The string to convert. * @returns {Array} Returns the property path array. */ var stringToPath = _memoizeCapped(function(string) { var result = []; if (string.charCodeAt(0) === 46 /* . */) { result.push(''); } string.replace(rePropName, function(match, number, quote, subString) { result.push(quote ? subString.replace(reEscapeChar, '$1') : (number || match)); }); return result; }); /* harmony default export */ const _stringToPath = (stringToPath); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_arrayMap.js /** * A specialized version of `_.map` for arrays without support for iteratee * shorthands. * * @private * @param {Array} [array] The array to iterate over. * @param {Function} iteratee The function invoked per iteration. * @returns {Array} Returns the new mapped array. */ function arrayMap(array, iteratee) { var index = -1, length = array == null ? 0 : array.length, result = Array(length); while (++index < length) { result[index] = iteratee(array[index], index, array); } return result; } /* harmony default export */ const _arrayMap = (arrayMap); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseToString.js /** Used as references for various `Number` constants. */ var INFINITY = 1 / 0; /** Used to convert symbols to primitives and strings. */ var _baseToString_symbolProto = _Symbol ? _Symbol.prototype : undefined, symbolToString = _baseToString_symbolProto ? _baseToString_symbolProto.toString : undefined; /** * The base implementation of `_.toString` which doesn't convert nullish * values to empty strings. * * @private * @param {*} value The value to process. * @returns {string} Returns the string. */ function baseToString(value) { // Exit early for strings to avoid a performance hit in some environments. if (typeof value == 'string') { return value; } if (lodash_es_isArray(value)) { // Recursively convert values (susceptible to call stack limits). return _arrayMap(value, baseToString) + ''; } if (lodash_es_isSymbol(value)) { return symbolToString ? symbolToString.call(value) : ''; } var result = (value + ''); return (result == '0' && (1 / value) == -INFINITY) ? '-0' : result; } /* harmony default export */ const _baseToString = (baseToString); ;// CONCATENATED MODULE: ./node_modules/lodash-es/toString.js /** * Converts `value` to a string. An empty string is returned for `null` * and `undefined` values. The sign of `-0` is preserved. * * @static * @memberOf _ * @since 4.0.0 * @category Lang * @param {*} value The value to convert. * @returns {string} Returns the converted string. * @example * * _.toString(null); * // => '' * * _.toString(-0); * // => '-0' * * _.toString([1, 2, 3]); * // => '1,2,3' */ function toString_toString(value) { return value == null ? '' : _baseToString(value); } /* harmony default export */ const lodash_es_toString = (toString_toString); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_castPath.js /** * Casts `value` to a path array if it's not one. * * @private * @param {*} value The value to inspect. * @param {Object} [object] The object to query keys on. * @returns {Array} Returns the cast property path array. */ function castPath(value, object) { if (lodash_es_isArray(value)) { return value; } return _isKey(value, object) ? [value] : _stringToPath(lodash_es_toString(value)); } /* harmony default export */ const _castPath = (castPath); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_toKey.js /** Used as references for various `Number` constants. */ var _toKey_INFINITY = 1 / 0; /** * Converts `value` to a string key if it's not a string or symbol. * * @private * @param {*} value The value to inspect. * @returns {string|symbol} Returns the key. */ function toKey(value) { if (typeof value == 'string' || lodash_es_isSymbol(value)) { return value; } var result = (value + ''); return (result == '0' && (1 / value) == -_toKey_INFINITY) ? '-0' : result; } /* harmony default export */ const _toKey = (toKey); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseGet.js /** * The base implementation of `_.get` without support for default values. * * @private * @param {Object} object The object to query. * @param {Array|string} path The path of the property to get. * @returns {*} Returns the resolved value. */ function baseGet(object, path) { path = _castPath(path, object); var index = 0, length = path.length; while (object != null && index < length) { object = object[_toKey(path[index++])]; } return (index && index == length) ? object : undefined; } /* harmony default export */ const _baseGet = (baseGet); ;// CONCATENATED MODULE: ./node_modules/lodash-es/get.js /** * Gets the value at `path` of `object`. If the resolved value is * `undefined`, the `defaultValue` is returned in its place. * * @static * @memberOf _ * @since 3.7.0 * @category Object * @param {Object} object The object to query. * @param {Array|string} path The path of the property to get. * @param {*} [defaultValue] The value returned for `undefined` resolved values. * @returns {*} Returns the resolved value. * @example * * var object = { 'a': [{ 'b': { 'c': 3 } }] }; * * _.get(object, 'a[0].b.c'); * // => 3 * * _.get(object, ['a', '0', 'b', 'c']); * // => 3 * * _.get(object, 'a.b.c', 'default'); * // => 'default' */ function get(object, path, defaultValue) { var result = object == null ? undefined : _baseGet(object, path); return result === undefined ? defaultValue : result; } /* harmony default export */ const lodash_es_get = (get); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseHasIn.js /** * The base implementation of `_.hasIn` without support for deep paths. * * @private * @param {Object} [object] The object to query. * @param {Array|string} key The key to check. * @returns {boolean} Returns `true` if `key` exists, else `false`. */ function baseHasIn(object, key) { return object != null && key in Object(object); } /* harmony default export */ const _baseHasIn = (baseHasIn); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_hasPath.js /** * Checks if `path` exists on `object`. * * @private * @param {Object} object The object to query. * @param {Array|string} path The path to check. * @param {Function} hasFunc The function to check properties. * @returns {boolean} Returns `true` if `path` exists, else `false`. */ function hasPath(object, path, hasFunc) { path = _castPath(path, object); var index = -1, length = path.length, result = false; while (++index < length) { var key = _toKey(path[index]); if (!(result = object != null && hasFunc(object, key))) { break; } object = object[key]; } if (result || ++index != length) { return result; } length = object == null ? 0 : object.length; return !!length && lodash_es_isLength(length) && _isIndex(key, length) && (lodash_es_isArray(object) || lodash_es_isArguments(object)); } /* harmony default export */ const _hasPath = (hasPath); ;// CONCATENATED MODULE: ./node_modules/lodash-es/hasIn.js /** * Checks if `path` is a direct or inherited property of `object`. * * @static * @memberOf _ * @since 4.0.0 * @category Object * @param {Object} object The object to query. * @param {Array|string} path The path to check. * @returns {boolean} Returns `true` if `path` exists, else `false`. * @example * * var object = _.create({ 'a': _.create({ 'b': 2 }) }); * * _.hasIn(object, 'a'); * // => true * * _.hasIn(object, 'a.b'); * // => true * * _.hasIn(object, ['a', 'b']); * // => true * * _.hasIn(object, 'b'); * // => false */ function hasIn(object, path) { return object != null && _hasPath(object, path, _baseHasIn); } /* harmony default export */ const lodash_es_hasIn = (hasIn); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseMatchesProperty.js /** Used to compose bitmasks for value comparisons. */ var _baseMatchesProperty_COMPARE_PARTIAL_FLAG = 1, _baseMatchesProperty_COMPARE_UNORDERED_FLAG = 2; /** * The base implementation of `_.matchesProperty` which doesn't clone `srcValue`. * * @private * @param {string} path The path of the property to get. * @param {*} srcValue The value to match. * @returns {Function} Returns the new spec function. */ function baseMatchesProperty(path, srcValue) { if (_isKey(path) && _isStrictComparable(srcValue)) { return _matchesStrictComparable(_toKey(path), srcValue); } return function(object) { var objValue = lodash_es_get(object, path); return (objValue === undefined && objValue === srcValue) ? lodash_es_hasIn(object, path) : _baseIsEqual(srcValue, objValue, _baseMatchesProperty_COMPARE_PARTIAL_FLAG | _baseMatchesProperty_COMPARE_UNORDERED_FLAG); }; } /* harmony default export */ const _baseMatchesProperty = (baseMatchesProperty); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseProperty.js /** * The base implementation of `_.property` without support for deep paths. * * @private * @param {string} key The key of the property to get. * @returns {Function} Returns the new accessor function. */ function baseProperty(key) { return function(object) { return object == null ? undefined : object[key]; }; } /* harmony default export */ const _baseProperty = (baseProperty); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_basePropertyDeep.js /** * A specialized version of `baseProperty` which supports deep paths. * * @private * @param {Array|string} path The path of the property to get. * @returns {Function} Returns the new accessor function. */ function basePropertyDeep(path) { return function(object) { return _baseGet(object, path); }; } /* harmony default export */ const _basePropertyDeep = (basePropertyDeep); ;// CONCATENATED MODULE: ./node_modules/lodash-es/property.js /** * Creates a function that returns the value at `path` of a given object. * * @static * @memberOf _ * @since 2.4.0 * @category Util * @param {Array|string} path The path of the property to get. * @returns {Function} Returns the new accessor function. * @example * * var objects = [ * { 'a': { 'b': 2 } }, * { 'a': { 'b': 1 } } * ]; * * _.map(objects, _.property('a.b')); * // => [2, 1] * * _.map(_.sortBy(objects, _.property(['a', 'b'])), 'a.b'); * // => [1, 2] */ function property(path) { return _isKey(path) ? _baseProperty(_toKey(path)) : _basePropertyDeep(path); } /* harmony default export */ const lodash_es_property = (property); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseIteratee.js /** * The base implementation of `_.iteratee`. * * @private * @param {*} [value=_.identity] The value to convert to an iteratee. * @returns {Function} Returns the iteratee. */ function baseIteratee(value) { // Don't store the `typeof` result in a variable to avoid a JIT bug in Safari 9. // See https://bugs.webkit.org/show_bug.cgi?id=156034 for more details. if (typeof value == 'function') { return value; } if (value == null) { return lodash_es_identity; } if (typeof value == 'object') { return lodash_es_isArray(value) ? _baseMatchesProperty(value[0], value[1]) : _baseMatches(value); } return lodash_es_property(value); } /* harmony default export */ const _baseIteratee = (baseIteratee); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_createBaseFor.js /** * Creates a base function for methods like `_.forIn` and `_.forOwn`. * * @private * @param {boolean} [fromRight] Specify iterating from right to left. * @returns {Function} Returns the new base function. */ function createBaseFor(fromRight) { return function(object, iteratee, keysFunc) { var index = -1, iterable = Object(object), props = keysFunc(object), length = props.length; while (length--) { var key = props[fromRight ? length : ++index]; if (iteratee(iterable[key], key, iterable) === false) { break; } } return object; }; } /* harmony default export */ const _createBaseFor = (createBaseFor); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseFor.js /** * The base implementation of `baseForOwn` which iterates over `object` * properties returned by `keysFunc` and invokes `iteratee` for each property. * Iteratee functions may exit iteration early by explicitly returning `false`. * * @private * @param {Object} object The object to iterate over. * @param {Function} iteratee The function invoked per iteration. * @param {Function} keysFunc The function to get the keys of `object`. * @returns {Object} Returns `object`. */ var baseFor = _createBaseFor(); /* harmony default export */ const _baseFor = (baseFor); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseForOwn.js /** * The base implementation of `_.forOwn` without support for iteratee shorthands. * * @private * @param {Object} object The object to iterate over. * @param {Function} iteratee The function invoked per iteration. * @returns {Object} Returns `object`. */ function baseForOwn(object, iteratee) { return object && _baseFor(object, iteratee, lodash_es_keys); } /* harmony default export */ const _baseForOwn = (baseForOwn); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_createBaseEach.js /** * Creates a `baseEach` or `baseEachRight` function. * * @private * @param {Function} eachFunc The function to iterate over a collection. * @param {boolean} [fromRight] Specify iterating from right to left. * @returns {Function} Returns the new base function. */ function createBaseEach(eachFunc, fromRight) { return function(collection, iteratee) { if (collection == null) { return collection; } if (!lodash_es_isArrayLike(collection)) { return eachFunc(collection, iteratee); } var length = collection.length, index = fromRight ? length : -1, iterable = Object(collection); while ((fromRight ? index-- : ++index < length)) { if (iteratee(iterable[index], index, iterable) === false) { break; } } return collection; }; } /* harmony default export */ const _createBaseEach = (createBaseEach); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseEach.js /** * The base implementation of `_.forEach` without support for iteratee shorthands. * * @private * @param {Array|Object} collection The collection to iterate over. * @param {Function} iteratee The function invoked per iteration. * @returns {Array|Object} Returns `collection`. */ var baseEach = _createBaseEach(_baseForOwn); /* harmony default export */ const _baseEach = (baseEach); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseSome.js /** * The base implementation of `_.some` without support for iteratee shorthands. * * @private * @param {Array|Object} collection The collection to iterate over. * @param {Function} predicate The function invoked per iteration. * @returns {boolean} Returns `true` if any element passes the predicate check, * else `false`. */ function baseSome(collection, predicate) { var result; _baseEach(collection, function(value, index, collection) { result = predicate(value, index, collection); return !result; }); return !!result; } /* harmony default export */ const _baseSome = (baseSome); ;// CONCATENATED MODULE: ./node_modules/lodash-es/some.js /** * Checks if `predicate` returns truthy for **any** element of `collection`. * Iteration is stopped once `predicate` returns truthy. The predicate is * invoked with three arguments: (value, index|key, collection). * * @static * @memberOf _ * @since 0.1.0 * @category Collection * @param {Array|Object} collection The collection to iterate over. * @param {Function} [predicate=_.identity] The function invoked per iteration. * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`. * @returns {boolean} Returns `true` if any element passes the predicate check, * else `false`. * @example * * _.some([null, 0, 'yes', false], Boolean); * // => true * * var users = [ * { 'user': 'barney', 'active': true }, * { 'user': 'fred', 'active': false } * ]; * * // The `_.matches` iteratee shorthand. * _.some(users, { 'user': 'barney', 'active': false }); * // => false * * // The `_.matchesProperty` iteratee shorthand. * _.some(users, ['active', false]); * // => true * * // The `_.property` iteratee shorthand. * _.some(users, 'active'); * // => true */ function some(collection, predicate, guard) { var func = lodash_es_isArray(collection) ? _arraySome : _baseSome; if (guard && _isIterateeCall(collection, predicate, guard)) { predicate = undefined; } return func(collection, _baseIteratee(predicate, 3)); } /* harmony default export */ const lodash_es_some = (some); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isEmpty.js /** `Object#toString` result references. */ var isEmpty_mapTag = '[object Map]', isEmpty_setTag = '[object Set]'; /** Used for built-in method references. */ var isEmpty_objectProto = Object.prototype; /** Used to check objects for own properties. */ var isEmpty_hasOwnProperty = isEmpty_objectProto.hasOwnProperty; /** * Checks if `value` is an empty object, collection, map, or set. * * Objects are considered empty if they have no own enumerable string keyed * properties. * * Array-like values such as `arguments` objects, arrays, buffers, strings, or * jQuery-like collections are considered empty if they have a `length` of `0`. * Similarly, maps and sets are considered empty if they have a `size` of `0`. * * @static * @memberOf _ * @since 0.1.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is empty, else `false`. * @example * * _.isEmpty(null); * // => true * * _.isEmpty(true); * // => true * * _.isEmpty(1); * // => true * * _.isEmpty([1, 2, 3]); * // => false * * _.isEmpty({ 'a': 1 }); * // => false */ function isEmpty(value) { if (value == null) { return true; } if (lodash_es_isArrayLike(value) && (lodash_es_isArray(value) || typeof value == 'string' || typeof value.splice == 'function' || lodash_es_isBuffer(value) || lodash_es_isTypedArray(value) || lodash_es_isArguments(value))) { return !value.length; } var tag = _getTag(value); if (tag == isEmpty_mapTag || tag == isEmpty_setTag) { return !value.size; } if (_isPrototype(value)) { return !_baseKeys(value).length; } for (var key in value) { if (isEmpty_hasOwnProperty.call(value, key)) { return false; } } return true; } /* harmony default export */ const lodash_es_isEmpty = (isEmpty); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_trimmedEndIndex.js /** Used to match a single whitespace character. */ var reWhitespace = /\s/; /** * Used by `_.trim` and `_.trimEnd` to get the index of the last non-whitespace * character of `string`. * * @private * @param {string} string The string to inspect. * @returns {number} Returns the index of the last non-whitespace character. */ function trimmedEndIndex(string) { var index = string.length; while (index-- && reWhitespace.test(string.charAt(index))) {} return index; } /* harmony default export */ const _trimmedEndIndex = (trimmedEndIndex); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseTrim.js /** Used to match leading whitespace. */ var reTrimStart = /^\s+/; /** * The base implementation of `_.trim`. * * @private * @param {string} string The string to trim. * @returns {string} Returns the trimmed string. */ function baseTrim(string) { return string ? string.slice(0, _trimmedEndIndex(string) + 1).replace(reTrimStart, '') : string; } /* harmony default export */ const _baseTrim = (baseTrim); ;// CONCATENATED MODULE: ./node_modules/lodash-es/toNumber.js /** Used as references for various `Number` constants. */ var NAN = 0 / 0; /** Used to detect bad signed hexadecimal string values. */ var reIsBadHex = /^[-+]0x[0-9a-f]+$/i; /** Used to detect binary string values. */ var reIsBinary = /^0b[01]+$/i; /** Used to detect octal string values. */ var reIsOctal = /^0o[0-7]+$/i; /** Built-in method references without a dependency on `root`. */ var freeParseInt = parseInt; /** * Converts `value` to a number. * * @static * @memberOf _ * @since 4.0.0 * @category Lang * @param {*} value The value to process. * @returns {number} Returns the number. * @example * * _.toNumber(3.2); * // => 3.2 * * _.toNumber(Number.MIN_VALUE); * // => 5e-324 * * _.toNumber(Infinity); * // => Infinity * * _.toNumber('3.2'); * // => 3.2 */ function toNumber(value) { if (typeof value == 'number') { return value; } if (lodash_es_isSymbol(value)) { return NAN; } if (lodash_es_isObject(value)) { var other = typeof value.valueOf == 'function' ? value.valueOf() : value; value = lodash_es_isObject(other) ? (other + '') : other; } if (typeof value != 'string') { return value === 0 ? value : +value; } value = _baseTrim(value); var isBinary = reIsBinary.test(value); return (isBinary || reIsOctal.test(value)) ? freeParseInt(value.slice(2), isBinary ? 2 : 8) : (reIsBadHex.test(value) ? NAN : +value); } /* harmony default export */ const lodash_es_toNumber = (toNumber); ;// CONCATENATED MODULE: ./node_modules/lodash-es/toFinite.js /** Used as references for various `Number` constants. */ var toFinite_INFINITY = 1 / 0, MAX_INTEGER = 1.7976931348623157e+308; /** * Converts `value` to a finite number. * * @static * @memberOf _ * @since 4.12.0 * @category Lang * @param {*} value The value to convert. * @returns {number} Returns the converted number. * @example * * _.toFinite(3.2); * // => 3.2 * * _.toFinite(Number.MIN_VALUE); * // => 5e-324 * * _.toFinite(Infinity); * // => 1.7976931348623157e+308 * * _.toFinite('3.2'); * // => 3.2 */ function toFinite(value) { if (!value) { return value === 0 ? value : 0; } value = lodash_es_toNumber(value); if (value === toFinite_INFINITY || value === -toFinite_INFINITY) { var sign = (value < 0 ? -1 : 1); return sign * MAX_INTEGER; } return value === value ? value : 0; } /* harmony default export */ const lodash_es_toFinite = (toFinite); ;// CONCATENATED MODULE: ./node_modules/lodash-es/toInteger.js /** * Converts `value` to an integer. * * **Note:** This method is loosely based on * [`ToInteger`](http://www.ecma-international.org/ecma-262/7.0/#sec-tointeger). * * @static * @memberOf _ * @since 4.0.0 * @category Lang * @param {*} value The value to convert. * @returns {number} Returns the converted integer. * @example * * _.toInteger(3.2); * // => 3 * * _.toInteger(Number.MIN_VALUE); * // => 0 * * _.toInteger(Infinity); * // => 1.7976931348623157e+308 * * _.toInteger('3.2'); * // => 3 */ function toInteger(value) { var result = lodash_es_toFinite(value), remainder = result % 1; return result === result ? (remainder ? result - remainder : result) : 0; } /* harmony default export */ const lodash_es_toInteger = (toInteger); ;// CONCATENATED MODULE: ./node_modules/lodash-es/before.js /** Error message constants. */ var before_FUNC_ERROR_TEXT = 'Expected a function'; /** * Creates a function that invokes `func`, with the `this` binding and arguments * of the created function, while it's called less than `n` times. Subsequent * calls to the created function return the result of the last `func` invocation. * * @static * @memberOf _ * @since 3.0.0 * @category Function * @param {number} n The number of calls at which `func` is no longer invoked. * @param {Function} func The function to restrict. * @returns {Function} Returns the new restricted function. * @example * * jQuery(element).on('click', _.before(5, addContactToList)); * // => Allows adding up to 4 contacts to the list. */ function before(n, func) { var result; if (typeof func != 'function') { throw new TypeError(before_FUNC_ERROR_TEXT); } n = lodash_es_toInteger(n); return function() { if (--n > 0) { result = func.apply(this, arguments); } if (n <= 1) { func = undefined; } return result; }; } /* harmony default export */ const lodash_es_before = (before); ;// CONCATENATED MODULE: ./node_modules/lodash-es/once.js /** * Creates a function that is restricted to invoking `func` once. Repeat calls * to the function return the value of the first invocation. The `func` is * invoked with the `this` binding and arguments of the created function. * * @static * @memberOf _ * @since 0.1.0 * @category Function * @param {Function} func The function to restrict. * @returns {Function} Returns the new restricted function. * @example * * var initialize = _.once(createApplication); * initialize(); * initialize(); * // => `createApplication` is invoked once */ function once(func) { return lodash_es_before(2, func); } /* harmony default export */ const lodash_es_once = (once); ;// CONCATENATED MODULE: ./node_modules/lodash-es/uniqueId.js /** Used to generate unique IDs. */ var idCounter = 0; /** * Generates a unique ID. If `prefix` is given, the ID is appended to it. * * @static * @since 0.1.0 * @memberOf _ * @category Util * @param {string} [prefix=''] The value to prefix the ID with. * @returns {string} Returns the unique ID. * @example * * _.uniqueId('contact_'); * // => 'contact_104' * * _.uniqueId(); * // => '105' */ function uniqueId(prefix) { var id = ++idCounter; return lodash_es_toString(prefix) + id; } /* harmony default export */ const lodash_es_uniqueId = (uniqueId); ;// CONCATENATED MODULE: ./node_modules/@converse/skeletor/src/events.js // Backbone.js 1.4.0 // (c) 2010-2019 Jeremy Ashkenas and DocumentCloud // Backbone may be freely distributed under the MIT license. // Events // ------ // A module that can be mixed in to *any object* in order to provide it with // a custom event channel. You may bind a callback to an event with `on` or // remove with `off`; `trigger`-ing an event fires all callbacks in // succession. // // let object = {}; // extend(object, Backbone.Events); // object.on('expand', function(){ alert('expanded'); }); // object.trigger('expand'); // const Events = {}; // Regular expression used to split event strings. const eventSplitter = /\s+/; // A private global variable to share between listeners and listenees. let _listening; // Iterates over the standard `event, callback` (as well as the fancy multiple // space-separated events `"change blur", callback` and jQuery-style event // maps `{event: callback}`). const eventsApi = function (iteratee, events, name, callback, opts) { let i = 0, names; if (name && typeof name === 'object') { // Handle event maps. if (callback !== undefined && 'context' in opts && opts.context === undefined) opts.context = callback; for (names = lodash_es_keys(name); i < names.length; i++) { events = eventsApi(iteratee, events, names[i], name[names[i]], opts); } } else if (name && eventSplitter.test(name)) { // Handle space-separated event names by delegating them individually. for (names = name.split(eventSplitter); i < names.length; i++) { events = iteratee(events, names[i], callback, opts); } } else { // Finally, standard events. events = iteratee(events, name, callback, opts); } return events; }; // Bind an event to a `callback` function. Passing `"all"` will bind // the callback to all events fired. Events.on = function (name, callback, context) { this._events = eventsApi(onApi, this._events || {}, name, callback, { context: context, ctx: this, listening: _listening }); if (_listening) { const listeners = this._listeners || (this._listeners = {}); listeners[_listening.id] = _listening; // Allow the listening to use a counter, instead of tracking // callbacks for library interop _listening.interop = false; } return this; }; // Inversion-of-control versions of `on`. Tell *this* object to listen to // an event in another object... keeping track of what it's listening to // for easier unbinding later. Events.listenTo = function (obj, name, callback) { if (!obj) return this; const id = obj._listenId || (obj._listenId = lodash_es_uniqueId('l')); const listeningTo = this._listeningTo || (this._listeningTo = {}); let listening = _listening = listeningTo[id]; // This object is not listening to any other events on `obj` yet. // Setup the necessary references to track the listening callbacks. if (!listening) { this._listenId || (this._listenId = lodash_es_uniqueId('l')); listening = _listening = listeningTo[id] = new Listening(this, obj); } // Bind callbacks on obj. const error = tryCatchOn(obj, name, callback, this); _listening = undefined; if (error) throw error; // If the target obj is not Backbone.Events, track events manually. if (listening.interop) listening.on(name, callback); return this; }; // The reducing API that adds a callback to the `events` object. const onApi = function (events, name, callback, options) { if (callback) { const handlers = events[name] || (events[name] = []); const context = options.context, ctx = options.ctx, listening = options.listening; if (listening) listening.count++; handlers.push({ callback: callback, context: context, ctx: context || ctx, listening: listening }); } return events; }; // An try-catch guarded #on function, to prevent poisoning the global // `_listening` variable. const tryCatchOn = function (obj, name, callback, context) { try { obj.on(name, callback, context); } catch (e) { return e; } }; // Remove one or many callbacks. If `context` is null, removes all // callbacks with that function. If `callback` is null, removes all // callbacks for the event. If `name` is null, removes all bound // callbacks for all events. Events.off = function (name, callback, context) { if (!this._events) return this; this._events = eventsApi(offApi, this._events, name, callback, { context: context, listeners: this._listeners }); return this; }; // Tell this object to stop listening to either specific events ... or // to every object it's currently listening to. Events.stopListening = function (obj, name, callback) { const listeningTo = this._listeningTo; if (!listeningTo) return this; const ids = obj ? [obj._listenId] : lodash_es_keys(listeningTo); for (let i = 0; i < ids.length; i++) { const listening = listeningTo[ids[i]]; // If listening doesn't exist, this object is not currently // listening to obj. Break out early. if (!listening) break; listening.obj.off(name, callback, this); if (listening.interop) listening.off(name, callback); } if (lodash_es_isEmpty(listeningTo)) this._listeningTo = undefined; return this; }; // The reducing API that removes a callback from the `events` object. const offApi = function (events, name, callback, options) { if (!events) return; const context = options.context, listeners = options.listeners; let i = 0, names; // Delete all event listeners and "drop" events. if (!name && !context && !callback) { for (names = lodash_es_keys(listeners); i < names.length; i++) { listeners[names[i]].cleanup(); } return; } names = name ? [name] : lodash_es_keys(events); for (; i < names.length; i++) { name = names[i]; const handlers = events[name]; // Bail out if there are no events stored. if (!handlers) { break; } // Find any remaining events. const remaining = []; for (let j = 0; j < handlers.length; j++) { const handler = handlers[j]; if (callback && callback !== handler.callback && callback !== handler.callback._callback || context && context !== handler.context) { remaining.push(handler); } else { const listening = handler.listening; if (listening) listening.off(name, callback); } } // Replace events if there are any remaining. Otherwise, clean up. if (remaining.length) { events[name] = remaining; } else { delete events[name]; } } return events; }; // Bind an event to only be triggered a single time. After the first time // the callback is invoked, its listener will be removed. If multiple events // are passed in using the space-separated syntax, the handler will fire // once for each event, not once for a combination of all events. Events.once = function (name, callback, context) { // Map the event into a `{event: once}` object. const events = eventsApi(onceMap, {}, name, callback, this.off.bind(this)); if (typeof name === 'string' && (context === null || context === undefined)) callback = undefined; return this.on(events, callback, context); }; // Inversion-of-control versions of `once`. Events.listenToOnce = function (obj, name, callback) { // Map the event into a `{event: once}` object. const events = eventsApi(onceMap, {}, name, callback, this.stopListening.bind(this, obj)); return this.listenTo(obj, events); }; // Reduces the event callbacks into a map of `{event: onceWrapper}`. // `offer` unbinds the `onceWrapper` after it has been called. const onceMap = function (map, name, callback, offer) { if (callback) { const _once = map[name] = lodash_es_once(function () { offer(name, _once); callback.apply(this, arguments); }); _once._callback = callback; } return map; }; // Trigger one or many events, firing all bound callbacks. Callbacks are // passed the same arguments as `trigger` is, apart from the event name // (unless you're listening on `"all"`, which will cause your callback to // receive the true name of the event as the first argument). Events.trigger = function (name) { if (!this._events) return this; const length = Math.max(0, arguments.length - 1); const args = Array(length); for (let i = 0; i < length; i++) args[i] = arguments[i + 1]; eventsApi(triggerApi, this._events, name, undefined, args); return this; }; // Handles triggering the appropriate event callbacks. const triggerApi = function (objEvents, name, callback, args) { if (objEvents) { const events = objEvents[name]; let allEvents = objEvents.all; if (events && allEvents) allEvents = allEvents.slice(); if (events) triggerEvents(events, args); if (allEvents) triggerEvents(allEvents, [name].concat(args)); } return objEvents; }; // A difficult-to-believe, but optimized internal dispatch function for // triggering events. Tries to keep the usual cases speedy (most internal // Backbone events have 3 arguments). const triggerEvents = function (events, args) { let ev, i = -1; const l = events.length, a1 = args[0], a2 = args[1], a3 = args[2]; switch (args.length) { case 0: while (++i < l) (ev = events[i]).callback.call(ev.ctx); return; case 1: while (++i < l) (ev = events[i]).callback.call(ev.ctx, a1); return; case 2: while (++i < l) (ev = events[i]).callback.call(ev.ctx, a1, a2); return; case 3: while (++i < l) (ev = events[i]).callback.call(ev.ctx, a1, a2, a3); return; default: while (++i < l) (ev = events[i]).callback.apply(ev.ctx, args); return; } }; // A listening class that tracks and cleans up memory bindings // when all callbacks have been offed. const Listening = function (listener, obj) { this.id = listener._listenId; this.listener = listener; this.obj = obj; this.interop = true; this.count = 0; this._events = undefined; }; Listening.prototype.on = Events.on; // Offs a callback (or several). // Uses an optimized counter if the listenee uses Backbone.Events. // Otherwise, falls back to manual tracking to support events // library interop. Listening.prototype.off = function (name, callback) { let cleanup; if (this.interop) { this._events = eventsApi(offApi, this._events, name, callback, { context: undefined, listeners: undefined }); cleanup = !this._events; } else { this.count--; cleanup = this.count === 0; } if (cleanup) this.cleanup(); }; // Cleans up memory bindings between the listener and the listenee. Listening.prototype.cleanup = function () { delete this.listener._listeningTo[this.obj._listenId]; if (!this.interop) delete this.obj._listeners[this.id]; }; // Aliases for backwards compatibility. Events.bind = Events.on; Events.unbind = Events.off; ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseAssign.js /** * The base implementation of `_.assign` without support for multiple sources * or `customizer` functions. * * @private * @param {Object} object The destination object. * @param {Object} source The source object. * @returns {Object} Returns `object`. */ function baseAssign(object, source) { return object && _copyObject(source, lodash_es_keys(source), object); } /* harmony default export */ const _baseAssign = (baseAssign); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseCreate.js /** Built-in value references. */ var objectCreate = Object.create; /** * The base implementation of `_.create` without support for assigning * properties to the created object. * * @private * @param {Object} proto The object to inherit from. * @returns {Object} Returns the new object. */ var baseCreate = (function() { function object() {} return function(proto) { if (!lodash_es_isObject(proto)) { return {}; } if (objectCreate) { return objectCreate(proto); } object.prototype = proto; var result = new object; object.prototype = undefined; return result; }; }()); /* harmony default export */ const _baseCreate = (baseCreate); ;// CONCATENATED MODULE: ./node_modules/lodash-es/create.js /** * Creates an object that inherits from the `prototype` object. If a * `properties` object is given, its own enumerable string keyed properties * are assigned to the created object. * * @static * @memberOf _ * @since 2.3.0 * @category Object * @param {Object} prototype The object to inherit from. * @param {Object} [properties] The properties to assign to the object. * @returns {Object} Returns the new object. * @example * * function Shape() { * this.x = 0; * this.y = 0; * } * * function Circle() { * Shape.call(this); * } * * Circle.prototype = _.create(Shape.prototype, { * 'constructor': Circle * }); * * var circle = new Circle; * circle instanceof Circle; * // => true * * circle instanceof Shape; * // => true */ function create(prototype, properties) { var result = _baseCreate(prototype); return properties == null ? result : _baseAssign(result, properties); } /* harmony default export */ const lodash_es_create = (create); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseHas.js /** Used for built-in method references. */ var _baseHas_objectProto = Object.prototype; /** Used to check objects for own properties. */ var _baseHas_hasOwnProperty = _baseHas_objectProto.hasOwnProperty; /** * The base implementation of `_.has` without support for deep paths. * * @private * @param {Object} [object] The object to query. * @param {Array|string} key The key to check. * @returns {boolean} Returns `true` if `key` exists, else `false`. */ function baseHas(object, key) { return object != null && _baseHas_hasOwnProperty.call(object, key); } /* harmony default export */ const _baseHas = (baseHas); ;// CONCATENATED MODULE: ./node_modules/lodash-es/has.js /** * Checks if `path` is a direct property of `object`. * * @static * @since 0.1.0 * @memberOf _ * @category Object * @param {Object} object The object to query. * @param {Array|string} path The path to check. * @returns {boolean} Returns `true` if `path` exists, else `false`. * @example * * var object = { 'a': { 'b': 2 } }; * var other = _.create({ 'a': _.create({ 'b': 2 }) }); * * _.has(object, 'a'); * // => true * * _.has(object, 'a.b'); * // => true * * _.has(object, ['a', 'b']); * // => true * * _.has(other, 'a'); * // => false */ function has(object, path) { return object != null && _hasPath(object, path, _baseHas); } /* harmony default export */ const lodash_es_has = (has); ;// CONCATENATED MODULE: ./node_modules/lodash-es/result.js /** * This method is like `_.get` except that if the resolved value is a * function it's invoked with the `this` binding of its parent object and * its result is returned. * * @static * @since 0.1.0 * @memberOf _ * @category Object * @param {Object} object The object to query. * @param {Array|string} path The path of the property to resolve. * @param {*} [defaultValue] The value returned for `undefined` resolved values. * @returns {*} Returns the resolved value. * @example * * var object = { 'a': [{ 'b': { 'c1': 3, 'c2': _.constant(4) } }] }; * * _.result(object, 'a[0].b.c1'); * // => 3 * * _.result(object, 'a[0].b.c2'); * // => 4 * * _.result(object, 'a[0].b.c3', 'default'); * // => 'default' * * _.result(object, 'a[0].b.c3', _.constant('default')); * // => 'default' */ function result(object, path, defaultValue) { path = _castPath(path, object); var index = -1, length = path.length; // Ensure the loop is entered when path is empty. if (!length) { length = 1; object = undefined; } while (++index < length) { var value = object == null ? undefined : object[_toKey(path[index])]; if (value === undefined) { index = length; value = defaultValue; } object = lodash_es_isFunction(value) ? value.call(object) : value; } return object; } /* harmony default export */ const lodash_es_result = (result); ;// CONCATENATED MODULE: ./node_modules/@converse/skeletor/src/helpers.js // (c) 2010-2019 Jeremy Ashkenas and DocumentCloud /** * Custom error for indicating timeouts * @namespace _converse */ class NotImplementedError extends Error {} // Helpers // ------- // Helper function to correctly set up the prototype chain for subclasses. // Similar to `goog.inherits`, but uses a hash of prototype properties and // class properties to be extended. // function inherits(protoProps, staticProps) { const parent = this; let child; // The constructor function for the new subclass is either defined by you // (the "constructor" property in your `extend` definition), or defaulted // by us to simply call the parent constructor. if (protoProps && lodash_es_has(protoProps, 'constructor')) { child = protoProps.constructor; } else { child = function () { return parent.apply(this, arguments); }; } // Add static properties to the constructor function, if supplied. lodash_es_assignIn(child, parent, staticProps); // Set the prototype chain to inherit from `parent`, without calling // `parent`'s constructor function and add the prototype properties. child.prototype = lodash_es_create(parent.prototype, protoProps); child.prototype.constructor = child; // Set a convenience property in case the parent's prototype is needed // later. child.__super__ = parent.prototype; return child; } function getResolveablePromise() { const wrapper = { isResolved: false, isPending: true, isRejected: false }; const promise = new Promise((resolve, reject) => { wrapper.resolve = resolve; wrapper.reject = reject; }); Object.assign(promise, wrapper); promise.then(function (v) { promise.isResolved = true; promise.isPending = false; promise.isRejected = false; return v; }, function (e) { promise.isResolved = false; promise.isPending = false; promise.isRejected = true; throw e; }); return promise; } // Throw an error when a URL is needed, and none is supplied. function urlError() { throw new Error('A "url" property or function must be specified'); } // Wrap an optional error callback with a fallback error event. function wrapError(model, options) { const error = options.error; options.error = function (resp) { if (error) error.call(options.context, model, resp, options); model.trigger('error', model, resp, options); }; } // Map from CRUD to HTTP for our default `sync` implementation. const methodMap = { create: 'POST', update: 'PUT', patch: 'PATCH', delete: 'DELETE', read: 'GET' }; function getSyncMethod(model) { const store = lodash_es_result(model, 'browserStorage') || lodash_es_result(model.collection, 'browserStorage'); return store ? store.sync() : sync; } // sync // ---- // Override this function to change the manner in which Backbone persists // models to the server. You will be passed the type of request, and the // model in question. By default, makes a RESTful Ajax request // to the model's `url()`. Some possible customizations could be: // // * Use `setTimeout` to batch rapid-fire updates into a single request. // * Send up the models as XML instead of JSON. // * Persist models via WebSockets instead of Ajax. // function sync(method, model) { let options = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : {}; const type = methodMap[method]; // Default JSON-request options. const params = { type: type, dataType: 'json' }; // Ensure that we have a URL. if (!options.url) { params.url = lodash_es_result(model, 'url') || urlError(); } // Ensure that we have the appropriate request data. if (options.data == null && model && (method === 'create' || method === 'update' || method === 'patch')) { params.contentType = 'application/json'; params.data = JSON.stringify(options.attrs || model.toJSON(options)); } // Don't process data on a non-GET request. if (params.type !== 'GET') { params.processData = false; } // Pass along `textStatus` and `errorThrown` from jQuery. const error = options.error; options.error = function (xhr, textStatus, errorThrown) { options.textStatus = textStatus; options.errorThrown = errorThrown; if (error) error.call(options.context, xhr, textStatus, errorThrown); }; // Make the request, allowing the user to override any Ajax options. const xhr = options.xhr = ajax(lodash_es_assignIn(params, options)); model.trigger('request', model, xhr, options); return xhr; } function ajax() { return fetch.apply(this, arguments); } ;// CONCATENATED MODULE: ./node_modules/@converse/skeletor/src/history.js // Backbone.js 1.4.0 // (c) 2010-2019 Jeremy Ashkenas and DocumentCloud // Backbone may be freely distributed under the MIT license. // History // ------- // Handles cross-browser history management, based on either // [pushState](http://diveintohtml5.info/history.html) and real URLs, or // [onhashchange](https://developer.mozilla.org/en-US/docs/DOM/window.onhashchange) // and URL fragments. If the browser supports neither (old IE, natch), // falls back to polling. const history_History = function () { this.handlers = []; this.checkUrl = this.checkUrl.bind(this); // Ensure that `History` can be used outside of the browser. if (typeof window !== 'undefined') { this.location = window.location; this.history = window.history; } }; history_History.extend = inherits; // Cached regex for stripping a leading hash/slash and trailing space. const routeStripper = /^[#\/]|\s+$/g; // Cached regex for stripping leading and trailing slashes. const rootStripper = /^\/+|\/+$/g; // Cached regex for stripping urls of hash. const pathStripper = /#.*$/; // Has the history handling already been started? history_History.started = false; // Set up all inheritable **History** properties and methods. Object.assign(history_History.prototype, Events, { // The default interval to poll for hash changes, if necessary, is // twenty times a second. interval: 50, // Are we at the app root? atRoot: function () { const path = this.location.pathname.replace(/[^\/]$/, '$&/'); return path === this.root && !this.getSearch(); }, // Does the pathname match the root? matchRoot: function () { const path = this.decodeFragment(this.location.pathname); const rootPath = path.slice(0, this.root.length - 1) + '/'; return rootPath === this.root; }, // Unicode characters in `location.pathname` are percent encoded so they're // decoded for comparison. `%25` should not be decoded since it may be part // of an encoded parameter. decodeFragment: function (fragment) { return decodeURI(fragment.replace(/%25/g, '%2525')); }, // In IE6, the hash fragment and search params are incorrect if the // fragment contains `?`. getSearch: function () { const match = this.location.href.replace(/#.*/, '').match(/\?.+/); return match ? match[0] : ''; }, // Gets the true hash value. Cannot use location.hash directly due to bug // in Firefox where location.hash will always be decoded. getHash: function (window) { const match = (window || this).location.href.match(/#(.*)$/); return match ? match[1] : ''; }, // Get the pathname and search params, without the root. getPath: function () { const path = this.decodeFragment(this.location.pathname + this.getSearch()).slice(this.root.length - 1); return path.charAt(0) === '/' ? path.slice(1) : path; }, // Get the cross-browser normalized URL fragment from the path or hash. getFragment: function (fragment) { if (fragment == null) { if (this._usePushState || !this._wantsHashChange) { fragment = this.getPath(); } else { fragment = this.getHash(); } } return fragment.replace(routeStripper, ''); }, // Start the hash change handling, returning `true` if the current URL matches // an existing route, and `false` otherwise. start: function (options) { if (history_History.started) throw new Error('history has already been started'); history_History.started = true; // Figure out the initial configuration. Do we need an iframe? // Is pushState desired ... is it available? this.options = lodash_es_assignIn({ root: '/' }, this.options, options); this.root = this.options.root; this._wantsHashChange = this.options.hashChange !== false; this._hasHashChange = 'onhashchange' in window && (document.documentMode === undefined || document.documentMode > 7); this._useHashChange = this._wantsHashChange && this._hasHashChange; this._wantsPushState = !!this.options.pushState; this._hasPushState = !!(this.history && this.history.pushState); this._usePushState = this._wantsPushState && this._hasPushState; this.fragment = this.getFragment(); // Normalize root to always include a leading and trailing slash. this.root = ('/' + this.root + '/').replace(rootStripper, '/'); // Transition from hashChange to pushState or vice versa if both are // requested. if (this._wantsHashChange && this._wantsPushState) { // If we've started off with a route from a `pushState`-enabled // browser, but we're currently in a browser that doesn't support it... if (!this._hasPushState && !this.atRoot()) { const rootPath = this.root.slice(0, -1) || '/'; this.location.replace(rootPath + '#' + this.getPath()); // Return immediately as browser will do redirect to new url return true; // Or if we've started out with a hash-based route, but we're currently // in a browser where it could be `pushState`-based instead... } else if (this._hasPushState && this.atRoot()) { this.navigate(this.getHash(), { replace: true }); } } // Proxy an iframe to handle location events if the browser doesn't // support the `hashchange` event, HTML5 history, or the user wants // `hashChange` but not `pushState`. if (!this._hasHashChange && this._wantsHashChange && !this._usePushState) { this.iframe = document.createElement('iframe'); this.iframe.src = 'javascript:0'; this.iframe.style.display = 'none'; this.iframe.tabIndex = -1; const body = document.body; // Using `appendChild` will throw on IE < 9 if the document is not ready. const iWindow = body.insertBefore(this.iframe, body.firstChild).contentWindow; iWindow.document.open(); iWindow.document.close(); iWindow.location.hash = '#' + this.fragment; } // Depending on whether we're using pushState or hashes, and whether // 'onhashchange' is supported, determine how we check the URL state. if (this._usePushState) { addEventListener('popstate', this.checkUrl, false); } else if (this._useHashChange && !this.iframe) { addEventListener('hashchange', this.checkUrl, false); } else if (this._wantsHashChange) { this._checkUrlInterval = setInterval(this.checkUrl, this.interval); } if (!this.options.silent) return this.loadUrl(); }, // Disable history, perhaps temporarily. Not useful in a real app, // but possibly useful for unit testing Routers. stop: function () { // Remove window listeners. if (this._usePushState) { removeEventListener('popstate', this.checkUrl, false); } else if (this._useHashChange && !this.iframe) { removeEventListener('hashchange', this.checkUrl, false); } // Clean up the iframe if necessary. if (this.iframe) { document.body.removeChild(this.iframe); this.iframe = null; } // Some environments will throw when clearing an undefined interval. if (this._checkUrlInterval) clearInterval(this._checkUrlInterval); history_History.started = false; }, // Add a route to be tested when the fragment changes. Routes added later // may override previous routes. route: function (route, callback) { this.handlers.unshift({ route: route, callback: callback }); }, // Checks the current URL to see if it has changed, and if it has, // calls `loadUrl`, normalizing across the hidden iframe. checkUrl: function (e) { let current = this.getFragment(); // If the user pressed the back button, the iframe's hash will have // changed and we should use that for comparison. if (current === this.fragment && this.iframe) { current = this.getHash(this.iframe.contentWindow); } if (current === this.fragment) return false; if (this.iframe) this.navigate(current); this.loadUrl(); }, // Attempt to load the current URL fragment. If a route succeeds with a // match, returns `true`. If no defined routes matches the fragment, // returns `false`. loadUrl: function (fragment) { // If the root doesn't match, no routes can match either. if (!this.matchRoot()) return false; fragment = this.fragment = this.getFragment(fragment); return lodash_es_some(this.handlers, function (handler) { if (handler.route.test(fragment)) { handler.callback(fragment); return true; } }); }, // Save a fragment into the hash history, or replace the URL state if the // 'replace' option is passed. You are responsible for properly URL-encoding // the fragment in advance. // // The options object can contain `trigger: true` if you wish to have the // route callback be fired (not usually desirable), or `replace: true`, if // you wish to modify the current URL without adding an entry to the history. navigate: function (fragment, options) { if (!history_History.started) return false; if (!options || options === true) options = { trigger: !!options }; // Normalize the fragment. fragment = this.getFragment(fragment || ''); // Don't include a trailing slash on the root. let rootPath = this.root; if (fragment === '' || fragment.charAt(0) === '?') { rootPath = rootPath.slice(0, -1) || '/'; } const url = rootPath + fragment; // Strip the fragment of the query and hash for matching. fragment = fragment.replace(pathStripper, ''); // Decode for matching. const decodedFragment = this.decodeFragment(fragment); if (this.fragment === decodedFragment) return; this.fragment = decodedFragment; // If pushState is available, we use it to set the fragment as a real URL. if (this._usePushState) { this.history[options.replace ? 'replaceState' : 'pushState']({}, document.title, url); // If hash changes haven't been explicitly disabled, update the hash // fragment to store history. } else if (this._wantsHashChange) { this._updateHash(this.location, fragment, options.replace); if (this.iframe && fragment !== this.getHash(this.iframe.contentWindow)) { const iWindow = this.iframe.contentWindow; // Opening and closing the iframe tricks IE7 and earlier to push a // history entry on hash-tag change. When replace is true, we don't // want this. if (!options.replace) { iWindow.document.open(); iWindow.document.close(); } this._updateHash(iWindow.location, fragment, options.replace); } // If you've told us that you explicitly don't want fallback hashchange- // based history, then `navigate` becomes a page refresh. } else { return this.location.assign(url); } if (options.trigger) return this.loadUrl(fragment); }, // Update the hash location, either replacing the current entry, or adding // a new one to the browser history. _updateHash: function (location, fragment, replace) { if (replace) { const href = location.href.replace(/(javascript:|#).*$/, ''); location.replace(href + '#' + fragment); } else { // Some browsers require that `hash` contains a leading #. location.hash = '#' + fragment; } } }); /* harmony default export */ const src_history = (history_History); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseIsRegExp.js /** `Object#toString` result references. */ var _baseIsRegExp_regexpTag = '[object RegExp]'; /** * The base implementation of `_.isRegExp` without Node.js optimizations. * * @private * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a regexp, else `false`. */ function baseIsRegExp(value) { return lodash_es_isObjectLike(value) && _baseGetTag(value) == _baseIsRegExp_regexpTag; } /* harmony default export */ const _baseIsRegExp = (baseIsRegExp); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isRegExp.js /* Node.js helper references. */ var nodeIsRegExp = _nodeUtil && _nodeUtil.isRegExp; /** * Checks if `value` is classified as a `RegExp` object. * * @static * @memberOf _ * @since 0.1.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a regexp, else `false`. * @example * * _.isRegExp(/abc/); * // => true * * _.isRegExp('/abc/'); * // => false */ var isRegExp = nodeIsRegExp ? _baseUnary(nodeIsRegExp) : _baseIsRegExp; /* harmony default export */ const lodash_es_isRegExp = (isRegExp); ;// CONCATENATED MODULE: ./node_modules/@converse/skeletor/src/router.js // Backbone.js 1.4.0 // (c) 2010-2019 Jeremy Ashkenas and DocumentCloud // Backbone may be freely distributed under the MIT license. // Router // ------ // Routers map faux-URLs to actions, and fire events when routes are // matched. Creating a new one sets its `routes` hash, if not set statically. const Router = function () { let options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; this.history = options.history || new src_history(); this.preinitialize.apply(this, arguments); if (options.routes) this.routes = options.routes; this._bindRoutes(); this.initialize.apply(this, arguments); }; Router.extend = inherits; // Cached regular expressions for matching named param parts and splatted // parts of route strings. const optionalParam = /\((.*?)\)/g; const namedParam = /(\(\?)?:\w+/g; const splatParam = /\*\w+/g; const escapeRegExp = /[\-{}\[\]+?.,\\\^$|#\s]/g; // Set up all inheritable **Router** properties and methods. Object.assign(Router.prototype, Events, { // preinitialize is an empty function by default. You can override it with a function // or object. preinitialize will run before any instantiation logic is run in the Router. preinitialize: function () {}, // Initialize is an empty function by default. Override it with your own // initialization logic. initialize: function () {}, // Manually bind a single named route to a callback. For example: // // this.route('search/:query/p:num', 'search', function(query, num) { // ... // }); // route: function (route, name, callback) { if (!lodash_es_isRegExp(route)) route = this._routeToRegExp(route); if (lodash_es_isFunction(name)) { callback = name; name = ''; } if (!callback) callback = this[name]; this.history.route(route, fragment => { const args = this._extractParameters(route, fragment); if (this.execute(callback, args, name) !== false) { this.trigger.apply(this, ['route:' + name].concat(args)); this.trigger('route', name, args); this.history.trigger('route', this, name, args); } }); return this; }, // Execute a route handler with the provided parameters. This is an // excellent place to do pre-route setup or post-route cleanup. execute: function (callback, args, name) { if (callback) callback.apply(this, args); }, // Simple proxy to `history` to save a fragment into the history. navigate: function (fragment, options) { this.history.navigate(fragment, options); return this; }, // Bind all defined routes to `history`. We have to reverse the // order of the routes here to support behavior where the most general // routes can be defined at the bottom of the route map. _bindRoutes: function () { if (!this.routes) return; this.routes = lodash_es_result(this, 'routes'); let route; const routes = lodash_es_keys(this.routes); while ((route = routes.pop()) != null) { this.route(route, this.routes[route]); } }, // Convert a route string into a regular expression, suitable for matching // against the current location hash. _routeToRegExp: function (route) { route = route.replace(escapeRegExp, '\\$&').replace(optionalParam, '(?:$1)?').replace(namedParam, function (match, optional) { return optional ? match : '([^/?]+)'; }).replace(splatParam, '([^?]*?)'); return new RegExp('^' + route + '(?:\\?([\\s\\S]*))?$'); }, // Given a route, and a URL fragment that it matches, return the array of // extracted decoded parameters. Empty or unmatched parameters will be // treated as `null` to normalize cross-browser behavior. _extractParameters: function (route, fragment) { const params = route.exec(fragment).slice(1); return params.map(function (param, i) { // Don't decode the search params. if (i === params.length - 1) return param || null; return param ? decodeURIComponent(param) : null; }); } }); ;// CONCATENATED MODULE: ./src/headless/shared/errors.js /** * Custom error for indicating timeouts * @namespace _converse */ class TimeoutError extends Error {} // EXTERNAL MODULE: ./node_modules/localforage-driver-memory/_bundle/umd.js var umd = __webpack_require__(3245); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_arrayEach.js /** * A specialized version of `_.forEach` for arrays without support for * iteratee shorthands. * * @private * @param {Array} [array] The array to iterate over. * @param {Function} iteratee The function invoked per iteration. * @returns {Array} Returns `array`. */ function arrayEach(array, iteratee) { var index = -1, length = array == null ? 0 : array.length; while (++index < length) { if (iteratee(array[index], index, array) === false) { break; } } return array; } /* harmony default export */ const _arrayEach = (arrayEach); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseAssignIn.js /** * The base implementation of `_.assignIn` without support for multiple sources * or `customizer` functions. * * @private * @param {Object} object The destination object. * @param {Object} source The source object. * @returns {Object} Returns `object`. */ function baseAssignIn(object, source) { return object && _copyObject(source, lodash_es_keysIn(source), object); } /* harmony default export */ const _baseAssignIn = (baseAssignIn); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_cloneBuffer.js /** Detect free variable `exports`. */ var _cloneBuffer_freeExports = typeof exports == 'object' && exports && !exports.nodeType && exports; /** Detect free variable `module`. */ var _cloneBuffer_freeModule = _cloneBuffer_freeExports && typeof module == 'object' && module && !module.nodeType && module; /** Detect the popular CommonJS extension `module.exports`. */ var _cloneBuffer_moduleExports = _cloneBuffer_freeModule && _cloneBuffer_freeModule.exports === _cloneBuffer_freeExports; /** Built-in value references. */ var _cloneBuffer_Buffer = _cloneBuffer_moduleExports ? _root.Buffer : undefined, allocUnsafe = _cloneBuffer_Buffer ? _cloneBuffer_Buffer.allocUnsafe : undefined; /** * Creates a clone of `buffer`. * * @private * @param {Buffer} buffer The buffer to clone. * @param {boolean} [isDeep] Specify a deep clone. * @returns {Buffer} Returns the cloned buffer. */ function cloneBuffer(buffer, isDeep) { if (isDeep) { return buffer.slice(); } var length = buffer.length, result = allocUnsafe ? allocUnsafe(length) : new buffer.constructor(length); buffer.copy(result); return result; } /* harmony default export */ const _cloneBuffer = (cloneBuffer); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_copyArray.js /** * Copies the values of `source` to `array`. * * @private * @param {Array} source The array to copy values from. * @param {Array} [array=[]] The array to copy values to. * @returns {Array} Returns `array`. */ function copyArray(source, array) { var index = -1, length = source.length; array || (array = Array(length)); while (++index < length) { array[index] = source[index]; } return array; } /* harmony default export */ const _copyArray = (copyArray); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_copySymbols.js /** * Copies own symbols of `source` to `object`. * * @private * @param {Object} source The object to copy symbols from. * @param {Object} [object={}] The object to copy symbols to. * @returns {Object} Returns `object`. */ function copySymbols(source, object) { return _copyObject(source, _getSymbols(source), object); } /* harmony default export */ const _copySymbols = (copySymbols); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_getSymbolsIn.js /* Built-in method references for those with the same name as other `lodash` methods. */ var _getSymbolsIn_nativeGetSymbols = Object.getOwnPropertySymbols; /** * Creates an array of the own and inherited enumerable symbols of `object`. * * @private * @param {Object} object The object to query. * @returns {Array} Returns the array of symbols. */ var getSymbolsIn = !_getSymbolsIn_nativeGetSymbols ? lodash_es_stubArray : function(object) { var result = []; while (object) { _arrayPush(result, _getSymbols(object)); object = _getPrototype(object); } return result; }; /* harmony default export */ const _getSymbolsIn = (getSymbolsIn); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_copySymbolsIn.js /** * Copies own and inherited symbols of `source` to `object`. * * @private * @param {Object} source The object to copy symbols from. * @param {Object} [object={}] The object to copy symbols to. * @returns {Object} Returns `object`. */ function copySymbolsIn(source, object) { return _copyObject(source, _getSymbolsIn(source), object); } /* harmony default export */ const _copySymbolsIn = (copySymbolsIn); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_getAllKeysIn.js /** * Creates an array of own and inherited enumerable property names and * symbols of `object`. * * @private * @param {Object} object The object to query. * @returns {Array} Returns the array of property names and symbols. */ function getAllKeysIn(object) { return _baseGetAllKeys(object, lodash_es_keysIn, _getSymbolsIn); } /* harmony default export */ const _getAllKeysIn = (getAllKeysIn); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_initCloneArray.js /** Used for built-in method references. */ var _initCloneArray_objectProto = Object.prototype; /** Used to check objects for own properties. */ var _initCloneArray_hasOwnProperty = _initCloneArray_objectProto.hasOwnProperty; /** * Initializes an array clone. * * @private * @param {Array} array The array to clone. * @returns {Array} Returns the initialized clone. */ function initCloneArray(array) { var length = array.length, result = new array.constructor(length); // Add properties assigned by `RegExp#exec`. if (length && typeof array[0] == 'string' && _initCloneArray_hasOwnProperty.call(array, 'index')) { result.index = array.index; result.input = array.input; } return result; } /* harmony default export */ const _initCloneArray = (initCloneArray); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_cloneArrayBuffer.js /** * Creates a clone of `arrayBuffer`. * * @private * @param {ArrayBuffer} arrayBuffer The array buffer to clone. * @returns {ArrayBuffer} Returns the cloned array buffer. */ function cloneArrayBuffer(arrayBuffer) { var result = new arrayBuffer.constructor(arrayBuffer.byteLength); new _Uint8Array(result).set(new _Uint8Array(arrayBuffer)); return result; } /* harmony default export */ const _cloneArrayBuffer = (cloneArrayBuffer); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_cloneDataView.js /** * Creates a clone of `dataView`. * * @private * @param {Object} dataView The data view to clone. * @param {boolean} [isDeep] Specify a deep clone. * @returns {Object} Returns the cloned data view. */ function cloneDataView(dataView, isDeep) { var buffer = isDeep ? _cloneArrayBuffer(dataView.buffer) : dataView.buffer; return new dataView.constructor(buffer, dataView.byteOffset, dataView.byteLength); } /* harmony default export */ const _cloneDataView = (cloneDataView); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_cloneRegExp.js /** Used to match `RegExp` flags from their coerced string values. */ var reFlags = /\w*$/; /** * Creates a clone of `regexp`. * * @private * @param {Object} regexp The regexp to clone. * @returns {Object} Returns the cloned regexp. */ function cloneRegExp(regexp) { var result = new regexp.constructor(regexp.source, reFlags.exec(regexp)); result.lastIndex = regexp.lastIndex; return result; } /* harmony default export */ const _cloneRegExp = (cloneRegExp); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_cloneSymbol.js /** Used to convert symbols to primitives and strings. */ var _cloneSymbol_symbolProto = _Symbol ? _Symbol.prototype : undefined, _cloneSymbol_symbolValueOf = _cloneSymbol_symbolProto ? _cloneSymbol_symbolProto.valueOf : undefined; /** * Creates a clone of the `symbol` object. * * @private * @param {Object} symbol The symbol object to clone. * @returns {Object} Returns the cloned symbol object. */ function cloneSymbol(symbol) { return _cloneSymbol_symbolValueOf ? Object(_cloneSymbol_symbolValueOf.call(symbol)) : {}; } /* harmony default export */ const _cloneSymbol = (cloneSymbol); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_cloneTypedArray.js /** * Creates a clone of `typedArray`. * * @private * @param {Object} typedArray The typed array to clone. * @param {boolean} [isDeep] Specify a deep clone. * @returns {Object} Returns the cloned typed array. */ function cloneTypedArray(typedArray, isDeep) { var buffer = isDeep ? _cloneArrayBuffer(typedArray.buffer) : typedArray.buffer; return new typedArray.constructor(buffer, typedArray.byteOffset, typedArray.length); } /* harmony default export */ const _cloneTypedArray = (cloneTypedArray); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_initCloneByTag.js /** `Object#toString` result references. */ var _initCloneByTag_boolTag = '[object Boolean]', _initCloneByTag_dateTag = '[object Date]', _initCloneByTag_mapTag = '[object Map]', _initCloneByTag_numberTag = '[object Number]', _initCloneByTag_regexpTag = '[object RegExp]', _initCloneByTag_setTag = '[object Set]', _initCloneByTag_stringTag = '[object String]', _initCloneByTag_symbolTag = '[object Symbol]'; var _initCloneByTag_arrayBufferTag = '[object ArrayBuffer]', _initCloneByTag_dataViewTag = '[object DataView]', _initCloneByTag_float32Tag = '[object Float32Array]', _initCloneByTag_float64Tag = '[object Float64Array]', _initCloneByTag_int8Tag = '[object Int8Array]', _initCloneByTag_int16Tag = '[object Int16Array]', _initCloneByTag_int32Tag = '[object Int32Array]', _initCloneByTag_uint8Tag = '[object Uint8Array]', _initCloneByTag_uint8ClampedTag = '[object Uint8ClampedArray]', _initCloneByTag_uint16Tag = '[object Uint16Array]', _initCloneByTag_uint32Tag = '[object Uint32Array]'; /** * Initializes an object clone based on its `toStringTag`. * * **Note:** This function only supports cloning values with tags of * `Boolean`, `Date`, `Error`, `Map`, `Number`, `RegExp`, `Set`, or `String`. * * @private * @param {Object} object The object to clone. * @param {string} tag The `toStringTag` of the object to clone. * @param {boolean} [isDeep] Specify a deep clone. * @returns {Object} Returns the initialized clone. */ function initCloneByTag(object, tag, isDeep) { var Ctor = object.constructor; switch (tag) { case _initCloneByTag_arrayBufferTag: return _cloneArrayBuffer(object); case _initCloneByTag_boolTag: case _initCloneByTag_dateTag: return new Ctor(+object); case _initCloneByTag_dataViewTag: return _cloneDataView(object, isDeep); case _initCloneByTag_float32Tag: case _initCloneByTag_float64Tag: case _initCloneByTag_int8Tag: case _initCloneByTag_int16Tag: case _initCloneByTag_int32Tag: case _initCloneByTag_uint8Tag: case _initCloneByTag_uint8ClampedTag: case _initCloneByTag_uint16Tag: case _initCloneByTag_uint32Tag: return _cloneTypedArray(object, isDeep); case _initCloneByTag_mapTag: return new Ctor; case _initCloneByTag_numberTag: case _initCloneByTag_stringTag: return new Ctor(object); case _initCloneByTag_regexpTag: return _cloneRegExp(object); case _initCloneByTag_setTag: return new Ctor; case _initCloneByTag_symbolTag: return _cloneSymbol(object); } } /* harmony default export */ const _initCloneByTag = (initCloneByTag); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_initCloneObject.js /** * Initializes an object clone. * * @private * @param {Object} object The object to clone. * @returns {Object} Returns the initialized clone. */ function initCloneObject(object) { return (typeof object.constructor == 'function' && !_isPrototype(object)) ? _baseCreate(_getPrototype(object)) : {}; } /* harmony default export */ const _initCloneObject = (initCloneObject); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseIsMap.js /** `Object#toString` result references. */ var _baseIsMap_mapTag = '[object Map]'; /** * The base implementation of `_.isMap` without Node.js optimizations. * * @private * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a map, else `false`. */ function baseIsMap(value) { return lodash_es_isObjectLike(value) && _getTag(value) == _baseIsMap_mapTag; } /* harmony default export */ const _baseIsMap = (baseIsMap); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isMap.js /* Node.js helper references. */ var nodeIsMap = _nodeUtil && _nodeUtil.isMap; /** * Checks if `value` is classified as a `Map` object. * * @static * @memberOf _ * @since 4.3.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a map, else `false`. * @example * * _.isMap(new Map); * // => true * * _.isMap(new WeakMap); * // => false */ var isMap = nodeIsMap ? _baseUnary(nodeIsMap) : _baseIsMap; /* harmony default export */ const lodash_es_isMap = (isMap); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseIsSet.js /** `Object#toString` result references. */ var _baseIsSet_setTag = '[object Set]'; /** * The base implementation of `_.isSet` without Node.js optimizations. * * @private * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a set, else `false`. */ function baseIsSet(value) { return lodash_es_isObjectLike(value) && _getTag(value) == _baseIsSet_setTag; } /* harmony default export */ const _baseIsSet = (baseIsSet); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isSet.js /* Node.js helper references. */ var nodeIsSet = _nodeUtil && _nodeUtil.isSet; /** * Checks if `value` is classified as a `Set` object. * * @static * @memberOf _ * @since 4.3.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a set, else `false`. * @example * * _.isSet(new Set); * // => true * * _.isSet(new WeakSet); * // => false */ var isSet = nodeIsSet ? _baseUnary(nodeIsSet) : _baseIsSet; /* harmony default export */ const lodash_es_isSet = (isSet); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseClone.js /** Used to compose bitmasks for cloning. */ var CLONE_DEEP_FLAG = 1, CLONE_FLAT_FLAG = 2, CLONE_SYMBOLS_FLAG = 4; /** `Object#toString` result references. */ var _baseClone_argsTag = '[object Arguments]', _baseClone_arrayTag = '[object Array]', _baseClone_boolTag = '[object Boolean]', _baseClone_dateTag = '[object Date]', _baseClone_errorTag = '[object Error]', _baseClone_funcTag = '[object Function]', _baseClone_genTag = '[object GeneratorFunction]', _baseClone_mapTag = '[object Map]', _baseClone_numberTag = '[object Number]', _baseClone_objectTag = '[object Object]', _baseClone_regexpTag = '[object RegExp]', _baseClone_setTag = '[object Set]', _baseClone_stringTag = '[object String]', _baseClone_symbolTag = '[object Symbol]', _baseClone_weakMapTag = '[object WeakMap]'; var _baseClone_arrayBufferTag = '[object ArrayBuffer]', _baseClone_dataViewTag = '[object DataView]', _baseClone_float32Tag = '[object Float32Array]', _baseClone_float64Tag = '[object Float64Array]', _baseClone_int8Tag = '[object Int8Array]', _baseClone_int16Tag = '[object Int16Array]', _baseClone_int32Tag = '[object Int32Array]', _baseClone_uint8Tag = '[object Uint8Array]', _baseClone_uint8ClampedTag = '[object Uint8ClampedArray]', _baseClone_uint16Tag = '[object Uint16Array]', _baseClone_uint32Tag = '[object Uint32Array]'; /** Used to identify `toStringTag` values supported by `_.clone`. */ var cloneableTags = {}; cloneableTags[_baseClone_argsTag] = cloneableTags[_baseClone_arrayTag] = cloneableTags[_baseClone_arrayBufferTag] = cloneableTags[_baseClone_dataViewTag] = cloneableTags[_baseClone_boolTag] = cloneableTags[_baseClone_dateTag] = cloneableTags[_baseClone_float32Tag] = cloneableTags[_baseClone_float64Tag] = cloneableTags[_baseClone_int8Tag] = cloneableTags[_baseClone_int16Tag] = cloneableTags[_baseClone_int32Tag] = cloneableTags[_baseClone_mapTag] = cloneableTags[_baseClone_numberTag] = cloneableTags[_baseClone_objectTag] = cloneableTags[_baseClone_regexpTag] = cloneableTags[_baseClone_setTag] = cloneableTags[_baseClone_stringTag] = cloneableTags[_baseClone_symbolTag] = cloneableTags[_baseClone_uint8Tag] = cloneableTags[_baseClone_uint8ClampedTag] = cloneableTags[_baseClone_uint16Tag] = cloneableTags[_baseClone_uint32Tag] = true; cloneableTags[_baseClone_errorTag] = cloneableTags[_baseClone_funcTag] = cloneableTags[_baseClone_weakMapTag] = false; /** * The base implementation of `_.clone` and `_.cloneDeep` which tracks * traversed objects. * * @private * @param {*} value The value to clone. * @param {boolean} bitmask The bitmask flags. * 1 - Deep clone * 2 - Flatten inherited properties * 4 - Clone symbols * @param {Function} [customizer] The function to customize cloning. * @param {string} [key] The key of `value`. * @param {Object} [object] The parent object of `value`. * @param {Object} [stack] Tracks traversed objects and their clone counterparts. * @returns {*} Returns the cloned value. */ function baseClone(value, bitmask, customizer, key, object, stack) { var result, isDeep = bitmask & CLONE_DEEP_FLAG, isFlat = bitmask & CLONE_FLAT_FLAG, isFull = bitmask & CLONE_SYMBOLS_FLAG; if (customizer) { result = object ? customizer(value, key, object, stack) : customizer(value); } if (result !== undefined) { return result; } if (!lodash_es_isObject(value)) { return value; } var isArr = lodash_es_isArray(value); if (isArr) { result = _initCloneArray(value); if (!isDeep) { return _copyArray(value, result); } } else { var tag = _getTag(value), isFunc = tag == _baseClone_funcTag || tag == _baseClone_genTag; if (lodash_es_isBuffer(value)) { return _cloneBuffer(value, isDeep); } if (tag == _baseClone_objectTag || tag == _baseClone_argsTag || (isFunc && !object)) { result = (isFlat || isFunc) ? {} : _initCloneObject(value); if (!isDeep) { return isFlat ? _copySymbolsIn(value, _baseAssignIn(result, value)) : _copySymbols(value, _baseAssign(result, value)); } } else { if (!cloneableTags[tag]) { return object ? value : {}; } result = _initCloneByTag(value, tag, isDeep); } } // Check for circular references and return its corresponding clone. stack || (stack = new _Stack); var stacked = stack.get(value); if (stacked) { return stacked; } stack.set(value, result); if (lodash_es_isSet(value)) { value.forEach(function(subValue) { result.add(baseClone(subValue, bitmask, customizer, subValue, value, stack)); }); } else if (lodash_es_isMap(value)) { value.forEach(function(subValue, key) { result.set(key, baseClone(subValue, bitmask, customizer, key, value, stack)); }); } var keysFunc = isFull ? (isFlat ? _getAllKeysIn : _getAllKeys) : (isFlat ? lodash_es_keysIn : lodash_es_keys); var props = isArr ? undefined : keysFunc(value); _arrayEach(props || value, function(subValue, key) { if (props) { key = subValue; subValue = value[key]; } // Recursively populate clone (susceptible to call stack limits). _assignValue(result, key, baseClone(subValue, bitmask, customizer, key, value, stack)); }); return result; } /* harmony default export */ const _baseClone = (baseClone); ;// CONCATENATED MODULE: ./node_modules/lodash-es/cloneDeep.js /** Used to compose bitmasks for cloning. */ var cloneDeep_CLONE_DEEP_FLAG = 1, cloneDeep_CLONE_SYMBOLS_FLAG = 4; /** * This method is like `_.clone` except that it recursively clones `value`. * * @static * @memberOf _ * @since 1.0.0 * @category Lang * @param {*} value The value to recursively clone. * @returns {*} Returns the deep cloned value. * @see _.clone * @example * * var objects = [{ 'a': 1 }, { 'b': 2 }]; * * var deep = _.cloneDeep(objects); * console.log(deep[0] === objects[0]); * // => false */ function cloneDeep(value) { return _baseClone(value, cloneDeep_CLONE_DEEP_FLAG | cloneDeep_CLONE_SYMBOLS_FLAG); } /* harmony default export */ const lodash_es_cloneDeep = (cloneDeep); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isString.js /** `Object#toString` result references. */ var isString_stringTag = '[object String]'; /** * Checks if `value` is classified as a `String` primitive or object. * * @static * @since 0.1.0 * @memberOf _ * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a string, else `false`. * @example * * _.isString('abc'); * // => true * * _.isString(1); * // => false */ function isString(value) { return typeof value == 'string' || (!lodash_es_isArray(value) && lodash_es_isObjectLike(value) && _baseGetTag(value) == isString_stringTag); } /* harmony default export */ const lodash_es_isString = (isString); ;// CONCATENATED MODULE: ./node_modules/localforage/src/utils/idb.js function getIDB() { /* global indexedDB,webkitIndexedDB,mozIndexedDB,OIndexedDB,msIndexedDB */ try { if (typeof indexedDB !== 'undefined') { return indexedDB; } if (typeof webkitIndexedDB !== 'undefined') { return webkitIndexedDB; } if (typeof mozIndexedDB !== 'undefined') { return mozIndexedDB; } if (typeof OIndexedDB !== 'undefined') { return OIndexedDB; } if (typeof msIndexedDB !== 'undefined') { return msIndexedDB; } } catch (e) { return; } } var idb = getIDB(); /* harmony default export */ const utils_idb = (idb); ;// CONCATENATED MODULE: ./node_modules/localforage/src/utils/isIndexedDBValid.js function isIndexedDBValid() { try { // Initialize IndexedDB; fall back to vendor-prefixed versions // if needed. if (!utils_idb || !utils_idb.open) { return false; } // We mimic PouchDB here; // // We test for openDatabase because IE Mobile identifies itself // as Safari. Oh the lulz... var isSafari = typeof openDatabase !== 'undefined' && /(Safari|iPhone|iPad|iPod)/.test(navigator.userAgent) && !/Chrome/.test(navigator.userAgent) && !/BlackBerry/.test(navigator.platform); var hasFetch = typeof fetch === 'function' && fetch.toString().indexOf('[native code') !== -1; // Safari <10.1 does not meet our requirements for IDB support // (see: https://github.com/pouchdb/pouchdb/issues/5572). // Safari 10.1 shipped with fetch, we can use that to detect it. // Note: this creates issues with `window.fetch` polyfills and // overrides; see: // https://github.com/localForage/localForage/issues/856 return (!isSafari || hasFetch) && typeof indexedDB !== 'undefined' && // some outdated implementations of IDB that appear on Samsung // and HTC Android devices <4.4 are missing IDBKeyRange // See: https://github.com/mozilla/localForage/issues/128 // See: https://github.com/mozilla/localForage/issues/272 typeof IDBKeyRange !== 'undefined'; } catch (e) { return false; } } /* harmony default export */ const utils_isIndexedDBValid = (isIndexedDBValid); ;// CONCATENATED MODULE: ./node_modules/localforage/src/utils/createBlob.js // Abstracts constructing a Blob object, so it also works in older // browsers that don't support the native Blob constructor. (i.e. // old QtWebKit versions, at least). // Abstracts constructing a Blob object, so it also works in older // browsers that don't support the native Blob constructor. (i.e. // old QtWebKit versions, at least). function createBlob(parts, properties) { /* global BlobBuilder,MSBlobBuilder,MozBlobBuilder,WebKitBlobBuilder */ parts = parts || []; properties = properties || {}; try { return new Blob(parts, properties); } catch (e) { if (e.name !== 'TypeError') { throw e; } var Builder = typeof BlobBuilder !== 'undefined' ? BlobBuilder : typeof MSBlobBuilder !== 'undefined' ? MSBlobBuilder : typeof MozBlobBuilder !== 'undefined' ? MozBlobBuilder : WebKitBlobBuilder; var builder = new Builder(); for (var i = 0; i < parts.length; i += 1) { builder.append(parts[i]); } return builder.getBlob(properties.type); } } /* harmony default export */ const utils_createBlob = (createBlob); ;// CONCATENATED MODULE: ./node_modules/localforage/src/utils/promise.js // This is CommonJS because lie is an external dependency, so Rollup // can just ignore it. if (typeof Promise === 'undefined') { // In the "nopromises" build this will just throw if you don't have // a global promise object, but it would throw anyway later. __webpack_require__(9236); } /* harmony default export */ const utils_promise = (Promise); ;// CONCATENATED MODULE: ./node_modules/localforage/src/utils/executeCallback.js function executeCallback(promise, callback) { if (callback) { promise.then(function (result) { callback(null, result); }, function (error) { callback(error); }); } } /* harmony default export */ const utils_executeCallback = (executeCallback); ;// CONCATENATED MODULE: ./node_modules/localforage/src/utils/executeTwoCallbacks.js function executeTwoCallbacks(promise, callback, errorCallback) { if (typeof callback === 'function') { promise.then(callback); } if (typeof errorCallback === 'function') { promise.catch(errorCallback); } } /* harmony default export */ const utils_executeTwoCallbacks = (executeTwoCallbacks); ;// CONCATENATED MODULE: ./node_modules/localforage/src/utils/normalizeKey.js function normalizeKey(key) { // Cast the key to a string, as that's all we can set as a key. if (typeof key !== 'string') { console.warn(`${key} used as a key, but it is not a string.`); key = String(key); } return key; } ;// CONCATENATED MODULE: ./node_modules/localforage/src/utils/getCallback.js function getCallback() { if (arguments.length && typeof arguments[arguments.length - 1] === 'function') { return arguments[arguments.length - 1]; } } ;// CONCATENATED MODULE: ./node_modules/localforage/src/drivers/indexeddb.js // Some code originally from async_storage.js in // [Gaia](https://github.com/mozilla-b2g/gaia). const DETECT_BLOB_SUPPORT_STORE = 'local-forage-detect-blob-support'; let supportsBlobs; const dbContexts = {}; const indexeddb_toString = Object.prototype.toString; // Transaction Modes const READ_ONLY = 'readonly'; const READ_WRITE = 'readwrite'; // Transform a binary string to an array buffer, because otherwise // weird stuff happens when you try to work with the binary string directly. // It is known. // From http://stackoverflow.com/questions/14967647/ (continues on next line) // encode-decode-image-with-base64-breaks-image (2013-04-21) function _binStringToArrayBuffer(bin) { var length = bin.length; var buf = new ArrayBuffer(length); var arr = new Uint8Array(buf); for (var i = 0; i < length; i++) { arr[i] = bin.charCodeAt(i); } return buf; } // // Blobs are not supported in all versions of IndexedDB, notably // Chrome <37 and Android <5. In those versions, storing a blob will throw. // // Various other blob bugs exist in Chrome v37-42 (inclusive). // Detecting them is expensive and confusing to users, and Chrome 37-42 // is at very low usage worldwide, so we do a hacky userAgent check instead. // // content-type bug: https://code.google.com/p/chromium/issues/detail?id=408120 // 404 bug: https://code.google.com/p/chromium/issues/detail?id=447916 // FileReader bug: https://code.google.com/p/chromium/issues/detail?id=447836 // // Code borrowed from PouchDB. See: // https://github.com/pouchdb/pouchdb/blob/master/packages/node_modules/pouchdb-adapter-idb/src/blobSupport.js // function _checkBlobSupportWithoutCaching(idb) { return new utils_promise(function (resolve) { var txn = idb.transaction(DETECT_BLOB_SUPPORT_STORE, READ_WRITE); var blob = utils_createBlob(['']); txn.objectStore(DETECT_BLOB_SUPPORT_STORE).put(blob, 'key'); txn.onabort = function (e) { // If the transaction aborts now its due to not being able to // write to the database, likely due to the disk being full e.preventDefault(); e.stopPropagation(); resolve(false); }; txn.oncomplete = function () { var matchedChrome = navigator.userAgent.match(/Chrome\/(\d+)/); var matchedEdge = navigator.userAgent.match(/Edge\//); // MS Edge pretends to be Chrome 42: // https://msdn.microsoft.com/en-us/library/hh869301%28v=vs.85%29.aspx resolve(matchedEdge || !matchedChrome || parseInt(matchedChrome[1], 10) >= 43); }; }).catch(function () { return false; // error, so assume unsupported }); } function _checkBlobSupport(idb) { if (typeof supportsBlobs === 'boolean') { return utils_promise.resolve(supportsBlobs); } return _checkBlobSupportWithoutCaching(idb).then(function (value) { supportsBlobs = value; return supportsBlobs; }); } function _deferReadiness(dbInfo) { var dbContext = dbContexts[dbInfo.name]; // Create a deferred object representing the current database operation. var deferredOperation = {}; deferredOperation.promise = new utils_promise(function (resolve, reject) { deferredOperation.resolve = resolve; deferredOperation.reject = reject; }); // Enqueue the deferred operation. dbContext.deferredOperations.push(deferredOperation); // Chain its promise to the database readiness. if (!dbContext.dbReady) { dbContext.dbReady = deferredOperation.promise; } else { dbContext.dbReady = dbContext.dbReady.then(function () { return deferredOperation.promise; }); } } function _advanceReadiness(dbInfo) { var dbContext = dbContexts[dbInfo.name]; // Dequeue a deferred operation. var deferredOperation = dbContext.deferredOperations.pop(); // Resolve its promise (which is part of the database readiness // chain of promises). if (deferredOperation) { deferredOperation.resolve(); return deferredOperation.promise; } } function _rejectReadiness(dbInfo, err) { var dbContext = dbContexts[dbInfo.name]; // Dequeue a deferred operation. var deferredOperation = dbContext.deferredOperations.pop(); // Reject its promise (which is part of the database readiness // chain of promises). if (deferredOperation) { deferredOperation.reject(err); return deferredOperation.promise; } } function _getConnection(dbInfo, upgradeNeeded) { return new utils_promise(function (resolve, reject) { dbContexts[dbInfo.name] = dbContexts[dbInfo.name] || createDbContext(); if (dbInfo.db) { if (upgradeNeeded) { _deferReadiness(dbInfo); dbInfo.db.close(); } else { return resolve(dbInfo.db); } } var dbArgs = [dbInfo.name]; if (upgradeNeeded) { dbArgs.push(dbInfo.version); } var openreq = utils_idb.open.apply(utils_idb, dbArgs); if (upgradeNeeded) { openreq.onupgradeneeded = function (e) { var db = openreq.result; try { db.createObjectStore(dbInfo.storeName); if (e.oldVersion <= 1) { // Added when support for blob shims was added db.createObjectStore(DETECT_BLOB_SUPPORT_STORE); } } catch (ex) { if (ex.name === 'ConstraintError') { console.warn('The database "' + dbInfo.name + '"' + ' has been upgraded from version ' + e.oldVersion + ' to version ' + e.newVersion + ', but the storage "' + dbInfo.storeName + '" already exists.'); } else { throw ex; } } }; } openreq.onerror = function (e) { e.preventDefault(); reject(openreq.error); }; openreq.onsuccess = function () { var db = openreq.result; db.onversionchange = function (e) { // Triggered when the database is modified (e.g. adding an objectStore) or // deleted (even when initiated by other sessions in different tabs). // Closing the connection here prevents those operations from being blocked. // If the database is accessed again later by this instance, the connection // will be reopened or the database recreated as needed. e.target.close(); }; resolve(db); _advanceReadiness(dbInfo); }; }); } function _getOriginalConnection(dbInfo) { return _getConnection(dbInfo, false); } function _getUpgradedConnection(dbInfo) { return _getConnection(dbInfo, true); } function _isUpgradeNeeded(dbInfo, defaultVersion) { if (!dbInfo.db) { return true; } var isNewStore = !dbInfo.db.objectStoreNames.contains(dbInfo.storeName); var isDowngrade = dbInfo.version < dbInfo.db.version; var isUpgrade = dbInfo.version > dbInfo.db.version; if (isDowngrade) { // If the version is not the default one // then warn for impossible downgrade. if (dbInfo.version !== defaultVersion) { console.warn('The database "' + dbInfo.name + '"' + " can't be downgraded from version " + dbInfo.db.version + ' to version ' + dbInfo.version + '.'); } // Align the versions to prevent errors. dbInfo.version = dbInfo.db.version; } if (isUpgrade || isNewStore) { // If the store is new then increment the version (if needed). // This will trigger an "upgradeneeded" event which is required // for creating a store. if (isNewStore) { var incVersion = dbInfo.db.version + 1; if (incVersion > dbInfo.version) { dbInfo.version = incVersion; } } return true; } return false; } // encode a blob for indexeddb engines that don't support blobs function _encodeBlob(blob) { return new utils_promise(function (resolve, reject) { var reader = new FileReader(); reader.onerror = reject; reader.onloadend = function (e) { var base64 = btoa(e.target.result || ''); resolve({ __local_forage_encoded_blob: true, data: base64, type: blob.type }); }; reader.readAsBinaryString(blob); }); } // decode an encoded blob function _decodeBlob(encodedBlob) { var arrayBuff = _binStringToArrayBuffer(atob(encodedBlob.data)); return utils_createBlob([arrayBuff], { type: encodedBlob.type }); } // is this one of our fancy encoded blobs? function _isEncodedBlob(value) { return value && value.__local_forage_encoded_blob; } // Specialize the default `ready()` function by making it dependent // on the current database operations. Thus, the driver will be actually // ready when it's been initialized (default) *and* there are no pending // operations on the database (initiated by some other instances). function _fullyReady(callback) { var self = this; var promise = self._initReady().then(function () { var dbContext = dbContexts[self._dbInfo.name]; if (dbContext && dbContext.dbReady) { return dbContext.dbReady; } }); utils_executeTwoCallbacks(promise, callback, callback); return promise; } // Try to establish a new db connection to replace the // current one which is broken (i.e. experiencing // InvalidStateError while creating a transaction). function _tryReconnect(dbInfo) { _deferReadiness(dbInfo); var dbContext = dbContexts[dbInfo.name]; var forages = dbContext.forages; for (var i = 0; i < forages.length; i++) { const forage = forages[i]; if (forage._dbInfo.db) { forage._dbInfo.db.close(); forage._dbInfo.db = null; } } dbInfo.db = null; return _getOriginalConnection(dbInfo).then(db => { dbInfo.db = db; if (_isUpgradeNeeded(dbInfo)) { // Reopen the database for upgrading. return _getUpgradedConnection(dbInfo); } return db; }).then(db => { // store the latest db reference // in case the db was upgraded dbInfo.db = dbContext.db = db; for (var i = 0; i < forages.length; i++) { forages[i]._dbInfo.db = db; } }).catch(err => { _rejectReadiness(dbInfo, err); throw err; }); } // FF doesn't like Promises (micro-tasks) and IDDB store operations, // so we have to do it with callbacks function createTransaction(dbInfo, mode, callback, retries) { if (retries === undefined) { retries = 1; } try { var tx = dbInfo.db.transaction(dbInfo.storeName, mode); callback(null, tx); } catch (err) { if (retries > 0 && (!dbInfo.db || err.name === 'InvalidStateError' || err.name === 'NotFoundError')) { return utils_promise.resolve().then(() => { if (!dbInfo.db || err.name === 'NotFoundError' && !dbInfo.db.objectStoreNames.contains(dbInfo.storeName) && dbInfo.version <= dbInfo.db.version) { // increase the db version, to create the new ObjectStore if (dbInfo.db) { dbInfo.version = dbInfo.db.version + 1; } // Reopen the database for upgrading. return _getUpgradedConnection(dbInfo); } }).then(() => { return _tryReconnect(dbInfo).then(function () { createTransaction(dbInfo, mode, callback, retries - 1); }); }).catch(callback); } callback(err); } } function createDbContext() { return { // Running localForages sharing a database. forages: [], // Shared database. db: null, // Database readiness (promise). dbReady: null, // Deferred operations on the database. deferredOperations: [] }; } // Open the IndexedDB database (automatically creates one if one didn't // previously exist), using any options set in the config. function _initStorage(options) { var self = this; var dbInfo = { db: null }; if (options) { for (var i in options) { dbInfo[i] = options[i]; } } // Get the current context of the database; var dbContext = dbContexts[dbInfo.name]; // ...or create a new context. if (!dbContext) { dbContext = createDbContext(); // Register the new context in the global container. dbContexts[dbInfo.name] = dbContext; } // Register itself as a running localForage in the current context. dbContext.forages.push(self); // Replace the default `ready()` function with the specialized one. if (!self._initReady) { self._initReady = self.ready; self.ready = _fullyReady; } // Create an array of initialization states of the related localForages. var initPromises = []; function ignoreErrors() { // Don't handle errors here, // just makes sure related localForages aren't pending. return utils_promise.resolve(); } for (var j = 0; j < dbContext.forages.length; j++) { var forage = dbContext.forages[j]; if (forage !== self) { // Don't wait for itself... initPromises.push(forage._initReady().catch(ignoreErrors)); } } // Take a snapshot of the related localForages. var forages = dbContext.forages.slice(0); // Initialize the connection process only when // all the related localForages aren't pending. return utils_promise.all(initPromises).then(function () { dbInfo.db = dbContext.db; // Get the connection or open a new one without upgrade. return _getOriginalConnection(dbInfo); }).then(function (db) { dbInfo.db = db; if (_isUpgradeNeeded(dbInfo, self._defaultConfig.version)) { // Reopen the database for upgrading. return _getUpgradedConnection(dbInfo); } return db; }).then(function (db) { dbInfo.db = dbContext.db = db; self._dbInfo = dbInfo; // Share the final connection amongst related localForages. for (var k = 0; k < forages.length; k++) { var forage = forages[k]; if (forage !== self) { // Self is already up-to-date. forage._dbInfo.db = dbInfo.db; forage._dbInfo.version = dbInfo.version; } } }); } function getItem(key, callback) { var self = this; key = normalizeKey(key); var promise = new utils_promise(function (resolve, reject) { self.ready().then(function () { createTransaction(self._dbInfo, READ_ONLY, function (err, transaction) { if (err) { return reject(err); } try { var store = transaction.objectStore(self._dbInfo.storeName); var req = store.get(key); req.onsuccess = function () { var value = req.result; if (value === undefined) { value = null; } if (_isEncodedBlob(value)) { value = _decodeBlob(value); } resolve(value); }; req.onerror = function () { reject(req.error); }; } catch (e) { reject(e); } }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } // Iterate over all items stored in database. function iterate(iterator, callback) { var self = this; var promise = new utils_promise(function (resolve, reject) { self.ready().then(function () { createTransaction(self._dbInfo, READ_ONLY, function (err, transaction) { if (err) { return reject(err); } try { var store = transaction.objectStore(self._dbInfo.storeName); var req = store.openCursor(); var iterationNumber = 1; req.onsuccess = function () { var cursor = req.result; if (cursor) { var value = cursor.value; if (_isEncodedBlob(value)) { value = _decodeBlob(value); } var result = iterator(value, cursor.key, iterationNumber++); // when the iterator callback returns any // (non-`undefined`) value, then we stop // the iteration immediately if (result !== void 0) { resolve(result); } else { cursor.continue(); } } else { resolve(); } }; req.onerror = function () { reject(req.error); }; } catch (e) { reject(e); } }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } function setItem(key, value, callback) { var self = this; key = normalizeKey(key); var promise = new utils_promise(function (resolve, reject) { var dbInfo; self.ready().then(function () { dbInfo = self._dbInfo; if (indexeddb_toString.call(value) === '[object Blob]') { return _checkBlobSupport(dbInfo.db).then(function (blobSupport) { if (blobSupport) { return value; } return _encodeBlob(value); }); } return value; }).then(function (value) { createTransaction(self._dbInfo, READ_WRITE, function (err, transaction) { if (err) { return reject(err); } try { var store = transaction.objectStore(self._dbInfo.storeName); // The reason we don't _save_ null is because IE 10 does // not support saving the `null` type in IndexedDB. How // ironic, given the bug below! // See: https://github.com/mozilla/localForage/issues/161 if (value === null) { value = undefined; } var req = store.put(value, key); transaction.oncomplete = function () { // Cast to undefined so the value passed to // callback/promise is the same as what one would get out // of `getItem()` later. This leads to some weirdness // (setItem('foo', undefined) will return `null`), but // it's not my fault localStorage is our baseline and that // it's weird. if (value === undefined) { value = null; } resolve(value); }; transaction.onabort = transaction.onerror = function () { var err = req.error ? req.error : req.transaction.error; reject(err); }; } catch (e) { reject(e); } }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } function removeItem(key, callback) { var self = this; key = normalizeKey(key); var promise = new utils_promise(function (resolve, reject) { self.ready().then(function () { createTransaction(self._dbInfo, READ_WRITE, function (err, transaction) { if (err) { return reject(err); } try { var store = transaction.objectStore(self._dbInfo.storeName); // We use a Grunt task to make this safe for IE and some // versions of Android (including those used by Cordova). // Normally IE won't like `.delete()` and will insist on // using `['delete']()`, but we have a build step that // fixes this for us now. var req = store.delete(key); transaction.oncomplete = function () { resolve(); }; transaction.onerror = function () { reject(req.error); }; // The request will be also be aborted if we've exceeded our storage // space. transaction.onabort = function () { var err = req.error ? req.error : req.transaction.error; reject(err); }; } catch (e) { reject(e); } }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } function clear(callback) { var self = this; var promise = new utils_promise(function (resolve, reject) { self.ready().then(function () { createTransaction(self._dbInfo, READ_WRITE, function (err, transaction) { if (err) { return reject(err); } try { var store = transaction.objectStore(self._dbInfo.storeName); var req = store.clear(); transaction.oncomplete = function () { resolve(); }; transaction.onabort = transaction.onerror = function () { var err = req.error ? req.error : req.transaction.error; reject(err); }; } catch (e) { reject(e); } }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } function indexeddb_length(callback) { var self = this; var promise = new utils_promise(function (resolve, reject) { self.ready().then(function () { createTransaction(self._dbInfo, READ_ONLY, function (err, transaction) { if (err) { return reject(err); } try { var store = transaction.objectStore(self._dbInfo.storeName); var req = store.count(); req.onsuccess = function () { resolve(req.result); }; req.onerror = function () { reject(req.error); }; } catch (e) { reject(e); } }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } function key(n, callback) { var self = this; var promise = new utils_promise(function (resolve, reject) { if (n < 0) { resolve(null); return; } self.ready().then(function () { createTransaction(self._dbInfo, READ_ONLY, function (err, transaction) { if (err) { return reject(err); } try { var store = transaction.objectStore(self._dbInfo.storeName); var advanced = false; var req = store.openKeyCursor(); req.onsuccess = function () { var cursor = req.result; if (!cursor) { // this means there weren't enough keys resolve(null); return; } if (n === 0) { // We have the first key, return it if that's what they // wanted. resolve(cursor.key); } else { if (!advanced) { // Otherwise, ask the cursor to skip ahead n // records. advanced = true; cursor.advance(n); } else { // When we get here, we've got the nth key. resolve(cursor.key); } } }; req.onerror = function () { reject(req.error); }; } catch (e) { reject(e); } }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } function indexeddb_keys(callback) { var self = this; var promise = new utils_promise(function (resolve, reject) { self.ready().then(function () { createTransaction(self._dbInfo, READ_ONLY, function (err, transaction) { if (err) { return reject(err); } try { var store = transaction.objectStore(self._dbInfo.storeName); var req = store.openKeyCursor(); var keys = []; req.onsuccess = function () { var cursor = req.result; if (!cursor) { resolve(keys); return; } keys.push(cursor.key); cursor.continue(); }; req.onerror = function () { reject(req.error); }; } catch (e) { reject(e); } }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } function dropInstance(options, callback) { callback = getCallback.apply(this, arguments); var currentConfig = this.config(); options = typeof options !== 'function' && options || {}; if (!options.name) { options.name = options.name || currentConfig.name; options.storeName = options.storeName || currentConfig.storeName; } var self = this; var promise; if (!options.name) { promise = utils_promise.reject('Invalid arguments'); } else { const isCurrentDb = options.name === currentConfig.name && self._dbInfo.db; const dbPromise = isCurrentDb ? utils_promise.resolve(self._dbInfo.db) : _getOriginalConnection(options).then(db => { const dbContext = dbContexts[options.name]; const forages = dbContext.forages; dbContext.db = db; for (var i = 0; i < forages.length; i++) { forages[i]._dbInfo.db = db; } return db; }); if (!options.storeName) { promise = dbPromise.then(db => { _deferReadiness(options); const dbContext = dbContexts[options.name]; const forages = dbContext.forages; db.close(); for (var i = 0; i < forages.length; i++) { const forage = forages[i]; forage._dbInfo.db = null; } const dropDBPromise = new utils_promise((resolve, reject) => { var req = utils_idb.deleteDatabase(options.name); req.onerror = () => { const db = req.result; if (db) { db.close(); } reject(req.error); }; req.onblocked = () => { // Closing all open connections in onversionchange handler should prevent this situation, but if // we do get here, it just means the request remains pending - eventually it will succeed or error console.warn('dropInstance blocked for database "' + options.name + '" until all open connections are closed'); }; req.onsuccess = () => { const db = req.result; if (db) { db.close(); } resolve(db); }; }); return dropDBPromise.then(db => { dbContext.db = db; for (var i = 0; i < forages.length; i++) { const forage = forages[i]; _advanceReadiness(forage._dbInfo); } }).catch(err => { (_rejectReadiness(options, err) || utils_promise.resolve()).catch(() => {}); throw err; }); }); } else { promise = dbPromise.then(db => { if (!db.objectStoreNames.contains(options.storeName)) { return; } const newVersion = db.version + 1; _deferReadiness(options); const dbContext = dbContexts[options.name]; const forages = dbContext.forages; db.close(); for (let i = 0; i < forages.length; i++) { const forage = forages[i]; forage._dbInfo.db = null; forage._dbInfo.version = newVersion; } const dropObjectPromise = new utils_promise((resolve, reject) => { const req = utils_idb.open(options.name, newVersion); req.onerror = err => { const db = req.result; db.close(); reject(err); }; req.onupgradeneeded = () => { var db = req.result; db.deleteObjectStore(options.storeName); }; req.onsuccess = () => { const db = req.result; db.close(); resolve(db); }; }); return dropObjectPromise.then(db => { dbContext.db = db; for (let j = 0; j < forages.length; j++) { const forage = forages[j]; forage._dbInfo.db = db; _advanceReadiness(forage._dbInfo); } }).catch(err => { (_rejectReadiness(options, err) || utils_promise.resolve()).catch(() => {}); throw err; }); }); } } utils_executeCallback(promise, callback); return promise; } var asyncStorage = { _driver: 'asyncStorage', _initStorage: _initStorage, _support: utils_isIndexedDBValid(), iterate: iterate, getItem: getItem, setItem: setItem, removeItem: removeItem, clear: clear, length: indexeddb_length, key: key, keys: indexeddb_keys, dropInstance: dropInstance }; /* harmony default export */ const indexeddb = (asyncStorage); ;// CONCATENATED MODULE: ./node_modules/localforage/src/utils/isWebSQLValid.js function isWebSQLValid() { return typeof openDatabase === 'function'; } /* harmony default export */ const utils_isWebSQLValid = (isWebSQLValid); ;// CONCATENATED MODULE: ./node_modules/localforage/src/utils/serializer.js /* eslint-disable no-bitwise */ // Sadly, the best way to save binary data in WebSQL/localStorage is serializing // it to Base64, so this is how we store it to prevent very strange errors with less // verbose ways of binary <-> string data storage. var BASE_CHARS = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/'; var BLOB_TYPE_PREFIX = '~~local_forage_type~'; var BLOB_TYPE_PREFIX_REGEX = /^~~local_forage_type~([^~]+)~/; var SERIALIZED_MARKER = '__lfsc__:'; var SERIALIZED_MARKER_LENGTH = SERIALIZED_MARKER.length; // OMG the serializations! var TYPE_ARRAYBUFFER = 'arbf'; var TYPE_BLOB = 'blob'; var TYPE_INT8ARRAY = 'si08'; var TYPE_UINT8ARRAY = 'ui08'; var TYPE_UINT8CLAMPEDARRAY = 'uic8'; var TYPE_INT16ARRAY = 'si16'; var TYPE_INT32ARRAY = 'si32'; var TYPE_UINT16ARRAY = 'ur16'; var TYPE_UINT32ARRAY = 'ui32'; var TYPE_FLOAT32ARRAY = 'fl32'; var TYPE_FLOAT64ARRAY = 'fl64'; var TYPE_SERIALIZED_MARKER_LENGTH = SERIALIZED_MARKER_LENGTH + TYPE_ARRAYBUFFER.length; var serializer_toString = Object.prototype.toString; function stringToBuffer(serializedString) { // Fill the string into a ArrayBuffer. var bufferLength = serializedString.length * 0.75; var len = serializedString.length; var i; var p = 0; var encoded1, encoded2, encoded3, encoded4; if (serializedString[serializedString.length - 1] === '=') { bufferLength--; if (serializedString[serializedString.length - 2] === '=') { bufferLength--; } } var buffer = new ArrayBuffer(bufferLength); var bytes = new Uint8Array(buffer); for (i = 0; i < len; i += 4) { encoded1 = BASE_CHARS.indexOf(serializedString[i]); encoded2 = BASE_CHARS.indexOf(serializedString[i + 1]); encoded3 = BASE_CHARS.indexOf(serializedString[i + 2]); encoded4 = BASE_CHARS.indexOf(serializedString[i + 3]); /*jslint bitwise: true */ bytes[p++] = encoded1 << 2 | encoded2 >> 4; bytes[p++] = (encoded2 & 15) << 4 | encoded3 >> 2; bytes[p++] = (encoded3 & 3) << 6 | encoded4 & 63; } return buffer; } // Converts a buffer to a string to store, serialized, in the backend // storage library. function bufferToString(buffer) { // base64-arraybuffer var bytes = new Uint8Array(buffer); var base64String = ''; var i; for (i = 0; i < bytes.length; i += 3) { /*jslint bitwise: true */ base64String += BASE_CHARS[bytes[i] >> 2]; base64String += BASE_CHARS[(bytes[i] & 3) << 4 | bytes[i + 1] >> 4]; base64String += BASE_CHARS[(bytes[i + 1] & 15) << 2 | bytes[i + 2] >> 6]; base64String += BASE_CHARS[bytes[i + 2] & 63]; } if (bytes.length % 3 === 2) { base64String = base64String.substring(0, base64String.length - 1) + '='; } else if (bytes.length % 3 === 1) { base64String = base64String.substring(0, base64String.length - 2) + '=='; } return base64String; } // Serialize a value, afterwards executing a callback (which usually // instructs the `setItem()` callback/promise to be executed). This is how // we store binary data with localStorage. function serialize(value, callback) { var valueType = ''; if (value) { valueType = serializer_toString.call(value); } // Cannot use `value instanceof ArrayBuffer` or such here, as these // checks fail when running the tests using casper.js... // // TODO: See why those tests fail and use a better solution. if (value && (valueType === '[object ArrayBuffer]' || value.buffer && serializer_toString.call(value.buffer) === '[object ArrayBuffer]')) { // Convert binary arrays to a string and prefix the string with // a special marker. var buffer; var marker = SERIALIZED_MARKER; if (value instanceof ArrayBuffer) { buffer = value; marker += TYPE_ARRAYBUFFER; } else { buffer = value.buffer; if (valueType === '[object Int8Array]') { marker += TYPE_INT8ARRAY; } else if (valueType === '[object Uint8Array]') { marker += TYPE_UINT8ARRAY; } else if (valueType === '[object Uint8ClampedArray]') { marker += TYPE_UINT8CLAMPEDARRAY; } else if (valueType === '[object Int16Array]') { marker += TYPE_INT16ARRAY; } else if (valueType === '[object Uint16Array]') { marker += TYPE_UINT16ARRAY; } else if (valueType === '[object Int32Array]') { marker += TYPE_INT32ARRAY; } else if (valueType === '[object Uint32Array]') { marker += TYPE_UINT32ARRAY; } else if (valueType === '[object Float32Array]') { marker += TYPE_FLOAT32ARRAY; } else if (valueType === '[object Float64Array]') { marker += TYPE_FLOAT64ARRAY; } else { callback(new Error('Failed to get type for BinaryArray')); } } callback(marker + bufferToString(buffer)); } else if (valueType === '[object Blob]') { // Conver the blob to a binaryArray and then to a string. var fileReader = new FileReader(); fileReader.onload = function () { // Backwards-compatible prefix for the blob type. var str = BLOB_TYPE_PREFIX + value.type + '~' + bufferToString(this.result); callback(SERIALIZED_MARKER + TYPE_BLOB + str); }; fileReader.readAsArrayBuffer(value); } else { try { callback(JSON.stringify(value)); } catch (e) { console.error("Couldn't convert value into a JSON string: ", value); callback(null, e); } } } // Deserialize data we've inserted into a value column/field. We place // special markers into our strings to mark them as encoded; this isn't // as nice as a meta field, but it's the only sane thing we can do whilst // keeping localStorage support intact. // // Oftentimes this will just deserialize JSON content, but if we have a // special marker (SERIALIZED_MARKER, defined above), we will extract // some kind of arraybuffer/binary data/typed array out of the string. function deserialize(value) { // If we haven't marked this string as being specially serialized (i.e. // something other than serialized JSON), we can just return it and be // done with it. if (value.substring(0, SERIALIZED_MARKER_LENGTH) !== SERIALIZED_MARKER) { return JSON.parse(value); } // The following code deals with deserializing some kind of Blob or // TypedArray. First we separate out the type of data we're dealing // with from the data itself. var serializedString = value.substring(TYPE_SERIALIZED_MARKER_LENGTH); var type = value.substring(SERIALIZED_MARKER_LENGTH, TYPE_SERIALIZED_MARKER_LENGTH); var blobType; // Backwards-compatible blob type serialization strategy. // DBs created with older versions of localForage will simply not have the blob type. if (type === TYPE_BLOB && BLOB_TYPE_PREFIX_REGEX.test(serializedString)) { var matcher = serializedString.match(BLOB_TYPE_PREFIX_REGEX); blobType = matcher[1]; serializedString = serializedString.substring(matcher[0].length); } var buffer = stringToBuffer(serializedString); // Return the right type based on the code/type set during // serialization. switch (type) { case TYPE_ARRAYBUFFER: return buffer; case TYPE_BLOB: return utils_createBlob([buffer], { type: blobType }); case TYPE_INT8ARRAY: return new Int8Array(buffer); case TYPE_UINT8ARRAY: return new Uint8Array(buffer); case TYPE_UINT8CLAMPEDARRAY: return new Uint8ClampedArray(buffer); case TYPE_INT16ARRAY: return new Int16Array(buffer); case TYPE_UINT16ARRAY: return new Uint16Array(buffer); case TYPE_INT32ARRAY: return new Int32Array(buffer); case TYPE_UINT32ARRAY: return new Uint32Array(buffer); case TYPE_FLOAT32ARRAY: return new Float32Array(buffer); case TYPE_FLOAT64ARRAY: return new Float64Array(buffer); default: throw new Error('Unkown type: ' + type); } } var localforageSerializer = { serialize: serialize, deserialize: deserialize, stringToBuffer: stringToBuffer, bufferToString: bufferToString }; /* harmony default export */ const serializer = (localforageSerializer); ;// CONCATENATED MODULE: ./node_modules/localforage/src/drivers/websql.js /* * Includes code from: * * base64-arraybuffer * https://github.com/niklasvh/base64-arraybuffer * * Copyright (c) 2012 Niklas von Hertzen * Licensed under the MIT license. */ function createDbTable(t, dbInfo, callback, errorCallback) { t.executeSql(`CREATE TABLE IF NOT EXISTS ${dbInfo.storeName} ` + '(id INTEGER PRIMARY KEY, key unique, value)', [], callback, errorCallback); } // Open the WebSQL database (automatically creates one if one didn't // previously exist), using any options set in the config. function websql_initStorage(options) { var self = this; var dbInfo = { db: null }; if (options) { for (var i in options) { dbInfo[i] = typeof options[i] !== 'string' ? options[i].toString() : options[i]; } } var dbInfoPromise = new utils_promise(function (resolve, reject) { // Open the database; the openDatabase API will automatically // create it for us if it doesn't exist. try { dbInfo.db = openDatabase(dbInfo.name, String(dbInfo.version), dbInfo.description, dbInfo.size); } catch (e) { return reject(e); } // Create our key/value table if it doesn't exist. dbInfo.db.transaction(function (t) { createDbTable(t, dbInfo, function () { self._dbInfo = dbInfo; resolve(); }, function (t, error) { reject(error); }); }, reject); }); dbInfo.serializer = serializer; return dbInfoPromise; } function tryExecuteSql(t, dbInfo, sqlStatement, args, callback, errorCallback) { t.executeSql(sqlStatement, args, callback, function (t, error) { if (error.code === error.SYNTAX_ERR) { t.executeSql('SELECT name FROM sqlite_master ' + "WHERE type='table' AND name = ?", [dbInfo.storeName], function (t, results) { if (!results.rows.length) { // if the table is missing (was deleted) // re-create it table and retry createDbTable(t, dbInfo, function () { t.executeSql(sqlStatement, args, callback, errorCallback); }, errorCallback); } else { errorCallback(t, error); } }, errorCallback); } else { errorCallback(t, error); } }, errorCallback); } function websql_getItem(key, callback) { var self = this; key = normalizeKey(key); var promise = new utils_promise(function (resolve, reject) { self.ready().then(function () { var dbInfo = self._dbInfo; dbInfo.db.transaction(function (t) { tryExecuteSql(t, dbInfo, `SELECT * FROM ${dbInfo.storeName} WHERE key = ? LIMIT 1`, [key], function (t, results) { var result = results.rows.length ? results.rows.item(0).value : null; // Check to see if this is serialized content we need to // unpack. if (result) { result = dbInfo.serializer.deserialize(result); } resolve(result); }, function (t, error) { reject(error); }); }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } function websql_iterate(iterator, callback) { var self = this; var promise = new utils_promise(function (resolve, reject) { self.ready().then(function () { var dbInfo = self._dbInfo; dbInfo.db.transaction(function (t) { tryExecuteSql(t, dbInfo, `SELECT * FROM ${dbInfo.storeName}`, [], function (t, results) { var rows = results.rows; var length = rows.length; for (var i = 0; i < length; i++) { var item = rows.item(i); var result = item.value; // Check to see if this is serialized content // we need to unpack. if (result) { result = dbInfo.serializer.deserialize(result); } result = iterator(result, item.key, i + 1); // void(0) prevents problems with redefinition // of `undefined`. if (result !== void 0) { resolve(result); return; } } resolve(); }, function (t, error) { reject(error); }); }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } function _setItem(key, value, callback, retriesLeft) { var self = this; key = normalizeKey(key); var promise = new utils_promise(function (resolve, reject) { self.ready().then(function () { // The localStorage API doesn't return undefined values in an // "expected" way, so undefined is always cast to null in all // drivers. See: https://github.com/mozilla/localForage/pull/42 if (value === undefined) { value = null; } // Save the original value to pass to the callback. var originalValue = value; var dbInfo = self._dbInfo; dbInfo.serializer.serialize(value, function (value, error) { if (error) { reject(error); } else { dbInfo.db.transaction(function (t) { tryExecuteSql(t, dbInfo, `INSERT OR REPLACE INTO ${dbInfo.storeName} ` + '(key, value) VALUES (?, ?)', [key, value], function () { resolve(originalValue); }, function (t, error) { reject(error); }); }, function (sqlError) { // The transaction failed; check // to see if it's a quota error. if (sqlError.code === sqlError.QUOTA_ERR) { // We reject the callback outright for now, but // it's worth trying to re-run the transaction. // Even if the user accepts the prompt to use // more storage on Safari, this error will // be called. // // Try to re-run the transaction. if (retriesLeft > 0) { resolve(_setItem.apply(self, [key, originalValue, callback, retriesLeft - 1])); return; } reject(sqlError); } }); } }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } function websql_setItem(key, value, callback) { return _setItem.apply(this, [key, value, callback, 1]); } function websql_removeItem(key, callback) { var self = this; key = normalizeKey(key); var promise = new utils_promise(function (resolve, reject) { self.ready().then(function () { var dbInfo = self._dbInfo; dbInfo.db.transaction(function (t) { tryExecuteSql(t, dbInfo, `DELETE FROM ${dbInfo.storeName} WHERE key = ?`, [key], function () { resolve(); }, function (t, error) { reject(error); }); }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } // Deletes every item in the table. // TODO: Find out if this resets the AUTO_INCREMENT number. function websql_clear(callback) { var self = this; var promise = new utils_promise(function (resolve, reject) { self.ready().then(function () { var dbInfo = self._dbInfo; dbInfo.db.transaction(function (t) { tryExecuteSql(t, dbInfo, `DELETE FROM ${dbInfo.storeName}`, [], function () { resolve(); }, function (t, error) { reject(error); }); }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } // Does a simple `COUNT(key)` to get the number of items stored in // localForage. function websql_length(callback) { var self = this; var promise = new utils_promise(function (resolve, reject) { self.ready().then(function () { var dbInfo = self._dbInfo; dbInfo.db.transaction(function (t) { // Ahhh, SQL makes this one soooooo easy. tryExecuteSql(t, dbInfo, `SELECT COUNT(key) as c FROM ${dbInfo.storeName}`, [], function (t, results) { var result = results.rows.item(0).c; resolve(result); }, function (t, error) { reject(error); }); }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } // Return the key located at key index X; essentially gets the key from a // `WHERE id = ?`. This is the most efficient way I can think to implement // this rarely-used (in my experience) part of the API, but it can seem // inconsistent, because we do `INSERT OR REPLACE INTO` on `setItem()`, so // the ID of each key will change every time it's updated. Perhaps a stored // procedure for the `setItem()` SQL would solve this problem? // TODO: Don't change ID on `setItem()`. function websql_key(n, callback) { var self = this; var promise = new utils_promise(function (resolve, reject) { self.ready().then(function () { var dbInfo = self._dbInfo; dbInfo.db.transaction(function (t) { tryExecuteSql(t, dbInfo, `SELECT key FROM ${dbInfo.storeName} WHERE id = ? LIMIT 1`, [n + 1], function (t, results) { var result = results.rows.length ? results.rows.item(0).key : null; resolve(result); }, function (t, error) { reject(error); }); }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } function websql_keys(callback) { var self = this; var promise = new utils_promise(function (resolve, reject) { self.ready().then(function () { var dbInfo = self._dbInfo; dbInfo.db.transaction(function (t) { tryExecuteSql(t, dbInfo, `SELECT key FROM ${dbInfo.storeName}`, [], function (t, results) { var keys = []; for (var i = 0; i < results.rows.length; i++) { keys.push(results.rows.item(i).key); } resolve(keys); }, function (t, error) { reject(error); }); }); }).catch(reject); }); utils_executeCallback(promise, callback); return promise; } // https://www.w3.org/TR/webdatabase/#databases // > There is no way to enumerate or delete the databases available for an origin from this API. function getAllStoreNames(db) { return new utils_promise(function (resolve, reject) { db.transaction(function (t) { t.executeSql('SELECT name FROM sqlite_master ' + "WHERE type='table' AND name <> '__WebKitDatabaseInfoTable__'", [], function (t, results) { var storeNames = []; for (var i = 0; i < results.rows.length; i++) { storeNames.push(results.rows.item(i).name); } resolve({ db, storeNames }); }, function (t, error) { reject(error); }); }, function (sqlError) { reject(sqlError); }); }); } function websql_dropInstance(options, callback) { callback = getCallback.apply(this, arguments); var currentConfig = this.config(); options = typeof options !== 'function' && options || {}; if (!options.name) { options.name = options.name || currentConfig.name; options.storeName = options.storeName || currentConfig.storeName; } var self = this; var promise; if (!options.name) { promise = utils_promise.reject('Invalid arguments'); } else { promise = new utils_promise(function (resolve) { var db; if (options.name === currentConfig.name) { // use the db reference of the current instance db = self._dbInfo.db; } else { db = openDatabase(options.name, '', '', 0); } if (!options.storeName) { // drop all database tables resolve(getAllStoreNames(db)); } else { resolve({ db, storeNames: [options.storeName] }); } }).then(function (operationInfo) { return new utils_promise(function (resolve, reject) { operationInfo.db.transaction(function (t) { function dropTable(storeName) { return new utils_promise(function (resolve, reject) { t.executeSql(`DROP TABLE IF EXISTS ${storeName}`, [], function () { resolve(); }, function (t, error) { reject(error); }); }); } var operations = []; for (var i = 0, len = operationInfo.storeNames.length; i < len; i++) { operations.push(dropTable(operationInfo.storeNames[i])); } utils_promise.all(operations).then(function () { resolve(); }).catch(function (e) { reject(e); }); }, function (sqlError) { reject(sqlError); }); }); }); } utils_executeCallback(promise, callback); return promise; } var webSQLStorage = { _driver: 'webSQLStorage', _initStorage: websql_initStorage, _support: utils_isWebSQLValid(), iterate: websql_iterate, getItem: websql_getItem, setItem: websql_setItem, removeItem: websql_removeItem, clear: websql_clear, length: websql_length, key: websql_key, keys: websql_keys, dropInstance: websql_dropInstance }; /* harmony default export */ const websql = (webSQLStorage); ;// CONCATENATED MODULE: ./node_modules/localforage/src/utils/isLocalStorageValid.js function isLocalStorageValid() { try { return typeof localStorage !== 'undefined' && 'setItem' in localStorage && // in IE8 typeof localStorage.setItem === 'object' !!localStorage.setItem; } catch (e) { return false; } } /* harmony default export */ const utils_isLocalStorageValid = (isLocalStorageValid); ;// CONCATENATED MODULE: ./node_modules/localforage/src/drivers/localstorage.js // If IndexedDB isn't available, we'll fall back to localStorage. // Note that this will have considerable performance and storage // side-effects (all data will be serialized on save and only data that // can be converted to a string via `JSON.stringify()` will be saved). function _getKeyPrefix(options, defaultConfig) { var keyPrefix = options.name + '/'; if (options.storeName !== defaultConfig.storeName) { keyPrefix += options.storeName + '/'; } return keyPrefix; } // Check if localStorage throws when saving an item function checkIfLocalStorageThrows() { var localStorageTestKey = '_localforage_support_test'; try { localStorage.setItem(localStorageTestKey, true); localStorage.removeItem(localStorageTestKey); return false; } catch (e) { return true; } } // Check if localStorage is usable and allows to save an item // This method checks if localStorage is usable in Safari Private Browsing // mode, or in any other case where the available quota for localStorage // is 0 and there wasn't any saved items yet. function _isLocalStorageUsable() { return !checkIfLocalStorageThrows() || localStorage.length > 0; } // Config the localStorage backend, using options set in the config. function localstorage_initStorage(options) { var self = this; var dbInfo = {}; if (options) { for (var i in options) { dbInfo[i] = options[i]; } } dbInfo.keyPrefix = _getKeyPrefix(options, self._defaultConfig); if (!_isLocalStorageUsable()) { return utils_promise.reject(); } self._dbInfo = dbInfo; dbInfo.serializer = serializer; return utils_promise.resolve(); } // Remove all keys from the datastore, effectively destroying all data in // the app's key/value store! function localstorage_clear(callback) { var self = this; var promise = self.ready().then(function () { var keyPrefix = self._dbInfo.keyPrefix; for (var i = localStorage.length - 1; i >= 0; i--) { var key = localStorage.key(i); if (key.indexOf(keyPrefix) === 0) { localStorage.removeItem(key); } } }); utils_executeCallback(promise, callback); return promise; } // Retrieve an item from the store. Unlike the original async_storage // library in Gaia, we don't modify return values at all. If a key's value // is `undefined`, we pass that value to the callback function. function localstorage_getItem(key, callback) { var self = this; key = normalizeKey(key); var promise = self.ready().then(function () { var dbInfo = self._dbInfo; var result = localStorage.getItem(dbInfo.keyPrefix + key); // If a result was found, parse it from the serialized // string into a JS object. If result isn't truthy, the key // is likely undefined and we'll pass it straight to the // callback. if (result) { result = dbInfo.serializer.deserialize(result); } return result; }); utils_executeCallback(promise, callback); return promise; } // Iterate over all items in the store. function localstorage_iterate(iterator, callback) { var self = this; var promise = self.ready().then(function () { var dbInfo = self._dbInfo; var keyPrefix = dbInfo.keyPrefix; var keyPrefixLength = keyPrefix.length; var length = localStorage.length; // We use a dedicated iterator instead of the `i` variable below // so other keys we fetch in localStorage aren't counted in // the `iterationNumber` argument passed to the `iterate()` // callback. // // See: github.com/mozilla/localForage/pull/435#discussion_r38061530 var iterationNumber = 1; for (var i = 0; i < length; i++) { var key = localStorage.key(i); if (key.indexOf(keyPrefix) !== 0) { continue; } var value = localStorage.getItem(key); // If a result was found, parse it from the serialized // string into a JS object. If result isn't truthy, the // key is likely undefined and we'll pass it straight // to the iterator. if (value) { value = dbInfo.serializer.deserialize(value); } value = iterator(value, key.substring(keyPrefixLength), iterationNumber++); if (value !== void 0) { return value; } } }); utils_executeCallback(promise, callback); return promise; } // Same as localStorage's key() method, except takes a callback. function localstorage_key(n, callback) { var self = this; var promise = self.ready().then(function () { var dbInfo = self._dbInfo; var result; try { result = localStorage.key(n); } catch (error) { result = null; } // Remove the prefix from the key, if a key is found. if (result) { result = result.substring(dbInfo.keyPrefix.length); } return result; }); utils_executeCallback(promise, callback); return promise; } function localstorage_keys(callback) { var self = this; var promise = self.ready().then(function () { var dbInfo = self._dbInfo; var length = localStorage.length; var keys = []; for (var i = 0; i < length; i++) { var itemKey = localStorage.key(i); if (itemKey.indexOf(dbInfo.keyPrefix) === 0) { keys.push(itemKey.substring(dbInfo.keyPrefix.length)); } } return keys; }); utils_executeCallback(promise, callback); return promise; } // Supply the number of keys in the datastore to the callback function. function localstorage_length(callback) { var self = this; var promise = self.keys().then(function (keys) { return keys.length; }); utils_executeCallback(promise, callback); return promise; } // Remove an item from the store, nice and simple. function localstorage_removeItem(key, callback) { var self = this; key = normalizeKey(key); var promise = self.ready().then(function () { var dbInfo = self._dbInfo; localStorage.removeItem(dbInfo.keyPrefix + key); }); utils_executeCallback(promise, callback); return promise; } // Set a key's value and run an optional callback once the value is set. // Unlike Gaia's implementation, the callback function is passed the value, // in case you want to operate on that value only after you're sure it // saved, or something like that. function localstorage_setItem(key, value, callback) { var self = this; key = normalizeKey(key); var promise = self.ready().then(function () { // Convert undefined values to null. // https://github.com/mozilla/localForage/pull/42 if (value === undefined) { value = null; } // Save the original value to pass to the callback. var originalValue = value; return new utils_promise(function (resolve, reject) { var dbInfo = self._dbInfo; dbInfo.serializer.serialize(value, function (value, error) { if (error) { reject(error); } else { try { localStorage.setItem(dbInfo.keyPrefix + key, value); resolve(originalValue); } catch (e) { // localStorage capacity exceeded. // TODO: Make this a specific error/event. if (e.name === 'QuotaExceededError' || e.name === 'NS_ERROR_DOM_QUOTA_REACHED') { reject(e); } reject(e); } } }); }); }); utils_executeCallback(promise, callback); return promise; } function localstorage_dropInstance(options, callback) { callback = getCallback.apply(this, arguments); options = typeof options !== 'function' && options || {}; if (!options.name) { var currentConfig = this.config(); options.name = options.name || currentConfig.name; options.storeName = options.storeName || currentConfig.storeName; } var self = this; var promise; if (!options.name) { promise = utils_promise.reject('Invalid arguments'); } else { promise = new utils_promise(function (resolve) { if (!options.storeName) { resolve(`${options.name}/`); } else { resolve(_getKeyPrefix(options, self._defaultConfig)); } }).then(function (keyPrefix) { for (var i = localStorage.length - 1; i >= 0; i--) { var key = localStorage.key(i); if (key.indexOf(keyPrefix) === 0) { localStorage.removeItem(key); } } }); } utils_executeCallback(promise, callback); return promise; } var localStorageWrapper = { _driver: 'localStorageWrapper', _initStorage: localstorage_initStorage, _support: utils_isLocalStorageValid(), iterate: localstorage_iterate, getItem: localstorage_getItem, setItem: localstorage_setItem, removeItem: localstorage_removeItem, clear: localstorage_clear, length: localstorage_length, key: localstorage_key, keys: localstorage_keys, dropInstance: localstorage_dropInstance }; /* harmony default export */ const localstorage = (localStorageWrapper); ;// CONCATENATED MODULE: ./node_modules/localforage/src/utils/includes.js const sameValue = (x, y) => x === y || typeof x === 'number' && typeof y === 'number' && isNaN(x) && isNaN(y); const includes = (array, searchElement) => { const len = array.length; let i = 0; while (i < len) { if (sameValue(array[i], searchElement)) { return true; } i++; } return false; }; /* harmony default export */ const utils_includes = (includes); ;// CONCATENATED MODULE: ./node_modules/localforage/src/utils/isArray.js const isArray_isArray = Array.isArray || function (arg) { return Object.prototype.toString.call(arg) === '[object Array]'; }; /* harmony default export */ const utils_isArray = (isArray_isArray); ;// CONCATENATED MODULE: ./node_modules/localforage/src/localforage.js // Drivers are stored here when `defineDriver()` is called. // They are shared across all instances of localForage. const DefinedDrivers = {}; const DriverSupport = {}; const DefaultDrivers = { INDEXEDDB: indexeddb, WEBSQL: websql, LOCALSTORAGE: localstorage }; const DefaultDriverOrder = [DefaultDrivers.INDEXEDDB._driver, DefaultDrivers.WEBSQL._driver, DefaultDrivers.LOCALSTORAGE._driver]; const OptionalDriverMethods = ['dropInstance']; const LibraryMethods = ['clear', 'getItem', 'iterate', 'key', 'keys', 'length', 'removeItem', 'setItem'].concat(OptionalDriverMethods); const DefaultConfig = { description: '', driver: DefaultDriverOrder.slice(), name: 'localforage', // Default DB size is _JUST UNDER_ 5MB, as it's the highest size // we can use without a prompt. size: 4980736, storeName: 'keyvaluepairs', version: 1.0 }; function callWhenReady(localForageInstance, libraryMethod) { localForageInstance[libraryMethod] = function () { const _args = arguments; return localForageInstance.ready().then(function () { return localForageInstance[libraryMethod].apply(localForageInstance, _args); }); }; } function extend() { for (let i = 1; i < arguments.length; i++) { const arg = arguments[i]; if (arg) { for (let key in arg) { if (arg.hasOwnProperty(key)) { if (utils_isArray(arg[key])) { arguments[0][key] = arg[key].slice(); } else { arguments[0][key] = arg[key]; } } } } } return arguments[0]; } class LocalForage { constructor(options) { for (let driverTypeKey in DefaultDrivers) { if (DefaultDrivers.hasOwnProperty(driverTypeKey)) { const driver = DefaultDrivers[driverTypeKey]; const driverName = driver._driver; this[driverTypeKey] = driverName; if (!DefinedDrivers[driverName]) { // we don't need to wait for the promise, // since the default drivers can be defined // in a blocking manner this.defineDriver(driver); } } } this._defaultConfig = extend({}, DefaultConfig); this._config = extend({}, this._defaultConfig, options); this._driverSet = null; this._initDriver = null; this._ready = false; this._dbInfo = null; this._wrapLibraryMethodsWithReady(); this.setDriver(this._config.driver).catch(() => {}); } // Set any config values for localForage; can be called anytime before // the first API call (e.g. `getItem`, `setItem`). // We loop through options so we don't overwrite existing config // values. config(options) { // If the options argument is an object, we use it to set values. // Otherwise, we return either a specified config value or all // config values. if (typeof options === 'object') { // If localforage is ready and fully initialized, we can't set // any new configuration values. Instead, we return an error. if (this._ready) { return new Error("Can't call config() after localforage " + 'has been used.'); } for (let i in options) { if (i === 'storeName') { options[i] = options[i].replace(/\W/g, '_'); } if (i === 'version' && typeof options[i] !== 'number') { return new Error('Database version must be a number.'); } this._config[i] = options[i]; } // after all config options are set and // the driver option is used, try setting it if ('driver' in options && options.driver) { return this.setDriver(this._config.driver); } return true; } else if (typeof options === 'string') { return this._config[options]; } else { return this._config; } } // Used to define a custom driver, shared across all instances of // localForage. defineDriver(driverObject, callback, errorCallback) { const promise = new utils_promise(function (resolve, reject) { try { const driverName = driverObject._driver; const complianceError = new Error('Custom driver not compliant; see ' + 'https://mozilla.github.io/localForage/#definedriver'); // A driver name should be defined and not overlap with the // library-defined, default drivers. if (!driverObject._driver) { reject(complianceError); return; } const driverMethods = LibraryMethods.concat('_initStorage'); for (let i = 0, len = driverMethods.length; i < len; i++) { const driverMethodName = driverMethods[i]; // when the property is there, // it should be a method even when optional const isRequired = !utils_includes(OptionalDriverMethods, driverMethodName); if ((isRequired || driverObject[driverMethodName]) && typeof driverObject[driverMethodName] !== 'function') { reject(complianceError); return; } } const configureMissingMethods = function () { const methodNotImplementedFactory = function (methodName) { return function () { const error = new Error(`Method ${methodName} is not implemented by the current driver`); const promise = utils_promise.reject(error); utils_executeCallback(promise, arguments[arguments.length - 1]); return promise; }; }; for (let i = 0, len = OptionalDriverMethods.length; i < len; i++) { const optionalDriverMethod = OptionalDriverMethods[i]; if (!driverObject[optionalDriverMethod]) { driverObject[optionalDriverMethod] = methodNotImplementedFactory(optionalDriverMethod); } } }; configureMissingMethods(); const setDriverSupport = function (support) { if (DefinedDrivers[driverName]) { console.info(`Redefining LocalForage driver: ${driverName}`); } DefinedDrivers[driverName] = driverObject; DriverSupport[driverName] = support; // don't use a then, so that we can define // drivers that have simple _support methods // in a blocking manner resolve(); }; if ('_support' in driverObject) { if (driverObject._support && typeof driverObject._support === 'function') { driverObject._support().then(setDriverSupport, reject); } else { setDriverSupport(!!driverObject._support); } } else { setDriverSupport(true); } } catch (e) { reject(e); } }); utils_executeTwoCallbacks(promise, callback, errorCallback); return promise; } driver() { return this._driver || null; } getDriver(driverName, callback, errorCallback) { const getDriverPromise = DefinedDrivers[driverName] ? utils_promise.resolve(DefinedDrivers[driverName]) : utils_promise.reject(new Error('Driver not found.')); utils_executeTwoCallbacks(getDriverPromise, callback, errorCallback); return getDriverPromise; } getSerializer(callback) { const serializerPromise = utils_promise.resolve(serializer); utils_executeTwoCallbacks(serializerPromise, callback); return serializerPromise; } ready(callback) { const self = this; const promise = self._driverSet.then(() => { if (self._ready === null) { self._ready = self._initDriver(); } return self._ready; }); utils_executeTwoCallbacks(promise, callback, callback); return promise; } setDriver(drivers, callback, errorCallback) { const self = this; if (!utils_isArray(drivers)) { drivers = [drivers]; } const supportedDrivers = this._getSupportedDrivers(drivers); function setDriverToConfig() { self._config.driver = self.driver(); } function extendSelfWithDriver(driver) { self._extend(driver); setDriverToConfig(); self._ready = self._initStorage(self._config); return self._ready; } function initDriver(supportedDrivers) { return function () { let currentDriverIndex = 0; function driverPromiseLoop() { while (currentDriverIndex < supportedDrivers.length) { let driverName = supportedDrivers[currentDriverIndex]; currentDriverIndex++; self._dbInfo = null; self._ready = null; return self.getDriver(driverName).then(extendSelfWithDriver).catch(driverPromiseLoop); } setDriverToConfig(); const error = new Error('No available storage method found.'); self._driverSet = utils_promise.reject(error); return self._driverSet; } return driverPromiseLoop(); }; } // There might be a driver initialization in progress // so wait for it to finish in order to avoid a possible // race condition to set _dbInfo const oldDriverSetDone = this._driverSet !== null ? this._driverSet.catch(() => utils_promise.resolve()) : utils_promise.resolve(); this._driverSet = oldDriverSetDone.then(() => { const driverName = supportedDrivers[0]; self._dbInfo = null; self._ready = null; return self.getDriver(driverName).then(driver => { self._driver = driver._driver; setDriverToConfig(); self._wrapLibraryMethodsWithReady(); self._initDriver = initDriver(supportedDrivers); }); }).catch(() => { setDriverToConfig(); const error = new Error('No available storage method found.'); self._driverSet = utils_promise.reject(error); return self._driverSet; }); utils_executeTwoCallbacks(this._driverSet, callback, errorCallback); return this._driverSet; } supports(driverName) { return !!DriverSupport[driverName]; } _extend(libraryMethodsAndProperties) { extend(this, libraryMethodsAndProperties); } _getSupportedDrivers(drivers) { const supportedDrivers = []; for (let i = 0, len = drivers.length; i < len; i++) { const driverName = drivers[i]; if (this.supports(driverName)) { supportedDrivers.push(driverName); } } return supportedDrivers; } _wrapLibraryMethodsWithReady() { // Add a stub for each driver API method that delays the call to the // corresponding driver method until localForage is ready. These stubs // will be replaced by the driver methods as soon as the driver is // loaded, so there is no performance impact. for (let i = 0, len = LibraryMethods.length; i < len; i++) { callWhenReady(this, LibraryMethods[i]); } } createInstance(options) { return new LocalForage(options); } } // The actual localForage object that we expose as a module or via a // global. It's extended by pulling in one of our other libraries. /* harmony default export */ const localforage = (new LocalForage()); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_assignMergeValue.js /** * This function is like `assignValue` except that it doesn't assign * `undefined` values. * * @private * @param {Object} object The object to modify. * @param {string} key The key of the property to assign. * @param {*} value The value to assign. */ function assignMergeValue(object, key, value) { if ((value !== undefined && !lodash_es_eq(object[key], value)) || (value === undefined && !(key in object))) { _baseAssignValue(object, key, value); } } /* harmony default export */ const _assignMergeValue = (assignMergeValue); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isArrayLikeObject.js /** * This method is like `_.isArrayLike` except that it also checks if `value` * is an object. * * @static * @memberOf _ * @since 4.0.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is an array-like object, * else `false`. * @example * * _.isArrayLikeObject([1, 2, 3]); * // => true * * _.isArrayLikeObject(document.body.children); * // => true * * _.isArrayLikeObject('abc'); * // => false * * _.isArrayLikeObject(_.noop); * // => false */ function isArrayLikeObject(value) { return lodash_es_isObjectLike(value) && lodash_es_isArrayLike(value); } /* harmony default export */ const lodash_es_isArrayLikeObject = (isArrayLikeObject); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_safeGet.js /** * Gets the value at `key`, unless `key` is "__proto__" or "constructor". * * @private * @param {Object} object The object to query. * @param {string} key The key of the property to get. * @returns {*} Returns the property value. */ function safeGet(object, key) { if (key === 'constructor' && typeof object[key] === 'function') { return; } if (key == '__proto__') { return; } return object[key]; } /* harmony default export */ const _safeGet = (safeGet); ;// CONCATENATED MODULE: ./node_modules/lodash-es/toPlainObject.js /** * Converts `value` to a plain object flattening inherited enumerable string * keyed properties of `value` to own properties of the plain object. * * @static * @memberOf _ * @since 3.0.0 * @category Lang * @param {*} value The value to convert. * @returns {Object} Returns the converted plain object. * @example * * function Foo() { * this.b = 2; * } * * Foo.prototype.c = 3; * * _.assign({ 'a': 1 }, new Foo); * // => { 'a': 1, 'b': 2 } * * _.assign({ 'a': 1 }, _.toPlainObject(new Foo)); * // => { 'a': 1, 'b': 2, 'c': 3 } */ function toPlainObject(value) { return _copyObject(value, lodash_es_keysIn(value)); } /* harmony default export */ const lodash_es_toPlainObject = (toPlainObject); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseMergeDeep.js /** * A specialized version of `baseMerge` for arrays and objects which performs * deep merges and tracks traversed objects enabling objects with circular * references to be merged. * * @private * @param {Object} object The destination object. * @param {Object} source The source object. * @param {string} key The key of the value to merge. * @param {number} srcIndex The index of `source`. * @param {Function} mergeFunc The function to merge values. * @param {Function} [customizer] The function to customize assigned values. * @param {Object} [stack] Tracks traversed source values and their merged * counterparts. */ function baseMergeDeep(object, source, key, srcIndex, mergeFunc, customizer, stack) { var objValue = _safeGet(object, key), srcValue = _safeGet(source, key), stacked = stack.get(srcValue); if (stacked) { _assignMergeValue(object, key, stacked); return; } var newValue = customizer ? customizer(objValue, srcValue, (key + ''), object, source, stack) : undefined; var isCommon = newValue === undefined; if (isCommon) { var isArr = lodash_es_isArray(srcValue), isBuff = !isArr && lodash_es_isBuffer(srcValue), isTyped = !isArr && !isBuff && lodash_es_isTypedArray(srcValue); newValue = srcValue; if (isArr || isBuff || isTyped) { if (lodash_es_isArray(objValue)) { newValue = objValue; } else if (lodash_es_isArrayLikeObject(objValue)) { newValue = _copyArray(objValue); } else if (isBuff) { isCommon = false; newValue = _cloneBuffer(srcValue, true); } else if (isTyped) { isCommon = false; newValue = _cloneTypedArray(srcValue, true); } else { newValue = []; } } else if (lodash_es_isPlainObject(srcValue) || lodash_es_isArguments(srcValue)) { newValue = objValue; if (lodash_es_isArguments(objValue)) { newValue = lodash_es_toPlainObject(objValue); } else if (!lodash_es_isObject(objValue) || lodash_es_isFunction(objValue)) { newValue = _initCloneObject(srcValue); } } else { isCommon = false; } } if (isCommon) { // Recursively merge objects and arrays (susceptible to call stack limits). stack.set(srcValue, newValue); mergeFunc(newValue, srcValue, srcIndex, customizer, stack); stack['delete'](srcValue); } _assignMergeValue(object, key, newValue); } /* harmony default export */ const _baseMergeDeep = (baseMergeDeep); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseMerge.js /** * The base implementation of `_.merge` without support for multiple sources. * * @private * @param {Object} object The destination object. * @param {Object} source The source object. * @param {number} srcIndex The index of `source`. * @param {Function} [customizer] The function to customize merged values. * @param {Object} [stack] Tracks traversed source values and their merged * counterparts. */ function baseMerge(object, source, srcIndex, customizer, stack) { if (object === source) { return; } _baseFor(source, function(srcValue, key) { stack || (stack = new _Stack); if (lodash_es_isObject(srcValue)) { _baseMergeDeep(object, source, key, srcIndex, baseMerge, customizer, stack); } else { var newValue = customizer ? customizer(_safeGet(object, key), srcValue, (key + ''), object, source, stack) : undefined; if (newValue === undefined) { newValue = srcValue; } _assignMergeValue(object, key, newValue); } }, lodash_es_keysIn); } /* harmony default export */ const _baseMerge = (baseMerge); ;// CONCATENATED MODULE: ./node_modules/lodash-es/merge.js /** * This method is like `_.assign` except that it recursively merges own and * inherited enumerable string keyed properties of source objects into the * destination object. Source properties that resolve to `undefined` are * skipped if a destination value exists. Array and plain object properties * are merged recursively. Other objects and value types are overridden by * assignment. Source objects are applied from left to right. Subsequent * sources overwrite property assignments of previous sources. * * **Note:** This method mutates `object`. * * @static * @memberOf _ * @since 0.5.0 * @category Object * @param {Object} object The destination object. * @param {...Object} [sources] The source objects. * @returns {Object} Returns `object`. * @example * * var object = { * 'a': [{ 'b': 2 }, { 'd': 4 }] * }; * * var other = { * 'a': [{ 'c': 3 }, { 'e': 5 }] * }; * * _.merge(object, other); * // => { 'a': [{ 'b': 2, 'c': 3 }, { 'd': 4, 'e': 5 }] } */ var merge = _createAssigner(function(object, source, srcIndex) { _baseMerge(object, source, srcIndex); }); /* harmony default export */ const lodash_es_merge = (merge); ;// CONCATENATED MODULE: ./node_modules/lodash-es/mergeWith.js /** * This method is like `_.merge` except that it accepts `customizer` which * is invoked to produce the merged values of the destination and source * properties. If `customizer` returns `undefined`, merging is handled by the * method instead. The `customizer` is invoked with six arguments: * (objValue, srcValue, key, object, source, stack). * * **Note:** This method mutates `object`. * * @static * @memberOf _ * @since 4.0.0 * @category Object * @param {Object} object The destination object. * @param {...Object} sources The source objects. * @param {Function} customizer The function to customize assigned values. * @returns {Object} Returns `object`. * @example * * function customizer(objValue, srcValue) { * if (_.isArray(objValue)) { * return objValue.concat(srcValue); * } * } * * var object = { 'a': [1], 'b': [2] }; * var other = { 'a': [3], 'b': [4] }; * * _.mergeWith(object, other, customizer); * // => { 'a': [1, 3], 'b': [2, 4] } */ var mergeWith = _createAssigner(function(object, source, srcIndex, customizer) { _baseMerge(object, source, srcIndex, customizer); }); /* harmony default export */ const lodash_es_mergeWith = (mergeWith); ;// CONCATENATED MODULE: ./node_modules/lodash-es/now.js /** * Gets the timestamp of the number of milliseconds that have elapsed since * the Unix epoch (1 January 1970 00:00:00 UTC). * * @static * @memberOf _ * @since 2.4.0 * @category Date * @returns {number} Returns the timestamp. * @example * * _.defer(function(stamp) { * console.log(_.now() - stamp); * }, _.now()); * // => Logs the number of milliseconds it took for the deferred invocation. */ var now = function() { return _root.Date.now(); }; /* harmony default export */ const lodash_es_now = (now); ;// CONCATENATED MODULE: ./node_modules/@converse/openpromise/openpromise.js function getOpenPromise() { const wrapper = { isResolved: false, isPending: true, isRejected: false }; const promise = new Promise((resolve, reject) => { wrapper.resolve = resolve; wrapper.reject = reject; }); Object.assign(promise, wrapper); promise.then(function (v) { promise.isResolved = true; promise.isPending = false; promise.isRejected = false; return v; }, function (e) { promise.isResolved = false; promise.isPending = false; promise.isRejected = true; throw e; }); return promise; } ;// CONCATENATED MODULE: ./node_modules/mergebounce/mergebounce.js /** Error message constants. */ const mergebounce_FUNC_ERROR_TEXT = 'Expected a function'; /* Built-in method references for those with the same name as other `lodash` methods. */ const mergebounce_nativeMax = Math.max; const nativeMin = Math.min; /** * Creates a debounced function that delays invoking `func` until after `wait` * milliseconds have elapsed since the last time the debounced function was * invoked. The debounced function comes with a `cancel` method to cancel * delayed `func` invocations and a `flush` method to immediately invoke them. * * This function differs from lodash's debounce by merging all passed objects * before passing them to the final invoked function. * * Because of this, invoking can only happen on the trailing edge, since * passed-in data would be discarded if invoking happened on the leading edge. * * If `wait` is `0`, `func` invocation is deferred until to the next tick, * similar to `setTimeout` with a timeout of `0`. * * @static * @category Function * @param {Function} func The function to mergebounce. * @param {number} [wait=0] The number of milliseconds to delay. * @param {Object} [options={}] The options object. * @param {number} [options.maxWait] * The maximum time `func` is allowed to be delayed before it's invoked. * @param {boolean} [options.concatArrays=false] * By default arrays will be treated as objects when being merged. When * merging two arrays, the values in the 2nd arrray will replace the * corresponding values (i.e. those with the same indexes) in the first array. * When `concatArrays` is set to `true`, arrays will be concatenated instead. * @param {boolean} [options.dedupeArrays=false] * This option is similar to `concatArrays`, except that the concatenated * array will also be deduplicated. Thus any entries that are concatenated to the * existing array, which are already contained in the existing array, will * first be removed. * @param {boolean} [options.promise=false] * By default, when calling a merge-debounced function that doesn't execute * immediately, you'll receive the result from its previous execution, or * `undefined` if it has never executed before. By setting the `promise` * option to `true`, a promise will be returned instead of the previous * execution result when the function is debounced. The promise will resolve * with the result of the next execution, as soon as it happens. * @returns {Function} Returns the new debounced function. * @example * * // Avoid costly calculations while the window size is in flux. * window.addEventListener('resize', mergebounce(calculateLayout, 150)); * * // Invoke `sendMail` when clicked, debouncing subsequent calls. * element.addEventListner('click', mergebounce(sendMail, 300)); * * // Ensure `batchLog` is invoked once after 1 second of debounced calls. * const mergebounced = mergebounce(batchLog, 250, { 'maxWait': 1000 }); * const source = new EventSource('/stream'); * jQuery(source).on('message', mergebounced); * * // Cancel the trailing debounced invocation. * window.addEventListener('popstate', mergebounced.cancel); */ function mergebounce(func, wait) { let options = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : {}; let lastArgs, lastThis, maxWait, result, timerId, lastCallTime, lastInvokeTime = 0, maxing = false; let promise = options.promise ? getOpenPromise() : null; if (typeof func != 'function') { throw new TypeError(mergebounce_FUNC_ERROR_TEXT); } wait = lodash_es_toNumber(wait) || 0; if (lodash_es_isObject(options)) { maxing = 'maxWait' in options; maxWait = maxing ? mergebounce_nativeMax(lodash_es_toNumber(options.maxWait) || 0, wait) : maxWait; } function invokeFunc(time) { const args = lastArgs; const thisArg = lastThis; const existingPromise = promise; lastArgs = lastThis = undefined; lastInvokeTime = time; result = func.apply(thisArg, args); if (options.promise) { existingPromise.resolve(result); promise = getOpenPromise(); } return options.promise ? existingPromise : result; } function leadingEdge(time) { // Reset any `maxWait` timer. lastInvokeTime = time; // Start the timer for the trailing edge. timerId = setTimeout(timerExpired, wait); return options.promise ? promise : result; } function remainingWait(time) { const timeSinceLastCall = time - lastCallTime; const timeSinceLastInvoke = time - lastInvokeTime; const timeWaiting = wait - timeSinceLastCall; return maxing ? nativeMin(timeWaiting, maxWait - timeSinceLastInvoke) : timeWaiting; } function shouldInvoke(time) { const timeSinceLastCall = time - lastCallTime; const timeSinceLastInvoke = time - lastInvokeTime; // Either this is the first call, activity has stopped and we're at the // trailing edge, the system time has gone backwards and we're treating // it as the trailing edge, or we've hit the `maxWait` limit. return lastCallTime === undefined || timeSinceLastCall >= wait || timeSinceLastCall < 0 || maxing && timeSinceLastInvoke >= maxWait; } function timerExpired() { const time = lodash_es_now(); if (shouldInvoke(time)) { return trailingEdge(time); } // Restart the timer. timerId = setTimeout(timerExpired, remainingWait(time)); } function trailingEdge(time) { timerId = undefined; // Only invoke if we have `lastArgs` which means `func` has been // debounced at least once. if (lastArgs) { return invokeFunc(time); } lastArgs = lastThis = undefined; return options.promise ? promise : result; } function cancel() { if (timerId !== undefined) { clearTimeout(timerId); } lastInvokeTime = 0; lastArgs = lastCallTime = lastThis = timerId = undefined; } function flush() { return timerId === undefined ? result : trailingEdge(lodash_es_now()); } function concatArrays(objValue, srcValue) { if (Array.isArray(objValue) && Array.isArray(srcValue)) { if (options !== null && options !== void 0 && options.dedupeArrays) { return objValue.concat(srcValue.filter(i => objValue.indexOf(i) === -1)); } else { return objValue.concat(srcValue); } } } function mergeArguments(args) { var _lastArgs; if ((_lastArgs = lastArgs) !== null && _lastArgs !== void 0 && _lastArgs.length) { if (!args.length) { return lastArgs; } if (options !== null && options !== void 0 && options.concatArrays || options !== null && options !== void 0 && options.dedupeArrays) { return lodash_es_mergeWith(lastArgs, args, concatArrays); } else { return lodash_es_merge(lastArgs, args); } } else { return args || []; } } function debounced() { const time = lodash_es_now(); const isInvoking = shouldInvoke(time); lastArgs = mergeArguments(Array.from(arguments)); lastThis = this; lastCallTime = time; if (isInvoking) { if (timerId === undefined) { return leadingEdge(lastCallTime); } if (maxing) { // Handle invocations in a tight loop. clearTimeout(timerId); timerId = setTimeout(timerExpired, wait); return invokeFunc(lastCallTime); } } if (timerId === undefined) { timerId = setTimeout(timerExpired, wait); } return options.promise ? promise : result; } debounced.cancel = cancel; debounced.flush = flush; return debounced; } /* harmony default export */ const mergebounce_mergebounce = (mergebounce); ;// CONCATENATED MODULE: ./node_modules/@converse/skeletor/src/drivers/sessionStorage.js // Copyright 2014 Mozilla // Copyright 2015 Thodoris Greasidis // Copyright 2018 JC Brand // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. const sessionStorage_serialize = serializer.serialize; const sessionStorage_deserialize = serializer.deserialize; function isSessionStorageValid() { // If the app is running inside a Google Chrome packaged webapp, or some // other context where sessionStorage isn't available, we don't use // sessionStorage. This feature detection is preferred over the old // `if (window.chrome && window.chrome.runtime)` code. // See: https://github.com/mozilla/localForage/issues/68 try { // If sessionStorage isn't available, we get outta here! // This should be inside a try catch if (sessionStorage && 'setItem' in sessionStorage) { return true; } } catch (e) { console.log(e); } return false; } function sessionStorage_getKeyPrefix(options, defaultConfig) { let keyPrefix = options.name + '/'; if (options.storeName !== defaultConfig.storeName) { keyPrefix += options.storeName + '/'; } return keyPrefix; } const dbInfo = { 'serializer': { 'serialize': sessionStorage_serialize, 'deserialize': sessionStorage_deserialize } }; function sessionStorage_initStorage(options) { dbInfo.keyPrefix = sessionStorage_getKeyPrefix(options, this._defaultConfig); if (options) { for (const i in options) { // eslint-disable-line guard-for-in dbInfo[i] = options[i]; } } } // Remove all keys from the datastore, effectively destroying all data in // the app's key/value store! function sessionStorage_clear(callback) { const promise = this.ready().then(function () { const keyPrefix = dbInfo.keyPrefix; for (let i = sessionStorage.length - 1; i >= 0; i--) { const key = sessionStorage.key(i); if (key.indexOf(keyPrefix) === 0) { sessionStorage.removeItem(key); } } }); utils_executeCallback(promise, callback); return promise; } // Retrieve an item from the store. Unlike the original async_storage // library in Gaia, we don't modify return values at all. If a key's value // is `undefined`, we pass that value to the callback function. function sessionStorage_getItem(key, callback) { key = normalizeKey(key); const promise = this.ready().then(function () { let result = sessionStorage.getItem(dbInfo.keyPrefix + key); // If a result was found, parse it from the serialized // string into a JS object. If result isn't truthy, the key // is likely undefined and we'll pass it straight to the // callback. if (result) { result = dbInfo.serializer.deserialize(result); } return result; }); utils_executeCallback(promise, callback); return promise; } // Iterate over all items in the store. function sessionStorage_iterate(iterator, callback) { const self = this; const promise = self.ready().then(function () { const keyPrefix = dbInfo.keyPrefix; const keyPrefixLength = keyPrefix.length; const length = sessionStorage.length; // We use a dedicated iterator instead of the `i` variable below // so other keys we fetch in sessionStorage aren't counted in // the `iterationNumber` argument passed to the `iterate()` // callback. // // See: github.com/mozilla/localForage/pull/435#discussion_r38061530 let iterationNumber = 1; for (let i = 0; i < length; i++) { const key = sessionStorage.key(i); if (key.indexOf(keyPrefix) !== 0) { continue; } let value = sessionStorage.getItem(key); // If a result was found, parse it from the serialized // string into a JS object. If result isn't truthy, the // key is likely undefined and we'll pass it straight // to the iterator. if (value) { value = dbInfo.serializer.deserialize(value); } value = iterator(value, key.substring(keyPrefixLength), iterationNumber++); if (value !== void 0) { // eslint-disable-line no-void return value; } } }); utils_executeCallback(promise, callback); return promise; } // Same as sessionStorage's key() method, except takes a callback. function sessionStorage_key(n, callback) { const self = this; const promise = self.ready().then(function () { let result; try { result = sessionStorage.key(n); } catch (error) { result = null; } // Remove the prefix from the key, if a key is found. if (result) { result = result.substring(dbInfo.keyPrefix.length); } return result; }); utils_executeCallback(promise, callback); return promise; } function sessionStorage_keys(callback) { const self = this; const promise = self.ready().then(function () { const length = sessionStorage.length; const keys = []; for (let i = 0; i < length; i++) { const itemKey = sessionStorage.key(i); if (itemKey.indexOf(dbInfo.keyPrefix) === 0) { keys.push(itemKey.substring(dbInfo.keyPrefix.length)); } } return keys; }); utils_executeCallback(promise, callback); return promise; } // Supply the number of keys in the datastore to the callback function. function sessionStorage_length(callback) { const self = this; const promise = self.keys().then(function (keys) { return keys.length; }); utils_executeCallback(promise, callback); return promise; } // Remove an item from the store, nice and simple. function sessionStorage_removeItem(key, callback) { key = normalizeKey(key); const promise = this.ready().then(function () { sessionStorage.removeItem(dbInfo.keyPrefix + key); }); utils_executeCallback(promise, callback); return promise; } // Set a key's value and run an optional callback once the value is set. // Unlike Gaia's implementation, the callback function is passed the value, // in case you want to operate on that value only after you're sure it // saved, or something like that. function sessionStorage_setItem(key, value, callback) { key = normalizeKey(key); const promise = this.ready().then(function () { // Convert undefined values to null. // https://github.com/mozilla/localForage/pull/42 if (value === undefined) { value = null; } // Save the original value to pass to the callback. const originalValue = value; return new Promise(function (resolve, reject) { dbInfo.serializer.serialize(value, function (value, error) { if (error) { reject(error); } else { try { sessionStorage.setItem(dbInfo.keyPrefix + key, value); resolve(originalValue); } catch (e) { // sessionStorage capacity exceeded. // TODO: Make this a specific error/event. if (e.name === 'QuotaExceededError' || e.name === 'NS_ERROR_DOM_QUOTA_REACHED') { reject(e); } reject(e); } } }); }); }); utils_executeCallback(promise, callback); return promise; } function sessionStorage_dropInstance(options, callback) { callback = getCallback.apply(this, arguments); options = typeof options !== 'function' && options || {}; if (!options.name) { const currentConfig = this.config(); options.name = options.name || currentConfig.name; options.storeName = options.storeName || currentConfig.storeName; } const self = this; let promise; if (!options.name) { promise = Promise.reject(new Error('Invalid arguments')); } else { promise = new Promise(function (resolve) { if (!options.storeName) { resolve(`${options.name}/`); } else { resolve(sessionStorage_getKeyPrefix(options, self._defaultConfig)); } }).then(function (keyPrefix) { for (let i = sessionStorage.length - 1; i >= 0; i--) { const key = sessionStorage.key(i); if (key.indexOf(keyPrefix) === 0) { sessionStorage.removeItem(key); } } }); } utils_executeCallback(promise, callback); return promise; } const sessionStorageWrapper = { _driver: 'sessionStorageWrapper', _initStorage: sessionStorage_initStorage, _support: isSessionStorageValid(), iterate: sessionStorage_iterate, getItem: sessionStorage_getItem, setItem: sessionStorage_setItem, removeItem: sessionStorage_removeItem, clear: sessionStorage_clear, length: sessionStorage_length, key: sessionStorage_key, keys: sessionStorage_keys, dropInstance: sessionStorage_dropInstance }; /* harmony default export */ const drivers_sessionStorage = (sessionStorageWrapper); // EXTERNAL MODULE: ./node_modules/localforage-setitems/dist/localforage-setitems.js var localforage_setitems = __webpack_require__(1459); // EXTERNAL MODULE: ./node_modules/localforage-getitems/dist/localforage-getitems.js var localforage_getitems = __webpack_require__(642); ;// CONCATENATED MODULE: ./node_modules/@converse/skeletor/src/storage.js /** * IndexedDB, localStorage and sessionStorage adapter */ const IN_MEMORY = umd._driver; localforage.defineDriver(umd); (0,localforage_setitems.extendPrototype)(localforage); (0,localforage_getitems.extendPrototype)(localforage); function S4() { // Generate four random hex digits. return ((1 + Math.random()) * 0x10000 | 0).toString(16).substring(1); } function guid() { // Generate a pseudo-GUID by concatenating random hexadecimal. return S4() + S4() + "-" + S4() + "-" + S4() + "-" + S4() + "-" + S4() + S4() + S4(); } class Storage { constructor(id, type) { let batchedWrites = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false; if (type === 'local' && !window.localStorage) { throw new Error("Skeletor.storage: Environment does not support localStorage."); } else if (type === 'session' && !window.sessionStorage) { throw new Error("Skeletor.storage: Environment does not support sessionStorage."); } if (lodash_es_isString(type)) { this.storeInitialized = this.initStore(type, batchedWrites); } else { this.store = type; if (batchedWrites) { this.store.debouncedSetItems = mergebounce_mergebounce(items => this.store.setItems(items), 50, { 'promise': true }); } this.storeInitialized = Promise.resolve(); } this.name = id; } async initStore(type, batchedWrites) { if (type === 'session') { localforage.setDriver(drivers_sessionStorage._driver); } else if (type === 'local') { await localforage.config({ 'driver': localforage.LOCALSTORAGE }); } else if (type === 'in_memory') { localforage.config({ 'driver': IN_MEMORY }); } else if (type !== 'indexed') { throw new Error("Skeletor.storage: No storage type was specified"); } this.store = localforage; if (batchedWrites) { this.store.debouncedSetItems = mergebounce_mergebounce(items => this.store.setItems(items), 50, { 'promise': true }); } } flush() { var _this$store$debounced; return (_this$store$debounced = this.store.debouncedSetItems) === null || _this$store$debounced === void 0 ? void 0 : _this$store$debounced.flush(); } async clear() { await this.store.removeItem(this.name).catch(e => console.error(e)); const re = new RegExp(`^${this.name}-`); const keys = await this.store.keys(); const removed_keys = keys.filter(k => re.test(k)); await Promise.all(removed_keys.map(k => this.store.removeItem(k).catch(e => console.error(e)))); } sync() { const that = this; async function localSync(method, model, options) { let resp, errorMessage, promise, new_attributes; // We get the collection (and if necessary the model attribute. // Waiting for storeInitialized will cause another iteration of // the event loop, after which the collection reference will // be removed from the model. const collection = model.collection; if (['patch', 'update'].includes(method)) { new_attributes = lodash_es_cloneDeep(model.attributes); } await that.storeInitialized; try { const original_attributes = model.attributes; switch (method) { case "read": if (model.id !== undefined) { resp = await that.find(model); } else { resp = await that.findAll(); } break; case "create": resp = await that.create(model, options); break; case 'patch': case "update": if (options.wait) { // When `wait` is set to true, Skeletor waits until // confirmation of storage before setting the values on // the model. // However, the new attributes needs to be sent, so it // sets them manually on the model and then removes // them after calling `sync`. // Because our `sync` method is asynchronous and we // wait for `storeInitialized`, the attributes are // already restored once we get here, so we need to do // the attributes dance again. model.attributes = new_attributes; } promise = that.update(model, options); if (options.wait) { model.attributes = original_attributes; } resp = await promise; break; case "delete": resp = await that.destroy(model, collection); break; } } catch (error) { if (error.code === 22 && that.getStorageSize() === 0) { errorMessage = "Private browsing is unsupported"; } else { errorMessage = error.message; } } if (resp) { if (options && options.success) { // When storing, we don't pass back the response (which is // the set attributes returned from localforage because // Skeletor sets them again on the model and due to the async // nature of localforage it can cause stale attributes to be // set on a model after it's been updated in the meantime. const data = method === "read" ? resp : null; options.success(data, options); } } else { errorMessage = errorMessage ? errorMessage : "Record Not Found"; if (options && options.error) { options.error(errorMessage); } } } localSync.__name__ = 'localSync'; return localSync; } removeCollectionReference(model, collection) { if (!collection) { return; } const ids = collection.filter(m => m.id !== model.id).map(m => this.getItemName(m.id)); return this.store.setItem(this.name, ids); } addCollectionReference(model, collection) { if (!collection) { return; } const ids = collection.map(m => this.getItemName(m.id)); const new_id = this.getItemName(model.id); if (!ids.includes(new_id)) { ids.push(new_id); } return this.store.setItem(this.name, ids); } getCollectionReferenceData(model) { if (!model.collection) { return {}; } const ids = model.collection.map(m => this.getItemName(m.id)); const new_id = this.getItemName(model.id); if (!ids.includes(new_id)) { ids.push(new_id); } const result = {}; result[this.name] = ids; return result; } async save(model) { if (this.store.setItems) { const items = {}; items[this.getItemName(model.id)] = model.toJSON(); Object.assign(items, this.getCollectionReferenceData(model)); return this.store.debouncedSetItems ? this.store.debouncedSetItems(items) : this.store.setItems(items); } else { const key = this.getItemName(model.id); const data = await this.store.setItem(key, model.toJSON()); await this.addCollectionReference(model, model.collection); return data; } } create(model, options) { /* Add a model, giving it a (hopefully)-unique GUID, if it doesn't already * have an id of it's own. */ if (!model.id) { model.id = guid(); model.set(model.idAttribute, model.id, options); } return this.save(model); } update(model) { return this.save(model); } find(model) { return this.store.getItem(this.getItemName(model.id)); } async findAll() { /* Return the array of all models currently in storage. */ const keys = await this.store.getItem(this.name); if (keys !== null && keys !== void 0 && keys.length) { const items = await this.store.getItems(keys); return Object.values(items); } return []; } async destroy(model, collection) { await this.flush(); await this.store.removeItem(this.getItemName(model.id)); await this.removeCollectionReference(model, collection); return model; } getStorageSize() { return this.store.length; } getItemName(id) { return this.name + "-" + id; } } Storage.sessionStorageInitialized = localforage.defineDriver(drivers_sessionStorage); Storage.localForage = localforage; /* harmony default export */ const storage = (Storage); ;// CONCATENATED MODULE: ./src/headless/utils/storage.js function getDefaultStore() { if (shared_converse.config.get('trusted')) { const is_non_persistent = core_api.settings.get('persistent_store') === 'sessionStorage'; return is_non_persistent ? 'session' : 'persistent'; } else { return 'session'; } } function storeUsesIndexedDB(store) { return store === 'persistent' && core_api.settings.get('persistent_store') === 'IndexedDB'; } function createStore(id, store) { const name = store || getDefaultStore(); const s = shared_converse.storage[name]; if (typeof s === 'undefined') { throw new TypeError(`createStore: Could not find store for ${id}`); } return new storage(id, s, storeUsesIndexedDB(store)); } function initStorage(model, id, type) { const store = type || getDefaultStore(); model.browserStorage = createStore(id, store); if (storeUsesIndexedDB(store)) { const flush = () => model.browserStorage.flush(); window.addEventListener(shared_converse.unloadevent, flush); model.on('destroy', () => window.removeEventListener(shared_converse.unloadevent, flush)); model.listenTo(shared_converse, 'beforeLogout', flush); } } ;// CONCATENATED MODULE: ./node_modules/lodash-es/isEqual.js /** * Performs a deep comparison between two values to determine if they are * equivalent. * * **Note:** This method supports comparing arrays, array buffers, booleans, * date objects, error objects, maps, numbers, `Object` objects, regexes, * sets, strings, symbols, and typed arrays. `Object` objects are compared * by their own, not inherited, enumerable properties. Functions and DOM * nodes are compared by strict equality, i.e. `===`. * * @static * @memberOf _ * @since 0.1.0 * @category Lang * @param {*} value The value to compare. * @param {*} other The other value to compare. * @returns {boolean} Returns `true` if the values are equivalent, else `false`. * @example * * var object = { 'a': 1 }; * var other = { 'a': 1 }; * * _.isEqual(object, other); * // => true * * object === other; * // => false */ function isEqual(value, other) { return _baseIsEqual(value, other); } /* harmony default export */ const lodash_es_isEqual = (isEqual); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseSet.js /** * The base implementation of `_.set`. * * @private * @param {Object} object The object to modify. * @param {Array|string} path The path of the property to set. * @param {*} value The value to set. * @param {Function} [customizer] The function to customize path creation. * @returns {Object} Returns `object`. */ function baseSet(object, path, value, customizer) { if (!lodash_es_isObject(object)) { return object; } path = _castPath(path, object); var index = -1, length = path.length, lastIndex = length - 1, nested = object; while (nested != null && ++index < length) { var key = _toKey(path[index]), newValue = value; if (key === '__proto__' || key === 'constructor' || key === 'prototype') { return object; } if (index != lastIndex) { var objValue = nested[key]; newValue = customizer ? customizer(objValue, key, nested) : undefined; if (newValue === undefined) { newValue = lodash_es_isObject(objValue) ? objValue : (_isIndex(path[index + 1]) ? [] : {}); } } _assignValue(nested, key, newValue); nested = nested[key]; } return object; } /* harmony default export */ const _baseSet = (baseSet); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_basePickBy.js /** * The base implementation of `_.pickBy` without support for iteratee shorthands. * * @private * @param {Object} object The source object. * @param {string[]} paths The property paths to pick. * @param {Function} predicate The function invoked per property. * @returns {Object} Returns the new object. */ function basePickBy(object, paths, predicate) { var index = -1, length = paths.length, result = {}; while (++index < length) { var path = paths[index], value = _baseGet(object, path); if (predicate(value, path)) { _baseSet(result, _castPath(path, object), value); } } return result; } /* harmony default export */ const _basePickBy = (basePickBy); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_basePick.js /** * The base implementation of `_.pick` without support for individual * property identifiers. * * @private * @param {Object} object The source object. * @param {string[]} paths The property paths to pick. * @returns {Object} Returns the new object. */ function basePick(object, paths) { return _basePickBy(object, paths, function(value, path) { return lodash_es_hasIn(object, path); }); } /* harmony default export */ const _basePick = (basePick); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_isFlattenable.js /** Built-in value references. */ var spreadableSymbol = _Symbol ? _Symbol.isConcatSpreadable : undefined; /** * Checks if `value` is a flattenable `arguments` object or array. * * @private * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is flattenable, else `false`. */ function isFlattenable(value) { return lodash_es_isArray(value) || lodash_es_isArguments(value) || !!(spreadableSymbol && value && value[spreadableSymbol]); } /* harmony default export */ const _isFlattenable = (isFlattenable); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseFlatten.js /** * The base implementation of `_.flatten` with support for restricting flattening. * * @private * @param {Array} array The array to flatten. * @param {number} depth The maximum recursion depth. * @param {boolean} [predicate=isFlattenable] The function invoked per iteration. * @param {boolean} [isStrict] Restrict to values that pass `predicate` checks. * @param {Array} [result=[]] The initial result value. * @returns {Array} Returns the new flattened array. */ function baseFlatten(array, depth, predicate, isStrict, result) { var index = -1, length = array.length; predicate || (predicate = _isFlattenable); result || (result = []); while (++index < length) { var value = array[index]; if (depth > 0 && predicate(value)) { if (depth > 1) { // Recursively flatten arrays (susceptible to call stack limits). baseFlatten(value, depth - 1, predicate, isStrict, result); } else { _arrayPush(result, value); } } else if (!isStrict) { result[result.length] = value; } } return result; } /* harmony default export */ const _baseFlatten = (baseFlatten); ;// CONCATENATED MODULE: ./node_modules/lodash-es/flatten.js /** * Flattens `array` a single level deep. * * @static * @memberOf _ * @since 0.1.0 * @category Array * @param {Array} array The array to flatten. * @returns {Array} Returns the new flattened array. * @example * * _.flatten([1, [2, [3, [4]], 5]]); * // => [1, 2, [3, [4]], 5] */ function flatten(array) { var length = array == null ? 0 : array.length; return length ? _baseFlatten(array, 1) : []; } /* harmony default export */ const lodash_es_flatten = (flatten); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_flatRest.js /** * A specialized version of `baseRest` which flattens the rest array. * * @private * @param {Function} func The function to apply a rest parameter to. * @returns {Function} Returns the new function. */ function flatRest(func) { return _setToString(_overRest(func, undefined, lodash_es_flatten), func + ''); } /* harmony default export */ const _flatRest = (flatRest); ;// CONCATENATED MODULE: ./node_modules/lodash-es/pick.js /** * Creates an object composed of the picked `object` properties. * * @static * @since 0.1.0 * @memberOf _ * @category Object * @param {Object} object The source object. * @param {...(string|string[])} [paths] The property paths to pick. * @returns {Object} Returns the new object. * @example * * var object = { 'a': 1, 'b': '2', 'c': 3 }; * * _.pick(object, ['a', 'c']); * // => { 'a': 1, 'c': 3 } */ var pick = _flatRest(function(object, paths) { return object == null ? {} : _basePick(object, paths); }); /* harmony default export */ const lodash_es_pick = (pick); // EXTERNAL MODULE: ./node_modules/dompurify/dist/purify.js var purify = __webpack_require__(7856); var purify_default = /*#__PURE__*/__webpack_require__.n(purify); ;// CONCATENATED MODULE: ./node_modules/lodash-es/compact.js /** * Creates an array with all falsey values removed. The values `false`, `null`, * `0`, `""`, `undefined`, and `NaN` are falsey. * * @static * @memberOf _ * @since 0.1.0 * @category Array * @param {Array} array The array to compact. * @returns {Array} Returns the new array of filtered values. * @example * * _.compact([0, 1, false, 2, '', 3]); * // => [1, 2, 3] */ function compact(array) { var index = -1, length = array == null ? 0 : array.length, resIndex = 0, result = []; while (++index < length) { var value = array[index]; if (value) { result[resIndex++] = value; } } return result; } /* harmony default export */ const lodash_es_compact = (compact); ;// CONCATENATED MODULE: ./node_modules/lodash-es/last.js /** * Gets the last element of `array`. * * @static * @memberOf _ * @since 0.1.0 * @category Array * @param {Array} array The array to query. * @returns {*} Returns the last element of `array`. * @example * * _.last([1, 2, 3]); * // => 3 */ function last(array) { var length = array == null ? 0 : array.length; return length ? array[length - 1] : undefined; } /* harmony default export */ const lodash_es_last = (last); // EXTERNAL MODULE: ./node_modules/sizzle/dist/sizzle.js var sizzle = __webpack_require__(1271); var sizzle_default = /*#__PURE__*/__webpack_require__.n(sizzle); ;// CONCATENATED MODULE: ./node_modules/lodash-es/clone.js /** Used to compose bitmasks for cloning. */ var clone_CLONE_SYMBOLS_FLAG = 4; /** * Creates a shallow clone of `value`. * * **Note:** This method is loosely based on the * [structured clone algorithm](https://mdn.io/Structured_clone_algorithm) * and supports cloning arrays, array buffers, booleans, date objects, maps, * numbers, `Object` objects, regexes, sets, strings, symbols, and typed * arrays. The own enumerable properties of `arguments` objects are cloned * as plain objects. An empty object is returned for uncloneable values such * as error objects, functions, DOM nodes, and WeakMaps. * * @static * @memberOf _ * @since 0.1.0 * @category Lang * @param {*} value The value to clone. * @returns {*} Returns the cloned value. * @see _.cloneDeep * @example * * var objects = [{ 'a': 1 }, { 'b': 2 }]; * * var shallow = _.clone(objects); * console.log(shallow[0] === objects[0]); * // => true */ function clone(value) { return _baseClone(value, clone_CLONE_SYMBOLS_FLAG); } /* harmony default export */ const lodash_es_clone = (clone); ;// CONCATENATED MODULE: ./node_modules/lodash-es/defaults.js /** Used for built-in method references. */ var defaults_objectProto = Object.prototype; /** Used to check objects for own properties. */ var defaults_hasOwnProperty = defaults_objectProto.hasOwnProperty; /** * Assigns own and inherited enumerable string keyed properties of source * objects to the destination object for all destination properties that * resolve to `undefined`. Source objects are applied from left to right. * Once a property is set, additional values of the same property are ignored. * * **Note:** This method mutates `object`. * * @static * @since 0.1.0 * @memberOf _ * @category Object * @param {Object} object The destination object. * @param {...Object} [sources] The source objects. * @returns {Object} Returns `object`. * @see _.defaultsDeep * @example * * _.defaults({ 'a': 1 }, { 'b': 2 }, { 'a': 3 }); * // => { 'a': 1, 'b': 2 } */ var defaults = _baseRest(function(object, sources) { object = Object(object); var index = -1; var length = sources.length; var guard = length > 2 ? sources[2] : undefined; if (guard && _isIterateeCall(sources[0], sources[1], guard)) { length = 1; } while (++index < length) { var source = sources[index]; var props = lodash_es_keysIn(source); var propsIndex = -1; var propsLength = props.length; while (++propsIndex < propsLength) { var key = props[propsIndex]; var value = object[key]; if (value === undefined || (lodash_es_eq(value, defaults_objectProto[key]) && !defaults_hasOwnProperty.call(object, key))) { object[key] = source[key]; } } } return object; }); /* harmony default export */ const lodash_es_defaults = (defaults); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseDelay.js /** Error message constants. */ var _baseDelay_FUNC_ERROR_TEXT = 'Expected a function'; /** * The base implementation of `_.delay` and `_.defer` which accepts `args` * to provide to `func`. * * @private * @param {Function} func The function to delay. * @param {number} wait The number of milliseconds to delay invocation. * @param {Array} args The arguments to provide to `func`. * @returns {number|Object} Returns the timer id or timeout object. */ function baseDelay(func, wait, args) { if (typeof func != 'function') { throw new TypeError(_baseDelay_FUNC_ERROR_TEXT); } return setTimeout(function() { func.apply(undefined, args); }, wait); } /* harmony default export */ const _baseDelay = (baseDelay); ;// CONCATENATED MODULE: ./node_modules/lodash-es/defer.js /** * Defers invoking the `func` until the current call stack has cleared. Any * additional arguments are provided to `func` when it's invoked. * * @static * @memberOf _ * @since 0.1.0 * @category Function * @param {Function} func The function to defer. * @param {...*} [args] The arguments to invoke `func` with. * @returns {number} Returns the timer id. * @example * * _.defer(function(text) { * console.log(text); * }, 'deferred'); * // => Logs 'deferred' after one millisecond. */ var defer = _baseRest(function(func, args) { return _baseDelay(func, 1, args); }); /* harmony default export */ const lodash_es_defer = (defer); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_basePropertyOf.js /** * The base implementation of `_.propertyOf` without support for deep paths. * * @private * @param {Object} object The object to query. * @returns {Function} Returns the new accessor function. */ function basePropertyOf(object) { return function(key) { return object == null ? undefined : object[key]; }; } /* harmony default export */ const _basePropertyOf = (basePropertyOf); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_escapeHtmlChar.js /** Used to map characters to HTML entities. */ var htmlEscapes = { '&': '&', '<': '<', '>': '>', '"': '"', "'": ''' }; /** * Used by `_.escape` to convert characters to HTML entities. * * @private * @param {string} chr The matched character to escape. * @returns {string} Returns the escaped character. */ var escapeHtmlChar = _basePropertyOf(htmlEscapes); /* harmony default export */ const _escapeHtmlChar = (escapeHtmlChar); ;// CONCATENATED MODULE: ./node_modules/lodash-es/escape.js /** Used to match HTML entities and HTML characters. */ var reUnescapedHtml = /[&<>"']/g, reHasUnescapedHtml = RegExp(reUnescapedHtml.source); /** * Converts the characters "&", "<", ">", '"', and "'" in `string` to their * corresponding HTML entities. * * **Note:** No other characters are escaped. To escape additional * characters use a third-party library like [_he_](https://mths.be/he). * * Though the ">" character is escaped for symmetry, characters like * ">" and "/" don't need escaping in HTML and have no special meaning * unless they're part of a tag or unquoted attribute value. See * [Mathias Bynens's article](https://mathiasbynens.be/notes/ambiguous-ampersands) * (under "semi-related fun fact") for more details. * * When working with HTML you should always * [quote attribute values](http://wonko.com/post/html-escaping) to reduce * XSS vectors. * * @static * @since 0.1.0 * @memberOf _ * @category String * @param {string} [string=''] The string to escape. * @returns {string} Returns the escaped string. * @example * * _.escape('fred, barney, & pebbles'); * // => 'fred, barney, & pebbles' */ function escape_escape(string) { string = lodash_es_toString(string); return (string && reHasUnescapedHtml.test(string)) ? string.replace(reUnescapedHtml, _escapeHtmlChar) : string; } /* harmony default export */ const lodash_es_escape = (escape_escape); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseInverter.js /** * The base implementation of `_.invert` and `_.invertBy` which inverts * `object` with values transformed by `iteratee` and set by `setter`. * * @private * @param {Object} object The object to iterate over. * @param {Function} setter The function to set `accumulator` values. * @param {Function} iteratee The iteratee to transform values. * @param {Object} accumulator The initial inverted object. * @returns {Function} Returns `accumulator`. */ function baseInverter(object, setter, iteratee, accumulator) { _baseForOwn(object, function(value, key, object) { setter(accumulator, iteratee(value), key, object); }); return accumulator; } /* harmony default export */ const _baseInverter = (baseInverter); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_createInverter.js /** * Creates a function like `_.invertBy`. * * @private * @param {Function} setter The function to set accumulator values. * @param {Function} toIteratee The function to resolve iteratees. * @returns {Function} Returns the new inverter function. */ function createInverter(setter, toIteratee) { return function(object, iteratee) { return _baseInverter(object, setter, toIteratee(iteratee), {}); }; } /* harmony default export */ const _createInverter = (createInverter); ;// CONCATENATED MODULE: ./node_modules/lodash-es/invert.js /** Used for built-in method references. */ var invert_objectProto = Object.prototype; /** * Used to resolve the * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring) * of values. */ var invert_nativeObjectToString = invert_objectProto.toString; /** * Creates an object composed of the inverted keys and values of `object`. * If `object` contains duplicate values, subsequent values overwrite * property assignments of previous values. * * @static * @memberOf _ * @since 0.7.0 * @category Object * @param {Object} object The object to invert. * @returns {Object} Returns the new inverted object. * @example * * var object = { 'a': 1, 'b': 2, 'c': 1 }; * * _.invert(object); * // => { '1': 'c', '2': 'b' } */ var invert = _createInverter(function(result, value, key) { if (value != null && typeof value.toString != 'function') { value = invert_nativeObjectToString.call(value); } result[value] = key; }, lodash_es_constant(lodash_es_identity)); /* harmony default export */ const lodash_es_invert = (invert); ;// CONCATENATED MODULE: ./node_modules/lodash-es/iteratee.js /** Used to compose bitmasks for cloning. */ var iteratee_CLONE_DEEP_FLAG = 1; /** * Creates a function that invokes `func` with the arguments of the created * function. If `func` is a property name, the created function returns the * property value for a given element. If `func` is an array or object, the * created function returns `true` for elements that contain the equivalent * source properties, otherwise it returns `false`. * * @static * @since 4.0.0 * @memberOf _ * @category Util * @param {*} [func=_.identity] The value to convert to a callback. * @returns {Function} Returns the callback. * @example * * var users = [ * { 'user': 'barney', 'age': 36, 'active': true }, * { 'user': 'fred', 'age': 40, 'active': false } * ]; * * // The `_.matches` iteratee shorthand. * _.filter(users, _.iteratee({ 'user': 'barney', 'active': true })); * // => [{ 'user': 'barney', 'age': 36, 'active': true }] * * // The `_.matchesProperty` iteratee shorthand. * _.filter(users, _.iteratee(['user', 'fred'])); * // => [{ 'user': 'fred', 'age': 40 }] * * // The `_.property` iteratee shorthand. * _.map(users, _.iteratee('user')); * // => ['barney', 'fred'] * * // Create custom iteratee shorthands. * _.iteratee = _.wrap(_.iteratee, function(iteratee, func) { * return !_.isRegExp(func) ? iteratee(func) : function(string) { * return func.test(string); * }; * }); * * _.filter(['abc', 'def'], /ef/); * // => ['def'] */ function iteratee(func) { return _baseIteratee(typeof func == 'function' ? func : _baseClone(func, iteratee_CLONE_DEEP_FLAG)); } /* harmony default export */ const lodash_es_iteratee = (iteratee); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseSlice.js /** * The base implementation of `_.slice` without an iteratee call guard. * * @private * @param {Array} array The array to slice. * @param {number} [start=0] The start position. * @param {number} [end=array.length] The end position. * @returns {Array} Returns the slice of `array`. */ function baseSlice(array, start, end) { var index = -1, length = array.length; if (start < 0) { start = -start > length ? 0 : (length + start); } end = end > length ? length : end; if (end < 0) { end += length; } length = start > end ? 0 : ((end - start) >>> 0); start >>>= 0; var result = Array(length); while (++index < length) { result[index] = array[index + start]; } return result; } /* harmony default export */ const _baseSlice = (baseSlice); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_parent.js /** * Gets the parent value at `path` of `object`. * * @private * @param {Object} object The object to query. * @param {Array} path The path to get the parent value of. * @returns {*} Returns the parent value. */ function _parent_parent(object, path) { return path.length < 2 ? object : _baseGet(object, _baseSlice(path, 0, -1)); } /* harmony default export */ const _parent = (_parent_parent); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseUnset.js /** * The base implementation of `_.unset`. * * @private * @param {Object} object The object to modify. * @param {Array|string} path The property path to unset. * @returns {boolean} Returns `true` if the property is deleted, else `false`. */ function baseUnset(object, path) { path = _castPath(path, object); object = _parent(object, path); return object == null || delete object[_toKey(lodash_es_last(path))]; } /* harmony default export */ const _baseUnset = (baseUnset); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_customOmitClone.js /** * Used by `_.omit` to customize its `_.cloneDeep` use to only clone plain * objects. * * @private * @param {*} value The value to inspect. * @param {string} key The key of the property to inspect. * @returns {*} Returns the uncloned value or `undefined` to defer cloning to `_.cloneDeep`. */ function customOmitClone(value) { return lodash_es_isPlainObject(value) ? undefined : value; } /* harmony default export */ const _customOmitClone = (customOmitClone); ;// CONCATENATED MODULE: ./node_modules/lodash-es/omit.js /** Used to compose bitmasks for cloning. */ var omit_CLONE_DEEP_FLAG = 1, omit_CLONE_FLAT_FLAG = 2, omit_CLONE_SYMBOLS_FLAG = 4; /** * The opposite of `_.pick`; this method creates an object composed of the * own and inherited enumerable property paths of `object` that are not omitted. * * **Note:** This method is considerably slower than `_.pick`. * * @static * @since 0.1.0 * @memberOf _ * @category Object * @param {Object} object The source object. * @param {...(string|string[])} [paths] The property paths to omit. * @returns {Object} Returns the new object. * @example * * var object = { 'a': 1, 'b': '2', 'c': 3 }; * * _.omit(object, ['a', 'c']); * // => { 'b': '2' } */ var omit = _flatRest(function(object, paths) { var result = {}; if (object == null) { return result; } var isDeep = false; paths = _arrayMap(paths, function(path) { path = _castPath(path, object); isDeep || (isDeep = path.length > 1); return path; }); _copyObject(object, _getAllKeysIn(object), result); if (isDeep) { result = _baseClone(result, omit_CLONE_DEEP_FLAG | omit_CLONE_FLAT_FLAG | omit_CLONE_SYMBOLS_FLAG, _customOmitClone); } var length = paths.length; while (length--) { _baseUnset(result, paths[length]); } return result; }); /* harmony default export */ const lodash_es_omit = (omit); ;// CONCATENATED MODULE: ./node_modules/@converse/skeletor/src/model.js // Backbone.js 1.4.0 // (c) 2010-2019 Jeremy Ashkenas and DocumentCloud // Backbone may be freely distributed under the MIT license. // Model // ----- // **Models** are the basic data object in the framework -- // frequently representing a row in a table in a database on your server. // A discrete chunk of data and a bunch of useful, related methods for // performing computations and transformations on that data. // Create a new model with the specified attributes. A client id (`cid`) // is automatically generated and assigned for you. const Model = function (attributes, options) { let attrs = attributes || {}; options || (options = {}); this.preinitialize.apply(this, arguments); this.cid = lodash_es_uniqueId(this.cidPrefix); this.attributes = {}; if (options.collection) this.collection = options.collection; if (options.parse) attrs = this.parse(attrs, options) || {}; const default_attrs = lodash_es_result(this, 'defaults'); attrs = lodash_es_defaults(lodash_es_assignIn({}, default_attrs, attrs), default_attrs); this.set(attrs, options); this.changed = {}; this.initialize.apply(this, arguments); }; Model.extend = inherits; // Attach all inheritable methods to the Model prototype. Object.assign(Model.prototype, Events, { // A hash of attributes whose current and previous value differ. changed: null, // The value returned during the last failed validation. validationError: null, // The default name for the JSON `id` attribute is `"id"`. MongoDB and // CouchDB users may want to set this to `"_id"`. idAttribute: 'id', // The prefix is used to create the client id which is used to identify models locally. // You may want to override this if you're experiencing name clashes with model ids. cidPrefix: 'c', // preinitialize is an empty function by default. You can override it with a function // or object. preinitialize will run before any instantiation logic is run in the Model. preinitialize: function () {}, // Initialize is an empty function by default. Override it with your own // initialization logic. initialize: function () {}, // Return a copy of the model's `attributes` object. toJSON: function (options) { return lodash_es_clone(this.attributes); }, // Proxy `Backbone.sync` by default -- but override this if you need // custom syncing semantics for *this* particular model. sync: function (method, model, options) { return getSyncMethod(this)(method, model, options); }, // Get the value of an attribute. get: function (attr) { return this.attributes[attr]; }, keys: function () { return Object.keys(this.attributes); }, values: function () { return Object.values(this.attributes); }, pairs: function () { return this.entries(); }, entries: function () { return Object.entries(this.attributes); }, invert: function () { return lodash_es_invert(this.attributes); }, pick: function () { for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } if (args.length === 1 && Array.isArray(args[0])) { args = args[0]; } return lodash_es_pick(this.attributes, args); }, omit: function () { for (var _len2 = arguments.length, args = new Array(_len2), _key2 = 0; _key2 < _len2; _key2++) { args[_key2] = arguments[_key2]; } if (args.length === 1 && Array.isArray(args[0])) { args = args[0]; } return lodash_es_omit(this.attributes, args); }, isEmpty: function () { return lodash_es_isEmpty(this.attributes); }, // Get the HTML-escaped value of an attribute. escape: function (attr) { return lodash_es_escape(this.get(attr)); }, // Returns `true` if the attribute contains a value that is not null // or undefined. has: function (attr) { return this.get(attr) != null; }, // Special-cased proxy to lodash's `matches` method. matches: function (attrs) { return !!lodash_es_iteratee(attrs, this)(this.attributes); }, // Set a hash of model attributes on the object, firing `"change"`. This is // the core primitive operation of a model, updating the data and notifying // anyone who needs to know about the change in state. The heart of the beast. set: function (key, val, options) { if (key == null) return this; // Handle both `"key", value` and `{key: value}` -style arguments. let attrs; if (typeof key === 'object') { attrs = key; options = val; } else { (attrs = {})[key] = val; } options || (options = {}); // Run validation. if (!this._validate(attrs, options)) return false; // Extract attributes and options. const unset = options.unset; const silent = options.silent; const changes = []; const changing = this._changing; this._changing = true; if (!changing) { this._previousAttributes = lodash_es_clone(this.attributes); this.changed = {}; } const current = this.attributes; const changed = this.changed; const prev = this._previousAttributes; // For each `set` attribute, update or delete the current value. for (const attr in attrs) { val = attrs[attr]; if (!lodash_es_isEqual(current[attr], val)) changes.push(attr); if (!lodash_es_isEqual(prev[attr], val)) { changed[attr] = val; } else { delete changed[attr]; } unset ? delete current[attr] : current[attr] = val; } // Update the `id`. if (this.idAttribute in attrs) this.id = this.get(this.idAttribute); // Trigger all relevant attribute changes. if (!silent) { if (changes.length) this._pending = options; for (let i = 0; i < changes.length; i++) { this.trigger('change:' + changes[i], this, current[changes[i]], options); } } // You might be wondering why there's a `while` loop here. Changes can // be recursively nested within `"change"` events. if (changing) return this; if (!silent) { while (this._pending) { options = this._pending; this._pending = false; this.trigger('change', this, options); } } this._pending = false; this._changing = false; return this; }, // Remove an attribute from the model, firing `"change"`. `unset` is a noop // if the attribute doesn't exist. unset: function (attr, options) { return this.set(attr, undefined, lodash_es_assignIn({}, options, { unset: true })); }, // Clear all attributes on the model, firing `"change"`. clear: function (options) { const attrs = {}; for (const key in this.attributes) attrs[key] = undefined; return this.set(attrs, lodash_es_assignIn({}, options, { unset: true })); }, // Determine if the model has changed since the last `"change"` event. // If you specify an attribute name, determine if that attribute has changed. hasChanged: function (attr) { if (attr == null) return !lodash_es_isEmpty(this.changed); return lodash_es_has(this.changed, attr); }, // Return an object containing all the attributes that have changed, or // false if there are no changed attributes. Useful for determining what // parts of a view need to be updated and/or what attributes need to be // persisted to the server. Unset attributes will be set to undefined. // You can also pass an attributes object to diff against the model, // determining if there *would be* a change. changedAttributes: function (diff) { if (!diff) return this.hasChanged() ? lodash_es_clone(this.changed) : false; const old = this._changing ? this._previousAttributes : this.attributes; const changed = {}; let hasChanged; for (const attr in diff) { const val = diff[attr]; if (lodash_es_isEqual(old[attr], val)) continue; changed[attr] = val; hasChanged = true; } return hasChanged ? changed : false; }, // Get the previous value of an attribute, recorded at the time the last // `"change"` event was fired. previous: function (attr) { if (attr == null || !this._previousAttributes) return null; return this._previousAttributes[attr]; }, // Get all of the attributes of the model at the time of the previous // `"change"` event. previousAttributes: function () { return lodash_es_clone(this._previousAttributes); }, // Fetch the model from the server, merging the response with the model's // local attributes. Any changed attributes will trigger a "change" event. fetch: function (options) { options = lodash_es_assignIn({ parse: true }, options); const model = this; const success = options.success; options.success = function (resp) { const serverAttrs = options.parse ? model.parse(resp, options) : resp; if (!model.set(serverAttrs, options)) return false; if (success) success.call(options.context, model, resp, options); model.trigger('sync', model, resp, options); }; wrapError(this, options); return this.sync('read', this, options); }, // Set a hash of model attributes, and sync the model to the server. // If the server returns an attributes hash that differs, the model's // state will be `set` again. save: function (key, val, options) { // Handle both `"key", value` and `{key: value}` -style arguments. let attrs; if (key == null || typeof key === 'object') { attrs = key; options = val; } else { (attrs = {})[key] = val; } options = lodash_es_assignIn({ validate: true, parse: true }, options); const wait = options.wait; const return_promise = options.promise; const promise = return_promise && getResolveablePromise(); // If we're not waiting and attributes exist, save acts as // `set(attr).save(null, opts)` with validation. Otherwise, check if // the model will be valid when the attributes, if any, are set. if (attrs && !wait) { if (!this.set(attrs, options)) return false; } else if (!this._validate(attrs, options)) { return false; } // After a successful server-side save, the client is (optionally) // updated with the server-side state. const model = this; const success = options.success; const error = options.error; const attributes = this.attributes; options.success = function (resp) { // Ensure attributes are restored during synchronous saves. model.attributes = attributes; let serverAttrs = options.parse ? model.parse(resp, options) : resp; if (wait) serverAttrs = lodash_es_assignIn({}, attrs, serverAttrs); if (serverAttrs && !model.set(serverAttrs, options)) return false; if (success) success.call(options.context, model, resp, options); model.trigger('sync', model, resp, options); return_promise && promise.resolve(); }; options.error = function (model, e, options) { error && error.call(options.context, model, e, options); return_promise && promise.reject(e); }; wrapError(this, options); // Set temporary attributes if `{wait: true}` to properly find new ids. if (attrs && wait) this.attributes = lodash_es_assignIn({}, attributes, attrs); const method = this.isNew() ? 'create' : options.patch ? 'patch' : 'update'; if (method === 'patch' && !options.attrs) options.attrs = attrs; const xhr = this.sync(method, this, options); // Restore attributes. this.attributes = attributes; if (return_promise) { return promise; } else { return xhr; } }, // Destroy this model on the server if it was already persisted. // Optimistically removes the model from its collection, if it has one. // If `wait: true` is passed, waits for the server to respond before removal. destroy: function (options) { options = options ? lodash_es_clone(options) : {}; const model = this; const success = options.success; const wait = options.wait; const destroy = function () { model.stopListening(); model.trigger('destroy', model, model.collection, options); }; options.success = function (resp) { if (wait) destroy(); if (success) success.call(options.context, model, resp, options); if (!model.isNew()) model.trigger('sync', model, resp, options); }; let xhr = false; if (this.isNew()) { lodash_es_defer(options.success); } else { wrapError(this, options); xhr = this.sync('delete', this, options); } if (!wait) destroy(); return xhr; }, // Default URL for the model's representation on the server -- if you're // using Backbone's restful methods, override this to change the endpoint // that will be called. url: function () { const base = lodash_es_result(this, 'urlRoot') || lodash_es_result(this.collection, 'url') || urlError(); if (this.isNew()) return base; const id = this.get(this.idAttribute); return base.replace(/[^\/]$/, '$&/') + encodeURIComponent(id); }, // **parse** converts a response into the hash of attributes to be `set` on // the model. The default implementation is just to pass the response along. parse: function (resp, options) { return resp; }, // Create a new model with identical attributes to this one. clone: function () { return new this.constructor(this.attributes); }, // A model is new if it has never been saved to the server, and lacks an id. isNew: function () { return !this.has(this.idAttribute); }, // Check if the model is currently in a valid state. isValid: function (options) { return this._validate({}, lodash_es_assignIn({}, options, { validate: true })); }, // Run validation against the next complete set of model attributes, // returning `true` if all is well. Otherwise, fire an `"invalid"` event. _validate: function (attrs, options) { if (!options.validate || !this.validate) return true; attrs = lodash_es_assignIn({}, this.attributes, attrs); const error = this.validationError = this.validate(attrs, options) || null; if (!error) return true; this.trigger('invalid', this, error, lodash_es_assignIn(options, { validationError: error })); return false; } }); ;// CONCATENATED MODULE: ./node_modules/lodash-es/debounce.js /** Error message constants. */ var debounce_FUNC_ERROR_TEXT = 'Expected a function'; /* Built-in method references for those with the same name as other `lodash` methods. */ var debounce_nativeMax = Math.max, debounce_nativeMin = Math.min; /** * Creates a debounced function that delays invoking `func` until after `wait` * milliseconds have elapsed since the last time the debounced function was * invoked. The debounced function comes with a `cancel` method to cancel * delayed `func` invocations and a `flush` method to immediately invoke them. * Provide `options` to indicate whether `func` should be invoked on the * leading and/or trailing edge of the `wait` timeout. The `func` is invoked * with the last arguments provided to the debounced function. Subsequent * calls to the debounced function return the result of the last `func` * invocation. * * **Note:** If `leading` and `trailing` options are `true`, `func` is * invoked on the trailing edge of the timeout only if the debounced function * is invoked more than once during the `wait` timeout. * * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred * until to the next tick, similar to `setTimeout` with a timeout of `0`. * * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/) * for details over the differences between `_.debounce` and `_.throttle`. * * @static * @memberOf _ * @since 0.1.0 * @category Function * @param {Function} func The function to debounce. * @param {number} [wait=0] The number of milliseconds to delay. * @param {Object} [options={}] The options object. * @param {boolean} [options.leading=false] * Specify invoking on the leading edge of the timeout. * @param {number} [options.maxWait] * The maximum time `func` is allowed to be delayed before it's invoked. * @param {boolean} [options.trailing=true] * Specify invoking on the trailing edge of the timeout. * @returns {Function} Returns the new debounced function. * @example * * // Avoid costly calculations while the window size is in flux. * jQuery(window).on('resize', _.debounce(calculateLayout, 150)); * * // Invoke `sendMail` when clicked, debouncing subsequent calls. * jQuery(element).on('click', _.debounce(sendMail, 300, { * 'leading': true, * 'trailing': false * })); * * // Ensure `batchLog` is invoked once after 1 second of debounced calls. * var debounced = _.debounce(batchLog, 250, { 'maxWait': 1000 }); * var source = new EventSource('/stream'); * jQuery(source).on('message', debounced); * * // Cancel the trailing debounced invocation. * jQuery(window).on('popstate', debounced.cancel); */ function debounce(func, wait, options) { var lastArgs, lastThis, maxWait, result, timerId, lastCallTime, lastInvokeTime = 0, leading = false, maxing = false, trailing = true; if (typeof func != 'function') { throw new TypeError(debounce_FUNC_ERROR_TEXT); } wait = lodash_es_toNumber(wait) || 0; if (lodash_es_isObject(options)) { leading = !!options.leading; maxing = 'maxWait' in options; maxWait = maxing ? debounce_nativeMax(lodash_es_toNumber(options.maxWait) || 0, wait) : maxWait; trailing = 'trailing' in options ? !!options.trailing : trailing; } function invokeFunc(time) { var args = lastArgs, thisArg = lastThis; lastArgs = lastThis = undefined; lastInvokeTime = time; result = func.apply(thisArg, args); return result; } function leadingEdge(time) { // Reset any `maxWait` timer. lastInvokeTime = time; // Start the timer for the trailing edge. timerId = setTimeout(timerExpired, wait); // Invoke the leading edge. return leading ? invokeFunc(time) : result; } function remainingWait(time) { var timeSinceLastCall = time - lastCallTime, timeSinceLastInvoke = time - lastInvokeTime, timeWaiting = wait - timeSinceLastCall; return maxing ? debounce_nativeMin(timeWaiting, maxWait - timeSinceLastInvoke) : timeWaiting; } function shouldInvoke(time) { var timeSinceLastCall = time - lastCallTime, timeSinceLastInvoke = time - lastInvokeTime; // Either this is the first call, activity has stopped and we're at the // trailing edge, the system time has gone backwards and we're treating // it as the trailing edge, or we've hit the `maxWait` limit. return (lastCallTime === undefined || (timeSinceLastCall >= wait) || (timeSinceLastCall < 0) || (maxing && timeSinceLastInvoke >= maxWait)); } function timerExpired() { var time = lodash_es_now(); if (shouldInvoke(time)) { return trailingEdge(time); } // Restart the timer. timerId = setTimeout(timerExpired, remainingWait(time)); } function trailingEdge(time) { timerId = undefined; // Only invoke if we have `lastArgs` which means `func` has been // debounced at least once. if (trailing && lastArgs) { return invokeFunc(time); } lastArgs = lastThis = undefined; return result; } function cancel() { if (timerId !== undefined) { clearTimeout(timerId); } lastInvokeTime = 0; lastArgs = lastCallTime = lastThis = timerId = undefined; } function flush() { return timerId === undefined ? result : trailingEdge(lodash_es_now()); } function debounced() { var time = lodash_es_now(), isInvoking = shouldInvoke(time); lastArgs = arguments; lastThis = this; lastCallTime = time; if (isInvoking) { if (timerId === undefined) { return leadingEdge(lastCallTime); } if (maxing) { // Handle invocations in a tight loop. clearTimeout(timerId); timerId = setTimeout(timerExpired, wait); return invokeFunc(lastCallTime); } } if (timerId === undefined) { timerId = setTimeout(timerExpired, wait); } return result; } debounced.cancel = cancel; debounced.flush = flush; return debounced; } /* harmony default export */ const lodash_es_debounce = (debounce); // EXTERNAL MODULE: ./node_modules/localforage-webextensionstorage-driver/local.js var local = __webpack_require__(7002); // EXTERNAL MODULE: ./node_modules/localforage-webextensionstorage-driver/sync.js var localforage_webextensionstorage_driver_sync = __webpack_require__(1063); ;// CONCATENATED MODULE: ./src/headless/shared/connection/index.js const i = Object.keys(Strophe.Status).reduce((max, k) => Math.max(max, Strophe.Status[k]), 0); Strophe.Status.RECONNECTING = i + 1; /** * The Connection class manages the connection to the XMPP server. It's * agnostic concerning the underlying protocol (i.e. websocket, long-polling * via BOSH or websocket inside a shared worker). */ class Connection extends Strophe.Connection { constructor(service, options) { super(service, options); this.debouncedReconnect = lodash_es_debounce(this.reconnect, 3000); } static generateResource() { return `/converse.js-${Math.floor(Math.random() * 139749528).toString()}`; } async bind() { /** * Synchronous event triggered before we send an IQ to bind the user's * JID resource for this session. * @event _converse#beforeResourceBinding */ await core_api.trigger('beforeResourceBinding', { 'synchronous': true }); super.bind(); } async onDomainDiscovered(response) { const text = await response.text(); const xrd = new window.DOMParser().parseFromString(text, "text/xml").firstElementChild; if (xrd.nodeName != "XRD" || xrd.namespaceURI != "http://docs.oasis-open.org/ns/xri/xrd-1.0") { return headless_log.warn("Could not discover XEP-0156 connection methods"); } const bosh_links = sizzle_default()(`Link[rel="urn:xmpp:alt-connections:xbosh"]`, xrd); const ws_links = sizzle_default()(`Link[rel="urn:xmpp:alt-connections:websocket"]`, xrd); const bosh_methods = bosh_links.map(el => el.getAttribute('href')); const ws_methods = ws_links.map(el => el.getAttribute('href')); if (bosh_methods.length === 0 && ws_methods.length === 0) { headless_log.warn("Neither BOSH nor WebSocket connection methods have been specified with XEP-0156."); } else { // TODO: support multiple endpoints core_api.settings.set("websocket_url", ws_methods.pop()); core_api.settings.set('bosh_service_url', bosh_methods.pop()); this.service = core_api.settings.get("websocket_url") || core_api.settings.get('bosh_service_url'); this.setProtocol(); } } /** * Adds support for XEP-0156 by quering the XMPP server for alternate * connection methods. This allows users to use the websocket or BOSH * connection of their own XMPP server instead of a proxy provided by the * host of Converse.js. * @method Connnection.discoverConnectionMethods */ async discoverConnectionMethods(domain) { // Use XEP-0156 to check whether this host advertises websocket or BOSH connection methods. const options = { 'mode': 'cors', 'headers': { 'Accept': 'application/xrd+xml, text/xml' } }; const url = `https://${domain}/.well-known/host-meta`; let response; try { response = await fetch(url, options); } catch (e) { headless_log.error(`Failed to discover alternative connection methods at ${url}`); headless_log.error(e); return; } if (response.status >= 200 && response.status < 400) { await this.onDomainDiscovered(response); } else { headless_log.warn("Could not discover XEP-0156 connection methods"); } } /** * Establish a new XMPP session by logging in with the supplied JID and * password. * @method Connnection.connect * @param { String } jid * @param { String } password * @param { Funtion } callback */ async connect(jid, password, callback) { if (core_api.settings.get("discover_connection_methods")) { const domain = Strophe.getDomainFromJid(jid); await this.discoverConnectionMethods(domain); } if (!core_api.settings.get('bosh_service_url') && !core_api.settings.get("websocket_url")) { throw new Error("You must supply a value for either the bosh_service_url or websocket_url or both."); } super.connect(jid, password, callback || this.onConnectStatusChanged, BOSH_WAIT); } /** * Switch to a different transport if a service URL is available for it. * * When reconnecting with a new transport, we call setUserJID * so that a new resource is generated, to avoid multiple * server-side sessions with the same resource. * * We also call `_proto._doDisconnect` so that connection event handlers * for the old transport are removed. */ async switchTransport() { if (core_api.connection.isType('websocket') && core_api.settings.get('bosh_service_url')) { await setUserJID(shared_converse.bare_jid); this._proto._doDisconnect(); this._proto = new Strophe.Bosh(this); this.service = core_api.settings.get('bosh_service_url'); } else if (core_api.connection.isType('bosh') && core_api.settings.get("websocket_url")) { if (core_api.settings.get("authentication") === shared_converse.ANONYMOUS) { // When reconnecting anonymously, we need to connect with only // the domain, not the full JID that we had in our previous // (now failed) session. await setUserJID(core_api.settings.get("jid")); } else { await setUserJID(shared_converse.bare_jid); } this._proto._doDisconnect(); this._proto = new Strophe.Websocket(this); this.service = core_api.settings.get("websocket_url"); } } async reconnect() { headless_log.debug('RECONNECTING: the connection has dropped, attempting to reconnect.'); this.reconnecting = true; await tearDown(); const conn_status = shared_converse.connfeedback.get('connection_status'); if (conn_status === Strophe.Status.CONNFAIL) { this.switchTransport(); } else if (conn_status === Strophe.Status.AUTHFAIL && core_api.settings.get("authentication") === shared_converse.ANONYMOUS) { // When reconnecting anonymously, we need to connect with only // the domain, not the full JID that we had in our previous // (now failed) session. await setUserJID(core_api.settings.get("jid")); } /** * Triggered when the connection has dropped, but Converse will attempt * to reconnect again. * @event _converse#will-reconnect */ core_api.trigger('will-reconnect'); if (core_api.settings.get("authentication") === shared_converse.ANONYMOUS) { await clearSession(); } return core_api.user.login(); } /** * Called as soon as a new connection has been established, either * by logging in or by attaching to an existing BOSH session. * @method Connection.onConnected * @param { Boolean } reconnecting - Whether Converse.js reconnected from an earlier dropped session. */ async onConnected(reconnecting) { delete this.reconnecting; this.flush(); // Solves problem of returned PubSub BOSH response not received by browser await setUserJID(this.jid); /** * Synchronous event triggered after we've sent an IQ to bind the * user's JID resource for this session. * @event _converse#afterResourceBinding */ await core_api.trigger('afterResourceBinding', reconnecting, { 'synchronous': true }); if (reconnecting) { /** * After the connection has dropped and converse.js has reconnected. * Any Strophe stanza handlers (as registered via `converse.listen.stanza`) will * have to be registered anew. * @event _converse#reconnected * @example _converse.api.listen.on('reconnected', () => { ... }); */ core_api.trigger('reconnected'); } else { /** * Triggered after the connection has been established and Converse * has got all its ducks in a row. * @event _converse#initialized */ core_api.trigger('connected'); } } /** * Used to keep track of why we got disconnected, so that we can * decide on what the next appropriate action is (in onDisconnected) * @method Connection.setDisconnectionCause * @param { Number } cause - The status number as received from Strophe. * @param { String } [reason] - An optional user-facing message as to why * there was a disconnection. * @param { Boolean } [override] - An optional flag to replace any previous * disconnection cause and reason. */ setDisconnectionCause(cause, reason, override) { if (cause === undefined) { delete this.disconnection_cause; delete this.disconnection_reason; } else if (this.disconnection_cause === undefined || override) { this.disconnection_cause = cause; this.disconnection_reason = reason; } } setConnectionStatus(status, message) { this.status = status; shared_converse.connfeedback.set({ 'connection_status': status, message }); } async finishDisconnection() { // Properly tear down the session so that it's possible to manually connect again. headless_log.debug('DISCONNECTED'); delete this.reconnecting; this.reset(); tearDown(); await clearSession(); delete shared_converse.connection; /** * Triggered after converse.js has disconnected from the XMPP server. * @event _converse#disconnected * @memberOf _converse * @example _converse.api.listen.on('disconnected', () => { ... }); */ core_api.trigger('disconnected'); } /** * Gets called once strophe's status reaches Strophe.Status.DISCONNECTED. * Will either start a teardown process for converse.js or attempt * to reconnect. * @method onDisconnected */ onDisconnected() { if (core_api.settings.get("auto_reconnect")) { const reason = this.disconnection_reason; if (this.disconnection_cause === Strophe.Status.AUTHFAIL) { if (core_api.settings.get("credentials_url") || core_api.settings.get("authentication") === shared_converse.ANONYMOUS) { // If `credentials_url` is set, we reconnect, because we might // be receiving expirable tokens from the credentials_url. // // If `authentication` is anonymous, we reconnect because we // might have tried to attach with stale BOSH session tokens // or with a cached JID and password return core_api.connection.reconnect(); } else { return this.finishDisconnection(); } } else if (this.status === Strophe.Status.CONNECTING) { // Don't try to reconnect if we were never connected to begin // with, otherwise an infinite loop can occur (e.g. when the // BOSH service URL returns a 404). const { __ } = shared_converse; this.setConnectionStatus(Strophe.Status.CONNFAIL, __('An error occurred while connecting to the chat server.')); return this.finishDisconnection(); } else if (this.disconnection_cause === shared_converse.LOGOUT || reason === Strophe.ErrorCondition.NO_AUTH_MECH || reason === "host-unknown" || reason === "remote-connection-failed") { return this.finishDisconnection(); } core_api.connection.reconnect(); } else { return this.finishDisconnection(); } } /** * Callback method called by Strophe as the Connection goes * through various states while establishing or tearing down a * connection. * @param { Number } status * @param { String } message */ onConnectStatusChanged(status, message) { const { __ } = shared_converse; headless_log.debug(`Status changed to: ${shared_converse.CONNECTION_STATUS[status]}`); if (status === Strophe.Status.ATTACHFAIL) { var _this$worker_attach_p; this.setConnectionStatus(status); (_this$worker_attach_p = this.worker_attach_promise) === null || _this$worker_attach_p === void 0 ? void 0 : _this$worker_attach_p.resolve(false); } else if (status === Strophe.Status.CONNECTED || status === Strophe.Status.ATTACHED) { var _this$worker_attach_p2, _this$worker_attach_p3; if ((_this$worker_attach_p2 = this.worker_attach_promise) !== null && _this$worker_attach_p2 !== void 0 && _this$worker_attach_p2.isResolved && this.status === Strophe.Status.ATTACHED) { // A different tab must have attached, so nothing to do for us here. return; } this.setConnectionStatus(status); (_this$worker_attach_p3 = this.worker_attach_promise) === null || _this$worker_attach_p3 === void 0 ? void 0 : _this$worker_attach_p3.resolve(true); // By default we always want to send out an initial presence stanza. shared_converse.send_initial_presence = true; this.setDisconnectionCause(); if (this.reconnecting) { headless_log.debug(status === Strophe.Status.CONNECTED ? 'Reconnected' : 'Reattached'); this.onConnected(true); } else { headless_log.debug(status === Strophe.Status.CONNECTED ? 'Connected' : 'Attached'); if (this.restored) { // No need to send an initial presence stanza when // we're restoring an existing session. shared_converse.send_initial_presence = false; } this.onConnected(); } } else if (status === Strophe.Status.DISCONNECTED) { this.setDisconnectionCause(status, message); this.onDisconnected(); } else if (status === Strophe.Status.BINDREQUIRED) { this.bind(); } else if (status === Strophe.Status.ERROR) { this.setConnectionStatus(status, __('An error occurred while connecting to the chat server.')); } else if (status === Strophe.Status.CONNECTING) { this.setConnectionStatus(status); } else if (status === Strophe.Status.AUTHENTICATING) { this.setConnectionStatus(status); } else if (status === Strophe.Status.AUTHFAIL) { if (!message) { message = __('Your XMPP address and/or password is incorrect. Please try again.'); } this.setConnectionStatus(status, message); this.setDisconnectionCause(status, message, true); this.onDisconnected(); } else if (status === Strophe.Status.CONNFAIL) { var _Strophe$ErrorConditi; let feedback = message; if (message === "host-unknown" || message == "remote-connection-failed") { feedback = __("Sorry, we could not connect to the XMPP host with domain: %1$s", `\"${Strophe.getDomainFromJid(this.jid)}\"`); } else if (message !== undefined && message === (Strophe === null || Strophe === void 0 ? void 0 : (_Strophe$ErrorConditi = Strophe.ErrorCondition) === null || _Strophe$ErrorConditi === void 0 ? void 0 : _Strophe$ErrorConditi.NO_AUTH_MECH)) { feedback = __("The XMPP server did not offer a supported authentication mechanism"); } this.setConnectionStatus(status, feedback); this.setDisconnectionCause(status, message); } else if (status === Strophe.Status.DISCONNECTING) { this.setDisconnectionCause(status, message); } } isType(type) { if (type.toLowerCase() === 'websocket') { return this._proto instanceof Strophe.Websocket; } else if (type.toLowerCase() === 'bosh') { return Strophe.Bosh && this._proto instanceof Strophe.Bosh; } } hasResumed() { var _api$settings$get; if ((_api$settings$get = core_api.settings.get("connection_options")) !== null && _api$settings$get !== void 0 && _api$settings$get.worker || this.isType('bosh')) { return shared_converse.connfeedback.get('connection_status') === Strophe.Status.ATTACHED; } else { // Not binding means that the session was resumed. return !this.do_bind; } } restoreWorkerSession() { this.attach(this.onConnectStatusChanged); this.worker_attach_promise = getOpenPromise(); return this.worker_attach_promise; } } /** * The MockConnection class is used during testing, to mock an XMPP connection. * @class */ class MockConnection extends Connection { constructor(service, options) { super(service, options); this.sent_stanzas = []; this.IQ_stanzas = []; this.IQ_ids = []; this.features = Strophe.xmlHtmlNode('' + '' + '' + '' + '' + '' + '' + `` + '' + '' + '' + '').firstChild; this._proto._processRequest = () => {}; this._proto._disconnect = () => this._onDisconnectTimeout(); this._proto._onDisconnectTimeout = () => {}; this._proto._connect = () => { this.connected = true; this.mock = true; this.jid = 'romeo@montague.lit/orchard'; this._changeConnectStatus(Strophe.Status.BINDREQUIRED); }; } _processRequest() {// eslint-disable-line class-methods-use-this // Don't attempt to send out stanzas } sendIQ(iq, callback, errback) { if (!lodash_es_isElement(iq)) { iq = iq.nodeTree; } this.IQ_stanzas.push(iq); const id = super.sendIQ(iq, callback, errback); this.IQ_ids.push(id); return id; } send(stanza) { if (lodash_es_isElement(stanza)) { this.sent_stanzas.push(stanza); } else { this.sent_stanzas.push(stanza.nodeTree); } return super.send(stanza); } async bind() { await core_api.trigger('beforeResourceBinding', { 'synchronous': true }); this.authenticated = true; if (!shared_converse.no_connection_on_bind) { this._changeConnectStatus(Strophe.Status.CONNECTED); } } } ;// CONCATENATED MODULE: ./src/headless/utils/init.js function initPlugins(_converse) { // 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 = []; const whitelist = CORE_PLUGINS.concat(_converse.api.settings.get("whitelisted_plugins")); if (_converse.api.settings.get("singleton")) { ['converse-bookmarks', 'converse-controlbox', 'converse-headline', 'converse-register'].forEach(name => _converse.api.settings.get("blacklisted_plugins").push(name)); } _converse.pluggable.initializePlugins({ _converse }, whitelist, _converse.api.settings.get("blacklisted_plugins")); /** * Triggered once all plugins have been initialized. This is a useful event if you want to * register event handlers but would like your own handlers to be overridable by * plugins. In that case, you need to first wait until all plugins have been * initialized, so that their overrides are active. One example where this is used * is in [converse-notifications.js](https://github.com/jcbrand/converse.js/blob/master/src/converse-notification.js)`. * * Also available as an [ES2015 Promise](http://es6-features.org/#PromiseUsage) * which can be listened to with `_converse.api.waitUntil`. * * @event _converse#pluginsInitialized * @memberOf _converse * @example _converse.api.listen.on('pluginsInitialized', () => { ... }); */ _converse.api.trigger('pluginsInitialized'); } async function initClientConfig(_converse) { /* The client config refers to configuration of the client which is * independent of any particular user. * What this means is that config values need to persist across * user sessions. */ const id = 'converse.client-config'; _converse.config = new Model({ id, 'trusted': true }); _converse.config.browserStorage = createStore(id, "session"); await new Promise(r => _converse.config.fetch({ 'success': r, 'error': r })); /** * Triggered once the XMPP-client configuration has been initialized. * The client configuration is independent of any particular and its values * persist across user sessions. * * @event _converse#clientConfigInitialized * @example * _converse.api.listen.on('clientConfigInitialized', () => { ... }); */ _converse.api.trigger('clientConfigInitialized'); } async function initSessionStorage(_converse) { await storage.sessionStorageInitialized; _converse.storage = { 'session': storage.localForage.createInstance({ 'name': _converse.isTestEnv() ? 'converse-test-session' : 'converse-session', 'description': 'sessionStorage instance', 'driver': ['sessionStorageWrapper'] }) }; } function initPersistentStorage(_converse, store_name) { if (_converse.api.settings.get('persistent_store') === 'sessionStorage') { return; } else if (_converse.api.settings.get("persistent_store") === 'BrowserExtLocal') { storage.localForage.defineDriver(local/* default */.Z).then(() => storage.localForage.setDriver('webExtensionLocalStorage')); _converse.storage['persistent'] = storage.localForage; return; } else if (_converse.api.settings.get("persistent_store") === 'BrowserExtSync') { storage.localForage.defineDriver(localforage_webextensionstorage_driver_sync/* default */.Z).then(() => storage.localForage.setDriver('webExtensionSyncStorage')); _converse.storage['persistent'] = storage.localForage; return; } const config = { 'name': _converse.isTestEnv() ? 'converse-test-persistent' : 'converse-persistent', 'storeName': store_name }; if (_converse.api.settings.get("persistent_store") === 'localStorage') { config['description'] = 'localStorage instance'; config['driver'] = [storage.localForage.LOCALSTORAGE]; } else if (_converse.api.settings.get("persistent_store") === 'IndexedDB') { config['description'] = 'indexedDB instance'; config['driver'] = [storage.localForage.INDEXEDDB]; } _converse.storage['persistent'] = storage.localForage.createInstance(config); } function saveJIDtoSession(_converse, jid) { jid = _converse.session.get('jid') || jid; if (_converse.api.settings.get("authentication") !== _converse.ANONYMOUS && !Strophe.getResourceFromJid(jid)) { jid = jid.toLowerCase() + Connection.generateResource(); } _converse.jid = jid; _converse.bare_jid = Strophe.getBareJidFromJid(jid); _converse.resource = Strophe.getResourceFromJid(jid); _converse.domain = Strophe.getDomainFromJid(jid); _converse.session.save({ 'jid': jid, 'bare_jid': _converse.bare_jid, 'resource': _converse.resource, 'domain': _converse.domain, // We use the `active` flag to determine whether we should use the values from sessionStorage. // When "cloning" a tab (e.g. via middle-click), the `active` flag will be set and we'll create // a new empty user session, otherwise it'll be false and we can re-use the user session. 'active': true }); // Set JID on the connection object so that when we call `connection.bind` // the new resource is found by Strophe.js and sent to the XMPP server. _converse.connection.jid = jid; } /** * Stores the passed in JID for the current user, potentially creating a * resource if the JID is bare. * * Given that we can only create an XMPP connection if we know the domain of * the server connect to and we only know this once we know the JID, we also * call {@link _converse.initConnection } (if necessary) to make sure that the * connection is set up. * * @emits _converse#setUserJID * @params { String } jid */ async function setUserJID(jid) { await initSession(shared_converse, jid); /** * Triggered whenever the user's JID has been updated * @event _converse#setUserJID */ shared_converse.api.trigger('setUserJID'); return jid; } async function initSession(_converse, jid) { var _converse$session; const is_shared_session = _converse.api.settings.get('connection_options').worker; const bare_jid = Strophe.getBareJidFromJid(jid).toLowerCase(); const id = `converse.session-${bare_jid}`; if (((_converse$session = _converse.session) === null || _converse$session === void 0 ? void 0 : _converse$session.get('id')) !== id) { initPersistentStorage(_converse, bare_jid); _converse.session = new Model({ id }); initStorage(_converse.session, id, is_shared_session ? "persistent" : "session"); await new Promise(r => _converse.session.fetch({ 'success': r, 'error': r })); if (!is_shared_session && _converse.session.get('active')) { // If the `active` flag is set, it means this tab was cloned from // another (e.g. via middle-click), and its session data was copied over. _converse.session.clear(); _converse.session.save({ id }); } saveJIDtoSession(_converse, jid); /** * Triggered once the user's session has been initialized. The session is a * cache which stores information about the user's current session. * @event _converse#userSessionInitialized * @memberOf _converse */ _converse.api.trigger('userSessionInitialized'); } else { saveJIDtoSession(_converse, jid); } } function registerGlobalEventHandlers(_converse) { document.addEventListener("visibilitychange", _converse.saveWindowState); _converse.saveWindowState({ 'type': document.hidden ? "blur" : "focus" }); // Set initial state /** * Called once Converse has registered its global event handlers * (for events such as window resize or unload). * Plugins can listen to this event as cue to register their own * global event handlers. * @event _converse#registeredGlobalEventHandlers * @example _converse.api.listen.on('registeredGlobalEventHandlers', () => { ... }); */ _converse.api.trigger('registeredGlobalEventHandlers'); } function unregisterGlobalEventHandlers(_converse) { const { api } = _converse; document.removeEventListener("visibilitychange", _converse.saveWindowState); api.trigger('unregisteredGlobalEventHandlers'); } // Make sure everything is reset in case this is a subsequent call to // converse.initialize (happens during tests). async function cleanup(_converse) { var _converse$connection; const { api } = _converse; await api.trigger('cleanup', { 'synchronous': true }); _converse.router.history.stop(); unregisterGlobalEventHandlers(_converse); (_converse$connection = _converse.connection) === null || _converse$connection === void 0 ? void 0 : _converse$connection.reset(); _converse.stopListening(); _converse.off(); if (_converse.promises['initialized'].isResolved) { api.promises.add('initialized'); } } async function getLoginCredentials() { let credentials; let wait = 0; while (!credentials) { try { credentials = await fetchLoginCredentials(wait); // eslint-disable-line no-await-in-loop } catch (e) { headless_log.error('Could not fetch login credentials'); headless_log.error(e); } // If unsuccessful, we wait 2 seconds between subsequent attempts to // fetch the credentials. wait = 2000; } return credentials; } function fetchLoginCredentials() { let wait = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 0; return new Promise(lodash_es_debounce(async (resolve, reject) => { let xhr = new XMLHttpRequest(); xhr.open('GET', shared_converse.api.settings.get("credentials_url"), true); xhr.setRequestHeader('Accept', 'application/json, text/javascript'); xhr.onload = () => { if (xhr.status >= 200 && xhr.status < 400) { const data = JSON.parse(xhr.responseText); setUserJID(data.jid).then(() => { resolve({ jid: data.jid, password: data.password }); }); } else { reject(new Error(`${xhr.status}: ${xhr.responseText}`)); } }; xhr.onerror = reject; /** * *Hook* which allows modifying the server request * @event _converse#beforeFetchLoginCredentials */ xhr = await shared_converse.api.hook('beforeFetchLoginCredentials', this, xhr); xhr.send(); }, wait)); } async function attemptNonPreboundSession(credentials, automatic) { const { api } = shared_converse; if (api.settings.get("authentication") === shared_converse.LOGIN) { // XXX: If EITHER ``keepalive`` or ``auto_login`` is ``true`` and // ``authentication`` is set to ``login``, then Converse will try to log the user in, // since we don't have a way to distinguish between wether we're // restoring a previous session (``keepalive``) or whether we're // automatically setting up a new session (``auto_login``). // So we can't do the check (!automatic || _converse.api.settings.get("auto_login")) here. if (credentials) { connect(credentials); } else if (api.settings.get("credentials_url")) { // We give credentials_url preference, because // _converse.connection.pass might be an expired token. connect(await getLoginCredentials()); } else if (shared_converse.jid && (api.settings.get("password") || shared_converse.connection.pass)) { connect(); } else if (!shared_converse.isTestEnv() && 'credentials' in navigator) { connect(await getLoginCredentialsFromBrowser()); } else { !shared_converse.isTestEnv() && headless_log.warn("attemptNonPreboundSession: Couldn't find credentials to log in with"); } } else if ([shared_converse.ANONYMOUS, shared_converse.EXTERNAL].includes(api.settings.get("authentication")) && (!automatic || api.settings.get("auto_login"))) { connect(); } } function getConnectionServiceURL() { const { api } = shared_converse; if (('WebSocket' in window || 'MozWebSocket' in window) && api.settings.get("websocket_url")) { return api.settings.get('websocket_url'); } else if (api.settings.get('bosh_service_url')) { return api.settings.get('bosh_service_url'); } return ''; } function connect(credentials) { const { api } = shared_converse; if ([shared_converse.ANONYMOUS, shared_converse.EXTERNAL].includes(api.settings.get("authentication"))) { if (!shared_converse.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 (!shared_converse.connection.reconnecting) { shared_converse.connection.reset(); } shared_converse.connection.connect(shared_converse.jid.toLowerCase()); } else if (api.settings.get("authentication") === shared_converse.LOGIN) { var _converse$connection2; const password = (credentials === null || credentials === void 0 ? void 0 : credentials.password) ?? (((_converse$connection2 = shared_converse.connection) === null || _converse$connection2 === void 0 ? void 0 : _converse$connection2.pass) || api.settings.get("password")); if (!password) { if (api.settings.get("auto_login")) { throw new Error("autoLogin: If you use auto_login and " + "authentication='login' then you also need to provide a password."); } shared_converse.connection.setDisconnectionCause(Strophe.Status.AUTHFAIL, undefined, true); api.connection.disconnect(); return; } if (!shared_converse.connection.reconnecting) { shared_converse.connection.reset(); shared_converse.connection.service = getConnectionServiceURL(); } shared_converse.connection.connect(shared_converse.jid, password); } } ;// CONCATENATED MODULE: ./src/headless/shared/settings/api.js /** * This grouping allows access to the * [configuration settings](/docs/html/configuration.html#configuration-settings) * of Converse. * * @namespace _converse.api.settings * @memberOf _converse.api */ const settings_api = { /** * Allows new configuration settings to be specified, or new default values for * existing configuration settings to be specified. * * Note, calling this method *after* converse.initialize has been * called will *not* change the initialization settings provided via * `converse.initialize`. * * @method _converse.api.settings.extend * @param {object} settings The configuration settings * @example * _converse.api.settings.extend({ * 'enable_foo': true * }); * * // The user can then override the default value of the configuration setting when * // calling `converse.initialize`. * converse.initialize({ * 'enable_foo': false * }); */ extend(settings) { return extendAppSettings(settings); }, update(settings) { headless_log.warn('The api.settings.update method has been deprecated and will be removed. ' + 'Please use api.settings.extend instead.'); return this.extend(settings); }, /** * @method _converse.api.settings.get * @returns {*} Value of the particular configuration setting. * @example _converse.api.settings.get("play_sounds"); */ get(key) { return getAppSetting(key); }, /** * Set one or many configuration settings. * * Note, this is not an alternative to calling {@link converse.initialize}, which still needs * to be called. Generally, you'd use this method after Converse is already * running and you want to change the configuration on-the-fly. * * @method _converse.api.settings.set * @param {Object} [settings] An object containing configuration settings. * @param {string} [key] Alternatively to passing in an object, you can pass in a key and a value. * @param {string} [value] * @example _converse.api.settings.set("play_sounds", true); * @example * _converse.api.settings.set({ * "play_sounds": true, * "hide_offline_users": true * }); */ set(key, val) { updateAppSettings(key, val); }, /** * The `listen` namespace exposes methods for creating event listeners * (aka handlers) for events related to settings. * * @namespace _converse.api.settings.listen * @memberOf _converse.api.settings */ listen: { /** * Register an event listener for the passed in event. * @method _converse.api.settings.listen.on * @param { ('change') } name - The name of the event to listen for. * Currently there is only the 'change' event. * @param { Function } handler - The event handler function * @param { Object } [context] - The context of the `this` attribute of the * handler function. * @example _converse.api.settings.listen.on('change', callback); */ on(name, handler, context) { registerListener(name, handler, context); }, /** * To stop listening to an event, you can use the `not` method. * @method _converse.api.settings.listen.not * @param { String } name The event's name * @param { Function } callback The callback method that is to no longer be called when the event fires * @example _converse.api.settings.listen.not('change', callback); */ not(name, handler) { unregisterListener(name, handler); } } }; /** * API for accessing and setting user settings. User settings are * different from the application settings from {@link _converse.api.settings} * because they are per-user and set via user action. * @namespace _converse.api.user.settings * @memberOf _converse.api.user */ const user_settings_api = { /** * Returns the user settings model. Useful when you want to listen for change events. * @async * @method _converse.api.user.settings.getModel * @returns {Promise} * @example const settings = await _converse.api.user.settings.getModel(); */ getModel() { return getUserSettings(); }, /** * Get the value of a particular user setting. * @method _converse.api.user.settings.get * @param {String} key - The setting name * @param {*} [fallback] - An optional fallback value if the user setting is undefined * @returns {Promise} Promise which resolves with the value of the particular configuration setting. * @example _converse.api.user.settings.get("foo"); */ async get(key, fallback) { const user_settings = await getUserSettings(); return user_settings.get(key) === undefined ? fallback : user_settings.get(key); }, /** * Set one or many user settings. * @async * @method _converse.api.user.settings.set * @param {Object} [settings] An object containing configuration settings. * @param {string} [key] Alternatively to passing in an object, you can pass in a key and a value. * @param {string} [value] * @example _converse.api.user.settings.set("foo", "bar"); * @example * _converse.api.user.settings.set({ * "foo": "bar", * "baz": "buz" * }); */ set(key, val) { if (lodash_es_isObject(key)) { return updateUserSettings(key, { 'promise': true }); } else { const o = {}; o[key] = val; return updateUserSettings(o, { 'promise': true }); } }, /** * Clears all the user settings * @async * @method _converse.api.user.settings.clear */ clear() { return clearUserSettings(); } }; ;// CONCATENATED MODULE: ./src/headless/utils/core.js /** * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) * @description This is the core utilities module. */ function isEmptyMessage(attrs) { if (attrs instanceof Model) { attrs = attrs.attributes; } return !attrs['oob_url'] && !attrs['file'] && !(attrs['is_encrypted'] && attrs['plaintext']) && !attrs['message']; } /* We distinguish between UniView and MultiView instances. * * UniView means that only one chat is visible, even though there might be multiple ongoing chats. * MultiView means that multiple chats may be visible simultaneously. */ function isUniView() { return ['mobile', 'fullscreen', 'embedded'].includes(settings_api.get("view_mode")); } async function tearDown() { await shared_converse.api.trigger('beforeTearDown', { 'synchronous': true }); window.removeEventListener('click', shared_converse.onUserActivity); window.removeEventListener('focus', shared_converse.onUserActivity); window.removeEventListener('keypress', shared_converse.onUserActivity); window.removeEventListener('mousemove', shared_converse.onUserActivity); window.removeEventListener(shared_converse.unloadevent, shared_converse.onUserActivity); window.clearInterval(shared_converse.everySecondTrigger); shared_converse.api.trigger('afterTearDown'); return shared_converse; } /** * The utils object * @namespace u */ const u = {}; u.isTagEqual = function (stanza, name) { if (stanza.nodeTree) { return u.isTagEqual(stanza.nodeTree, name); } else if (!(stanza instanceof Element)) { throw Error("isTagEqual called with value which isn't " + "an element or Strophe.Builder instance"); } else { return Strophe.isTagEqual(stanza, name); } }; const parser = new DOMParser(); const parserErrorNS = parser.parseFromString('invalid', 'text/xml').getElementsByTagName("parsererror")[0].namespaceURI; u.getJIDFromURI = function (jid) { return jid.startsWith('xmpp:') && jid.endsWith('?join') ? jid.replace(/^xmpp:/, '').replace(/\?join$/, '') : jid; }; u.toStanza = function (string) { const node = parser.parseFromString(string, "text/xml"); if (node.getElementsByTagNameNS(parserErrorNS, 'parsererror').length) { throw new Error(`Parser Error: ${string}`); } return node.firstElementChild; }; u.getLongestSubstring = function (string, candidates) { function reducer(accumulator, current_value) { if (string.startsWith(current_value)) { if (current_value.length > accumulator.length) { return current_value; } else { return accumulator; } } else { return accumulator; } } return candidates.reduce(reducer, ''); }; /** * Given a message object, return its text with @ chars * inserted before the mentioned nicknames. */ function prefixMentions(message) { let text = message.getMessageText(); (message.get('references') || []).sort((a, b) => b.begin - a.begin).forEach(ref => { text = `${text.slice(0, ref.begin)}@${text.slice(ref.begin)}`; }); return text; } u.isValidJID = function (jid) { if (typeof jid === 'string') { return lodash_es_compact(jid.split('@')).length === 2 && !jid.startsWith('@') && !jid.endsWith('@'); } return false; }; u.isValidMUCJID = function (jid) { return !jid.startsWith('@') && !jid.endsWith('@'); }; u.isSameBareJID = function (jid1, jid2) { if (typeof jid1 !== 'string' || typeof jid2 !== 'string') { return false; } return Strophe.getBareJidFromJid(jid1).toLowerCase() === Strophe.getBareJidFromJid(jid2).toLowerCase(); }; u.isSameDomain = function (jid1, jid2) { if (typeof jid1 !== 'string' || typeof jid2 !== 'string') { return false; } return Strophe.getDomainFromJid(jid1).toLowerCase() === Strophe.getDomainFromJid(jid2).toLowerCase(); }; u.isNewMessage = function (message) { /* Given a stanza, determine whether it's a new * message, i.e. not a MAM archived one. */ if (message instanceof Element) { return !(sizzle_default()(`result[xmlns="${Strophe.NS.MAM}"]`, message).length && sizzle_default()(`delay[xmlns="${Strophe.NS.DELAY}"]`, message).length); } else if (message instanceof Model) { message = message.attributes; } return !(message['is_delayed'] && message['is_archived']); }; u.shouldCreateMessage = function (attrs) { return attrs['retracted'] || // Retraction received *before* the message !isEmptyMessage(attrs); }; u.shouldCreateGroupchatMessage = function (attrs) { return attrs.nick && (u.shouldCreateMessage(attrs) || attrs.is_tombstone); }; u.isChatRoom = function (model) { return model && model.get('type') === 'chatroom'; }; u.isErrorObject = function (o) { return o instanceof Error; }; u.isErrorStanza = function (stanza) { if (!lodash_es_isElement(stanza)) { return false; } return stanza.getAttribute('type') === 'error'; }; u.isForbiddenError = function (stanza) { if (!lodash_es_isElement(stanza)) { return false; } return sizzle_default()(`error[type="auth"] forbidden[xmlns="${Strophe.NS.STANZAS}"]`, stanza).length > 0; }; u.isServiceUnavailableError = function (stanza) { if (!lodash_es_isElement(stanza)) { return false; } return sizzle_default()(`error[type="cancel"] service-unavailable[xmlns="${Strophe.NS.STANZAS}"]`, stanza).length > 0; }; /** * Merge the second object into the first one. * @private * @method u#merge * @param { Object } first * @param { Object } second */ u.merge = function merge(first, second) { for (const k in second) { if (lodash_es_isObject(first[k])) { merge(first[k], second[k]); } else { first[k] = second[k]; } } }; u.getOuterWidth = function (el) { let include_margin = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false; let width = el.offsetWidth; if (!include_margin) { return width; } const style = window.getComputedStyle(el); width += parseInt(style.marginLeft ? style.marginLeft : 0, 10) + parseInt(style.marginRight ? style.marginRight : 0, 10); return width; }; /** * Converts an HTML string into a DOM element. * Expects that the HTML string has only one top-level element, * i.e. not multiple ones. * @private * @method u#stringToElement * @param { String } s - The HTML string */ u.stringToElement = function (s) { var div = document.createElement('div'); div.innerHTML = s; return div.firstElementChild; }; /** * Checks whether the DOM element matches the given selector. * @private * @method u#matchesSelector * @param { DOMElement } el - The DOM element * @param { String } selector - The selector */ u.matchesSelector = function (el, selector) { const match = el.matches || el.matchesSelector || el.msMatchesSelector || el.mozMatchesSelector || el.webkitMatchesSelector || el.oMatchesSelector; return match ? match.call(el, selector) : false; }; /** * Returns a list of children of the DOM element that match the selector. * @private * @method u#queryChildren * @param { DOMElement } el - the DOM element * @param { String } selector - the selector they should be matched against */ u.queryChildren = function (el, selector) { return Array.from(el.childNodes).filter(el => u.matchesSelector(el, selector)); }; u.contains = function (attr, query) { const checker = (item, key) => item.get(key).toLowerCase().includes(query.toLowerCase()); return function (item) { if (typeof attr === 'object') { return Object.keys(attr).reduce((acc, k) => acc || checker(item, k), false); } else if (typeof attr === 'string') { return checker(item, attr); } else { throw new TypeError('contains: wrong attribute type. Must be string or array.'); } }; }; u.isOfType = function (type, item) { return item.get('type') == type; }; u.isInstance = function (type, item) { return item instanceof type; }; u.getAttribute = function (key, item) { return item.get(key); }; u.contains.not = function (attr, query) { return function (item) { return !u.contains(attr, query)(item); }; }; u.rootContains = function (root, el) { // The document element does not have the contains method in IE. if (root === document && !root.contains) { return document.head.contains(el) || document.body.contains(el); } return root.contains ? root.contains(el) : window.HTMLElement.prototype.contains.call(root, el); }; u.createFragmentFromText = function (markup) { /* Returns a DocumentFragment containing DOM nodes based on the * passed-in markup text. */ // http://stackoverflow.com/questions/9334645/create-node-from-markup-string var frag = document.createDocumentFragment(), tmp = document.createElement('body'), child; tmp.innerHTML = markup; // Append elements in a loop to a DocumentFragment, so that the // browser does not re-render the document for each node. while (child = tmp.firstChild) { // eslint-disable-line no-cond-assign frag.appendChild(child); } return frag; }; u.isPersistableModel = function (model) { return model.collection && model.collection.browserStorage; }; u.getResolveablePromise = getOpenPromise; u.getOpenPromise = getOpenPromise; u.interpolate = function (string, o) { return string.replace(/{{{([^{}]*)}}}/g, (a, b) => { var r = o[b]; return typeof r === 'string' || typeof r === 'number' ? r : a; }); }; /** * Call the callback once all the events have been triggered * @private * @method u#onMultipleEvents * @param { Array } events: An array of objects, with keys `object` and * `event`, representing the event name and the object it's triggered upon. * @param { Function } callback: The function to call once all events have * been triggered. */ u.onMultipleEvents = function () { let events = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : []; let callback = arguments.length > 1 ? arguments[1] : undefined; let triggered = []; function handler(result) { triggered.push(result); if (events.length === triggered.length) { callback(triggered); triggered = []; } } events.forEach(e => e.object.on(e.event, handler)); }; function safeSave(model, attributes, options) { if (u.isPersistableModel(model)) { model.save(attributes, options); } else { model.set(attributes, options); } } u.safeSave = safeSave; u.siblingIndex = function (el) { /* eslint-disable no-cond-assign */ for (var i = 0; el = el.previousElementSibling; i++); return i; }; /** * Returns the current word being written in the input element * @method u#getCurrentWord * @param {HTMLElement} input - The HTMLElement in which text is being entered * @param {integer} [index] - An optional rightmost boundary index. If given, the text * value of the input element will only be considered up until this index. * @param {string} [delineator] - An optional string delineator to * differentiate between words. * @private */ u.getCurrentWord = function (input, index, delineator) { if (!index) { index = input.selectionEnd || undefined; } let [word] = input.value.slice(0, index).split(/\s/).slice(-1); if (delineator) { [word] = word.split(delineator).slice(-1); } return word; }; u.isMentionBoundary = s => s !== '@' && RegExp(`(\\p{Z}|\\p{P})`, 'u').test(s); u.replaceCurrentWord = function (input, new_value) { const caret = input.selectionEnd || undefined; const current_word = lodash_es_last(input.value.slice(0, caret).split(/\s/)); const value = input.value; const mention_boundary = u.isMentionBoundary(current_word[0]) ? current_word[0] : ''; input.value = value.slice(0, caret - current_word.length) + mention_boundary + `${new_value} ` + value.slice(caret); const selection_end = caret - current_word.length + new_value.length + 1; input.selectionEnd = mention_boundary ? selection_end + 1 : selection_end; }; u.triggerEvent = function (el, name) { let type = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : "Event"; let bubbles = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : true; let cancelable = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : true; const evt = document.createEvent(type); evt.initEvent(name, bubbles, cancelable); el.dispatchEvent(evt); }; u.getSelectValues = function (select) { const result = []; const options = select && select.options; for (var i = 0, iLen = options.length; i < iLen; i++) { const opt = options[i]; if (opt.selected) { result.push(opt.value || opt.text); } } return result; }; u.getRandomInt = function (max) { return Math.floor(Math.random() * Math.floor(max)); }; u.placeCaretAtEnd = function (textarea) { if (textarea !== document.activeElement) { textarea.focus(); } // Double the length because Opera is inconsistent about whether a carriage return is one character or two. const len = textarea.value.length * 2; // Timeout seems to be required for Blink setTimeout(() => textarea.setSelectionRange(len, len), 1); // Scroll to the bottom, in case we're in a tall textarea // (Necessary for Firefox and Chrome) this.scrollTop = 999999; }; function getUniqueId(suffix) { const uuid = 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) { const r = Math.random() * 16 | 0; const v = c === 'x' ? r : r & 0x3 | 0x8; return v.toString(16); }); if (typeof suffix === "string" || typeof suffix === "number") { return uuid + ":" + suffix; } else { return uuid; } } u.httpToGeoUri = function (text) { const replacement = 'geo:$1,$2'; return text.replace(settings_api.get("geouri_regex"), replacement); }; /** * Clears the specified timeout and interval. * @method u#clearTimers * @param {number} timeout - Id if the timeout to clear. * @param {number} interval - Id of the interval to clear. * @private * @copyright Simen Bekkhus 2016 * @license MIT */ function clearTimers(timeout, interval) { clearTimeout(timeout); clearInterval(interval); } /** * Creates a {@link Promise} that resolves if the passed in function returns a truthy value. * Rejects if it throws or does not return truthy within the given max_wait. * @method u#waitUntil * @param {Function} func - The function called every check_delay, * and the result of which is the resolved value of the promise. * @param {number} [max_wait=300] - The time to wait before rejecting the promise. * @param {number} [check_delay=3] - The time to wait before each invocation of {func}. * @returns {Promise} A promise resolved with the value of func, * or rejected with the exception thrown by it or it times out. * @copyright Simen Bekkhus 2016 * @license MIT */ u.waitUntil = function (func) { let max_wait = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 300; let check_delay = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 3; // Run the function once without setting up any listeners in case it's already true try { const result = func(); if (result) { return Promise.resolve(result); } } catch (e) { return Promise.reject(e); } const promise = getOpenPromise(); const timeout_err = new Error(); function checker() { try { const result = func(); if (result) { clearTimers(max_wait_timeout, interval); promise.resolve(result); } } catch (e) { clearTimers(max_wait_timeout, interval); promise.reject(e); } } const interval = setInterval(checker, check_delay); function handler() { clearTimers(max_wait_timeout, interval); const err_msg = `Wait until promise timed out: \n\n${timeout_err.stack}`; console.trace(); headless_log.error(err_msg); promise.reject(new Error(err_msg)); } const max_wait_timeout = setTimeout(handler, max_wait); return promise; }; function setUnloadEvent() { 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/ shared_converse.unloadevent = 'pagehide'; } else if ('onbeforeunload' in window) { shared_converse.unloadevent = 'beforeunload'; } else if ('onunload' in window) { shared_converse.unloadevent = 'unload'; } } async function getLoginCredentialsFromBrowser() { try { const creds = await navigator.credentials.get({ 'password': true }); if (creds && creds.type == 'password' && u.isValidJID(creds.id)) { await setUserJID(creds.id); return { 'jid': creds.id, 'password': creds.password }; } } catch (e) { headless_log.error(e); } } function replacePromise(name) { const existing_promise = shared_converse.promises[name]; if (!existing_promise) { throw new Error(`Tried to replace non-existing promise: ${name}`); } if (existing_promise.replace) { const promise = getOpenPromise(); promise.replace = existing_promise.replace; shared_converse.promises[name] = promise; } else { headless_log.debug(`Not replacing promise "${name}"`); } } const core_element = document.createElement('div'); function decodeHTMLEntities(str) { if (str && typeof str === 'string') { core_element.innerHTML = purify_default().sanitize(str); str = core_element.textContent; core_element.textContent = ''; } return str; } /* harmony default export */ const utils_core = (Object.assign({ prefixMentions, isEmptyMessage, getUniqueId }, u)); ;// CONCATENATED MODULE: ./src/headless/shared/settings/constants.js /** * @typedef { Object } ConfigurationSettings * Converse's core configuration values * @property { Boolean } [allow_non_roster_messaging=false] * @property { Boolean } [allow_url_history_change=true] * @property { String } [assets_path='/dist'] * @property { ('login'|'prebind'|'anonymous'|'external') } [authentication='login'] * @property { Boolean } [auto_login=false] - Currently only used in connection with anonymous login * @property { Boolean } [auto_reconnect=true] * @property { Array} [blacklisted_plugins] * @property { Boolean } [clear_cache_on_logout=false] * @property { Object } [connection_options] * @property { String } [credentials_url] - URL from where login credentials can be fetched * @property { Boolean } [discover_connection_methods=true] * @property { RegExp } [geouri_regex] * @property { RegExp } [geouri_replacement='https - //www.openstreetmap.org/?mlat=$1&mlon=$2#map=18/$1/$2'] * @property { String } [i18n] * @property { String } [jid] * @property { Boolean } [keepalive=true] * @property { ('debug'|'info'|'eror') } [loglevel='info'] * @property { Array } [locales] * @property { String } [nickname] * @property { String } [password] * @property { ('IndexedDB'|'localStorage') } [persistent_store='IndexedDB'] * @property { String } [rid] * @property { Element } [root=window.document] * @property { String } [sid] * @property { Boolean } [singleton=false] * @property { Boolean } [strict_plugin_dependencies=false] * @property { ('overlayed'|'fullscreen'|'mobile') } [view_mode='overlayed'] * @property { String } [websocket_url] * @property { Array} [whitelisted_plugins] */ const DEFAULT_SETTINGS = { allow_non_roster_messaging: false, allow_url_history_change: true, assets_path: '/dist', authentication: 'login', // Available values are "login", "prebind", "anonymous" and "external". auto_login: false, // Currently only used in connection with anonymous login auto_reconnect: true, blacklisted_plugins: [], clear_cache_on_logout: false, connection_options: {}, credentials_url: null, // URL from where login credentials can be fetched discover_connection_methods: true, geouri_regex: /https\:\/\/www.openstreetmap.org\/.*#map=[0-9]+\/([\-0-9.]+)\/([\-0-9.]+)\S*/g, geouri_replacement: 'https://www.openstreetmap.org/?mlat=$1&mlon=$2#map=18/$1/$2', i18n: undefined, jid: undefined, keepalive: true, loglevel: 'info', locales: ['af', 'ar', 'bg', 'ca', 'cs', 'da', 'de', 'el', 'eo', 'es', 'eu', 'en', 'fa', 'fi', 'fr', 'gl', 'he', 'hi', 'hu', 'id', 'it', 'ja', 'lt', 'nb', 'nl', 'mr', 'oc', 'pl', 'pt', 'pt_BR', 'ro', 'ru', 'sv', 'th', 'tr', 'uk', 'vi', 'zh_CN', 'zh_TW'], nickname: undefined, password: undefined, persistent_store: 'IndexedDB', rid: undefined, root: window.document, sid: undefined, singleton: false, strict_plugin_dependencies: false, view_mode: 'overlayed', // Choices are 'overlayed', 'fullscreen', 'mobile' websocket_url: undefined, whitelisted_plugins: [] }; ;// CONCATENATED MODULE: ./src/headless/shared/settings/utils.js let app_settings; let init_settings = {}; // Container for settings passed in via converse.initialize let user_settings; // User settings, populated via api.users.settings function getAppSettings() { return app_settings; } function initAppSettings(settings) { init_settings = settings; app_settings = {}; Object.assign(app_settings, Events); // Allow only whitelisted settings to be overwritten via converse.initialize const allowed_settings = lodash_es_pick(settings, Object.keys(DEFAULT_SETTINGS)); lodash_es_assignIn(app_settings, DEFAULT_SETTINGS, allowed_settings); } function getInitSettings() { return init_settings; } function getAppSetting(key) { if (Object.keys(DEFAULT_SETTINGS).includes(key)) { return app_settings[key]; } } function extendAppSettings(settings) { utils_core.merge(DEFAULT_SETTINGS, settings); // When updating the settings, we need to avoid overwriting the // initialization_settings (i.e. the settings passed in via converse.initialize). const allowed_keys = Object.keys(lodash_es_pick(settings, Object.keys(DEFAULT_SETTINGS))); const allowed_site_settings = lodash_es_pick(init_settings, allowed_keys); const updated_settings = lodash_es_assignIn(lodash_es_pick(settings, allowed_keys), allowed_site_settings); utils_core.merge(app_settings, updated_settings); } function registerListener(name, func, context) { app_settings.on(name, func, context); } function unregisterListener(name, func) { app_settings.off(name, func); } function updateAppSettings(key, val) { if (key == null) return this; // eslint-disable-line no-eq-null let attrs; if (lodash_es_isObject(key)) { attrs = key; } else if (typeof key === 'string') { attrs = {}; attrs[key] = val; } const allowed_keys = Object.keys(lodash_es_pick(attrs, Object.keys(DEFAULT_SETTINGS))); const changed = {}; allowed_keys.forEach(k => { const val = attrs[k]; if (!lodash_es_isEqual(app_settings[k], val)) { changed[k] = val; app_settings[k] = val; } }); Object.keys(changed).forEach(k => app_settings.trigger('change:' + k, changed[k])); app_settings.trigger('change', changed); } /** * @async */ function initUserSettings() { var _user_settings; if (!shared_converse.bare_jid) { const msg = "No JID to fetch user settings for"; headless_log.error(msg); throw Error(msg); } if (!((_user_settings = user_settings) !== null && _user_settings !== void 0 && _user_settings.fetched)) { const id = `converse.user-settings.${shared_converse.bare_jid}`; user_settings = new Model({ id }); initStorage(user_settings, id); user_settings.fetched = user_settings.fetch({ 'promise': true }); } return user_settings.fetched; } async function getUserSettings() { await initUserSettings(); return user_settings; } async function updateUserSettings(data, options) { await initUserSettings(); return user_settings.save(data, options); } async function clearUserSettings() { await initUserSettings(); return user_settings.clear(); } ;// CONCATENATED MODULE: ./src/headless/shared/_converse.js /** * A private, closured object containing the private api (via {@link _converse.api}) * as well as private methods and internal data-structures. * @global * @namespace _converse */ const _converse = { log: headless_log, CONNECTION_STATUS: CONNECTION_STATUS, templates: {}, promises: { 'initialized': getOpenPromise() }, 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 }, ANONYMOUS: 'anonymous', CLOSED: 'closed', EXTERNAL: 'external', LOGIN: 'login', LOGOUT: 'logout', OPENED: 'opened', PREBIND: 'prebind', /** * @constant * @type { integer } */ STANZA_TIMEOUT: 20000, SUCCESS: 'success', FAILURE: 'failure', // Generated from css/images/user.svg DEFAULT_IMAGE_TYPE: 'image/svg+xml', DEFAULT_IMAGE: "PD94bWwgdmVyc2lvbj0iMS4wIj8+CjxzdmcgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiB3aWR0aD0iMTI4IiBoZWlnaHQ9IjEyOCI+CiA8cmVjdCB3aWR0aD0iMTI4IiBoZWlnaHQ9IjEyOCIgZmlsbD0iIzU1NSIvPgogPGNpcmNsZSBjeD0iNjQiIGN5PSI0MSIgcj0iMjQiIGZpbGw9IiNmZmYiLz4KIDxwYXRoIGQ9Im0yOC41IDExMiB2LTEyIGMwLTEyIDEwLTI0IDI0LTI0IGgyMyBjMTQgMCAyNCAxMiAyNCAyNCB2MTIiIGZpbGw9IiNmZmYiLz4KPC9zdmc+Cg==", TIMEOUTS: { // Set as module attr so that we can override in tests. PAUSED: 10000, INACTIVE: 90000 }, // XEP-0085 Chat states // https://xmpp.org/extensions/xep-0085.html INACTIVE: 'inactive', ACTIVE: 'active', COMPOSING: 'composing', PAUSED: 'paused', GONE: 'gone', // Chat types PRIVATE_CHAT_TYPE: 'chatbox', CHATROOMS_TYPE: 'chatroom', HEADLINES_TYPE: 'headline', CONTROLBOX_TYPE: 'controlbox', default_connection_options: { 'explicitResourceBinding': true }, router: new Router(), TimeoutError: TimeoutError, isTestEnv: () => { return getInitSettings()['bosh_service_url'] === 'montague.lit/http-bind'; }, getDefaultStore: getDefaultStore, createStore: createStore, /** * Translate the given string based on the current locale. * @method __ * @private * @memberOf _converse * @param { String } str */ '__': function () { return i18n.__(...arguments); }, /** * A no-op method which is used to signal to gettext that the passed in string * should be included in the pot translation file. * * In contrast to the double-underscore method, the triple underscore method * doesn't actually translate the strings. * * One reason for this method might be because we're using strings we cannot * send to the translation function because they require variable interpolation * and we don't yet have the variables at scan time. * * @method ___ * @private * @memberOf _converse * @param { String } str */ '___': str => str }; /* harmony default export */ const shared_converse = (_converse); // EXTERNAL MODULE: ./node_modules/dayjs/plugin/advancedFormat.js var advancedFormat = __webpack_require__(8734); var advancedFormat_default = /*#__PURE__*/__webpack_require__.n(advancedFormat); ;// CONCATENATED MODULE: ./src/headless/shared/connection/api.js /** * This grouping collects API functions related to the XMPP connection. * * @namespace _converse.api.connection * @memberOf _converse.api */ /* harmony default export */ const api = ({ /** * @method _converse.api.connection.connected * @memberOf _converse.api.connection * @returns {boolean} Whether there is an established connection or not. */ connected() { var _converse$connection; return (shared_converse === null || shared_converse === void 0 ? void 0 : (_converse$connection = shared_converse.connection) === null || _converse$connection === void 0 ? void 0 : _converse$connection.connected) && true; }, /** * Terminates the connection. * * @method _converse.api.connection.disconnect * @memberOf _converse.api.connection */ disconnect() { if (shared_converse.connection) { shared_converse.connection.disconnect(); } }, /** * Can be called once the XMPP connection has dropped and we want * to attempt reconnection. * Only needs to be called once, if reconnect fails Converse will * attempt to reconnect every two seconds, alternating between BOSH and * Websocket if URLs for both were provided. * @method reconnect * @memberOf _converse.api.connection */ reconnect() { const { __, connection } = shared_converse; connection.setConnectionStatus(Strophe.Status.RECONNECTING, __('The connection has dropped, attempting to reconnect.')); if (connection !== null && connection !== void 0 && connection.reconnecting) { return connection.debouncedReconnect(); } else { return connection.reconnect(); } }, /** * Utility method to determine the type of connection we have * @method isType * @memberOf _converse.api.connection * @returns {boolean} */ isType(type) { return shared_converse.connection.isType(type); } }); // EXTERNAL MODULE: ./node_modules/dayjs/dayjs.min.js var dayjs_min = __webpack_require__(7484); var dayjs_min_default = /*#__PURE__*/__webpack_require__.n(dayjs_min); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseInvoke.js /** * The base implementation of `_.invoke` without support for individual * method arguments. * * @private * @param {Object} object The object to query. * @param {Array|string} path The path of the method to invoke. * @param {Array} args The arguments to invoke the method with. * @returns {*} Returns the result of the invoked method. */ function baseInvoke(object, path, args) { path = _castPath(path, object); object = _parent(object, path); var func = object == null ? object : object[_toKey(lodash_es_last(path))]; return func == null ? undefined : _apply(func, object, args); } /* harmony default export */ const _baseInvoke = (baseInvoke); ;// CONCATENATED MODULE: ./node_modules/lodash-es/invoke.js /** * Invokes the method at `path` of `object`. * * @static * @memberOf _ * @since 4.0.0 * @category Object * @param {Object} object The object to query. * @param {Array|string} path The path of the method to invoke. * @param {...*} [args] The arguments to invoke the method with. * @returns {*} Returns the result of the invoked method. * @example * * var object = { 'a': [{ 'b': { 'c': [1, 2, 3, 4] } }] }; * * _.invoke(object, 'a[0].b.c.slice', 1, 3); * // => [2, 3] */ var invoke = _baseRest(_baseInvoke); /* harmony default export */ const lodash_es_invoke = (invoke); ;// CONCATENATED MODULE: ./node_modules/pluggable.js/src/pluggable.js /* ____ __ __ __ _ / __ \/ /_ __ ___ ___ ____ _/ /_ / /__ (_)____ / /_/ / / / / / __ \/ __ \/ __/ / __ \/ / _ \ / / ___/ / ____/ / /_/ / /_/ / /_/ / /_/ / /_/ / / __/ / (__ ) /_/ /_/\__,_/\__, /\__, /\__/_/_.___/_/\___(_)_/ /____/ /____//____/ /___/ */ // Pluggable.js lets you to make your Javascript code pluggable while still // keeping sensitive objects and data private through closures. // `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. 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); } for (var _len = arguments.length, args = new Array(_len > 4 ? _len - 4 : 0), _key = 4; _key < _len; _key++) { args[_key - 4] = arguments[_key]; } return value.apply(this, args); } // 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). class PluginSocket { constructor(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 = []; } // `_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(key, plugin) { const value = plugin.overrides[key]; if (typeof value === "function") { const default_super = {}; default_super[this.name] = this.plugged; const super_method = this.plugged[key]; this.plugged[key] = function () { for (var _len2 = arguments.length, args = new Array(_len2), _key2 = 0; _key2 < _len2; _key2++) { args[_key2] = arguments[_key2]; } return wrappedOverride.apply(this, [key, value, super_method, default_super, ...args]); }; } else { this.plugged[key] = value; } } _extendObject(obj, attributes) { if (!obj.prototype.__super__) { obj.prototype.__super__ = {}; obj.prototype.__super__[this.name] = this.plugged; } for (const [key, value] of Object.entries(attributes)) { if (key === 'events') { obj.prototype[key] = Object.assign(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. const default_super = {}; default_super[this.name] = this.plugged; const super_method = obj.prototype[key]; obj.prototype[key] = function () { for (var _len3 = arguments.length, args = new Array(_len3), _key3 = 0; _key3 < _len3; _key3++) { args[_key3] = arguments[_key3]; } return wrappedOverride.apply(this, [key, value, super_method, default_super, ...args]); }; } else { obj.prototype[key] = value; } } } // Plugins can specify dependencies (by means of the // `dependencies` list attribute) which refers to dependencies // which will be initialized first, before the plugin itself gets initialized. // // If `strict_plugin_dependencies` is set to `false` (on the object being // made pluggable), then no error will be thrown if any of these plugins aren't // available. loadPluginDependencies(plugin) { var _plugin$dependencies; (_plugin$dependencies = plugin.dependencies) === null || _plugin$dependencies === void 0 ? void 0 : _plugin$dependencies.forEach(name => { const dep = this.plugins[name]; if (dep) { var _dep$dependencies; if ((_dep$dependencies = dep.dependencies) !== null && _dep$dependencies !== void 0 && _dep$dependencies.includes(plugin.__name__)) { /* FIXME: circular dependency checking is only one level deep. */ throw "Found a circular dependency between the plugins \"" + plugin.__name__ + "\" and \"" + name + "\""; } this.initializePlugin(dep); } else { this.throwUndefinedDependencyError("Could not find dependency \"" + name + "\" " + "for the plugin \"" + plugin.__name__ + "\". " + "If it's needed, make sure it's loaded by require.js"); } }); } throwUndefinedDependencyError(msg) { if (this.plugged.strict_plugin_dependencies) { throw msg; } else { if (console.warn) { console.warn(msg); } else { console.log(msg); } } } // `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(plugin) { Object.keys(plugin.overrides || {}).forEach(key => { const override = plugin.overrides[key]; if (typeof override === "object") { if (typeof this.plugged[key] === 'undefined') { this.throwUndefinedDependencyError(`Plugin "${plugin.__name__}" tried to override "${key}" but it's not found.`); } else { this._extendObject(this.plugged[key], override); } } else { this._overrideAttribute(key, plugin); } }); } // `initializePlugin` applies the overrides (if any) defined on all // the registered plugins and then calls the initialize method of the plugin initializePlugin(plugin) { var _plugin$enabled; if (!Object.keys(this.allowed_plugins).includes(plugin.__name__)) { /* Don't initialize disallowed plugins. */ return; } if (this.initialized_plugins.includes(plugin.__name__)) { /* Don't initialize plugins twice, otherwise we get * infinite recursion in overridden methods. */ return; } if (typeof plugin.enabled === 'boolean' && plugin.enabled || (_plugin$enabled = plugin.enabled) !== null && _plugin$enabled !== void 0 && _plugin$enabled.call(plugin, this.plugged) || plugin.enabled == null) { // isNil Object.assign(plugin, this.properties); if (plugin.dependencies) { this.loadPluginDependencies(plugin); } this.applyOverrides(plugin); if (typeof plugin.initialize === "function") { plugin.initialize.bind(plugin)(this); } this.initialized_plugins.push(plugin.__name__); } } // `registerPlugin` registers (or inserts, if you'd like) a plugin, // by adding it to the `plugins` map on the PluginSocket instance. 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; } // `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() { let properties = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; let whitelist = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : []; let blacklist = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : []; if (!Object.keys(this.plugins).length) { return; } this.properties = properties; this.allowed_plugins = {}; for (const [key, plugin] of Object.entries(this.plugins)) { if ((!whitelist.length || whitelist.includes(key)) && !blacklist.includes(key)) { this.allowed_plugins[key] = plugin; } } Object.values(this.allowed_plugins).forEach(o => this.initializePlugin(o)); } } 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'; } object[attrname] = new PluginSocket(object, name); return object; } /* harmony default export */ const pluggable = ({ enable }); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_arrayAggregator.js /** * A specialized version of `baseAggregator` for arrays. * * @private * @param {Array} [array] The array to iterate over. * @param {Function} setter The function to set `accumulator` values. * @param {Function} iteratee The iteratee to transform keys. * @param {Object} accumulator The initial aggregated object. * @returns {Function} Returns `accumulator`. */ function arrayAggregator(array, setter, iteratee, accumulator) { var index = -1, length = array == null ? 0 : array.length; while (++index < length) { var value = array[index]; setter(accumulator, value, iteratee(value), array); } return accumulator; } /* harmony default export */ const _arrayAggregator = (arrayAggregator); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseAggregator.js /** * Aggregates elements of `collection` on `accumulator` with keys transformed * by `iteratee` and values set by `setter`. * * @private * @param {Array|Object} collection The collection to iterate over. * @param {Function} setter The function to set `accumulator` values. * @param {Function} iteratee The iteratee to transform keys. * @param {Object} accumulator The initial aggregated object. * @returns {Function} Returns `accumulator`. */ function baseAggregator(collection, setter, iteratee, accumulator) { _baseEach(collection, function(value, key, collection) { setter(accumulator, value, iteratee(value), collection); }); return accumulator; } /* harmony default export */ const _baseAggregator = (baseAggregator); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_createAggregator.js /** * Creates a function like `_.groupBy`. * * @private * @param {Function} setter The function to set accumulator values. * @param {Function} [initializer] The accumulator object initializer. * @returns {Function} Returns the new aggregator function. */ function createAggregator(setter, initializer) { return function(collection, iteratee) { var func = lodash_es_isArray(collection) ? _arrayAggregator : _baseAggregator, accumulator = initializer ? initializer() : {}; return func(collection, setter, _baseIteratee(iteratee, 2), accumulator); }; } /* harmony default export */ const _createAggregator = (createAggregator); ;// CONCATENATED MODULE: ./node_modules/lodash-es/countBy.js /** Used for built-in method references. */ var countBy_objectProto = Object.prototype; /** Used to check objects for own properties. */ var countBy_hasOwnProperty = countBy_objectProto.hasOwnProperty; /** * Creates an object composed of keys generated from the results of running * each element of `collection` thru `iteratee`. The corresponding value of * each key is the number of times the key was returned by `iteratee`. The * iteratee is invoked with one argument: (value). * * @static * @memberOf _ * @since 0.5.0 * @category Collection * @param {Array|Object} collection The collection to iterate over. * @param {Function} [iteratee=_.identity] The iteratee to transform keys. * @returns {Object} Returns the composed aggregate object. * @example * * _.countBy([6.1, 4.2, 6.3], Math.floor); * // => { '4': 1, '6': 2 } * * // The `_.property` iteratee shorthand. * _.countBy(['one', 'two', 'three'], 'length'); * // => { '3': 2, '5': 1 } */ var countBy = _createAggregator(function(result, value, key) { if (countBy_hasOwnProperty.call(result, key)) { ++result[key]; } else { _baseAssignValue(result, key, 1); } }); /* harmony default export */ const lodash_es_countBy = (countBy); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseFindIndex.js /** * The base implementation of `_.findIndex` and `_.findLastIndex` without * support for iteratee shorthands. * * @private * @param {Array} array The array to inspect. * @param {Function} predicate The function invoked per iteration. * @param {number} fromIndex The index to search from. * @param {boolean} [fromRight] Specify iterating from right to left. * @returns {number} Returns the index of the matched value, else `-1`. */ function baseFindIndex(array, predicate, fromIndex, fromRight) { var length = array.length, index = fromIndex + (fromRight ? 1 : -1); while ((fromRight ? index-- : ++index < length)) { if (predicate(array[index], index, array)) { return index; } } return -1; } /* harmony default export */ const _baseFindIndex = (baseFindIndex); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseIsNaN.js /** * The base implementation of `_.isNaN` without support for number objects. * * @private * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is `NaN`, else `false`. */ function baseIsNaN(value) { return value !== value; } /* harmony default export */ const _baseIsNaN = (baseIsNaN); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_strictIndexOf.js /** * A specialized version of `_.indexOf` which performs strict equality * comparisons of values, i.e. `===`. * * @private * @param {Array} array The array to inspect. * @param {*} value The value to search for. * @param {number} fromIndex The index to search from. * @returns {number} Returns the index of the matched value, else `-1`. */ function strictIndexOf(array, value, fromIndex) { var index = fromIndex - 1, length = array.length; while (++index < length) { if (array[index] === value) { return index; } } return -1; } /* harmony default export */ const _strictIndexOf = (strictIndexOf); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseIndexOf.js /** * The base implementation of `_.indexOf` without `fromIndex` bounds checks. * * @private * @param {Array} array The array to inspect. * @param {*} value The value to search for. * @param {number} fromIndex The index to search from. * @returns {number} Returns the index of the matched value, else `-1`. */ function baseIndexOf(array, value, fromIndex) { return value === value ? _strictIndexOf(array, value, fromIndex) : _baseFindIndex(array, _baseIsNaN, fromIndex); } /* harmony default export */ const _baseIndexOf = (baseIndexOf); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_arrayIncludes.js /** * A specialized version of `_.includes` for arrays without support for * specifying an index to search from. * * @private * @param {Array} [array] The array to inspect. * @param {*} target The value to search for. * @returns {boolean} Returns `true` if `target` is found, else `false`. */ function arrayIncludes(array, value) { var length = array == null ? 0 : array.length; return !!length && _baseIndexOf(array, value, 0) > -1; } /* harmony default export */ const _arrayIncludes = (arrayIncludes); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_arrayIncludesWith.js /** * This function is like `arrayIncludes` except that it accepts a comparator. * * @private * @param {Array} [array] The array to inspect. * @param {*} target The value to search for. * @param {Function} comparator The comparator invoked per element. * @returns {boolean} Returns `true` if `target` is found, else `false`. */ function arrayIncludesWith(array, value, comparator) { var index = -1, length = array == null ? 0 : array.length; while (++index < length) { if (comparator(value, array[index])) { return true; } } return false; } /* harmony default export */ const _arrayIncludesWith = (arrayIncludesWith); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseDifference.js /** Used as the size to enable large array optimizations. */ var _baseDifference_LARGE_ARRAY_SIZE = 200; /** * The base implementation of methods like `_.difference` without support * for excluding multiple arrays or iteratee shorthands. * * @private * @param {Array} array The array to inspect. * @param {Array} values The values to exclude. * @param {Function} [iteratee] The iteratee invoked per element. * @param {Function} [comparator] The comparator invoked per element. * @returns {Array} Returns the new array of filtered values. */ function baseDifference(array, values, iteratee, comparator) { var index = -1, includes = _arrayIncludes, isCommon = true, length = array.length, result = [], valuesLength = values.length; if (!length) { return result; } if (iteratee) { values = _arrayMap(values, _baseUnary(iteratee)); } if (comparator) { includes = _arrayIncludesWith; isCommon = false; } else if (values.length >= _baseDifference_LARGE_ARRAY_SIZE) { includes = _cacheHas; isCommon = false; values = new _SetCache(values); } outer: while (++index < length) { var value = array[index], computed = iteratee == null ? value : iteratee(value); value = (comparator || value !== 0) ? value : 0; if (isCommon && computed === computed) { var valuesIndex = valuesLength; while (valuesIndex--) { if (values[valuesIndex] === computed) { continue outer; } } result.push(value); } else if (!includes(values, computed, comparator)) { result.push(value); } } return result; } /* harmony default export */ const _baseDifference = (baseDifference); ;// CONCATENATED MODULE: ./node_modules/lodash-es/difference.js /** * Creates an array of `array` values not included in the other given arrays * using [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero) * for equality comparisons. The order and references of result values are * determined by the first array. * * **Note:** Unlike `_.pullAll`, this method returns a new array. * * @static * @memberOf _ * @since 0.1.0 * @category Array * @param {Array} array The array to inspect. * @param {...Array} [values] The values to exclude. * @returns {Array} Returns the new array of filtered values. * @see _.without, _.xor * @example * * _.difference([2, 1], [2, 3]); * // => [1] */ var difference = _baseRest(function(array, values) { return lodash_es_isArrayLikeObject(array) ? _baseDifference(array, _baseFlatten(values, 1, lodash_es_isArrayLikeObject, true)) : []; }); /* harmony default export */ const lodash_es_difference = (difference); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_arrayEvery.js /** * A specialized version of `_.every` for arrays without support for * iteratee shorthands. * * @private * @param {Array} [array] The array to iterate over. * @param {Function} predicate The function invoked per iteration. * @returns {boolean} Returns `true` if all elements pass the predicate check, * else `false`. */ function arrayEvery(array, predicate) { var index = -1, length = array == null ? 0 : array.length; while (++index < length) { if (!predicate(array[index], index, array)) { return false; } } return true; } /* harmony default export */ const _arrayEvery = (arrayEvery); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseEvery.js /** * The base implementation of `_.every` without support for iteratee shorthands. * * @private * @param {Array|Object} collection The collection to iterate over. * @param {Function} predicate The function invoked per iteration. * @returns {boolean} Returns `true` if all elements pass the predicate check, * else `false` */ function baseEvery(collection, predicate) { var result = true; _baseEach(collection, function(value, index, collection) { result = !!predicate(value, index, collection); return result; }); return result; } /* harmony default export */ const _baseEvery = (baseEvery); ;// CONCATENATED MODULE: ./node_modules/lodash-es/every.js /** * Checks if `predicate` returns truthy for **all** elements of `collection`. * Iteration is stopped once `predicate` returns falsey. The predicate is * invoked with three arguments: (value, index|key, collection). * * **Note:** This method returns `true` for * [empty collections](https://en.wikipedia.org/wiki/Empty_set) because * [everything is true](https://en.wikipedia.org/wiki/Vacuous_truth) of * elements of empty collections. * * @static * @memberOf _ * @since 0.1.0 * @category Collection * @param {Array|Object} collection The collection to iterate over. * @param {Function} [predicate=_.identity] The function invoked per iteration. * @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`. * @returns {boolean} Returns `true` if all elements pass the predicate check, * else `false`. * @example * * _.every([true, 1, null, 'yes'], Boolean); * // => false * * var users = [ * { 'user': 'barney', 'age': 36, 'active': false }, * { 'user': 'fred', 'age': 40, 'active': false } * ]; * * // The `_.matches` iteratee shorthand. * _.every(users, { 'user': 'barney', 'active': false }); * // => false * * // The `_.matchesProperty` iteratee shorthand. * _.every(users, ['active', false]); * // => true * * // The `_.property` iteratee shorthand. * _.every(users, 'active'); * // => false */ function every(collection, predicate, guard) { var func = lodash_es_isArray(collection) ? _arrayEvery : _baseEvery; if (guard && _isIterateeCall(collection, predicate, guard)) { predicate = undefined; } return func(collection, _baseIteratee(predicate, 3)); } /* harmony default export */ const lodash_es_every = (every); ;// CONCATENATED MODULE: ./node_modules/lodash-es/findIndex.js /* Built-in method references for those with the same name as other `lodash` methods. */ var findIndex_nativeMax = Math.max; /** * This method is like `_.find` except that it returns the index of the first * element `predicate` returns truthy for instead of the element itself. * * @static * @memberOf _ * @since 1.1.0 * @category Array * @param {Array} array The array to inspect. * @param {Function} [predicate=_.identity] The function invoked per iteration. * @param {number} [fromIndex=0] The index to search from. * @returns {number} Returns the index of the found element, else `-1`. * @example * * var users = [ * { 'user': 'barney', 'active': false }, * { 'user': 'fred', 'active': false }, * { 'user': 'pebbles', 'active': true } * ]; * * _.findIndex(users, function(o) { return o.user == 'barney'; }); * // => 0 * * // The `_.matches` iteratee shorthand. * _.findIndex(users, { 'user': 'fred', 'active': false }); * // => 1 * * // The `_.matchesProperty` iteratee shorthand. * _.findIndex(users, ['active', false]); * // => 0 * * // The `_.property` iteratee shorthand. * _.findIndex(users, 'active'); * // => 2 */ function findIndex(array, predicate, fromIndex) { var length = array == null ? 0 : array.length; if (!length) { return -1; } var index = fromIndex == null ? 0 : lodash_es_toInteger(fromIndex); if (index < 0) { index = findIndex_nativeMax(length + index, 0); } return _baseFindIndex(array, _baseIteratee(predicate, 3), index); } /* harmony default export */ const lodash_es_findIndex = (findIndex); ;// CONCATENATED MODULE: ./node_modules/lodash-es/findLastIndex.js /* Built-in method references for those with the same name as other `lodash` methods. */ var findLastIndex_nativeMax = Math.max, findLastIndex_nativeMin = Math.min; /** * This method is like `_.findIndex` except that it iterates over elements * of `collection` from right to left. * * @static * @memberOf _ * @since 2.0.0 * @category Array * @param {Array} array The array to inspect. * @param {Function} [predicate=_.identity] The function invoked per iteration. * @param {number} [fromIndex=array.length-1] The index to search from. * @returns {number} Returns the index of the found element, else `-1`. * @example * * var users = [ * { 'user': 'barney', 'active': true }, * { 'user': 'fred', 'active': false }, * { 'user': 'pebbles', 'active': false } * ]; * * _.findLastIndex(users, function(o) { return o.user == 'pebbles'; }); * // => 2 * * // The `_.matches` iteratee shorthand. * _.findLastIndex(users, { 'user': 'barney', 'active': true }); * // => 0 * * // The `_.matchesProperty` iteratee shorthand. * _.findLastIndex(users, ['active', false]); * // => 2 * * // The `_.property` iteratee shorthand. * _.findLastIndex(users, 'active'); * // => 0 */ function findLastIndex(array, predicate, fromIndex) { var length = array == null ? 0 : array.length; if (!length) { return -1; } var index = length - 1; if (fromIndex !== undefined) { index = lodash_es_toInteger(fromIndex); index = fromIndex < 0 ? findLastIndex_nativeMax(length + index, 0) : findLastIndex_nativeMin(index, length - 1); } return _baseFindIndex(array, _baseIteratee(predicate, 3), index, true); } /* harmony default export */ const lodash_es_findLastIndex = (findLastIndex); ;// CONCATENATED MODULE: ./node_modules/lodash-es/groupBy.js /** Used for built-in method references. */ var groupBy_objectProto = Object.prototype; /** Used to check objects for own properties. */ var groupBy_hasOwnProperty = groupBy_objectProto.hasOwnProperty; /** * Creates an object composed of keys generated from the results of running * each element of `collection` thru `iteratee`. The order of grouped values * is determined by the order they occur in `collection`. The corresponding * value of each key is an array of elements responsible for generating the * key. The iteratee is invoked with one argument: (value). * * @static * @memberOf _ * @since 0.1.0 * @category Collection * @param {Array|Object} collection The collection to iterate over. * @param {Function} [iteratee=_.identity] The iteratee to transform keys. * @returns {Object} Returns the composed aggregate object. * @example * * _.groupBy([6.1, 4.2, 6.3], Math.floor); * // => { '4': [4.2], '6': [6.1, 6.3] } * * // The `_.property` iteratee shorthand. * _.groupBy(['one', 'two', 'three'], 'length'); * // => { '3': ['one', 'two'], '5': ['three'] } */ var groupBy = _createAggregator(function(result, value, key) { if (groupBy_hasOwnProperty.call(result, key)) { result[key].push(value); } else { _baseAssignValue(result, key, [value]); } }); /* harmony default export */ const lodash_es_groupBy = (groupBy); ;// CONCATENATED MODULE: ./node_modules/lodash-es/indexOf.js /* Built-in method references for those with the same name as other `lodash` methods. */ var indexOf_nativeMax = Math.max; /** * Gets the index at which the first occurrence of `value` is found in `array` * using [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero) * for equality comparisons. If `fromIndex` is negative, it's used as the * offset from the end of `array`. * * @static * @memberOf _ * @since 0.1.0 * @category Array * @param {Array} array The array to inspect. * @param {*} value The value to search for. * @param {number} [fromIndex=0] The index to search from. * @returns {number} Returns the index of the matched value, else `-1`. * @example * * _.indexOf([1, 2, 1, 2], 2); * // => 1 * * // Search from the `fromIndex`. * _.indexOf([1, 2, 1, 2], 2, 2); * // => 3 */ function indexOf(array, value, fromIndex) { var length = array == null ? 0 : array.length; if (!length) { return -1; } var index = fromIndex == null ? 0 : lodash_es_toInteger(fromIndex); if (index < 0) { index = indexOf_nativeMax(length + index, 0); } return _baseIndexOf(array, value, index); } /* harmony default export */ const lodash_es_indexOf = (indexOf); ;// CONCATENATED MODULE: ./node_modules/lodash-es/keyBy.js /** * Creates an object composed of keys generated from the results of running * each element of `collection` thru `iteratee`. The corresponding value of * each key is the last element responsible for generating the key. The * iteratee is invoked with one argument: (value). * * @static * @memberOf _ * @since 4.0.0 * @category Collection * @param {Array|Object} collection The collection to iterate over. * @param {Function} [iteratee=_.identity] The iteratee to transform keys. * @returns {Object} Returns the composed aggregate object. * @example * * var array = [ * { 'dir': 'left', 'code': 97 }, * { 'dir': 'right', 'code': 100 } * ]; * * _.keyBy(array, function(o) { * return String.fromCharCode(o.code); * }); * // => { 'a': { 'dir': 'left', 'code': 97 }, 'd': { 'dir': 'right', 'code': 100 } } * * _.keyBy(array, 'dir'); * // => { 'left': { 'dir': 'left', 'code': 97 }, 'right': { 'dir': 'right', 'code': 100 } } */ var keyBy = _createAggregator(function(result, value, key) { _baseAssignValue(result, key, value); }); /* harmony default export */ const lodash_es_keyBy = (keyBy); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_strictLastIndexOf.js /** * A specialized version of `_.lastIndexOf` which performs strict equality * comparisons of values, i.e. `===`. * * @private * @param {Array} array The array to inspect. * @param {*} value The value to search for. * @param {number} fromIndex The index to search from. * @returns {number} Returns the index of the matched value, else `-1`. */ function strictLastIndexOf(array, value, fromIndex) { var index = fromIndex + 1; while (index--) { if (array[index] === value) { return index; } } return index; } /* harmony default export */ const _strictLastIndexOf = (strictLastIndexOf); ;// CONCATENATED MODULE: ./node_modules/lodash-es/lastIndexOf.js /* Built-in method references for those with the same name as other `lodash` methods. */ var lastIndexOf_nativeMax = Math.max, lastIndexOf_nativeMin = Math.min; /** * This method is like `_.indexOf` except that it iterates over elements of * `array` from right to left. * * @static * @memberOf _ * @since 0.1.0 * @category Array * @param {Array} array The array to inspect. * @param {*} value The value to search for. * @param {number} [fromIndex=array.length-1] The index to search from. * @returns {number} Returns the index of the matched value, else `-1`. * @example * * _.lastIndexOf([1, 2, 1, 2], 2); * // => 3 * * // Search from the `fromIndex`. * _.lastIndexOf([1, 2, 1, 2], 2, 2); * // => 1 */ function lastIndexOf(array, value, fromIndex) { var length = array == null ? 0 : array.length; if (!length) { return -1; } var index = length; if (fromIndex !== undefined) { index = lodash_es_toInteger(fromIndex); index = index < 0 ? lastIndexOf_nativeMax(length + index, 0) : lastIndexOf_nativeMin(index, length - 1); } return value === value ? _strictLastIndexOf(array, value, index) : _baseFindIndex(array, _baseIsNaN, index, true); } /* harmony default export */ const lodash_es_lastIndexOf = (lastIndexOf); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseMap.js /** * The base implementation of `_.map` without support for iteratee shorthands. * * @private * @param {Array|Object} collection The collection to iterate over. * @param {Function} iteratee The function invoked per iteration. * @returns {Array} Returns the new mapped array. */ function baseMap(collection, iteratee) { var index = -1, result = lodash_es_isArrayLike(collection) ? Array(collection.length) : []; _baseEach(collection, function(value, key, collection) { result[++index] = iteratee(value, key, collection); }); return result; } /* harmony default export */ const _baseMap = (baseMap); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseSortBy.js /** * The base implementation of `_.sortBy` which uses `comparer` to define the * sort order of `array` and replaces criteria objects with their corresponding * values. * * @private * @param {Array} array The array to sort. * @param {Function} comparer The function to define sort order. * @returns {Array} Returns `array`. */ function baseSortBy(array, comparer) { var length = array.length; array.sort(comparer); while (length--) { array[length] = array[length].value; } return array; } /* harmony default export */ const _baseSortBy = (baseSortBy); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_compareAscending.js /** * Compares values to sort them in ascending order. * * @private * @param {*} value The value to compare. * @param {*} other The other value to compare. * @returns {number} Returns the sort order indicator for `value`. */ function compareAscending(value, other) { if (value !== other) { var valIsDefined = value !== undefined, valIsNull = value === null, valIsReflexive = value === value, valIsSymbol = lodash_es_isSymbol(value); var othIsDefined = other !== undefined, othIsNull = other === null, othIsReflexive = other === other, othIsSymbol = lodash_es_isSymbol(other); if ((!othIsNull && !othIsSymbol && !valIsSymbol && value > other) || (valIsSymbol && othIsDefined && othIsReflexive && !othIsNull && !othIsSymbol) || (valIsNull && othIsDefined && othIsReflexive) || (!valIsDefined && othIsReflexive) || !valIsReflexive) { return 1; } if ((!valIsNull && !valIsSymbol && !othIsSymbol && value < other) || (othIsSymbol && valIsDefined && valIsReflexive && !valIsNull && !valIsSymbol) || (othIsNull && valIsDefined && valIsReflexive) || (!othIsDefined && valIsReflexive) || !othIsReflexive) { return -1; } } return 0; } /* harmony default export */ const _compareAscending = (compareAscending); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_compareMultiple.js /** * Used by `_.orderBy` to compare multiple properties of a value to another * and stable sort them. * * If `orders` is unspecified, all values are sorted in ascending order. Otherwise, * specify an order of "desc" for descending or "asc" for ascending sort order * of corresponding values. * * @private * @param {Object} object The object to compare. * @param {Object} other The other object to compare. * @param {boolean[]|string[]} orders The order to sort by for each property. * @returns {number} Returns the sort order indicator for `object`. */ function compareMultiple(object, other, orders) { var index = -1, objCriteria = object.criteria, othCriteria = other.criteria, length = objCriteria.length, ordersLength = orders.length; while (++index < length) { var result = _compareAscending(objCriteria[index], othCriteria[index]); if (result) { if (index >= ordersLength) { return result; } var order = orders[index]; return result * (order == 'desc' ? -1 : 1); } } // Fixes an `Array#sort` bug in the JS engine embedded in Adobe applications // that causes it, under certain circumstances, to provide the same value for // `object` and `other`. See https://github.com/jashkenas/underscore/pull/1247 // for more details. // // This also ensures a stable sort in V8 and other engines. // See https://bugs.chromium.org/p/v8/issues/detail?id=90 for more details. return object.index - other.index; } /* harmony default export */ const _compareMultiple = (compareMultiple); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseOrderBy.js /** * The base implementation of `_.orderBy` without param guards. * * @private * @param {Array|Object} collection The collection to iterate over. * @param {Function[]|Object[]|string[]} iteratees The iteratees to sort by. * @param {string[]} orders The sort orders of `iteratees`. * @returns {Array} Returns the new sorted array. */ function baseOrderBy(collection, iteratees, orders) { if (iteratees.length) { iteratees = _arrayMap(iteratees, function(iteratee) { if (lodash_es_isArray(iteratee)) { return function(value) { return _baseGet(value, iteratee.length === 1 ? iteratee[0] : iteratee); } } return iteratee; }); } else { iteratees = [lodash_es_identity]; } var index = -1; iteratees = _arrayMap(iteratees, _baseUnary(_baseIteratee)); var result = _baseMap(collection, function(value, key, collection) { var criteria = _arrayMap(iteratees, function(iteratee) { return iteratee(value); }); return { 'criteria': criteria, 'index': ++index, 'value': value }; }); return _baseSortBy(result, function(object, other) { return _compareMultiple(object, other, orders); }); } /* harmony default export */ const _baseOrderBy = (baseOrderBy); ;// CONCATENATED MODULE: ./node_modules/lodash-es/sortBy.js /** * Creates an array of elements, sorted in ascending order by the results of * running each element in a collection thru each iteratee. This method * performs a stable sort, that is, it preserves the original sort order of * equal elements. The iteratees are invoked with one argument: (value). * * @static * @memberOf _ * @since 0.1.0 * @category Collection * @param {Array|Object} collection The collection to iterate over. * @param {...(Function|Function[])} [iteratees=[_.identity]] * The iteratees to sort by. * @returns {Array} Returns the new sorted array. * @example * * var users = [ * { 'user': 'fred', 'age': 48 }, * { 'user': 'barney', 'age': 36 }, * { 'user': 'fred', 'age': 30 }, * { 'user': 'barney', 'age': 34 } * ]; * * _.sortBy(users, [function(o) { return o.user; }]); * // => objects for [['barney', 36], ['barney', 34], ['fred', 48], ['fred', 30]] * * _.sortBy(users, ['user', 'age']); * // => objects for [['barney', 34], ['barney', 36], ['fred', 30], ['fred', 48]] */ var sortBy = _baseRest(function(collection, iteratees) { if (collection == null) { return []; } var length = iteratees.length; if (length > 1 && _isIterateeCall(collection, iteratees[0], iteratees[1])) { iteratees = []; } else if (length > 2 && _isIterateeCall(iteratees[0], iteratees[1], iteratees[2])) { iteratees = [iteratees[0]]; } return _baseOrderBy(collection, _baseFlatten(iteratees, 1), []); }); /* harmony default export */ const lodash_es_sortBy = (sortBy); ;// CONCATENATED MODULE: ./node_modules/@converse/skeletor/src/collection.js // Backbone.js 1.4.0 // (c) 2010-2019 Jeremy Ashkenas and DocumentCloud // Backbone may be freely distributed under the MIT license. // Collection // ---------- // If models tend to represent a single row of data, a Collection is // more analogous to a table full of data ... or a small slice or page of that // table, or a collection of rows that belong together for a particular reason // -- all of the messages in this particular folder, all of the documents // belonging to this particular author, and so on. Collections maintain // indexes of their models, both in order, and for lookup by `id`. const slice = Array.prototype.slice; // Create a new **Collection**, perhaps to contain a specific type of `model`. // If a `comparator` is specified, the Collection will maintain // its models in sort order, as they're added and removed. const Collection = function (models, options) { options || (options = {}); this.preinitialize.apply(this, arguments); if (options.model) this.model = options.model; if (options.comparator !== undefined) this.comparator = options.comparator; this._reset(); this.initialize.apply(this, arguments); if (models) this.reset(models, lodash_es_assignIn({ silent: true }, options)); }; Collection.extend = inherits; // Default options for `Collection#set`. const setOptions = { add: true, remove: true, merge: true }; const addOptions = { add: true, remove: false }; // Splices `insert` into `array` at index `at`. const collection_splice = function (array, insert, at) { at = Math.min(Math.max(at, 0), array.length); const tail = Array(array.length - at); const length = insert.length; let i; for (i = 0; i < tail.length; i++) tail[i] = array[i + at]; for (i = 0; i < length; i++) array[i + at] = insert[i]; for (i = 0; i < tail.length; i++) array[i + length + at] = tail[i]; }; // Define the Collection's inheritable methods. Object.assign(Collection.prototype, Events, { // The default model for a collection is just a **Backbone.Model**. // This should be overridden in most cases. model: Model, // preinitialize is an empty function by default. You can override it with a function // or object. preinitialize will run before any instantiation logic is run in the Collection. preinitialize: function () {}, // Initialize is an empty function by default. Override it with your own // initialization logic. initialize: function () {}, // The JSON representation of a Collection is an array of the // models' attributes. toJSON: function (options) { return this.map(function (model) { return model.toJSON(options); }); }, // Proxy `Backbone.sync` by default. sync: function (method, model, options) { return getSyncMethod(this)(method, model, options); }, // Add a model, or list of models to the set. `models` may be Backbone // Models or raw JavaScript objects to be converted to Models, or any // combination of the two. add: function (models, options) { return this.set(models, lodash_es_assignIn({ merge: false }, options, addOptions)); }, // Remove a model, or a list of models from the set. remove: function (models, options) { options = lodash_es_assignIn({}, options); const singular = !Array.isArray(models); models = singular ? [models] : models.slice(); const removed = this._removeModels(models, options); if (!options.silent && removed.length) { options.changes = { added: [], merged: [], removed: removed }; this.trigger('update', this, options); } return singular ? removed[0] : removed; }, // Update a collection by `set`-ing a new list of models, adding new ones, // removing models that are no longer present, and merging models that // already exist in the collection, as necessary. Similar to **Model#set**, // the core operation for updating the data contained by the collection. set: function (models, options) { if (models == null) return; options = lodash_es_assignIn({}, setOptions, options); if (options.parse && !this._isModel(models)) { models = this.parse(models, options) || []; } const singular = !Array.isArray(models); models = singular ? [models] : models.slice(); let at = options.at; if (at != null) at = +at; if (at > this.length) at = this.length; if (at < 0) at += this.length + 1; const set = []; const toAdd = []; const toMerge = []; const toRemove = []; const modelMap = {}; const add = options.add; const merge = options.merge; const remove = options.remove; let sort = false; const sortable = this.comparator && at == null && options.sort !== false; const sortAttr = lodash_es_isString(this.comparator) ? this.comparator : null; // Turn bare objects into model references, and prevent invalid models // from being added. let model, i; for (i = 0; i < models.length; i++) { model = models[i]; // If a duplicate is found, prevent it from being added and // optionally merge it into the existing model. const existing = this.get(model); if (existing) { if (merge && model !== existing) { let attrs = this._isModel(model) ? model.attributes : model; if (options.parse) attrs = existing.parse(attrs, options); existing.set(attrs, options); toMerge.push(existing); if (sortable && !sort) sort = existing.hasChanged(sortAttr); } if (!modelMap[existing.cid]) { modelMap[existing.cid] = true; set.push(existing); } models[i] = existing; // If this is a new, valid model, push it to the `toAdd` list. } else if (add) { model = models[i] = this._prepareModel(model, options); if (model) { toAdd.push(model); this._addReference(model, options); modelMap[model.cid] = true; set.push(model); } } } // Remove stale models. if (remove) { for (i = 0; i < this.length; i++) { model = this.models[i]; if (!modelMap[model.cid]) toRemove.push(model); } if (toRemove.length) this._removeModels(toRemove, options); } // See if sorting is needed, update `length` and splice in new models. let orderChanged = false; const replace = !sortable && add && remove; if (set.length && replace) { orderChanged = this.length !== set.length || lodash_es_some(this.models, (m, index) => m !== set[index]); this.models.length = 0; collection_splice(this.models, set, 0); this.length = this.models.length; } else if (toAdd.length) { if (sortable) sort = true; collection_splice(this.models, toAdd, at == null ? this.length : at); this.length = this.models.length; } // Silently sort the collection if appropriate. if (sort) this.sort({ silent: true }); // Unless silenced, it's time to fire all appropriate add/sort/update events. if (!options.silent) { for (i = 0; i < toAdd.length; i++) { if (at != null) options.index = at + i; model = toAdd[i]; model.trigger('add', model, this, options); } if (sort || orderChanged) this.trigger('sort', this, options); if (toAdd.length || toRemove.length || toMerge.length) { options.changes = { added: toAdd, removed: toRemove, merged: toMerge }; this.trigger('update', this, options); } } // Return the added (or merged) model (or models). return singular ? models[0] : models; }, clearStore: async function () { let options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; let filter = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : o => o; await Promise.all(this.models.filter(filter).map(m => { return new Promise(resolve => { m.destroy(Object.assign(options, { 'success': resolve, 'error': (m, e) => { console.error(e); resolve(); } })); }); })); await this.browserStorage.clear(); this.reset(); }, // When you have more items than you want to add or remove individually, // you can reset the entire set with a new list of models, without firing // any granular `add` or `remove` events. Fires `reset` when finished. // Useful for bulk operations and optimizations. reset: function (models, options) { options = options ? lodash_es_clone(options) : {}; for (let i = 0; i < this.models.length; i++) { this._removeReference(this.models[i], options); } options.previousModels = this.models; this._reset(); models = this.add(models, lodash_es_assignIn({ silent: true }, options)); if (!options.silent) this.trigger('reset', this, options); return models; }, // Add a model to the end of the collection. push: function (model, options) { return this.add(model, lodash_es_assignIn({ at: this.length }, options)); }, // Remove a model from the end of the collection. pop: function (options) { const model = this.at(this.length - 1); return this.remove(model, options); }, // Add a model to the beginning of the collection. unshift: function (model, options) { return this.add(model, lodash_es_assignIn({ at: 0 }, options)); }, // Remove a model from the beginning of the collection. shift: function (options) { const model = this.at(0); return this.remove(model, options); }, // Slice out a sub-array of models from the collection. slice: function () { return slice.apply(this.models, arguments); }, filter: function (callback, thisArg) { return this.models.filter(lodash_es_isFunction(callback) ? callback : m => m.matches(callback), thisArg); }, every: function (pred) { return lodash_es_every(this.models.map(m => m.attributes), pred); }, difference: function (values) { return lodash_es_difference(this.models, values); }, max: function () { return Math.max.apply(Math, this.models); }, min: function () { return Math.min.apply(Math, this.models); }, drop: function () { let n = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 1; return this.models.slice(n); }, some: function (pred) { return lodash_es_some(this.models.map(m => m.attributes), pred); }, sortBy: function (iteratee) { return lodash_es_sortBy(this.models, lodash_es_isFunction(iteratee) ? iteratee : m => lodash_es_isString(iteratee) ? m.get(iteratee) : m.matches(iteratee)); }, isEmpty: function () { return lodash_es_isEmpty(this.models); }, keyBy: function (iteratee) { return lodash_es_keyBy(this.models, iteratee); }, each: function (callback, thisArg) { return this.forEach(callback, thisArg); }, forEach: function (callback, thisArg) { return this.models.forEach(callback, thisArg); }, includes: function (item) { return this.models.includes(item); }, size: function () { return this.models.length; }, countBy: function (f) { return lodash_es_countBy(this.models, lodash_es_isFunction(f) ? f : m => lodash_es_isString(f) ? m.get(f) : m.matches(f)); }, groupBy: function (pred) { return lodash_es_groupBy(this.models, lodash_es_isFunction(pred) ? pred : m => lodash_es_isString(pred) ? m.get(pred) : m.matches(pred)); }, indexOf: function (fromIndex) { return lodash_es_indexOf(this.models, fromIndex); }, findLastIndex: function (pred, fromIndex) { return lodash_es_findLastIndex(this.models, lodash_es_isFunction(pred) ? pred : m => lodash_es_isString(pred) ? m.get(pred) : m.matches(pred), fromIndex); }, lastIndexOf: function (fromIndex) { return lodash_es_lastIndexOf(this.models, fromIndex); }, findIndex: function (pred) { return lodash_es_findIndex(this.models, lodash_es_isFunction(pred) ? pred : m => lodash_es_isString(pred) ? m.get(pred) : m.matches(pred)); }, last: function () { const length = this.models == null ? 0 : this.models.length; return length ? this.models[length - 1] : undefined; }, head: function () { return this.models[0]; }, first: function () { return this.head(); }, map: function (cb, thisArg) { return this.models.map(lodash_es_isFunction(cb) ? cb : m => lodash_es_isString(cb) ? m.get(cb) : m.matches(cb), thisArg); }, reduce: function (callback, initialValue) { return this.models.reduce(callback, initialValue || this.models[0]); }, reduceRight: function (callback, initialValue) { return this.models.reduceRight(callback, initialValue || this.models[0]); }, toArray: function () { return Array.from(this.models); }, // Get a model from the set by id, cid, model object with id or cid // properties, or an attributes object that is transformed through modelId. get: function (obj) { if (obj == null) return undefined; return this._byId[obj] || this._byId[this.modelId(this._isModel(obj) ? obj.attributes : obj)] || obj.cid && this._byId[obj.cid]; }, // Returns `true` if the model is in the collection. has: function (obj) { return this.get(obj) != null; }, // Get the model at the given index. at: function (index) { if (index < 0) index += this.length; return this.models[index]; }, // Return models with matching attributes. Useful for simple cases of // `filter`. where: function (attrs, first) { return this[first ? 'find' : 'filter'](attrs); }, // Return the first model with matching attributes. Useful for simple cases // of `find`. findWhere: function (attrs) { return this.where(attrs, true); }, find: function (predicate, fromIndex) { const pred = lodash_es_isFunction(predicate) ? predicate : m => m.matches(predicate); return this.models.find(pred, fromIndex); }, // Force the collection to re-sort itself. You don't need to call this under // normal circumstances, as the set will maintain sort order as each item // is added. sort: function (options) { let comparator = this.comparator; if (!comparator) throw new Error('Cannot sort a set without a comparator'); options || (options = {}); const length = comparator.length; if (lodash_es_isFunction(comparator)) comparator = comparator.bind(this); // Run sort based on type of `comparator`. if (length === 1 || lodash_es_isString(comparator)) { this.models = this.sortBy(comparator); } else { this.models.sort(comparator); } if (!options.silent) this.trigger('sort', this, options); return this; }, // Pluck an attribute from each model in the collection. pluck: function (attr) { return this.map(attr + ''); }, // Fetch the default set of models for this collection, resetting the // collection when they arrive. If `reset: true` is passed, the response // data will be passed through the `reset` method instead of `set`. fetch: function (options) { options = lodash_es_assignIn({ parse: true }, options); const success = options.success; const collection = this; const promise = options.promise && getResolveablePromise(); options.success = function (resp) { const method = options.reset ? 'reset' : 'set'; collection[method](resp, options); if (success) success.call(options.context, collection, resp, options); promise && promise.resolve(); collection.trigger('sync', collection, resp, options); }; wrapError(this, options); return promise ? promise : this.sync('read', this, options); }, // Create a new instance of a model in this collection. Add the model to the // collection immediately, unless `wait: true` is passed, in which case we // wait for the server to agree. create: function (model, options) { options = options ? lodash_es_clone(options) : {}; const wait = options.wait; const return_promise = options.promise; const promise = return_promise && getResolveablePromise(); model = this._prepareModel(model, options); if (!model) return false; if (!wait) this.add(model, options); const collection = this; const success = options.success; const error = options.error; options.success = function (m, resp, callbackOpts) { if (wait) { collection.add(m, callbackOpts); } if (success) { success.call(callbackOpts.context, m, resp, callbackOpts); } if (return_promise) { promise.resolve(m); } }; options.error = function (model, e, options) { error && error.call(options.context, model, e, options); return_promise && promise.reject(e); }; model.save(null, Object.assign(options, { 'promise': false })); if (return_promise) { return promise; } else { return model; } }, // **parse** converts a response into a list of models to be added to the // collection. The default implementation is just to pass it through. parse: function (resp, options) { return resp; }, // Create a new collection with an identical list of models as this one. clone: function () { return new this.constructor(this.models, { model: this.model, comparator: this.comparator }); }, // Define how to uniquely identify models in the collection. modelId: function (attrs) { var _this$model$prototype; return attrs[((_this$model$prototype = this.model.prototype) === null || _this$model$prototype === void 0 ? void 0 : _this$model$prototype.idAttribute) || 'id']; }, // Get an iterator of all models in this collection. values: function () { return new CollectionIterator(this, ITERATOR_VALUES); }, // Get an iterator of all model IDs in this collection. keys: function () { return new CollectionIterator(this, ITERATOR_KEYS); }, // Get an iterator of all [ID, model] tuples in this collection. entries: function () { return new CollectionIterator(this, ITERATOR_KEYSVALUES); }, // Private method to reset all internal state. Called when the collection // is first initialized or reset. _reset: function () { this.length = 0; this.models = []; this._byId = {}; }, // Prepare a hash of attributes (or other model) to be added to this // collection. _prepareModel: function (attrs, options) { if (this._isModel(attrs)) { if (!attrs.collection) attrs.collection = this; return attrs; } options = options ? lodash_es_clone(options) : {}; options.collection = this; const model = new this.model(attrs, options); if (!model.validationError) return model; this.trigger('invalid', this, model.validationError, options); return false; }, // Internal method called by both remove and set. _removeModels: function (models, options) { const removed = []; for (let i = 0; i < models.length; i++) { const model = this.get(models[i]); if (!model) continue; const index = this.indexOf(model); this.models.splice(index, 1); this.length--; // Remove references before triggering 'remove' event to prevent an // infinite loop. #3693 delete this._byId[model.cid]; const id = this.modelId(model.attributes); if (id != null) delete this._byId[id]; if (!options.silent) { options.index = index; model.trigger('remove', model, this, options); } removed.push(model); this._removeReference(model, options); } return removed; }, // Method for checking whether an object should be considered a model for // the purposes of adding to the collection. _isModel: function (model) { return model instanceof Model; }, // Internal method to create a model's ties to a collection. _addReference: function (model, options) { this._byId[model.cid] = model; const id = this.modelId(model.attributes); if (id != null) this._byId[id] = model; model.on('all', this._onModelEvent, this); }, // Internal method to sever a model's ties to a collection. _removeReference: function (model, options) { delete this._byId[model.cid]; const id = this.modelId(model.attributes); if (id != null) delete this._byId[id]; if (this === model.collection) delete model.collection; model.off('all', this._onModelEvent, this); }, // Internal method called every time a model in the set fires an event. // Sets need to update their indexes when models change ids. All other // events simply proxy through. "add" and "remove" events that originate // in other collections are ignored. _onModelEvent: function (event, model, collection, options) { if (model) { if ((event === 'add' || event === 'remove') && collection !== this) return; if (event === 'destroy') this.remove(model, options); if (event === 'change') { const prevId = this.modelId(model.previousAttributes()); const id = this.modelId(model.attributes); if (prevId !== id) { if (prevId != null) delete this._byId[prevId]; if (id != null) this._byId[id] = model; } } } this.trigger.apply(this, arguments); } }); // Defining an @@iterator method implements JavaScript's Iterable protocol. // In modern ES2015 browsers, this value is found at Symbol.iterator. /* global Symbol */ const $$iterator = typeof Symbol === 'function' && Symbol.iterator; if ($$iterator) { Collection.prototype[$$iterator] = Collection.prototype.values; } // CollectionIterator // ------------------ // A CollectionIterator implements JavaScript's Iterator protocol, allowing the // use of `for of` loops in modern browsers and interoperation between // Collection and other JavaScript functions and third-party libraries // which can operate on Iterables. const CollectionIterator = function (collection, kind) { this._collection = collection; this._kind = kind; this._index = 0; }; // This "enum" defines the three possible kinds of values which can be emitted // by a CollectionIterator that correspond to the values(), keys() and entries() // methods on Collection, respectively. const ITERATOR_VALUES = 1; const ITERATOR_KEYS = 2; const ITERATOR_KEYSVALUES = 3; // All Iterators should themselves be Iterable. if ($$iterator) { CollectionIterator.prototype[$$iterator] = function () { return this; }; } CollectionIterator.prototype.next = function () { if (this._collection) { // Only continue iterating if the iterated collection is long enough. if (this._index < this._collection.length) { const model = this._collection.at(this._index); this._index++; // Construct a value depending on what kind of values should be iterated. let value; if (this._kind === ITERATOR_VALUES) { value = model; } else { const id = this._collection.modelId(model.attributes); if (this._kind === ITERATOR_KEYS) { value = id; } else { // ITERATOR_KEYSVALUES value = [id, model]; } } return { value: value, done: false }; } // Once exhausted, remove the reference to the collection so future // calls to the next method always return done. this._collection = undefined; } return { value: undefined, done: true }; }; ;// CONCATENATED MODULE: ./node_modules/@lit/reactive-element/css-tag.js /** * @license * Copyright 2019 Google LLC * SPDX-License-Identifier: BSD-3-Clause */ const css_tag_t=window.ShadowRoot&&(void 0===window.ShadyCSS||window.ShadyCSS.nativeShadow)&&"adoptedStyleSheets"in Document.prototype&&"replace"in CSSStyleSheet.prototype,e=Symbol(),n=new Map;class s{constructor(t,n){if(this._$cssResult$=!0,n!==e)throw Error("CSSResult is not constructable. Use `unsafeCSS` or `css` instead.");this.cssText=t}get styleSheet(){let e=n.get(this.cssText);return css_tag_t&&void 0===e&&(n.set(this.cssText,e=new CSSStyleSheet),e.replaceSync(this.cssText)),e}toString(){return this.cssText}}const o=t=>new s("string"==typeof t?t:t+"",e),r=(t,...n)=>{const o=1===t.length?t[0]:n.reduce(((e,n,s)=>e+(t=>{if(!0===t._$cssResult$)return t.cssText;if("number"==typeof t)return t;throw Error("Value passed to 'css' function must be a 'css' function result: "+t+". Use 'unsafeCSS' to pass non-literal values, but take care to ensure page security.")})(n)+t[s+1]),t[0]);return new s(o,e)},css_tag_i=(e,n)=>{css_tag_t?e.adoptedStyleSheets=n.map((t=>t instanceof CSSStyleSheet?t:t.styleSheet)):n.forEach((t=>{const n=document.createElement("style"),s=window.litNonce;void 0!==s&&n.setAttribute("nonce",s),n.textContent=t.cssText,e.appendChild(n)}))},S=css_tag_t?t=>t:t=>t instanceof CSSStyleSheet?(t=>{let e="";for(const n of t.cssRules)e+=n.cssText;return o(e)})(t):t; //# sourceMappingURL=css-tag.js.map ;// CONCATENATED MODULE: ./node_modules/@lit/reactive-element/reactive-element.js /** * @license * Copyright 2017 Google LLC * SPDX-License-Identifier: BSD-3-Clause */var reactive_element_s;const reactive_element_e=window.trustedTypes,reactive_element_r=reactive_element_e?reactive_element_e.emptyScript:"",h=window.reactiveElementPolyfillSupport,reactive_element_o={toAttribute(t,i){switch(i){case Boolean:t=t?reactive_element_r:null;break;case Object:case Array:t=null==t?t:JSON.stringify(t)}return t},fromAttribute(t,i){let s=t;switch(i){case Boolean:s=null!==t;break;case Number:s=null===t?null:Number(t);break;case Object:case Array:try{s=JSON.parse(t)}catch(t){s=null}}return s}},reactive_element_n=(t,i)=>i!==t&&(i==i||t==t),l={attribute:!0,type:String,converter:reactive_element_o,reflect:!1,hasChanged:reactive_element_n};class a extends HTMLElement{constructor(){super(),this._$Et=new Map,this.isUpdatePending=!1,this.hasUpdated=!1,this._$Ei=null,this.o()}static addInitializer(t){var i;null!==(i=this.l)&&void 0!==i||(this.l=[]),this.l.push(t)}static get observedAttributes(){this.finalize();const t=[];return this.elementProperties.forEach(((i,s)=>{const e=this._$Eh(s,i);void 0!==e&&(this._$Eu.set(e,s),t.push(e))})),t}static createProperty(t,i=l){if(i.state&&(i.attribute=!1),this.finalize(),this.elementProperties.set(t,i),!i.noAccessor&&!this.prototype.hasOwnProperty(t)){const s="symbol"==typeof t?Symbol():"__"+t,e=this.getPropertyDescriptor(t,s,i);void 0!==e&&Object.defineProperty(this.prototype,t,e)}}static getPropertyDescriptor(t,i,s){return{get(){return this[i]},set(e){const r=this[t];this[i]=e,this.requestUpdate(t,r,s)},configurable:!0,enumerable:!0}}static getPropertyOptions(t){return this.elementProperties.get(t)||l}static finalize(){if(this.hasOwnProperty("finalized"))return!1;this.finalized=!0;const t=Object.getPrototypeOf(this);if(t.finalize(),this.elementProperties=new Map(t.elementProperties),this._$Eu=new Map,this.hasOwnProperty("properties")){const t=this.properties,i=[...Object.getOwnPropertyNames(t),...Object.getOwnPropertySymbols(t)];for(const s of i)this.createProperty(s,t[s])}return this.elementStyles=this.finalizeStyles(this.styles),!0}static finalizeStyles(i){const s=[];if(Array.isArray(i)){const e=new Set(i.flat(1/0).reverse());for(const i of e)s.unshift(S(i))}else void 0!==i&&s.push(S(i));return s}static _$Eh(t,i){const s=i.attribute;return!1===s?void 0:"string"==typeof s?s:"string"==typeof t?t.toLowerCase():void 0}o(){var t;this._$Ep=new Promise((t=>this.enableUpdating=t)),this._$AL=new Map,this._$Em(),this.requestUpdate(),null===(t=this.constructor.l)||void 0===t||t.forEach((t=>t(this)))}addController(t){var i,s;(null!==(i=this._$Eg)&&void 0!==i?i:this._$Eg=[]).push(t),void 0!==this.renderRoot&&this.isConnected&&(null===(s=t.hostConnected)||void 0===s||s.call(t))}removeController(t){var i;null===(i=this._$Eg)||void 0===i||i.splice(this._$Eg.indexOf(t)>>>0,1)}_$Em(){this.constructor.elementProperties.forEach(((t,i)=>{this.hasOwnProperty(i)&&(this._$Et.set(i,this[i]),delete this[i])}))}createRenderRoot(){var t;const s=null!==(t=this.shadowRoot)&&void 0!==t?t:this.attachShadow(this.constructor.shadowRootOptions);return css_tag_i(s,this.constructor.elementStyles),s}connectedCallback(){var t;void 0===this.renderRoot&&(this.renderRoot=this.createRenderRoot()),this.enableUpdating(!0),null===(t=this._$Eg)||void 0===t||t.forEach((t=>{var i;return null===(i=t.hostConnected)||void 0===i?void 0:i.call(t)}))}enableUpdating(t){}disconnectedCallback(){var t;null===(t=this._$Eg)||void 0===t||t.forEach((t=>{var i;return null===(i=t.hostDisconnected)||void 0===i?void 0:i.call(t)}))}attributeChangedCallback(t,i,s){this._$AK(t,s)}_$ES(t,i,s=l){var e,r;const h=this.constructor._$Eh(t,s);if(void 0!==h&&!0===s.reflect){const n=(null!==(r=null===(e=s.converter)||void 0===e?void 0:e.toAttribute)&&void 0!==r?r:reactive_element_o.toAttribute)(i,s.type);this._$Ei=t,null==n?this.removeAttribute(h):this.setAttribute(h,n),this._$Ei=null}}_$AK(t,i){var s,e,r;const h=this.constructor,n=h._$Eu.get(t);if(void 0!==n&&this._$Ei!==n){const t=h.getPropertyOptions(n),l=t.converter,a=null!==(r=null!==(e=null===(s=l)||void 0===s?void 0:s.fromAttribute)&&void 0!==e?e:"function"==typeof l?l:null)&&void 0!==r?r:reactive_element_o.fromAttribute;this._$Ei=n,this[n]=a(i,t.type),this._$Ei=null}}requestUpdate(t,i,s){let e=!0;void 0!==t&&(((s=s||this.constructor.getPropertyOptions(t)).hasChanged||reactive_element_n)(this[t],i)?(this._$AL.has(t)||this._$AL.set(t,i),!0===s.reflect&&this._$Ei!==t&&(void 0===this._$EC&&(this._$EC=new Map),this._$EC.set(t,s))):e=!1),!this.isUpdatePending&&e&&(this._$Ep=this._$E_())}async _$E_(){this.isUpdatePending=!0;try{await this._$Ep}catch(t){Promise.reject(t)}const t=this.scheduleUpdate();return null!=t&&await t,!this.isUpdatePending}scheduleUpdate(){return this.performUpdate()}performUpdate(){var t;if(!this.isUpdatePending)return;this.hasUpdated,this._$Et&&(this._$Et.forEach(((t,i)=>this[i]=t)),this._$Et=void 0);let i=!1;const s=this._$AL;try{i=this.shouldUpdate(s),i?(this.willUpdate(s),null===(t=this._$Eg)||void 0===t||t.forEach((t=>{var i;return null===(i=t.hostUpdate)||void 0===i?void 0:i.call(t)})),this.update(s)):this._$EU()}catch(t){throw i=!1,this._$EU(),t}i&&this._$AE(s)}willUpdate(t){}_$AE(t){var i;null===(i=this._$Eg)||void 0===i||i.forEach((t=>{var i;return null===(i=t.hostUpdated)||void 0===i?void 0:i.call(t)})),this.hasUpdated||(this.hasUpdated=!0,this.firstUpdated(t)),this.updated(t)}_$EU(){this._$AL=new Map,this.isUpdatePending=!1}get updateComplete(){return this.getUpdateComplete()}getUpdateComplete(){return this._$Ep}shouldUpdate(t){return!0}update(t){void 0!==this._$EC&&(this._$EC.forEach(((t,i)=>this._$ES(i,this[i],t))),this._$EC=void 0),this._$EU()}updated(t){}firstUpdated(t){}}a.finalized=!0,a.elementProperties=new Map,a.elementStyles=[],a.shadowRootOptions={mode:"open"},null==h||h({ReactiveElement:a}),(null!==(reactive_element_s=globalThis.reactiveElementVersions)&&void 0!==reactive_element_s?reactive_element_s:globalThis.reactiveElementVersions=[]).push("1.3.0"); //# sourceMappingURL=reactive-element.js.map ;// CONCATENATED MODULE: ./node_modules/lit-html/lit-html.js /** * @license * Copyright 2017 Google LLC * SPDX-License-Identifier: BSD-3-Clause */ var lit_html_t; const lit_html_i = globalThis.trustedTypes, lit_html_s = lit_html_i ? lit_html_i.createPolicy("lit-html", { createHTML: t => t }) : void 0, lit_html_e = `lit$${(Math.random() + "").slice(9)}$`, lit_html_o = "?" + lit_html_e, lit_html_n = `<${lit_html_o}>`, lit_html_l = document, lit_html_h = function () { let t = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : ""; return lit_html_l.createComment(t); }, lit_html_r = t => null === t || "object" != typeof t && "function" != typeof t, d = Array.isArray, lit_html_u = t => { var i; return d(t) || "function" == typeof (null === (i = t) || void 0 === i ? void 0 : i[Symbol.iterator]); }, c = /<(?:(!--|\/[^a-zA-Z])|(\/?[a-zA-Z][^>\s]*)|(\/?$))/g, v = /-->/g, lit_html_a = />/g, f = />|[ \n \r](?:([^\s"'>=/]+)([ \n \r]*=[ \n \r]*(?:[^ \n \r"'`<>=]|("|')|))|$)/g, _ = /'/g, m = /"/g, g = /^(?:script|style|textarea|title)$/i, p = t => function (i) { for (var _len = arguments.length, s = new Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) { s[_key - 1] = arguments[_key]; } return { _$litType$: t, strings: i, values: s }; }, $ = p(1), y = p(2), b = Symbol.for("lit-noChange"), w = Symbol.for("lit-nothing"), T = new WeakMap(), x = (t, i, s) => { var e, o; const n = null !== (e = null == s ? void 0 : s.renderBefore) && void 0 !== e ? e : i; let l = n._$litPart$; if (void 0 === l) { const t = null !== (o = null == s ? void 0 : s.renderBefore) && void 0 !== o ? o : null; n._$litPart$ = l = new N(i.insertBefore(lit_html_h(), t), t, void 0, null != s ? s : {}); } return l._$AI(t), l; }, A = lit_html_l.createTreeWalker(lit_html_l, 129, null, !1), C = (t, i) => { const o = t.length - 1, l = []; let h, r = 2 === i ? "" : "", d = c; for (let i = 0; i < o; i++) { const s = t[i]; let o, u, p = -1, $ = 0; for (; $ < s.length && (d.lastIndex = $, u = d.exec(s), null !== u);) $ = d.lastIndex, d === c ? "!--" === u[1] ? d = v : void 0 !== u[1] ? d = lit_html_a : void 0 !== u[2] ? (g.test(u[2]) && (h = RegExp("" === u[0] ? (d = null != h ? h : c, p = -1) : void 0 === u[1] ? p = -2 : (p = d.lastIndex - u[2].length, o = u[1], d = void 0 === u[3] ? f : '"' === u[3] ? m : _) : d === m || d === _ ? d = f : d === v || d === lit_html_a ? d = c : (d = f, h = void 0); const y = d === f && t[i + 1].startsWith("/>") ? " " : ""; r += d === c ? s + lit_html_n : p >= 0 ? (l.push(o), s.slice(0, p) + "$lit$" + s.slice(p) + lit_html_e + y) : s + lit_html_e + (-2 === p ? (l.push(void 0), i) : y); } const u = r + (t[o] || "") + (2 === i ? "" : ""); if (!Array.isArray(t) || !t.hasOwnProperty("raw")) throw Error("invalid template strings array"); return [void 0 !== lit_html_s ? lit_html_s.createHTML(u) : u, l]; }; class E { constructor(_ref, n) { let { strings: t, _$litType$: s } = _ref; let l; this.parts = []; let r = 0, d = 0; const u = t.length - 1, c = this.parts, [v, a] = C(t, s); if (this.el = E.createElement(v, n), A.currentNode = this.el.content, 2 === s) { const t = this.el.content, i = t.firstChild; i.remove(), t.append(...i.childNodes); } for (; null !== (l = A.nextNode()) && c.length < u;) { if (1 === l.nodeType) { if (l.hasAttributes()) { const t = []; for (const i of l.getAttributeNames()) if (i.endsWith("$lit$") || i.startsWith(lit_html_e)) { const s = a[d++]; if (t.push(i), void 0 !== s) { const t = l.getAttribute(s.toLowerCase() + "$lit$").split(lit_html_e), i = /([.?@])?(.*)/.exec(s); c.push({ type: 1, index: r, name: i[2], strings: t, ctor: "." === i[1] ? M : "?" === i[1] ? H : "@" === i[1] ? I : lit_html_S }); } else c.push({ type: 6, index: r }); } for (const i of t) l.removeAttribute(i); } if (g.test(l.tagName)) { const t = l.textContent.split(lit_html_e), s = t.length - 1; if (s > 0) { l.textContent = lit_html_i ? lit_html_i.emptyScript : ""; for (let i = 0; i < s; i++) l.append(t[i], lit_html_h()), A.nextNode(), c.push({ type: 2, index: ++r }); l.append(t[s], lit_html_h()); } } } else if (8 === l.nodeType) if (l.data === lit_html_o) c.push({ type: 2, index: r });else { let t = -1; for (; -1 !== (t = l.data.indexOf(lit_html_e, t + 1));) c.push({ type: 7, index: r }), t += lit_html_e.length - 1; } r++; } } static createElement(t, i) { const s = lit_html_l.createElement("template"); return s.innerHTML = t, s; } } function P(t, i) { let s = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : t; let e = arguments.length > 3 ? arguments[3] : undefined; var o, n, l, h; if (i === b) return i; let d = void 0 !== e ? null === (o = s._$Cl) || void 0 === o ? void 0 : o[e] : s._$Cu; const u = lit_html_r(i) ? void 0 : i._$litDirective$; return (null == d ? void 0 : d.constructor) !== u && (null === (n = null == d ? void 0 : d._$AO) || void 0 === n || n.call(d, !1), void 0 === u ? d = void 0 : (d = new u(t), d._$AT(t, s, e)), void 0 !== e ? (null !== (l = (h = s)._$Cl) && void 0 !== l ? l : h._$Cl = [])[e] = d : s._$Cu = d), void 0 !== d && (i = P(t, d._$AS(t, i.values), d, e)), i; } class V { constructor(t, i) { this.v = [], this._$AN = void 0, this._$AD = t, this._$AM = i; } get parentNode() { return this._$AM.parentNode; } get _$AU() { return this._$AM._$AU; } p(t) { var i; const { el: { content: s }, parts: e } = this._$AD, o = (null !== (i = null == t ? void 0 : t.creationScope) && void 0 !== i ? i : lit_html_l).importNode(s, !0); A.currentNode = o; let n = A.nextNode(), h = 0, r = 0, d = e[0]; for (; void 0 !== d;) { if (h === d.index) { let i; 2 === d.type ? i = new N(n, n.nextSibling, this, t) : 1 === d.type ? i = new d.ctor(n, d.name, d.strings, this, t) : 6 === d.type && (i = new L(n, this, t)), this.v.push(i), d = e[++r]; } h !== (null == d ? void 0 : d.index) && (n = A.nextNode(), h++); } return o; } m(t) { let i = 0; for (const s of this.v) void 0 !== s && (void 0 !== s.strings ? (s._$AI(t, s, i), i += s.strings.length - 2) : s._$AI(t[i])), i++; } } class N { constructor(t, i, s, e) { var o; this.type = 2, this._$AH = w, this._$AN = void 0, this._$AA = t, this._$AB = i, this._$AM = s, this.options = e, this._$Cg = null === (o = null == e ? void 0 : e.isConnected) || void 0 === o || o; } get _$AU() { var t, i; return null !== (i = null === (t = this._$AM) || void 0 === t ? void 0 : t._$AU) && void 0 !== i ? i : this._$Cg; } get parentNode() { let t = this._$AA.parentNode; const i = this._$AM; return void 0 !== i && 11 === t.nodeType && (t = i.parentNode), t; } get startNode() { return this._$AA; } get endNode() { return this._$AB; } _$AI(t) { let i = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : this; t = P(this, t, i), lit_html_r(t) ? t === w || null == t || "" === t ? (this._$AH !== w && this._$AR(), this._$AH = w) : t !== this._$AH && t !== b && this.$(t) : void 0 !== t._$litType$ ? this.T(t) : void 0 !== t.nodeType ? this.k(t) : lit_html_u(t) ? this.S(t) : this.$(t); } A(t) { let i = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : this._$AB; return this._$AA.parentNode.insertBefore(t, i); } k(t) { this._$AH !== t && (this._$AR(), this._$AH = this.A(t)); } $(t) { this._$AH !== w && lit_html_r(this._$AH) ? this._$AA.nextSibling.data = t : this.k(lit_html_l.createTextNode(t)), this._$AH = t; } T(t) { var i; const { values: s, _$litType$: e } = t, o = "number" == typeof e ? this._$AC(t) : (void 0 === e.el && (e.el = E.createElement(e.h, this.options)), e); if ((null === (i = this._$AH) || void 0 === i ? void 0 : i._$AD) === o) this._$AH.m(s);else { const t = new V(o, this), i = t.p(this.options); t.m(s), this.k(i), this._$AH = t; } } _$AC(t) { let i = T.get(t.strings); return void 0 === i && T.set(t.strings, i = new E(t)), i; } S(t) { d(this._$AH) || (this._$AH = [], this._$AR()); const i = this._$AH; let s, e = 0; for (const o of t) e === i.length ? i.push(s = new N(this.A(lit_html_h()), this.A(lit_html_h()), this, this.options)) : s = i[e], s._$AI(o), e++; e < i.length && (this._$AR(s && s._$AB.nextSibling, e), i.length = e); } _$AR() { let t = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : this._$AA.nextSibling; let i = arguments.length > 1 ? arguments[1] : undefined; var s; for (null === (s = this._$AP) || void 0 === s || s.call(this, !1, !0, i); t && t !== this._$AB;) { const i = t.nextSibling; t.remove(), t = i; } } setConnected(t) { var i; void 0 === this._$AM && (this._$Cg = t, null === (i = this._$AP) || void 0 === i || i.call(this, t)); } } class lit_html_S { constructor(t, i, s, e, o) { this.type = 1, this._$AH = w, this._$AN = void 0, this.element = t, this.name = i, this._$AM = e, this.options = o, s.length > 2 || "" !== s[0] || "" !== s[1] ? (this._$AH = Array(s.length - 1).fill(new String()), this.strings = s) : this._$AH = w; } get tagName() { return this.element.tagName; } get _$AU() { return this._$AM._$AU; } _$AI(t) { let i = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : this; let s = arguments.length > 2 ? arguments[2] : undefined; let e = arguments.length > 3 ? arguments[3] : undefined; const o = this.strings; let n = !1; if (void 0 === o) t = P(this, t, i, 0), n = !lit_html_r(t) || t !== this._$AH && t !== b, n && (this._$AH = t);else { const e = t; let l, h; for (t = o[0], l = 0; l < o.length - 1; l++) h = P(this, e[s + l], i, l), h === b && (h = this._$AH[l]), n || (n = !lit_html_r(h) || h !== this._$AH[l]), h === w ? t = w : t !== w && (t += (null != h ? h : "") + o[l + 1]), this._$AH[l] = h; } n && !e && this.C(t); } C(t) { t === w ? this.element.removeAttribute(this.name) : this.element.setAttribute(this.name, null != t ? t : ""); } } class M extends lit_html_S { constructor() { super(...arguments), this.type = 3; } C(t) { this.element[this.name] = t === w ? void 0 : t; } } const k = lit_html_i ? lit_html_i.emptyScript : ""; class H extends lit_html_S { constructor() { super(...arguments), this.type = 4; } C(t) { t && t !== w ? this.element.setAttribute(this.name, k) : this.element.removeAttribute(this.name); } } class I extends lit_html_S { constructor(t, i, s, e, o) { super(t, i, s, e, o), this.type = 5; } _$AI(t) { let i = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : this; var s; if ((t = null !== (s = P(this, t, i, 0)) && void 0 !== s ? s : w) === b) return; const e = this._$AH, o = t === w && e !== w || t.capture !== e.capture || t.once !== e.once || t.passive !== e.passive, n = t !== w && (e === w || o); o && this.element.removeEventListener(this.name, this, e), n && this.element.addEventListener(this.name, this, t), this._$AH = t; } handleEvent(t) { var i, s; "function" == typeof this._$AH ? this._$AH.call(null !== (s = null === (i = this.options) || void 0 === i ? void 0 : i.host) && void 0 !== s ? s : this.element, t) : this._$AH.handleEvent(t); } } class L { constructor(t, i, s) { this.element = t, this.type = 6, this._$AN = void 0, this._$AM = i, this.options = s; } get _$AU() { return this._$AM._$AU; } _$AI(t) { P(this, t); } } const R = { P: "$lit$", L: lit_html_e, V: lit_html_o, I: 1, N: C, R: V, D: lit_html_u, j: P, H: N, O: lit_html_S, F: H, B: I, W: M, Z: L }, z = window.litHtmlPolyfillSupport; null == z || z(E, N), (null !== (lit_html_t = globalThis.litHtmlVersions) && void 0 !== lit_html_t ? lit_html_t : globalThis.litHtmlVersions = []).push("2.2.0"); ;// CONCATENATED MODULE: ./node_modules/lit-element/lit-element.js /** * @license * Copyright 2017 Google LLC * SPDX-License-Identifier: BSD-3-Clause */var lit_element_l,lit_element_o;const lit_element_r=(/* unused pure expression or super */ null && (t));class lit_element_s extends a{constructor(){super(...arguments),this.renderOptions={host:this},this._$Dt=void 0}createRenderRoot(){var t,e;const i=super.createRenderRoot();return null!==(t=(e=this.renderOptions).renderBefore)&&void 0!==t||(e.renderBefore=i.firstChild),i}update(t){const i=this.render();this.hasUpdated||(this.renderOptions.isConnected=this.isConnected),super.update(t),this._$Dt=x(i,this.renderRoot,this.renderOptions)}connectedCallback(){var t;super.connectedCallback(),null===(t=this._$Dt)||void 0===t||t.setConnected(!0)}disconnectedCallback(){var t;super.disconnectedCallback(),null===(t=this._$Dt)||void 0===t||t.setConnected(!1)}render(){return b}}lit_element_s.finalized=!0,lit_element_s._$litElement$=!0,null===(lit_element_l=globalThis.litElementHydrateSupport)||void 0===lit_element_l||lit_element_l.call(globalThis,{LitElement:lit_element_s});const lit_element_n=globalThis.litElementPolyfillSupport;null==lit_element_n||lit_element_n({LitElement:lit_element_s});const lit_element_h={_$AK:(t,e,i)=>{t._$AK(e,i)},_$AL:t=>t._$AL};(null!==(lit_element_o=globalThis.litElementVersions)&&void 0!==lit_element_o?lit_element_o:globalThis.litElementVersions=[]).push("3.2.0"); //# sourceMappingURL=lit-element.js.map ;// CONCATENATED MODULE: ./node_modules/lit/index.js //# sourceMappingURL=index.js.map ;// CONCATENATED MODULE: ./src/headless/core.js /** * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) */ dayjs_min_default().extend((advancedFormat_default())); // Add Strophe Namespaces Strophe.addNamespace('ACTIVITY', 'http://jabber.org/protocol/activity'); 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('EME', 'urn:xmpp:eme:0'); Strophe.addNamespace('FASTEN', 'urn:xmpp:fasten:0'); Strophe.addNamespace('FORWARD', 'urn:xmpp:forward:0'); Strophe.addNamespace('HINTS', 'urn:xmpp:hints'); Strophe.addNamespace('HTTPUPLOAD', 'urn:xmpp:http:upload:0'); Strophe.addNamespace('MAM', 'urn:xmpp:mam:2'); Strophe.addNamespace('MARKERS', 'urn:xmpp:chat-markers:0'); Strophe.addNamespace('MENTIONS', 'urn:xmpp:mmn:0'); Strophe.addNamespace('MESSAGE_CORRECT', 'urn:xmpp:message-correct:0'); Strophe.addNamespace('MODERATE', 'urn:xmpp:message-moderate:0'); Strophe.addNamespace('NICK', 'http://jabber.org/protocol/nick'); Strophe.addNamespace('OCCUPANTID', 'urn:xmpp:occupant-id:0'); Strophe.addNamespace('OMEMO', 'eu.siacs.conversations.axolotl'); Strophe.addNamespace('OUTOFBAND', 'jabber:x:oob'); Strophe.addNamespace('PUBSUB', 'http://jabber.org/protocol/pubsub'); Strophe.addNamespace('RAI', 'urn:xmpp:rai:0'); Strophe.addNamespace('RECEIPTS', 'urn:xmpp:receipts'); Strophe.addNamespace('REFERENCE', 'urn:xmpp:reference:0'); Strophe.addNamespace('REGISTER', 'jabber:iq:register'); Strophe.addNamespace('RETRACT', 'urn:xmpp:message-retract:0'); Strophe.addNamespace('ROSTERX', 'http://jabber.org/protocol/rosterx'); Strophe.addNamespace('RSM', 'http://jabber.org/protocol/rsm'); Strophe.addNamespace('SID', 'urn:xmpp:sid:0'); Strophe.addNamespace('SPOILER', 'urn:xmpp:spoiler:0'); Strophe.addNamespace('STANZAS', 'urn:ietf:params:xml:ns:xmpp-stanzas'); Strophe.addNamespace('STYLING', 'urn:xmpp:styling:0'); Strophe.addNamespace('VCARD', 'vcard-temp'); Strophe.addNamespace('VCARDUPDATE', 'vcard-temp:x:update'); Strophe.addNamespace('XFORM', 'jabber:x:data'); Strophe.addNamespace('XHTML', 'http://www.w3.org/1999/xhtml'); shared_converse.VERSION_NAME = "v9.1.1"; Object.assign(shared_converse, Events); // Make converse pluggable pluggable.enable(shared_converse, '_converse', 'pluggable'); /** * ### The private API * * The private API methods are only accessible via the closured {@link _converse} * object, which is only available to plugins. * * These methods are kept private (i.e. not global) because they may return * sensitive data which should be kept off-limits to other 3rd-party scripts * that might be running in the page. * * @namespace _converse.api * @memberOf _converse */ const core_api = shared_converse.api = { connection: api, settings: settings_api, /** * Lets you trigger events, which can be listened to via * {@link _converse.api.listen.on} or {@link _converse.api.listen.once} * (see [_converse.api.listen](http://localhost:8000/docs/html/api/-_converse.api.listen.html)). * * Some events also double as promises and can be waited on via {@link _converse.api.waitUntil}. * * @method _converse.api.trigger * @param {string} name - The event name * @param {...any} [argument] - Argument to be passed to the event handler * @param {object} [options] * @param {boolean} [options.synchronous] - Whether the event is synchronous or not. * When a synchronous event is fired, a promise will be returned * by {@link _converse.api.trigger} which resolves once all the * event handlers' promises have been resolved. */ async trigger(name) { if (!shared_converse._events) { return; } const args = Array.from(arguments); const options = args.pop(); if (options && options.synchronous) { const events = shared_converse._events[name] || []; const event_args = args.splice(1); await Promise.all(events.map(e => e.callback.apply(e.ctx, event_args))); } else { shared_converse.trigger.apply(shared_converse, arguments); } const promise = shared_converse.promises[name]; if (promise !== undefined) { promise.resolve(); } }, /** * Triggers a hook which can be intercepted by registered listeners via * {@link _converse.api.listen.on} or {@link _converse.api.listen.once}. * (see [_converse.api.listen](http://localhost:8000/docs/html/api/-_converse.api.listen.html)). * A hook is a special kind of event which allows you to intercept a data * structure in order to modify it, before passing it back. * @async * @param {string} name - The hook name * @param {...any} context - The context to which the hook applies (could be for example, a {@link _converse.ChatBox)). * @param {...any} data - The data structure to be intercepted and modified by the hook listeners. * @returns {Promise} - A promise that resolves with the modified data structure. */ hook(name, context, data) { const events = shared_converse._events[name] || []; if (events.length) { // Create a chain of promises, with each one feeding its output to // the next. The first input is a promise with the original data // sent to this hook. return events.reduce((o, e) => o.then(d => e.callback(context, d)), Promise.resolve(data)); } else { return data; } }, /** * This grouping collects API functions related to the current logged in user. * * @namespace _converse.api.user * @memberOf _converse.api */ user: { settings: user_settings_api, /** * @method _converse.api.user.jid * @returns {string} The current user's full JID (Jabber ID) * @example _converse.api.user.jid()) */ jid() { return shared_converse.connection.jid; }, /** * Logs the user in. * * If called without any parameters, Converse will try * to log the user in by calling the `prebind_url` or `credentials_url` depending * on whether prebinding is used or not. * * @method _converse.api.user.login * @param {string} [jid] * @param {string} [password] * @param {boolean} [automatic=false] - An internally used flag that indicates whether * this method was called automatically once the connection has been * initialized. It's used together with the `auto_login` configuration flag * to determine whether Converse should try to log the user in if it * fails to restore a previous auth'd session. * @returns {void} */ async login(jid, password) { var _converse$connection, _api$settings$get; let automatic = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false; jid = jid || core_api.settings.get('jid'); if (!((_converse$connection = shared_converse.connection) !== null && _converse$connection !== void 0 && _converse$connection.jid) || jid && !utils_core.isSameDomain(shared_converse.connection.jid, jid)) { await shared_converse.initConnection(); } if ((_api$settings$get = core_api.settings.get("connection_options")) !== null && _api$settings$get !== void 0 && _api$settings$get.worker && (await shared_converse.connection.restoreWorkerSession())) { return; } if (jid) { jid = await setUserJID(jid); } // See whether there is a BOSH session to re-attach to const bosh_plugin = shared_converse.pluggable.plugins["converse-bosh"]; if (bosh_plugin !== null && bosh_plugin !== void 0 && bosh_plugin.enabled()) { if (await shared_converse.restoreBOSHSession()) { return; } else if (core_api.settings.get("authentication") === shared_converse.PREBIND && (!automatic || core_api.settings.get("auto_login"))) { return shared_converse.startNewPreboundBOSHSession(); } } password = password || core_api.settings.get("password"); const credentials = jid && password ? { jid, password } : null; attemptNonPreboundSession(credentials, automatic); }, /** * Logs the user out of the current XMPP session. * @method _converse.api.user.logout * @example _converse.api.user.logout(); */ async logout() { /** * Triggered before the user is logged out * @event _converse#beforeLogout */ await core_api.trigger('beforeLogout', { 'synchronous': true }); const promise = getOpenPromise(); const complete = () => { // Recreate all the promises Object.keys(shared_converse.promises).forEach(replacePromise); delete shared_converse.jid; /** * Triggered once the user has logged out. * @event _converse#logout */ core_api.trigger('logout'); promise.resolve(); }; shared_converse.connection.setDisconnectionCause(shared_converse.LOGOUT, undefined, true); if (shared_converse.connection !== undefined) { core_api.listen.once('disconnected', () => complete()); shared_converse.connection.disconnect(); } else { complete(); } return promise; } }, /** * Converse and its plugins trigger various events which you can listen to via the * {@link _converse.api.listen} namespace. * * Some of these events are also available as [ES2015 Promises](http://es6-features.org/#PromiseUsage) * although not all of them could logically act as promises, since some events * might be fired multpile times whereas promises are to be resolved (or * rejected) only once. * * Events which are also promises include: * * * [cachedRoster](/docs/html/events.html#cachedroster) * * [chatBoxesFetched](/docs/html/events.html#chatBoxesFetched) * * [pluginsInitialized](/docs/html/events.html#pluginsInitialized) * * [roster](/docs/html/events.html#roster) * * [rosterContactsFetched](/docs/html/events.html#rosterContactsFetched) * * [rosterGroupsFetched](/docs/html/events.html#rosterGroupsFetched) * * [rosterInitialized](/docs/html/events.html#rosterInitialized) * * The various plugins might also provide promises, and they do this by using the * `promises.add` api method. * * @namespace _converse.api.promises * @memberOf _converse.api */ promises: { /** * By calling `promises.add`, a new [Promise](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise) * is made available for other code or plugins to depend on via the * {@link _converse.api.waitUntil} method. * * Generally, it's the responsibility of the plugin which adds the promise to * also resolve it. * * This is done by calling {@link _converse.api.trigger}, which not only resolves the * promise, but also emits an event with the same name (which can be listened to * via {@link _converse.api.listen}). * * @method _converse.api.promises.add * @param {string|array} [name|names] The name or an array of names for the promise(s) to be added * @param {boolean} [replace=true] Whether this promise should be replaced with a new one when the user logs out. * @example _converse.api.promises.add('foo-completed'); */ add(promises) { let replace = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : true; promises = Array.isArray(promises) ? promises : [promises]; promises.forEach(name => { const promise = getOpenPromise(); promise.replace = replace; shared_converse.promises[name] = promise; }); } }, /** * Converse emits events to which you can subscribe to. * * The `listen` namespace exposes methods for creating event listeners * (aka handlers) for these events. * * @namespace _converse.api.listen * @memberOf _converse */ listen: { /** * Lets you listen to an event exactly once. * @method _converse.api.listen.once * @param {string} name The event's name * @param {function} callback The callback method to be called when the event is emitted. * @param {object} [context] The value of the `this` parameter for the callback. * @example _converse.api.listen.once('message', function (messageXML) { ... }); */ once: shared_converse.once.bind(shared_converse), /** * Lets you subscribe to an event. * Every time the event fires, the callback method specified by `callback` will be called. * @method _converse.api.listen.on * @param {string} name The event's name * @param {function} callback The callback method to be called when the event is emitted. * @param {object} [context] The value of the `this` parameter for the callback. * @example _converse.api.listen.on('message', function (messageXML) { ... }); */ on: shared_converse.on.bind(shared_converse), /** * To stop listening to an event, you can use the `not` method. * @method _converse.api.listen.not * @param {string} name The event's name * @param {function} callback The callback method that is to no longer be called when the event fires * @example _converse.api.listen.not('message', function (messageXML); */ not: shared_converse.off.bind(shared_converse), /** * Subscribe to an incoming stanza * Every a matched stanza is received, the callback method specified by * `callback` will be called. * @method _converse.api.listen.stanza * @param {string} name The stanza's name * @param {object} options Matching options (e.g. 'ns' for namespace, 'type' for stanza type, also 'id' and 'from'); * @param {function} handler The callback method to be called when the stanza appears */ stanza(name, options, handler) { if (lodash_es_isFunction(options)) { handler = options; options = {}; } else { options = options || {}; } shared_converse.connection.addHandler(handler, options.ns, name, options.type, options.id, options.from, options); } }, /** * Wait until a promise is resolved or until the passed in function returns * a truthy value. * @method _converse.api.waitUntil * @param {string|function} condition - The name of the promise to wait for, * or a function which should eventually return a truthy value. * @returns {Promise} */ waitUntil(condition) { if (lodash_es_isFunction(condition)) { return utils_core.waitUntil(condition); } else { const promise = shared_converse.promises[condition]; if (promise === undefined) { return null; } return promise; } }, /** * Allows you to send XML stanzas. * @method _converse.api.send * @param {XMLElement} stanza * @return {void} * @example * const msg = converse.env.$msg({ * 'from': 'juliet@example.com/balcony', * 'to': 'romeo@example.net', * 'type':'chat' * }); * _converse.api.send(msg); */ send(stanza) { var _stanza; if (!core_api.connection.connected()) { headless_log.warn("Not sending stanza because we're not connected!"); headless_log.warn(Strophe.serialize(stanza)); return; } if (typeof stanza === 'string') { stanza = utils_core.toStanza(stanza); } else if ((_stanza = stanza) !== null && _stanza !== void 0 && _stanza.nodeTree) { stanza = stanza.nodeTree; } if (stanza.tagName === 'iq') { return core_api.sendIQ(stanza); } else { shared_converse.connection.send(stanza); core_api.trigger('send', stanza); } }, /** * Send an IQ stanza * @method _converse.api.sendIQ * @param {XMLElement} stanza * @param {Integer} [timeout=_converse.STANZA_TIMEOUT] * @param {Boolean} [reject=true] - Whether an error IQ should cause the promise * to be rejected. If `false`, the promise will resolve instead of being rejected. * @returns {Promise} A promise which resolves (or potentially rejected) once we * receive a `result` or `error` stanza or once a timeout is reached. * If the IQ stanza being sent is of type `result` or `error`, there's * nothing to wait for, so an already resolved promise is returned. */ sendIQ(stanza) { var _stanza2; let timeout = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : shared_converse.STANZA_TIMEOUT; let reject = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : true; let promise; stanza = ((_stanza2 = stanza) === null || _stanza2 === void 0 ? void 0 : _stanza2.nodeTree) ?? stanza; if (['get', 'set'].includes(stanza.getAttribute('type'))) { timeout = timeout || shared_converse.STANZA_TIMEOUT; if (reject) { promise = new Promise((resolve, reject) => shared_converse.connection.sendIQ(stanza, resolve, reject, timeout)); promise.catch(e => { if (e === null) { throw new TimeoutError(`Timeout error after ${timeout}ms for the following IQ stanza: ${Strophe.serialize(stanza)}`); } }); } else { promise = new Promise(resolve => shared_converse.connection.sendIQ(stanza, resolve, resolve, timeout)); } } else { shared_converse.connection.sendIQ(stanza); promise = Promise.resolve(); } core_api.trigger('send', stanza); return promise; } }; shared_converse.shouldClearCache = () => !shared_converse.config.get('trusted') || core_api.settings.get('clear_cache_on_logout') || shared_converse.isTestEnv(); function clearSession() { var _converse$session; (_converse$session = shared_converse.session) === null || _converse$session === void 0 ? void 0 : _converse$session.destroy(); delete shared_converse.session; shared_converse.shouldClearCache() && shared_converse.api.user.settings.clear(); /** * Synchronouse event triggered once the user session has been cleared, * for example when the user has logged out or when Converse has * disconnected for some other reason. * @event _converse#clearSession */ return shared_converse.api.trigger('clearSession', { 'synchronous': true }); } shared_converse.initConnection = function () { const api = shared_converse.api; if (!api.settings.get('bosh_service_url')) { if (api.settings.get("authentication") === shared_converse.PREBIND) { throw new Error("authentication is set to 'prebind' but we don't have a BOSH connection"); } } const XMPPConnection = shared_converse.isTestEnv() ? MockConnection : Connection; shared_converse.connection = new XMPPConnection(getConnectionServiceURL(), Object.assign(shared_converse.default_connection_options, api.settings.get("connection_options"), { 'keepalive': api.settings.get("keepalive") })); setUpXMLLogging(); /** * Triggered once the `Connection` constructor has been initialized, which * will be responsible for managing the connection to the XMPP server. * * @event _converse#connectionInitialized */ api.trigger('connectionInitialized'); }; function setUpXMLLogging() { const lmap = {}; lmap[Strophe.LogLevel.DEBUG] = 'debug'; lmap[Strophe.LogLevel.INFO] = 'info'; lmap[Strophe.LogLevel.WARN] = 'warn'; lmap[Strophe.LogLevel.ERROR] = 'error'; lmap[Strophe.LogLevel.FATAL] = 'fatal'; Strophe.log = (level, msg) => headless_log.log(msg, lmap[level]); Strophe.error = msg => headless_log.error(msg); shared_converse.connection.xmlInput = body => headless_log.debug(body.outerHTML, 'color: darkgoldenrod'); shared_converse.connection.xmlOutput = body => headless_log.debug(body.outerHTML, 'color: darkcyan'); } shared_converse.saveWindowState = function (ev) { // XXX: eventually we should be able to just use // document.visibilityState (when we drop support for older // browsers). let state; const event_map = { 'focus': "visible", 'focusin': "visible", 'pageshow': "visible", 'blur': "hidden", 'focusout': "hidden", 'pagehide': "hidden" }; ev = ev || document.createEvent('Events'); if (ev.type in event_map) { state = event_map[ev.type]; } else { state = document.hidden ? "hidden" : "visible"; } shared_converse.windowState = state; /** * Triggered when window state has changed. * Used to determine when a user left the page and when came back. * @event _converse#windowStateChanged * @type { object } * @property{ string } state - Either "hidden" or "visible" * @example _converse.api.listen.on('windowStateChanged', obj => { ... }); */ core_api.trigger('windowStateChanged', { state }); }; shared_converse.ConnectionFeedback = Model.extend({ defaults: { 'connection_status': Strophe.Status.DISCONNECTED, 'message': '' }, initialize() { this.on('change', () => core_api.trigger('connfeedback', shared_converse.connfeedback)); } }); const core_converse = window.converse || {}; /** * ### The Public API * * This namespace contains public API methods which are are * accessible on the global `converse` object. * They are public, because any JavaScript in the * page can call them. Public methods therefore don’t expose any sensitive * or closured data. To do that, you’ll need to create a plugin, which has * access to the private API method. * * @global * @namespace converse */ Object.assign(core_converse, { CHAT_STATES: CHAT_STATES, keycodes: KEYCODES, /** * Public API method which initializes Converse. * This method must always be called when using Converse. * @async * @memberOf converse * @method initialize * @param {object} config A map of [configuration-settings](https://conversejs.org/docs/html/configuration.html#configuration-settings). * @example * converse.initialize({ * auto_list_rooms: false, * auto_subscribe: false, * bosh_service_url: 'https://bind.example.com', * hide_muc_server: false, * i18n: 'en', * play_sounds: true, * show_controlbox_by_default: true, * debug: false, * roster_groups: true * }); */ async initialize(settings) { var _api$elements; await cleanup(shared_converse); setUnloadEvent(); initAppSettings(settings); shared_converse.strict_plugin_dependencies = settings.strict_plugin_dependencies; // Needed by pluggable.js headless_log.setLogLevel(core_api.settings.get("loglevel")); if (core_api.settings.get("authentication") === shared_converse.ANONYMOUS) { if (core_api.settings.get("auto_login") && !core_api.settings.get('jid')) { throw new Error("Config Error: you need to provide the server's " + "domain via the 'jid' option when using anonymous " + "authentication with auto_login."); } } shared_converse.router.route(/^converse\?loglevel=(debug|info|warn|error|fatal)$/, 'loglevel', l => headless_log.setLogLevel(l)); shared_converse.connfeedback = new shared_converse.ConnectionFeedback(); /* When reloading the page: * For new sessions, we need to send out a presence stanza to notify * the server/network that we're online. * When re-attaching to an existing session we don't need to again send out a presence stanza, * because it's as if "we never left" (see onConnectStatusChanged). * https://github.com/conversejs/converse.js/issues/521 */ shared_converse.send_initial_presence = true; await initSessionStorage(shared_converse); await initClientConfig(shared_converse); await i18n.initialize(); initPlugins(shared_converse); // Register all custom elements // XXX: api.elements is defined in the UI part of Converse, outside of @converse/headless. // This line should probably be moved to the UI code as part of a larger refactoring. (_api$elements = core_api.elements) === null || _api$elements === void 0 ? void 0 : _api$elements.register(); registerGlobalEventHandlers(shared_converse); try { !History.started && shared_converse.router.history.start(); } catch (e) { headless_log.error(e); } const plugins = shared_converse.pluggable.plugins; if (core_api.settings.get("auto_login") || core_api.settings.get("keepalive") && lodash_es_invoke(plugins['converse-bosh'], 'enabled')) { await core_api.user.login(null, null, true); } /** * Triggered once converse.initialize has finished. * @event _converse#initialized */ core_api.trigger('initialized'); if (shared_converse.isTestEnv()) { return shared_converse; } }, /** * Exposes methods for adding and removing plugins. You'll need to write a plugin * if you want to have access to the private API methods defined further down below. * * For more information on plugins, read the documentation on [writing a plugin](/docs/html/plugin_development.html). * @namespace plugins * @memberOf converse */ plugins: { /** * Registers a new plugin. * @method converse.plugins.add * @param {string} name The name of the plugin * @param {object} plugin The plugin object * @example * const plugin = { * initialize: function () { * // Gets called as soon as the plugin has been loaded. * * // Inside this method, you have access to the private * // API via `_covnerse.api`. * * // The private _converse object contains the core logic * // and data-structures of Converse. * } * } * converse.plugins.add('myplugin', plugin); */ add(name, plugin) { plugin.__name__ = name; if (shared_converse.pluggable.plugins[name] !== undefined) { throw new TypeError(`Error: plugin with name "${name}" has already been ` + 'registered!'); } else { shared_converse.pluggable.plugins[name] = plugin; } } }, /** * Utility methods and globals from bundled 3rd party libraries. * @typedef ConverseEnv * @property {function} converse.env.$build - Creates a Strophe.Builder, for creating stanza objects. * @property {function} converse.env.$iq - Creates a Strophe.Builder with an element as the root. * @property {function} converse.env.$msg - Creates a Strophe.Builder with an element as the root. * @property {function} converse.env.$pres - Creates a Strophe.Builder with an element as the root. * @property {function} converse.env.Promise - The Promise implementation used by Converse. * @property {function} converse.env.Strophe - The [Strophe](http://strophe.im/strophejs) XMPP library used by Converse. * @property {function} converse.env.f - And instance of Lodash with its methods wrapped to produce immutable auto-curried iteratee-first data-last methods. * @property {function} converse.env.sizzle - [Sizzle](https://sizzlejs.com) CSS selector engine. * @property {function} converse.env.sprintf * @property {object} converse.env._ - The instance of [lodash-es](http://lodash.com) used by Converse. * @property {object} converse.env.dayjs - [DayJS](https://github.com/iamkun/dayjs) date manipulation library. * @property {object} converse.env.utils - Module containing common utility methods used by Converse. * @memberOf converse */ 'env': { $build: $build, $iq: $iq, $msg: $msg, $pres: $pres, 'utils': utils_core, Collection: Collection, Model: Model, Promise, Strophe: Strophe, URI: (URI_default()), dayjs: (dayjs_min_default()), html: $, log: headless_log, sizzle: (sizzle_default()), sprintf: sprintf.sprintf, u: utils_core } }); ;// CONCATENATED MODULE: ./src/headless/shared/actions.js const actions_u = core_converse.env.utils; function rejectMessage(stanza, text) { // Reject an incoming message by replying with an error message of type "cancel". core_api.send($msg({ 'to': stanza.getAttribute('from'), 'type': 'error', 'id': stanza.getAttribute('id') }).c('error', { 'type': 'cancel' }).c('not-allowed', { xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas' }).up().c('text', { xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas' }).t(text)); headless_log.warn(`Rejecting message stanza with the following reason: ${text}`); headless_log.warn(stanza); } /** * Send out a XEP-0333 chat marker * @param { String } to_jid * @param { String } id - The id of the message being marked * @param { String } type - The marker type * @param { String } msg_type */ function sendMarker(to_jid, id, type, msg_type) { const stanza = $msg({ 'from': shared_converse.connection.jid, 'id': actions_u.getUniqueId(), 'to': to_jid, 'type': msg_type ? msg_type : 'chat' }).c(type, { 'xmlns': Strophe.NS.MARKERS, 'id': id }); core_api.send(stanza); } ;// CONCATENATED MODULE: ./src/headless/utils/url.js const { u: url_u } = core_converse.env; /** * Given a url, check whether the protocol being used is allowed for rendering * the media in the chat (as opposed to just rendering a URL hyperlink). * @param { String } url * @returns { Boolean } */ function isAllowedProtocolForMedia(url) { const uri = getURI(url); const { protocol } = window.location; if (['chrome-extension:', 'file:'].includes(protocol)) { return true; } return protocol === 'http:' || protocol === 'https:' && ['https', 'aesgcm'].includes(uri.protocol().toLowerCase()); } function getURI(url) { try { return url instanceof (URI_default()) ? url : new (URI_default())(url); } catch (error) { headless_log.debug(error); return null; } } /** * Given the an array of file extensions, check whether a URL points to a file * ending in one of them. * @param { String[] } types - An array of file extensions * @param { String } url * @returns { Boolean } * @example * checkFileTypes(['.gif'], 'https://conversejs.org/cat.gif?foo=bar'); */ function checkFileTypes(types, url) { const uri = getURI(url); if (uri === null) { throw new Error(`checkFileTypes: could not parse url ${url}`); } const filename = uri.filename().toLowerCase(); return !!types.filter(ext => filename.endsWith(ext)).length; } function isDomainWhitelisted(whitelist, url) { const uri = getURI(url); const subdomain = uri.subdomain(); const domain = uri.domain(); const fulldomain = `${subdomain ? `${subdomain}.` : ''}${domain}`; return whitelist.includes(domain) || whitelist.includes(fulldomain); } function shouldRenderMediaFromURL(url_text, type) { if (!isAllowedProtocolForMedia(url_text)) { return false; } const may_render = core_api.settings.get('render_media'); const is_domain_allowed = isDomainAllowed(url_text, `allowed_${type}_domains`); if (Array.isArray(may_render)) { return is_domain_allowed && isDomainWhitelisted(may_render, url_text); } else { return is_domain_allowed && may_render; } } function filterQueryParamsFromURL(url) { const paramsArray = core_api.settings.get('filter_url_query_params'); if (!paramsArray) return url; const parsed_uri = getURI(url); return parsed_uri.removeQuery(paramsArray).toString(); } function isDomainAllowed(url, setting) { const allowed_domains = core_api.settings.get(setting); if (!Array.isArray(allowed_domains)) { return true; } try { return isDomainWhitelisted(allowed_domains, url); } catch (error) { headless_log.debug(error); return false; } } /** * Accepts a {@link MediaURL} object and then checks whether its domain is * allowed for rendering in the chat. * @param { MediaURL } o * @returns { Bool } */ function isMediaURLDomainAllowed(o) { return o.is_audio && isDomainAllowed(o.url, 'allowed_audio_domains') || o.is_video && isDomainAllowed(o.url, 'allowed_video_domains') || o.is_image && isDomainAllowed(o.url, 'allowed_image_domains'); } function isURLWithImageExtension(url) { return checkFileTypes(['.jpg', '.jpeg', '.png', '.gif', '.bmp', '.tiff', '.svg'], url); } function isGIFURL(url) { return checkFileTypes(['.gif'], url); } function isAudioURL(url) { return checkFileTypes(['.ogg', '.mp3', '.m4a'], url); } function isVideoURL(url) { return checkFileTypes(['.mp4', '.webm'], url); } function isImageURL(url) { const regex = core_api.settings.get('image_urls_regex'); return (regex === null || regex === void 0 ? void 0 : regex.test(url)) || isURLWithImageExtension(url); } function isEncryptedFileURL(url) { return url.startsWith('aesgcm://'); } Object.assign(url_u, { isAudioURL, isGIFURL, isVideoURL, isImageURL, isURLWithImageExtension, checkFileTypes, getURI, shouldRenderMediaFromURL, isAllowedProtocolForMedia }); ;// CONCATENATED MODULE: ./src/headless/shared/parsers.js const { NS } = Strophe; class StanzaParseError extends Error { constructor(message, stanza) { super(message, stanza); this.name = 'StanzaParseError'; this.stanza = stanza; } } /** * Extract the XEP-0359 stanza IDs from the passed in stanza * and return a map containing them. * @private * @param { XMLElement } stanza - The message stanza * @returns { Object } */ function getStanzaIDs(stanza, original_stanza) { const attrs = {}; // Store generic stanza ids const sids = sizzle_default()(`stanza-id[xmlns="${Strophe.NS.SID}"]`, stanza); const sid_attrs = sids.reduce((acc, s) => { acc[`stanza_id ${s.getAttribute('by')}`] = s.getAttribute('id'); return acc; }, {}); Object.assign(attrs, sid_attrs); // Store the archive id const result = sizzle_default()(`message > result[xmlns="${Strophe.NS.MAM}"]`, original_stanza).pop(); if (result) { const by_jid = original_stanza.getAttribute('from') || shared_converse.bare_jid; attrs[`stanza_id ${by_jid}`] = result.getAttribute('id'); } // Store the origin id const origin_id = sizzle_default()(`origin-id[xmlns="${Strophe.NS.SID}"]`, stanza).pop(); if (origin_id) { attrs['origin_id'] = origin_id.getAttribute('id'); } return attrs; } function getEncryptionAttributes(stanza) { const eme_tag = sizzle_default()(`encryption[xmlns="${Strophe.NS.EME}"]`, stanza).pop(); const namespace = eme_tag === null || eme_tag === void 0 ? void 0 : eme_tag.getAttribute('namespace'); const attrs = {}; if (namespace) { attrs.is_encrypted = true; attrs.encryption_namespace = namespace; } else if (sizzle_default()(`encrypted[xmlns="${Strophe.NS.OMEMO}"]`, stanza).pop()) { attrs.is_encrypted = true; attrs.encryption_namespace = Strophe.NS.OMEMO; } return attrs; } /** * @private * @param { XMLElement } stanza - The message stanza * @param { XMLElement } original_stanza - The original stanza, that contains the * message stanza, if it was contained, otherwise it's the message stanza itself. * @returns { Object } */ function getRetractionAttributes(stanza, original_stanza) { const fastening = sizzle_default()(`> apply-to[xmlns="${Strophe.NS.FASTEN}"]`, stanza).pop(); if (fastening) { const applies_to_id = fastening.getAttribute('id'); const retracted = sizzle_default()(`> retract[xmlns="${Strophe.NS.RETRACT}"]`, fastening).pop(); if (retracted) { const delay = sizzle_default()(`delay[xmlns="${Strophe.NS.DELAY}"]`, original_stanza).pop(); const time = delay ? dayjs_min_default()(delay.getAttribute('stamp')).toISOString() : new Date().toISOString(); return { 'editable': false, 'retracted': time, 'retracted_id': applies_to_id }; } } else { const tombstone = sizzle_default()(`> retracted[xmlns="${Strophe.NS.RETRACT}"]`, stanza).pop(); if (tombstone) { return { 'editable': false, 'is_tombstone': true, 'retracted': tombstone.getAttribute('stamp') }; } } return {}; } function getCorrectionAttributes(stanza, original_stanza) { const el = sizzle_default()(`replace[xmlns="${Strophe.NS.MESSAGE_CORRECT}"]`, stanza).pop(); if (el) { const replace_id = el.getAttribute('id'); if (replace_id) { const delay = sizzle_default()(`delay[xmlns="${Strophe.NS.DELAY}"]`, original_stanza).pop(); const time = delay ? dayjs_min_default()(delay.getAttribute('stamp')).toISOString() : new Date().toISOString(); return { replace_id, 'edited': time }; } } return {}; } function getOpenGraphMetadata(stanza) { const fastening = sizzle_default()(`> apply-to[xmlns="${Strophe.NS.FASTEN}"]`, stanza).pop(); if (fastening) { const applies_to_id = fastening.getAttribute('id'); const meta = sizzle_default()(`> meta[xmlns="${Strophe.NS.XHTML}"]`, fastening); if (meta.length) { const msg_limit = core_api.settings.get('message_limit'); const data = meta.reduce((acc, el) => { const property = el.getAttribute('property'); if (property) { let value = decodeHTMLEntities(el.getAttribute('content') || ''); if (msg_limit && property === 'og:description' && value.length >= msg_limit) { value = `${value.slice(0, msg_limit)}${decodeHTMLEntities('…')}`; } acc[property] = value; } return acc; }, { 'ogp_for_id': applies_to_id }); if ("og:description" in data || "og:title" in data || "og:image" in data) { return data; } } } return {}; } function getMediaURLsMetadata(text) { let offset = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 0; const objs = []; if (!text) { return {}; } try { URI_default().withinString(text, (url, start, end) => { if (url.startsWith('_')) { url = url.slice(1); start += 1; } if (url.endsWith('_')) { url = url.slice(0, url.length - 1); end -= 1; } objs.push({ url, 'start': start + offset, 'end': end + offset }); return url; }, URL_PARSE_OPTIONS); } catch (error) { headless_log.debug(error); } /** * @typedef { Object } MediaURLMetadata * An object representing the metadata of a URL found in a chat message * The actual URL is not saved, it can be extracted via the `start` and `end` indexes. * @property { Boolean } is_audio * @property { Boolean } is_image * @property { Boolean } is_video * @property { String } end * @property { String } start */ const media_urls = objs.map(o => ({ 'end': o.end, 'is_audio': isAudioURL(o.url), 'is_image': isImageURL(o.url), 'is_video': isVideoURL(o.url), 'is_encrypted': isEncryptedFileURL(o.url), 'start': o.start })); return media_urls.length ? { media_urls } : {}; } function getSpoilerAttributes(stanza) { const spoiler = sizzle_default()(`spoiler[xmlns="${Strophe.NS.SPOILER}"]`, stanza).pop(); return { 'is_spoiler': !!spoiler, 'spoiler_hint': spoiler === null || spoiler === void 0 ? void 0 : spoiler.textContent }; } function getOutOfBandAttributes(stanza) { const xform = sizzle_default()(`x[xmlns="${Strophe.NS.OUTOFBAND}"]`, stanza).pop(); if (xform) { var _xform$querySelector, _xform$querySelector2; return { 'oob_url': (_xform$querySelector = xform.querySelector('url')) === null || _xform$querySelector === void 0 ? void 0 : _xform$querySelector.textContent, 'oob_desc': (_xform$querySelector2 = xform.querySelector('desc')) === null || _xform$querySelector2 === void 0 ? void 0 : _xform$querySelector2.textContent }; } return {}; } /** * Returns the human readable error message contained in a `groupchat` message stanza of type `error`. * @private * @param { XMLElement } stanza - The message stanza */ function getErrorAttributes(stanza) { if (stanza.getAttribute('type') === 'error') { const error = stanza.querySelector('error'); const text = sizzle_default()(`text[xmlns="${Strophe.NS.STANZAS}"]`, error).pop(); return { 'is_error': true, 'error_text': text === null || text === void 0 ? void 0 : text.textContent, 'error_type': error.getAttribute('type'), 'error_condition': error.firstElementChild.nodeName }; } return {}; } function getReferences(stanza) { return sizzle_default()(`reference[xmlns="${Strophe.NS.REFERENCE}"]`, stanza).map(ref => { var _stanza$querySelector; const anchor = ref.getAttribute('anchor'); const text = (_stanza$querySelector = stanza.querySelector(anchor ? `#${anchor}` : 'body')) === null || _stanza$querySelector === void 0 ? void 0 : _stanza$querySelector.textContent; if (!text) { headless_log.warn(`Could not find referenced text for ${ref}`); return null; } const begin = ref.getAttribute('begin'); const end = ref.getAttribute('end'); return { 'begin': begin, 'end': end, 'type': ref.getAttribute('type'), 'value': text.slice(begin, end), 'uri': ref.getAttribute('uri') }; }).filter(r => r); } function getReceiptId(stanza) { const receipt = sizzle_default()(`received[xmlns="${Strophe.NS.RECEIPTS}"]`, stanza).pop(); return receipt === null || receipt === void 0 ? void 0 : receipt.getAttribute('id'); } /** * Determines whether the passed in stanza is a XEP-0280 Carbon * @private * @param { XMLElement } stanza - The message stanza * @returns { Boolean } */ function isCarbon(stanza) { const xmlns = Strophe.NS.CARBONS; return sizzle_default()(`message > received[xmlns="${xmlns}"]`, stanza).length > 0 || sizzle_default()(`message > sent[xmlns="${xmlns}"]`, stanza).length > 0; } /** * Returns the XEP-0085 chat state contained in a message stanza * @private * @param { XMLElement } stanza - The message stanza */ function getChatState(stanza) { var _sizzle$pop; return (_sizzle$pop = sizzle_default()(` composing[xmlns="${NS.CHATSTATES}"], paused[xmlns="${NS.CHATSTATES}"], inactive[xmlns="${NS.CHATSTATES}"], active[xmlns="${NS.CHATSTATES}"], gone[xmlns="${NS.CHATSTATES}"]`, stanza).pop()) === null || _sizzle$pop === void 0 ? void 0 : _sizzle$pop.nodeName; } function isValidReceiptRequest(stanza, attrs) { return attrs.sender !== 'me' && !attrs.is_carbon && !attrs.is_archived && sizzle_default()(`request[xmlns="${Strophe.NS.RECEIPTS}"]`, stanza).length; } /** * Check whether the passed-in stanza is a forwarded message that is "bare", * i.e. it's not forwarded as part of a larger protocol, like MAM. * @param { XMLElement } stanza */ function throwErrorIfInvalidForward(stanza) { const bare_forward = sizzle_default()(`message > forwarded[xmlns="${Strophe.NS.FORWARD}"]`, stanza).length; if (bare_forward) { rejectMessage(stanza, 'Forwarded messages not part of an encapsulating protocol are not supported'); const from_jid = stanza.getAttribute('from'); throw new StanzaParseError(`Ignoring unencapsulated forwarded message from ${from_jid}`, stanza); } } /** * Determines whether the passed in stanza is a XEP-0333 Chat Marker * @private * @method getChatMarker * @param { XMLElement } stanza - The message stanza * @returns { Boolean } */ function getChatMarker(stanza) { // If we receive more than one marker (which shouldn't happen), we take // the highest level of acknowledgement. return sizzle_default()(` acknowledged[xmlns="${Strophe.NS.MARKERS}"], displayed[xmlns="${Strophe.NS.MARKERS}"], received[xmlns="${Strophe.NS.MARKERS}"]`, stanza).pop(); } function isHeadline(stanza) { return stanza.getAttribute('type') === 'headline'; } function isServerMessage(stanza) { if (sizzle_default()(`mentions[xmlns="${Strophe.NS.MENTIONS}"]`, stanza).pop()) { return false; } const from_jid = stanza.getAttribute('from'); if (stanza.getAttribute('type') !== 'error' && from_jid && !from_jid.includes('@')) { // Some servers (e.g. Prosody) don't set the stanza // 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 stanza. return true; } return false; } /** * Determines whether the passed in stanza is a XEP-0313 MAM stanza * @private * @method isArchived * @param { XMLElement } stanza - The message stanza * @returns { Boolean } */ function isArchived(original_stanza) { return !!sizzle_default()(`message > result[xmlns="${Strophe.NS.MAM}"]`, original_stanza).pop(); } /** * Returns an object containing all attribute names and values for a particular element. * @method getAttributes * @param { XMLElement } stanza * @returns { Object } */ function getAttributes(stanza) { return stanza.getAttributeNames().reduce((acc, name) => { acc[name] = Strophe.xmlunescape(stanza.getAttribute(name)); return acc; }, {}); } ;// CONCATENATED MODULE: ./src/headless/plugins/adhoc.js const { Strophe: adhoc_Strophe } = core_converse.env; let adhoc_converse, adhoc_api; adhoc_Strophe.addNamespace('ADHOC', 'http://jabber.org/protocol/commands'); function parseForCommands(stanza) { const items = sizzle_default()(`query[xmlns="${adhoc_Strophe.NS.DISCO_ITEMS}"][node="${adhoc_Strophe.NS.ADHOC}"] item`, stanza); return items.map(getAttributes); } const adhoc_adhoc_api = { /** * The XEP-0050 Ad-Hoc Commands API * * This API lets you discover ad-hoc commands available for an entity in the XMPP network. * * @namespace api.adhoc * @memberOf api */ adhoc: { /** * @method api.adhoc.getCommands * @param { String } to_jid */ async getCommands(to_jid) { let commands = []; try { commands = parseForCommands(await adhoc_api.disco.items(to_jid, adhoc_Strophe.NS.ADHOC)); } catch (e) { if (e === null) { headless_log.error(`Error: timeout while fetching ad-hoc commands for ${to_jid}`); } else { headless_log.error(`Error while fetching ad-hoc commands for ${to_jid}`); headless_log.error(e); } } return commands; } } }; core_converse.plugins.add('converse-adhoc', { dependencies: ["converse-disco"], initialize() { adhoc_converse = this._converse; adhoc_api = adhoc_converse.api; Object.assign(adhoc_api, adhoc_adhoc_api); } }); /* harmony default export */ const adhoc = ((/* unused pure expression or super */ null && (adhoc_adhoc_api))); ;// CONCATENATED MODULE: ./src/headless/plugins/chat/model-with-contact.js const ModelWithContact = Model.extend({ initialize() { this.rosterContactAdded = getOpenPromise(); }, async setRosterContact(jid) { const contact = await core_api.contacts.get(jid); if (contact) { this.contact = contact; this.set('nickname', contact.get('nickname')); this.rosterContactAdded.resolve(); } } }); /* harmony default export */ const model_with_contact = (ModelWithContact); // EXTERNAL MODULE: ./node_modules/filesize/lib/filesize.min.js var filesize_min = __webpack_require__(6755); var filesize_min_default = /*#__PURE__*/__webpack_require__.n(filesize_min); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isMatch.js /** * Performs a partial deep comparison between `object` and `source` to * determine if `object` contains equivalent property values. * * **Note:** This method is equivalent to `_.matches` when `source` is * partially applied. * * Partial comparisons will match empty array and empty object `source` * values against any array or object value, respectively. See `_.isEqual` * for a list of supported value comparisons. * * @static * @memberOf _ * @since 3.0.0 * @category Lang * @param {Object} object The object to inspect. * @param {Object} source The object of property values to match. * @returns {boolean} Returns `true` if `object` is a match, else `false`. * @example * * var object = { 'a': 1, 'b': 2 }; * * _.isMatch(object, { 'b': 2 }); * // => true * * _.isMatch(object, { 'b': 1 }); * // => false */ function isMatch(object, source) { return object === source || _baseIsMatch(object, source, _getMatchData(source)); } /* harmony default export */ const lodash_es_isMatch = (isMatch); ;// CONCATENATED MODULE: ./src/headless/shared/chat/utils.js const { u: utils_u } = core_converse.env; function pruneHistory(model) { const max_history = core_api.settings.get('prune_messages_above'); if (max_history && typeof max_history === 'number') { if (model.messages.length > max_history) { const non_empty_messages = model.messages.filter(m => !utils_u.isEmptyMessage(m)); if (non_empty_messages.length > max_history) { while (non_empty_messages.length > max_history) { non_empty_messages.shift().destroy(); } /** * Triggered once the message history has been pruned, i.e. * once older messages have been removed to keep the * number of messages below the value set in `prune_messages_above`. * @event _converse#historyPruned * @type { _converse.ChatBox | _converse.ChatRoom } * @example _converse.api.listen.on('historyPruned', this => { ... }); */ core_api.trigger('historyPruned', model); } } } } /** * Given an array of {@link MediaURLMetadata} objects and text, return an * array of {@link MediaURL} objects. * @param { Array } arr * @param { String } text * @returns{ Array } */ function getMediaURLs(arr, text) { let offset = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 0; /** * @typedef { Object } MediaURLData * An object representing a URL found in a chat message * @property { Boolean } is_audio * @property { Boolean } is_image * @property { Boolean } is_video * @property { String } end * @property { String } start * @property { String } url */ return arr.map(o => { const start = o.start - offset; const end = o.end - offset; if (start < 0 || start >= text.length) { return null; } return Object.assign({}, o, { start, end, 'url': text.substring(o.start - offset, o.end - offset) }); }).filter(o => o); } const debouncedPruneHistory = lodash_es_debounce(pruneHistory, 500); ;// CONCATENATED MODULE: ./src/headless/plugins/chat/parsers.js const { Strophe: parsers_Strophe, sizzle: parsers_sizzle } = core_converse.env; /** * Parses a passed in message stanza and returns an object of attributes. * @method st#parseMessage * @param { XMLElement } stanza - The message stanza * @param { _converse } _converse * @returns { (MessageAttributes|Error) } */ async function parseMessage(stanza) { var _stanza$querySelector, _stanza$querySelector2, _contact, _contact$attributes, _stanza$querySelector3, _stanza$querySelector4; throwErrorIfInvalidForward(stanza); let to_jid = stanza.getAttribute('to'); const to_resource = parsers_Strophe.getResourceFromJid(to_jid); if (core_api.settings.get('filter_by_resource') && to_resource && to_resource !== shared_converse.resource) { return new StanzaParseError(`Ignoring incoming message intended for a different resource: ${to_jid}`, stanza); } const original_stanza = stanza; let from_jid = stanza.getAttribute('from') || shared_converse.bare_jid; if (isCarbon(stanza)) { if (from_jid === shared_converse.bare_jid) { const selector = `[xmlns="${parsers_Strophe.NS.CARBONS}"] > forwarded[xmlns="${parsers_Strophe.NS.FORWARD}"] > message`; stanza = parsers_sizzle(selector, stanza).pop(); to_jid = stanza.getAttribute('to'); from_jid = stanza.getAttribute('from'); } else { // Prevent message forging via carbons: https://xmpp.org/extensions/xep-0280.html#security rejectMessage(stanza, 'Rejecting carbon from invalid JID'); return new StanzaParseError(`Rejecting carbon from invalid JID ${to_jid}`, stanza); } } const is_archived = isArchived(stanza); if (is_archived) { if (from_jid === shared_converse.bare_jid) { const selector = `[xmlns="${parsers_Strophe.NS.MAM}"] > forwarded[xmlns="${parsers_Strophe.NS.FORWARD}"] > message`; stanza = parsers_sizzle(selector, stanza).pop(); to_jid = stanza.getAttribute('to'); from_jid = stanza.getAttribute('from'); } else { return new StanzaParseError(`Invalid Stanza: alleged MAM message from ${stanza.getAttribute('from')}`, stanza); } } const from_bare_jid = parsers_Strophe.getBareJidFromJid(from_jid); const is_me = from_bare_jid === shared_converse.bare_jid; if (is_me && to_jid === null) { return new StanzaParseError(`Don't know how to handle message stanza without 'to' attribute. ${stanza.outerHTML}`, stanza); } const is_headline = isHeadline(stanza); const is_server_message = isServerMessage(stanza); let contact, contact_jid; if (!is_headline && !is_server_message) { contact_jid = is_me ? parsers_Strophe.getBareJidFromJid(to_jid) : from_bare_jid; contact = await core_api.contacts.get(contact_jid); if (contact === undefined && !core_api.settings.get('allow_non_roster_messaging')) { headless_log.error(stanza); return new StanzaParseError(`Blocking messaging with a JID not in our roster because allow_non_roster_messaging is false.`, stanza); } } /** * @typedef { Object } MessageAttributes * The object which {@link parseMessage} returns * @property { ('me'|'them') } sender - Whether the message was sent by the current user or someone else * @property { Array } references - A list of objects representing XEP-0372 references * @property { Boolean } editable - Is this message editable via XEP-0308? * @property { Boolean } is_archived - Is this message from a XEP-0313 MAM archive? * @property { Boolean } is_carbon - Is this message a XEP-0280 Carbon? * @property { Boolean } is_delayed - Was delivery of this message was delayed as per XEP-0203? * @property { Boolean } is_encrypted - Is this message XEP-0384 encrypted? * @property { Boolean } is_error - Whether an error was received for this message * @property { Boolean } is_headline - Is this a "headline" message? * @property { Boolean } is_markable - Can this message be marked with a XEP-0333 chat marker? * @property { Boolean } is_marker - Is this message a XEP-0333 Chat Marker? * @property { Boolean } is_only_emojis - Does the message body contain only emojis? * @property { Boolean } is_spoiler - Is this a XEP-0382 spoiler message? * @property { Boolean } is_tombstone - Is this a XEP-0424 tombstone? * @property { Boolean } is_unstyled - Whether XEP-0393 styling hints should be ignored * @property { Boolean } is_valid_receipt_request - Does this message request a XEP-0184 receipt (and is not from us or a carbon or archived message) * @property { Object } encrypted - XEP-0384 encryption payload attributes * @property { String } body - The contents of the tag of the message stanza * @property { String } chat_state - The XEP-0085 chat state notification contained in this message * @property { String } contact_jid - The JID of the other person or entity * @property { String } edited - An ISO8601 string recording the time that the message was edited per XEP-0308 * @property { String } error_condition - The defined error condition * @property { String } error_text - The error text received from the server * @property { String } error_type - The type of error received from the server * @property { String } from - The sender JID * @property { String } fullname - The full name of the sender * @property { String } marker - The XEP-0333 Chat Marker value * @property { String } marker_id - The `id` attribute of a XEP-0333 chat marker * @property { String } msgid - The root `id` attribute of the stanza * @property { String } nick - The roster nickname of the sender * @property { String } oob_desc - The description of the XEP-0066 out of band data * @property { String } oob_url - The URL of the XEP-0066 out of band data * @property { String } origin_id - The XEP-0359 Origin ID * @property { String } receipt_id - The `id` attribute of a XEP-0184 element * @property { String } received - An ISO8601 string recording the time that the message was received * @property { String } replace_id - The `id` attribute of a XEP-0308 element * @property { String } retracted - An ISO8601 string recording the time that the message was retracted * @property { String } retracted_id - The `id` attribute of a XEP-424 element * @property { String } spoiler_hint The XEP-0382 spoiler hint * @property { String } stanza_id - The XEP-0359 Stanza ID. Note: the key is actualy `stanza_id ${by_jid}` and there can be multiple. * @property { String } subject - The element value * @property { String } thread - The element value * @property { String } time - The time (in ISO8601 format), either given by the XEP-0203 element, or of receipt. * @property { String } to - The recipient JID * @property { String } type - The type of message */ const delay = parsers_sizzle(`delay[xmlns="${parsers_Strophe.NS.DELAY}"]`, original_stanza).pop(); const marker = getChatMarker(stanza); const now = new Date().toISOString(); let attrs = Object.assign({ contact_jid, is_archived, is_headline, is_server_message, 'body': (_stanza$querySelector = stanza.querySelector('body')) === null || _stanza$querySelector === void 0 ? void 0 : (_stanza$querySelector2 = _stanza$querySelector.textContent) === null || _stanza$querySelector2 === void 0 ? void 0 : _stanza$querySelector2.trim(), 'chat_state': getChatState(stanza), 'from': parsers_Strophe.getBareJidFromJid(stanza.getAttribute('from')), 'is_carbon': isCarbon(original_stanza), 'is_delayed': !!delay, 'is_markable': !!parsers_sizzle(`markable[xmlns="${parsers_Strophe.NS.MARKERS}"]`, stanza).length, 'is_marker': !!marker, 'is_unstyled': !!parsers_sizzle(`unstyled[xmlns="${parsers_Strophe.NS.STYLING}"]`, stanza).length, 'marker_id': marker && marker.getAttribute('id'), 'msgid': stanza.getAttribute('id') || original_stanza.getAttribute('id'), 'nick': (_contact = contact) === null || _contact === void 0 ? void 0 : (_contact$attributes = _contact.attributes) === null || _contact$attributes === void 0 ? void 0 : _contact$attributes.nickname, 'receipt_id': getReceiptId(stanza), 'received': new Date().toISOString(), 'references': getReferences(stanza), 'sender': is_me ? 'me' : 'them', 'subject': (_stanza$querySelector3 = stanza.querySelector('subject')) === null || _stanza$querySelector3 === void 0 ? void 0 : _stanza$querySelector3.textContent, 'thread': (_stanza$querySelector4 = stanza.querySelector('thread')) === null || _stanza$querySelector4 === void 0 ? void 0 : _stanza$querySelector4.textContent, 'time': delay ? dayjs_min_default()(delay.getAttribute('stamp')).toISOString() : now, 'to': stanza.getAttribute('to'), 'type': stanza.getAttribute('type') }, getErrorAttributes(stanza), getOutOfBandAttributes(stanza), getSpoilerAttributes(stanza), getCorrectionAttributes(stanza, original_stanza), getStanzaIDs(stanza, original_stanza), getRetractionAttributes(stanza, original_stanza), getEncryptionAttributes(stanza, shared_converse)); if (attrs.is_archived) { const from = original_stanza.getAttribute('from'); if (from && from !== shared_converse.bare_jid) { return new StanzaParseError(`Invalid Stanza: Forged MAM message from ${from}`, stanza); } } await core_api.emojis.initialize(); attrs = Object.assign({ 'message': attrs.body || attrs.error, // TODO: Remove and use body and error attributes instead 'is_only_emojis': attrs.body ? utils_core.isOnlyEmojis(attrs.body) : false, 'is_valid_receipt_request': isValidReceiptRequest(stanza, attrs) }, attrs); // We prefer to use one of the XEP-0359 unique and stable stanza IDs // as the Model id, to avoid duplicates. attrs['id'] = attrs['origin_id'] || attrs[`stanza_id ${attrs.from}`] || utils_core.getUniqueId(); /** * *Hook* which allows plugins to add additional parsing * @event _converse#parseMessage */ attrs = await core_api.hook('parseMessage', stanza, attrs); // We call this after the hook, to allow plugins (like omemo) to decrypt encrypted // messages, since we need to parse the message text to determine whether // there are media urls. return Object.assign(attrs, getMediaURLsMetadata(attrs.is_encrypted ? attrs.plaintext : attrs.body)); } ;// CONCATENATED MODULE: ./src/headless/plugins/chat/model.js const { Strophe: model_Strophe, $msg: model_$msg } = core_converse.env; const model_u = core_converse.env.utils; /** * Represents an open/ongoing chat conversation. * * @class * @namespace _converse.ChatBox * @memberOf _converse */ const ChatBox = model_with_contact.extend({ defaults() { return { 'bookmarked': false, 'chat_state': undefined, 'hidden': isUniView() && !core_api.settings.get('singleton'), 'message_type': 'chat', 'nickname': undefined, 'num_unread': 0, 'time_opened': this.get('time_opened') || new Date().getTime(), 'time_sent': new Date(0).toISOString(), 'type': shared_converse.PRIVATE_CHAT_TYPE, 'url': '' }; }, async initialize() { this.initialized = getOpenPromise(); model_with_contact.prototype.initialize.apply(this, arguments); const jid = this.get('jid'); if (!jid) { // XXX: The `validate` method will prevent this model // from being persisted if there's no jid, but that gets // called after model instantiation, so we have to deal // with invalid models here also. // This happens when the controlbox is in browser storage, // but we're in embedded mode. return; } this.set({ 'box_id': `box-${jid}` }); this.initNotifications(); this.initUI(); this.initMessages(); if (this.get('type') === shared_converse.PRIVATE_CHAT_TYPE) { this.presence = shared_converse.presences.get(jid) || shared_converse.presences.create({ jid }); await this.setRosterContact(jid); this.presence.on('change:show', item => this.onPresenceChanged(item)); } this.on('change:chat_state', this.sendChatState, this); this.ui.on('change:scrolled', this.onScrolledChanged, this); await this.fetchMessages(); /** * Triggered once a {@link _converse.ChatBox} has been created and initialized. * @event _converse#chatBoxInitialized * @type { _converse.ChatBox} * @example _converse.api.listen.on('chatBoxInitialized', model => { ... }); */ await core_api.trigger('chatBoxInitialized', this, { 'Synchronous': true }); this.initialized.resolve(); }, getMessagesCollection() { return new shared_converse.Messages(); }, getMessagesCacheKey() { return `converse.messages-${this.get('jid')}-${shared_converse.bare_jid}`; }, initMessages() { this.messages = this.getMessagesCollection(); this.messages.fetched = getOpenPromise(); this.messages.fetched.then(() => { this.pruneHistoryWhenScrolledDown(); /** * Triggered whenever a { @link _converse.ChatBox } or ${ @link _converse.ChatRoom } * has fetched its messages from the local cache. * @event _converse#afterMessagesFetched * @type { _converse.ChatBox| _converse.ChatRoom } * @example _converse.api.listen.on('afterMessagesFetched', (chat) => { ... }); */ core_api.trigger('afterMessagesFetched', this); }); this.messages.chatbox = this; initStorage(this.messages, this.getMessagesCacheKey()); this.listenTo(this.messages, 'change:upload', this.onMessageUploadChanged, this); this.listenTo(this.messages, 'add', this.onMessageAdded, this); }, initUI() { this.ui = new Model(); }, initNotifications() { this.notifications = new Model(); }, getNotificationsText() { var _this$notifications, _this$notifications2, _this$notifications3; const { __ } = shared_converse; if (((_this$notifications = this.notifications) === null || _this$notifications === void 0 ? void 0 : _this$notifications.get('chat_state')) === shared_converse.COMPOSING) { return __('%1$s is typing', this.getDisplayName()); } else if (((_this$notifications2 = this.notifications) === null || _this$notifications2 === void 0 ? void 0 : _this$notifications2.get('chat_state')) === shared_converse.PAUSED) { return __('%1$s has stopped typing', this.getDisplayName()); } else if (((_this$notifications3 = this.notifications) === null || _this$notifications3 === void 0 ? void 0 : _this$notifications3.get('chat_state')) === shared_converse.GONE) { return __('%1$s has gone away', this.getDisplayName()); } else { return ''; } }, afterMessagesFetched() { /** * Triggered whenever a `_converse.ChatBox` instance has fetched its messages from * `sessionStorage` but **NOT** from the server. * @event _converse#afterMessagesFetched * @type {_converse.ChatBox | _converse.ChatRoom} * @example _converse.api.listen.on('afterMessagesFetched', view => { ... }); */ core_api.trigger('afterMessagesFetched', this); }, fetchMessages() { if (this.messages.fetched_flag) { headless_log.info(`Not re-fetching messages for ${this.get('jid')}`); return; } this.messages.fetched_flag = true; const resolve = this.messages.fetched.resolve; this.messages.fetch({ 'add': true, 'success': msgs => { this.afterMessagesFetched(msgs); resolve(); }, 'error': () => { this.afterMessagesFetched(); resolve(); } }); return this.messages.fetched; }, async handleErrorMessageStanza(stanza) { const { __ } = shared_converse; const attrs = await parseMessage(stanza, shared_converse); if (!(await this.shouldShowErrorMessage(attrs))) { return; } const message = this.getMessageReferencedByError(attrs); if (message) { const new_attrs = { 'error': attrs.error, 'error_condition': attrs.error_condition, 'error_text': attrs.error_text, 'error_type': attrs.error_type, 'editable': false }; if (attrs.msgid === message.get('retraction_id')) { // The error message refers to a retraction new_attrs.retraction_id = undefined; if (!attrs.error) { if (attrs.error_condition === 'forbidden') { new_attrs.error = __("You're not allowed to retract your message."); } else { new_attrs.error = __('Sorry, an error occurred while trying to retract your message.'); } } } else if (!attrs.error) { if (attrs.error_condition === 'forbidden') { new_attrs.error = __("You're not allowed to send a message."); } else { new_attrs.error = __('Sorry, an error occurred while trying to send your message.'); } } message.save(new_attrs); } else { this.createMessage(attrs); } }, /** * Queue an incoming `chat` message stanza for processing. * @async * @private * @method _converse.ChatBox#queueMessage * @param { Promise } attrs - A promise which resolves to the message attributes */ queueMessage(attrs) { this.msg_chain = (this.msg_chain || this.messages.fetched).then(() => this.onMessage(attrs)).catch(e => headless_log.error(e)); return this.msg_chain; }, /** * @async * @private * @method _converse.ChatBox#onMessage * @param { MessageAttributes } attrs_promse - A promise which resolves to the message attributes. */ async onMessage(attrs) { attrs = await attrs; if (model_u.isErrorObject(attrs)) { attrs.stanza && headless_log.error(attrs.stanza); return headless_log.error(attrs.message); } const message = this.getDuplicateMessage(attrs); if (message) { this.updateMessage(message, attrs); } else if (!this.handleReceipt(attrs) && !this.handleChatMarker(attrs) && !(await this.handleRetraction(attrs))) { this.setEditable(attrs, attrs.time); if (attrs['chat_state'] && attrs.sender === 'them') { this.notifications.set('chat_state', attrs.chat_state); } if (model_u.shouldCreateMessage(attrs)) { const msg = this.handleCorrection(attrs) || (await this.createMessage(attrs)); this.notifications.set({ 'chat_state': null }); this.handleUnreadMessage(msg); } } }, async onMessageUploadChanged(message) { if (message.get('upload') === shared_converse.SUCCESS) { const attrs = { 'body': message.get('body'), 'spoiler_hint': message.get('spoiler_hint'), 'oob_url': message.get('oob_url') }; await this.sendMessage(attrs); message.destroy(); } }, onMessageAdded(message) { if (core_api.settings.get('prune_messages_above') && (core_api.settings.get('pruning_behavior') === 'scrolled' || !this.ui.get('scrolled')) && !model_u.isEmptyMessage(message)) { debouncedPruneHistory(this); } }, async clearMessages() { try { await this.messages.clearStore(); } catch (e) { this.messages.trigger('reset'); headless_log.error(e); } finally { // No point in fetching messages from the cache if it's been cleared. // Make sure to resolve the fetched promise to avoid freezes. this.messages.fetched.resolve(); } }, async close() { if (core_api.connection.connected()) { // Immediately sending the chat state, because the // model is going to be destroyed afterwards. this.setChatState(shared_converse.INACTIVE); this.sendChatState(); } try { await new Promise((success, reject) => { return this.destroy({ success, 'error': (m, e) => reject(e) }); }); } catch (e) { headless_log.error(e); } finally { if (core_api.settings.get('clear_messages_on_reconnection')) { await this.clearMessages(); } } /** * Triggered once a chatbox has been closed. * @event _converse#chatBoxClosed * @type {_converse.ChatBox | _converse.ChatRoom} * @example _converse.api.listen.on('chatBoxClosed', chat => { ... }); */ core_api.trigger('chatBoxClosed', this); }, announceReconnection() { /** * Triggered whenever a `_converse.ChatBox` instance has reconnected after an outage * @event _converse#onChatReconnected * @type {_converse.ChatBox | _converse.ChatRoom} * @example _converse.api.listen.on('onChatReconnected', chat => { ... }); */ core_api.trigger('chatReconnected', this); }, async onReconnection() { if (core_api.settings.get('clear_messages_on_reconnection')) { await this.clearMessages(); } this.announceReconnection(); }, onPresenceChanged(item) { const { __ } = shared_converse; const show = item.get('show'); const fullname = this.getDisplayName(); let text; if (show === 'offline') { text = __('%1$s has gone offline', fullname); } else if (show === 'away') { text = __('%1$s has gone away', fullname); } else if (show === 'dnd') { text = __('%1$s is busy', fullname); } else if (show === 'online') { text = __('%1$s is online', fullname); } text && this.createMessage({ 'message': text, 'type': 'info' }); }, onScrolledChanged() { if (!this.ui.get('scrolled')) { this.clearUnreadMsgCounter(); this.pruneHistoryWhenScrolledDown(); } }, pruneHistoryWhenScrolledDown() { if (core_api.settings.get('prune_messages_above') && core_api.settings.get('pruning_behavior') === 'unscrolled' && !this.ui.get('scrolled')) { debouncedPruneHistory(this); } }, validate(attrs) { if (!attrs.jid) { return 'Ignored ChatBox without JID'; } const room_jids = core_api.settings.get('auto_join_rooms').map(s => lodash_es_isObject(s) ? s.jid : s); const auto_join = core_api.settings.get('auto_join_private_chats').concat(room_jids); if (core_api.settings.get("singleton") && !auto_join.includes(attrs.jid) && !core_api.settings.get('auto_join_on_invite')) { const msg = `${attrs.jid} is not allowed because singleton is true and it's not being auto_joined`; headless_log.warn(msg); return msg; } }, getDisplayName() { if (this.contact) { return this.contact.getDisplayName(); } else if (this.vcard) { return this.vcard.getDisplayName(); } else { return this.get('jid'); } }, async createMessageFromError(error) { if (error instanceof shared_converse.TimeoutError) { const msg = await this.createMessage({ 'type': 'error', 'message': error.message, 'retry_event_id': error.retry_event_id, 'is_ephemeral': 30000 }); msg.error = error; } }, editEarlierMessage() { let message; let idx = this.messages.findLastIndex('correcting'); if (idx >= 0) { this.messages.at(idx).save('correcting', false); while (idx > 0) { idx -= 1; const candidate = this.messages.at(idx); if (candidate.get('editable')) { message = candidate; break; } } } message = message || this.messages.filter({ 'sender': 'me' }).reverse().find(m => m.get('editable')); if (message) { message.save('correcting', true); } }, editLaterMessage() { let message; let idx = this.messages.findLastIndex('correcting'); if (idx >= 0) { this.messages.at(idx).save('correcting', false); while (idx < this.messages.length - 1) { idx += 1; const candidate = this.messages.at(idx); if (candidate.get('editable')) { message = candidate; message.save('correcting', true); break; } } } return message; }, getOldestMessage() { for (let i = 0; i < this.messages.length; i++) { const message = this.messages.at(i); if (message.get('type') === this.get('message_type')) { return message; } } }, getMostRecentMessage() { for (let i = this.messages.length - 1; i >= 0; i--) { const message = this.messages.at(i); if (message.get('type') === this.get('message_type')) { return message; } } }, getUpdatedMessageAttributes(message, attrs) { if (!attrs.error_type && message.get('error_type') === 'Decryption') { // Looks like we have a failed decrypted message stored, and now // we have a properly decrypted version of the same message. // See issue: https://github.com/conversejs/converse.js/issues/2733#issuecomment-1035493594 return Object.assign({}, attrs, { error_condition: undefined, error_message: undefined, error_text: undefined, error_type: undefined, is_archived: attrs.is_archived, is_ephemeral: false, is_error: false }); } else { return { is_archived: attrs.is_archived }; } }, updateMessage(message, attrs) { const new_attrs = this.getUpdatedMessageAttributes(message, attrs); new_attrs && message.save(new_attrs); }, /** * 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. * @private * @method _converse.ChatBox#setChatState * @param { string } state - The chat state (consts ACTIVE, COMPOSING, PAUSED, INACTIVE, GONE) */ setChatState(state, options) { if (this.chat_state_timeout !== undefined) { window.clearTimeout(this.chat_state_timeout); delete this.chat_state_timeout; } if (state === shared_converse.COMPOSING) { this.chat_state_timeout = window.setTimeout(this.setChatState.bind(this), shared_converse.TIMEOUTS.PAUSED, shared_converse.PAUSED); } else if (state === shared_converse.PAUSED) { this.chat_state_timeout = window.setTimeout(this.setChatState.bind(this), shared_converse.TIMEOUTS.INACTIVE, shared_converse.INACTIVE); } this.set('chat_state', state, options); return this; }, /** * Given an error `` stanza's attributes, find the saved message model which is * referenced by that error. * @param { Object } attrs */ getMessageReferencedByError(attrs) { const id = attrs.msgid; return id && this.messages.models.find(m => [m.get('msgid'), m.get('retraction_id')].includes(id)); }, /** * @private * @method _converse.ChatBox#shouldShowErrorMessage * @returns {boolean} */ shouldShowErrorMessage(attrs) { const msg = this.getMessageReferencedByError(attrs); if (!msg && attrs.chat_state) { // If the error refers to a message not included in our store, // and it has a chat state tag, we assume that this was a // CSI message (which we don't store). // See https://github.com/conversejs/converse.js/issues/1317 return; } // Gets overridden in ChatRoom return true; }, isSameUser(jid1, jid2) { return model_u.isSameBareJID(jid1, jid2); }, /** * Looks whether we already have a retraction for this * incoming message. If so, it's considered "dangling" because it * probably hasn't been applied to anything yet, given that the * relevant message is only coming in now. * @private * @method _converse.ChatBox#findDanglingRetraction * @param { object } attrs - Attributes representing a received * message, as returned by {@link parseMessage} * @returns { _converse.Message } */ findDanglingRetraction(attrs) { if (!attrs.origin_id || !this.messages.length) { return null; } // Only look for dangling retractions if there are newer // messages than this one, since retractions come after. if (this.messages.last().get('time') > attrs.time) { // Search from latest backwards const messages = Array.from(this.messages.models); messages.reverse(); return messages.find(_ref => { let { attributes } = _ref; return attributes.retracted_id === attrs.origin_id && attributes.from === attrs.from && !attributes.moderated_by; }); } }, /** * Handles message retraction based on the passed in attributes. * @private * @method _converse.ChatBox#handleRetraction * @param { object } attrs - Attributes representing a received * message, as returned by {@link parseMessage} * @returns { Boolean } Returns `true` or `false` depending on * whether a message was retracted or not. */ async handleRetraction(attrs) { const RETRACTION_ATTRIBUTES = ['retracted', 'retracted_id', 'editable']; if (attrs.retracted) { if (attrs.is_tombstone) { return false; } const message = this.messages.findWhere({ 'origin_id': attrs.retracted_id, 'from': attrs.from }); if (!message) { attrs['dangling_retraction'] = true; await this.createMessage(attrs); return true; } message.save(lodash_es_pick(attrs, RETRACTION_ATTRIBUTES)); return true; } else { // Check if we have dangling retraction const message = this.findDanglingRetraction(attrs); if (message) { const retraction_attrs = lodash_es_pick(message.attributes, RETRACTION_ATTRIBUTES); const new_attrs = Object.assign({ 'dangling_retraction': false }, attrs, retraction_attrs); delete new_attrs['id']; // Delete id, otherwise a new cache entry gets created message.save(new_attrs); return true; } } return false; }, /** * Determines whether the passed in message attributes represent a * message which corrects a previously received message, or an * older message which has already been corrected. * In both cases, update the corrected message accordingly. * @private * @method _converse.ChatBox#handleCorrection * @param { object } attrs - Attributes representing a received * message, as returned by {@link parseMessage} * @returns { _converse.Message|undefined } Returns the corrected * message or `undefined` if not applicable. */ handleCorrection(attrs) { if (!attrs.replace_id || !attrs.from) { return; } const message = this.messages.findWhere({ 'msgid': attrs.replace_id, 'from': attrs.from }); if (!message) { return; } const older_versions = message.get('older_versions') || {}; if (attrs.time < message.get('time') && message.get('edited')) { // This is an older message which has been corrected afterwards older_versions[attrs.time] = attrs['message']; message.save({ 'older_versions': older_versions }); } else { // This is a correction of an earlier message we already received if (Object.keys(older_versions).length) { older_versions[message.get('edited')] = message.getMessageText(); } else { older_versions[message.get('time')] = message.getMessageText(); } attrs = Object.assign(attrs, { older_versions }); delete attrs['msgid']; // We want to keep the msgid of the original message delete attrs['id']; // Delete id, otherwise a new cache entry gets created attrs['time'] = message.get('time'); message.save(attrs); } return message; }, /** * Returns an already cached message (if it exists) based on the * passed in attributes map. * @private * @method _converse.ChatBox#getDuplicateMessage * @param { object } attrs - Attributes representing a received * message, as returned by {@link parseMessage} * @returns {Promise<_converse.Message>} */ getDuplicateMessage(attrs) { const queries = [...this.getStanzaIdQueryAttrs(attrs), this.getOriginIdQueryAttrs(attrs), this.getMessageBodyQueryAttrs(attrs)].filter(s => s); const msgs = this.messages.models; return msgs.find(m => queries.reduce((out, q) => out || lodash_es_isMatch(m.attributes, q), false)); }, getOriginIdQueryAttrs(attrs) { return attrs.origin_id && { 'origin_id': attrs.origin_id, 'from': attrs.from }; }, getStanzaIdQueryAttrs(attrs) { const keys = Object.keys(attrs).filter(k => k.startsWith('stanza_id ')); return keys.map(key => { const by_jid = key.replace(/^stanza_id /, ''); const query = {}; query[`stanza_id ${by_jid}`] = attrs[key]; return query; }); }, getMessageBodyQueryAttrs(attrs) { if (attrs.msgid) { const query = { 'from': attrs.from, 'msgid': attrs.msgid }; // XXX: Need to take XEP-428 into consideration if (!attrs.is_encrypted && attrs.body) { // We can't match the message if it's a reflected // encrypted message (e.g. via MAM or in a MUC) query['body'] = attrs.body; } return query; } }, /** * Retract one of your messages in this chat * @private * @method _converse.ChatBoxView#retractOwnMessage * @param { _converse.Message } message - The message which we're retracting. */ retractOwnMessage(message) { this.sendRetractionMessage(message); message.save({ 'retracted': new Date().toISOString(), 'retracted_id': message.get('origin_id'), 'retraction_id': message.get('id'), 'is_ephemeral': true, 'editable': false }); }, /** * Sends a message stanza to retract a message in this chat * @private * @method _converse.ChatBox#sendRetractionMessage * @param { _converse.Message } message - The message which we're retracting. */ sendRetractionMessage(message) { const origin_id = message.get('origin_id'); if (!origin_id) { throw new Error("Can't retract message without a XEP-0359 Origin ID"); } const msg = model_$msg({ 'id': model_u.getUniqueId(), 'to': this.get('jid'), 'type': "chat" }).c('store', { xmlns: model_Strophe.NS.HINTS }).up().c("apply-to", { 'id': origin_id, 'xmlns': model_Strophe.NS.FASTEN }).c('retract', { xmlns: model_Strophe.NS.RETRACT }); return shared_converse.connection.send(msg); }, /** * Finds the last eligible message and then sends a XEP-0333 chat marker for it. * @param { ('received'|'displayed'|'acknowledged') } [type='displayed'] * @param { Boolean } force - Whether a marker should be sent for the * message, even if it didn't include a `markable` element. */ sendMarkerForLastMessage() { let type = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 'displayed'; let force = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false; const msgs = Array.from(this.messages.models); msgs.reverse(); const msg = msgs.find(m => m.get('sender') === 'them' && (force || m.get('is_markable'))); msg && this.sendMarkerForMessage(msg, type, force); }, /** * Given the passed in message object, send a XEP-0333 chat marker. * @param { _converse.Message } msg * @param { ('received'|'displayed'|'acknowledged') } [type='displayed'] * @param { Boolean } force - Whether a marker should be sent for the * message, even if it didn't include a `markable` element. */ sendMarkerForMessage(msg) { let type = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 'displayed'; let force = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false; if (!msg || !core_api.settings.get('send_chat_markers').includes(type)) { return; } if (msg !== null && msg !== void 0 && msg.get('is_markable') || force) { const from_jid = model_Strophe.getBareJidFromJid(msg.get('from')); sendMarker(from_jid, msg.get('msgid'), type, msg.get('type')); } }, handleChatMarker(attrs) { const to_bare_jid = model_Strophe.getBareJidFromJid(attrs.to); if (to_bare_jid !== shared_converse.bare_jid) { return false; } if (attrs.is_markable) { if (this.contact && !attrs.is_archived && !attrs.is_carbon) { sendMarker(attrs.from, attrs.msgid, 'received'); } return false; } else if (attrs.marker_id) { const message = this.messages.findWhere({ 'msgid': attrs.marker_id }); const field_name = `marker_${attrs.marker}`; if (message && !message.get(field_name)) { message.save({ field_name: new Date().toISOString() }); } return true; } }, sendReceiptStanza(to_jid, id) { const receipt_stanza = model_$msg({ 'from': shared_converse.connection.jid, 'id': model_u.getUniqueId(), 'to': to_jid, 'type': 'chat' }).c('received', { 'xmlns': model_Strophe.NS.RECEIPTS, 'id': id }).up().c('store', { 'xmlns': model_Strophe.NS.HINTS }).up(); core_api.send(receipt_stanza); }, handleReceipt(attrs) { if (attrs.sender === 'them') { if (attrs.is_valid_receipt_request) { this.sendReceiptStanza(attrs.from, attrs.msgid); } else if (attrs.receipt_id) { const message = this.messages.findWhere({ 'msgid': attrs.receipt_id }); if (message && !message.get('received')) { message.save({ 'received': new Date().toISOString() }); } return true; } } return false; }, /** * Given a {@link _converse.Message} return the XML stanza that represents it. * @private * @method _converse.ChatBox#createMessageStanza * @param { _converse.Message } message - The message object */ async createMessageStanza(message) { const stanza = model_$msg({ 'from': shared_converse.connection.jid, 'to': this.get('jid'), 'type': this.get('message_type'), 'id': message.get('edited') && model_u.getUniqueId() || message.get('msgid') }).c('body').t(message.get('body')).up().c(shared_converse.ACTIVE, { 'xmlns': model_Strophe.NS.CHATSTATES }).root(); if (message.get('type') === 'chat') { stanza.c('request', { 'xmlns': model_Strophe.NS.RECEIPTS }).root(); } if (!message.get('is_encrypted')) { if (message.get('is_spoiler')) { if (message.get('spoiler_hint')) { stanza.c('spoiler', { 'xmlns': model_Strophe.NS.SPOILER }, message.get('spoiler_hint')).root(); } else { stanza.c('spoiler', { 'xmlns': model_Strophe.NS.SPOILER }).root(); } } (message.get('references') || []).forEach(reference => { const attrs = { 'xmlns': model_Strophe.NS.REFERENCE, 'begin': reference.begin, 'end': reference.end, 'type': reference.type }; if (reference.uri) { attrs.uri = reference.uri; } stanza.c('reference', attrs).root(); }); if (message.get('oob_url')) { stanza.c('x', { 'xmlns': model_Strophe.NS.OUTOFBAND }).c('url').t(message.get('oob_url')).root(); } } if (message.get('edited')) { stanza.c('replace', { 'xmlns': model_Strophe.NS.MESSAGE_CORRECT, 'id': message.get('msgid') }).root(); } if (message.get('origin_id')) { stanza.c('origin-id', { 'xmlns': model_Strophe.NS.SID, 'id': message.get('origin_id') }).root(); } stanza.root(); /** * *Hook* which allows plugins to update an outgoing message stanza * @event _converse#createMessageStanza * @param { _converse.ChatBox | _converse.ChatRoom } - The chat from * which this message stanza is being sent. * @param { Object } data - Message data * @param { _converse.Message | _converse.ChatRoomMessage } data.message * The message object from which the stanza is created and which gets persisted to storage. * @param { Strophe.Builder } data.stanza * The stanza that will be sent out, as a Strophe.Builder object. * You can use the Strophe.Builder functions to extend the stanza. * See http://strophe.im/strophejs/doc/1.4.3/files/strophe-umd-js.html#Strophe.Builder.Functions */ const data = await core_api.hook('createMessageStanza', this, { message, stanza }); return data.stanza; }, async getOutgoingMessageAttributes(attrs) { var _attrs; const is_spoiler = !!this.get('composing_spoiler'); const origin_id = model_u.getUniqueId(); const text = (_attrs = attrs) === null || _attrs === void 0 ? void 0 : _attrs.body; const body = text ? model_u.httpToGeoUri(model_u.shortnamesToUnicode(text), shared_converse) : undefined; attrs = Object.assign({}, attrs, { 'from': shared_converse.bare_jid, 'fullname': shared_converse.xmppstatus.get('fullname'), 'id': origin_id, 'is_only_emojis': text ? model_u.isOnlyEmojis(text) : false, 'jid': this.get('jid'), 'message': body, 'msgid': origin_id, 'nickname': this.get('nickname'), 'sender': 'me', 'time': new Date().toISOString(), 'type': this.get('message_type'), body, is_spoiler, origin_id }, getMediaURLsMetadata(text)); /** * *Hook* which allows plugins to update the attributes of an outgoing message. * These attributes get set on the { @link _converse.Message } or * { @link _converse.ChatRoomMessage } and persisted to storage. * @event _converse#getOutgoingMessageAttributes * @param { _converse.ChatBox | _converse.ChatRoom } chat * The chat from which this message will be sent. * @param { MessageAttributes } attrs * The message attributes, from which the stanza will be created. */ attrs = await core_api.hook('getOutgoingMessageAttributes', this, attrs); return attrs; }, /** * Responsible for setting the editable attribute of messages. * If api.settings.get('allow_message_corrections') is "last", then only the last * message sent from me will be editable. If set to "all" all messages * will be editable. Otherwise no messages will be editable. * @method _converse.ChatBox#setEditable * @memberOf _converse.ChatBox * @param { Object } attrs An object containing message attributes. * @param { String } send_time - time when the message was sent */ setEditable(attrs, send_time) { if (attrs.is_headline || model_u.isEmptyMessage(attrs) || attrs.sender !== 'me') { return; } if (core_api.settings.get('allow_message_corrections') === 'all') { attrs.editable = !(attrs.file || attrs.retracted || 'oob_url' in attrs); } else if (core_api.settings.get('allow_message_corrections') === 'last' && send_time > this.get('time_sent')) { this.set({ 'time_sent': send_time }); const msg = this.messages.findWhere({ 'editable': true }); if (msg) { msg.save({ 'editable': false }); } attrs.editable = !(attrs.file || attrs.retracted || 'oob_url' in attrs); } }, /** * Queue the creation of a message, to make sure that we don't run * into a race condition whereby we're creating a new message * before the collection has been fetched. * @async * @private * @method _converse.ChatBox#createMessage * @param { Object } attrs */ async createMessage(attrs, options) { attrs.time = attrs.time || new Date().toISOString(); await this.messages.fetched; return this.messages.create(attrs, options); }, /** * Responsible for sending off a text message inside an ongoing chat conversation. * @private * @method _converse.ChatBox#sendMessage * @memberOf _converse.ChatBox * @param { Object } [attrs] - A map of attributes to be saved on the message * @returns { _converse.Message } * @example * const chat = api.chats.get('buddy1@example.org'); * chat.sendMessage({'body': 'hello world'}); */ async sendMessage(attrs) { attrs = await this.getOutgoingMessageAttributes(attrs); let message = this.messages.findWhere('correcting'); if (message) { const older_versions = message.get('older_versions') || {}; const edited_time = message.get('edited') || message.get('time'); older_versions[edited_time] = message.getMessageText(); const plaintext = attrs.is_encrypted ? attrs.message : undefined; message.save({ 'body': attrs.body, 'message': attrs.body, 'correcting': false, 'edited': new Date().toISOString(), 'is_only_emojis': attrs.is_only_emojis, 'origin_id': model_u.getUniqueId(), 'received': undefined, 'references': attrs.references, older_versions, plaintext }); } else { this.setEditable(attrs, new Date().toISOString()); message = await this.createMessage(attrs); } try { const stanza = await this.createMessageStanza(message); core_api.send(stanza); } catch (e) { message.destroy(); headless_log.error(e); return; } /** * Triggered when a message is being sent out * @event _converse#sendMessage * @type { Object } * @param { Object } data * @property { (_converse.ChatBox | _converse.ChatRoom) } data.chatbox * @property { (_converse.Message | _converse.ChatRoomMessage) } data.message */ core_api.trigger('sendMessage', { 'chatbox': this, message }); return message; }, /** * Sends a message with the current XEP-0085 chat state of the user * as taken from the `chat_state` attribute of the {@link _converse.ChatBox}. * @private * @method _converse.ChatBox#sendChatState */ sendChatState() { if (core_api.settings.get('send_chat_state_notifications') && this.get('chat_state')) { const allowed = core_api.settings.get('send_chat_state_notifications'); if (Array.isArray(allowed) && !allowed.includes(this.get('chat_state'))) { return; } core_api.send(model_$msg({ 'id': model_u.getUniqueId(), 'to': this.get('jid'), 'type': 'chat' }).c(this.get('chat_state'), { 'xmlns': model_Strophe.NS.CHATSTATES }).up().c('no-store', { 'xmlns': model_Strophe.NS.HINTS }).up().c('no-permanent-store', { 'xmlns': model_Strophe.NS.HINTS })); } }, async sendFiles(files) { var _maxFileSize; const { __ } = shared_converse; const result = await core_api.disco.features.get(model_Strophe.NS.HTTPUPLOAD, shared_converse.domain); const item = result.pop(); if (!item) { this.createMessage({ 'message': __("Sorry, looks like file upload is not supported by your server."), 'type': 'error', 'is_ephemeral': true }); return; } const data = item.dataforms.where({ 'FORM_TYPE': { 'value': model_Strophe.NS.HTTPUPLOAD, 'type': "hidden" } }).pop(); const max_file_size = window.parseInt((_maxFileSize = ((data === null || data === void 0 ? void 0 : data.attributes) || {})['max-file-size']) === null || _maxFileSize === void 0 ? void 0 : _maxFileSize.value); const slot_request_url = item === null || item === void 0 ? void 0 : item.id; if (!slot_request_url) { this.createMessage({ 'message': __("Sorry, looks like file upload is not supported by your server."), 'type': 'error', 'is_ephemeral': true }); return; } Array.from(files).forEach(async file => { /** * *Hook* which allows plugins to transform files before they'll be * uploaded. The main use-case is to encrypt the files. * @event _converse#beforeFileUpload * @param { _converse.ChatBox | _converse.ChatRoom } chat * The chat from which this file will be uploaded. * @param { File } file * The file that will be uploaded */ file = await core_api.hook('beforeFileUpload', this, file); if (!window.isNaN(max_file_size) && window.parseInt(file.size) > max_file_size) { return this.createMessage({ 'message': __('The size of your file, %1$s, exceeds the maximum allowed by your server, which is %2$s.', file.name, filesize_min_default()(max_file_size)), 'type': 'error', 'is_ephemeral': true }); } else { const initial_attrs = await this.getOutgoingMessageAttributes(); const attrs = Object.assign(initial_attrs, { 'file': true, 'progress': 0, 'slot_request_url': slot_request_url }); this.setEditable(attrs, new Date().toISOString()); const message = await this.createMessage(attrs, { 'silent': true }); message.file = file; this.messages.trigger('add', message); message.getRequestSlotURL(); } }); }, maybeShow(force) { if (isUniView()) { const filter = c => !c.get('hidden') && c.get('jid') !== this.get('jid') && c.get('id') !== 'controlbox'; const other_chats = shared_converse.chatboxes.filter(filter); if (force || other_chats.length === 0) { // We only have one chat visible at any one time. // So before opening a chat, we make sure all other chats are hidden. other_chats.forEach(c => model_u.safeSave(c, { 'hidden': true })); model_u.safeSave(this, { 'hidden': false }); } return; } model_u.safeSave(this, { 'hidden': false }); this.trigger('show'); return this; }, /** * Indicates whether the chat is hidden and therefore * whether a newly received message will be visible * to the user or not. * @returns {boolean} */ isHidden() { // Note: This methods gets overridden by converse-minimize return this.get('hidden') || this.isScrolledUp() || shared_converse.windowState === 'hidden'; }, /** * Given a newly received {@link _converse.Message} instance, * update the unread counter if necessary. * @private * @method _converse.ChatBox#handleUnreadMessage * @param {_converse.Message} message */ handleUnreadMessage(message) { if (!(message !== null && message !== void 0 && message.get('body'))) { return; } if (model_u.isNewMessage(message)) { if (message.get('sender') === 'me') { // 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.ui.set('scrolled', false); } else if (this.isHidden()) { this.incrementUnreadMsgsCounter(message); } else { this.sendMarkerForMessage(message); } } }, incrementUnreadMsgsCounter(message) { const settings = { 'num_unread': this.get('num_unread') + 1 }; if (this.get('num_unread') === 0) { settings['first_unread_id'] = message.get('id'); } this.save(settings); }, clearUnreadMsgCounter() { if (this.get('num_unread') > 0) { this.sendMarkerForMessage(this.messages.last()); } model_u.safeSave(this, { 'num_unread': 0 }); }, isScrolledUp() { return this.ui.get('scrolled'); } }); /* harmony default export */ const model = (ChatBox); ;// CONCATENATED MODULE: ./src/headless/plugins/chat/message.js const { Strophe: message_Strophe, sizzle: message_sizzle, u: message_u } = core_converse.env; /** * Mixin which turns a `ModelWithContact` model into a non-MUC message. These can be either `chat` messages or `headline` messages. * @mixin * @namespace _converse.Message * @memberOf _converse * @example const msg = new _converse.Message({'message': 'hello world!'}); */ const MessageMixin = { defaults() { return { 'msgid': message_u.getUniqueId(), 'time': new Date().toISOString(), 'is_ephemeral': false }; }, async initialize() { if (!this.checkValidity()) { return; } this.initialized = getOpenPromise(); if (this.get('file')) { this.on('change:put', () => this.uploadFile()); } // If `type` changes from `error` to `chat`, we want to set the contact. See #2733 this.on('change:type', () => this.setContact()); this.on('change:is_ephemeral', () => this.setTimerForEphemeralMessage()); await this.setContact(); this.setTimerForEphemeralMessage(); /** * Triggered once a {@link _converse.Message} has been created and initialized. * @event _converse#messageInitialized * @type { _converse.Message} * @example _converse.api.listen.on('messageInitialized', model => { ... }); */ await core_api.trigger('messageInitialized', this, { 'Synchronous': true }); this.initialized.resolve(); }, setContact() { if (this.get('type') === 'chat') { model_with_contact.prototype.initialize.apply(this, arguments); this.setRosterContact(message_Strophe.getBareJidFromJid(this.get('from'))); } }, /** * Sets an auto-destruct timer for this message, if it's is_ephemeral. * @private * @method _converse.Message#setTimerForEphemeralMessage */ setTimerForEphemeralMessage() { if (this.ephemeral_timer) { clearTimeout(this.ephemeral_timer); } const is_ephemeral = this.isEphemeral(); if (is_ephemeral) { const timeout = typeof is_ephemeral === "number" ? is_ephemeral : 10000; this.ephemeral_timer = window.setTimeout(() => this.safeDestroy(), timeout); } }, checkValidity() { if (Object.keys(this.attributes).length === 3) { // XXX: This is an empty message with only the 3 default values. // This seems to happen when saving a newly created message // fails for some reason. // TODO: This is likely fixable by setting `wait` when // creating messages. See the wait-for-messages branch. this.validationError = 'Empty message'; this.safeDestroy(); return false; } return true; }, /** * Determines whether this messsage may be retracted by the current user. * @private * @method _converse.Messages#mayBeRetracted * @returns { Boolean } */ mayBeRetracted() { const is_own_message = this.get('sender') === 'me'; const not_canceled = this.get('error_type') !== 'cancel'; return is_own_message && not_canceled && ['all', 'own'].includes(core_api.settings.get('allow_message_retraction')); }, safeDestroy() { try { this.destroy(); } catch (e) { headless_log.warn(`safeDestroy: ${e}`); } }, /** * Returns a boolean indicating whether this message is ephemeral, * meaning it will get automatically removed after ten seconds. * @returns { boolean } */ isEphemeral() { return this.get('is_ephemeral'); }, /** * Returns a boolean indicating whether this message is a XEP-0245 /me command. * @returns { boolean } */ isMeCommand() { const text = this.getMessageText(); if (!text) { return false; } return text.startsWith('/me '); }, /** * Returns a boolean indicating whether this message is considered a followup * message from the previous one. Followup messages are shown grouped together * under one author heading. * A message is considered a followup of it's predecessor when it's a chat * message from the same author, within 10 minutes. * @returns { boolean } */ isFollowup() { const messages = this.collection.models; const idx = messages.indexOf(this); const prev_model = idx ? messages[idx - 1] : null; if (prev_model === null) { return false; } const date = dayjs_min_default()(this.get('time')); return this.get('from') === prev_model.get('from') && !this.isMeCommand() && !prev_model.isMeCommand() && this.get('type') !== 'info' && prev_model.get('type') !== 'info' && date.isBefore(dayjs_min_default()(prev_model.get('time')).add(10, 'minutes')) && !!this.get('is_encrypted') === !!prev_model.get('is_encrypted'); }, getDisplayName() { if (this.contact) { return this.contact.getDisplayName(); } else if (this.vcard) { return this.vcard.getDisplayName(); } else { return this.get('from'); } }, getMessageText() { if (this.get('is_encrypted')) { const { __ } = shared_converse; return this.get('plaintext') || this.get('body') || __('Undecryptable OMEMO message'); } else if (['groupchat', 'chat'].includes(this.get('type'))) { return this.get('body'); } else { return this.get('message'); } }, /** * Send out an IQ stanza to request a file upload slot. * https://xmpp.org/extensions/xep-0363.html#request * @private * @method _converse.Message#sendSlotRequestStanza */ sendSlotRequestStanza() { if (!this.file) { return Promise.reject(new Error('file is undefined')); } const iq = core_converse.env.$iq({ 'from': shared_converse.jid, 'to': this.get('slot_request_url'), 'type': 'get' }).c('request', { 'xmlns': message_Strophe.NS.HTTPUPLOAD, 'filename': this.file.name, 'size': this.file.size, 'content-type': this.file.type }); return core_api.sendIQ(iq); }, getUploadRequestMetadata(stanza) { const headers = message_sizzle(`slot[xmlns="${message_Strophe.NS.HTTPUPLOAD}"] put header`, stanza); // https://xmpp.org/extensions/xep-0363.html#request // TODO: Can't set the Cookie header in JavaScipt, instead cookies need // to be manually set via document.cookie, so we're leaving it out here. return { 'headers': headers.map(h => ({ 'name': h.getAttribute('name'), 'value': h.textContent })).filter(h => ['Authorization', 'Expires'].includes(h.name)) }; }, async getRequestSlotURL() { const { __ } = shared_converse; let stanza; try { stanza = await this.sendSlotRequestStanza(); } catch (e) { headless_log.error(e); return this.save({ 'type': 'error', 'message': __('Sorry, could not determine upload URL.'), 'is_ephemeral': true }); } const slot = message_sizzle(`slot[xmlns="${message_Strophe.NS.HTTPUPLOAD}"]`, stanza).pop(); if (slot) { this.upload_metadata = this.getUploadRequestMetadata(stanza); this.save({ 'get': slot.querySelector('get').getAttribute('url'), 'put': slot.querySelector('put').getAttribute('url') }); } else { return this.save({ 'type': 'error', 'message': __('Sorry, could not determine file upload URL.'), 'is_ephemeral': true }); } }, uploadFile() { var _this$upload_metadata; const xhr = new XMLHttpRequest(); xhr.onreadystatechange = async () => { if (xhr.readyState === XMLHttpRequest.DONE) { headless_log.info('Status: ' + xhr.status); if (xhr.status === 200 || xhr.status === 201) { let attrs = { 'upload': shared_converse.SUCCESS, 'oob_url': this.get('get'), 'message': this.get('get'), 'body': this.get('get') }; /** * *Hook* which allows plugins to change the attributes * saved on the message once a file has been uploaded. * @event _converse#afterFileUploaded */ attrs = await core_api.hook('afterFileUploaded', this, attrs); this.save(attrs); } else { xhr.onerror(); } } }; xhr.upload.addEventListener('progress', evt => { if (evt.lengthComputable) { this.set('progress', evt.loaded / evt.total); } }, false); xhr.onerror = () => { const { __ } = shared_converse; let message; if (xhr.responseText) { message = __('Sorry, could not succesfully upload your file. Your server’s response: "%1$s"', xhr.responseText); } else { message = __('Sorry, could not succesfully upload your file.'); } this.save({ 'type': 'error', 'upload': shared_converse.FAILURE, 'message': message, 'is_ephemeral': true }); }; xhr.open('PUT', this.get('put'), true); xhr.setRequestHeader('Content-type', this.file.type); (_this$upload_metadata = this.upload_metadata.headers) === null || _this$upload_metadata === void 0 ? void 0 : _this$upload_metadata.forEach(h => xhr.setRequestHeader(h.name, h.value)); xhr.send(this.file); } }; /* harmony default export */ const message = (MessageMixin); ;// CONCATENATED MODULE: ./src/headless/plugins/chat/api.js /* harmony default export */ const chat_api = ({ /** * The "chats" namespace (used for one-on-one chats) * * @namespace api.chats * @memberOf api */ chats: { /** * @method api.chats.create * @param {string|string[]} jid|jids An jid or array of jids * @param {object} [attrs] An object containing configuration attributes. */ async create(jids, attrs) { if (typeof jids === 'string') { if (attrs && !(attrs !== null && attrs !== void 0 && attrs.fullname)) { var _contact$attributes; const contact = await core_api.contacts.get(jids); attrs.fullname = contact === null || contact === void 0 ? void 0 : (_contact$attributes = contact.attributes) === null || _contact$attributes === void 0 ? void 0 : _contact$attributes.fullname; } const chatbox = core_api.chats.get(jids, attrs, true); if (!chatbox) { headless_log.error("Could not open chatbox for JID: " + jids); return; } return chatbox; } if (Array.isArray(jids)) { return Promise.all(jids.forEach(async jid => { var _contact$attributes2; const contact = await core_api.contacts.get(jids); attrs.fullname = contact === null || contact === void 0 ? void 0 : (_contact$attributes2 = contact.attributes) === null || _contact$attributes2 === void 0 ? void 0 : _contact$attributes2.fullname; return core_api.chats.get(jid, attrs, true).maybeShow(); })); } headless_log.error("chats.create: You need to provide at least one JID"); return null; }, /** * Opens a new one-on-one chat. * * @method api.chats.open * @param {String|string[]} name - e.g. 'buddy@example.com' or ['buddy1@example.com', 'buddy2@example.com'] * @param {Object} [attrs] - Attributes to be set on the _converse.ChatBox model. * @param {Boolean} [attrs.minimized] - Should the chat be created in minimized state. * @param {Boolean} [force=false] - By default, a minimized * chat won't be maximized (in `overlayed` view mode) and in * `fullscreen` view mode a newly opened chat won't replace * another chat already in the foreground. * Set `force` to `true` if you want to force the chat to be * maximized or shown. * @returns {Promise} Promise which resolves with the * _converse.ChatBox representing the chat. * * @example * // To open a single chat, provide the JID of the contact you're chatting with in that chat: * converse.plugins.add('myplugin', { * initialize: function() { * const _converse = this._converse; * // Note, buddy@example.org must be in your contacts roster! * api.chats.open('buddy@example.com').then(chat => { * // Now you can do something with the chat model * }); * } * }); * * @example * // To open an array of chats, provide an array of JIDs: * converse.plugins.add('myplugin', { * initialize: function () { * const _converse = this._converse; * // Note, these users must first be in your contacts roster! * api.chats.open(['buddy1@example.com', 'buddy2@example.com']).then(chats => { * // Now you can do something with the chat models * }); * } * }); */ async open(jids, attrs, force) { if (typeof jids === 'string') { const chat = await core_api.chats.get(jids, attrs, true); if (chat) { return chat.maybeShow(force); } return chat; } else if (Array.isArray(jids)) { return Promise.all(jids.map(j => core_api.chats.get(j, attrs, true).then(c => c && c.maybeShow(force))).filter(c => c)); } const err_msg = "chats.open: You need to provide at least one JID"; headless_log.error(err_msg); throw new Error(err_msg); }, /** * Retrieves a chat or all chats. * * @method api.chats.get * @param {String|string[]} jids - e.g. 'buddy@example.com' or ['buddy1@example.com', 'buddy2@example.com'] * @param {Object} [attrs] - Attributes to be set on the _converse.ChatBox model. * @param {Boolean} [create=false] - Whether the chat should be created if it's not found. * @returns { Promise<_converse.ChatBox> } * * @example * // To return a single chat, provide the JID of the contact you're chatting with in that chat: * const model = await api.chats.get('buddy@example.com'); * * @example * // To return an array of chats, provide an array of JIDs: * const models = await api.chats.get(['buddy1@example.com', 'buddy2@example.com']); * * @example * // To return all open chats, call the method without any parameters:: * const models = await api.chats.get(); * */ async get(jids) { let attrs = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; let create = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false; await core_api.waitUntil('chatBoxesFetched'); async function _get(jid) { let model = await core_api.chatboxes.get(jid); if (!model && create) { model = await core_api.chatboxes.create(jid, attrs, shared_converse.ChatBox); } else { model = model && model.get('type') === shared_converse.PRIVATE_CHAT_TYPE ? model : null; if (model && Object.keys(attrs).length) { model.save(attrs); } } return model; } if (jids === undefined) { const chats = await core_api.chatboxes.get(); return chats.filter(c => c.get('type') === shared_converse.PRIVATE_CHAT_TYPE); } else if (typeof jids === 'string') { return _get(jids); } return Promise.all(jids.map(jid => _get(jid))); } } }); ;// CONCATENATED MODULE: ./src/headless/plugins/chat/utils.js const { Strophe: utils_Strophe, sizzle: utils_sizzle, u: chat_utils_u } = core_converse.env; function openChat(jid) { if (!chat_utils_u.isValidJID(jid)) { return headless_log.warn(`Invalid JID "${jid}" provided in URL fragment`); } core_api.chats.open(jid); } async function onClearSession() { if (shared_converse.shouldClearCache()) { await Promise.all(shared_converse.chatboxes.map(c => c.messages && c.messages.clearStore({ 'silent': true }))); const filter = o => o.get('type') !== shared_converse.CONTROLBOX_TYPE; shared_converse.chatboxes.clearStore({ 'silent': true }, filter); } } async function handleErrorMessage(stanza) { const from_jid = utils_Strophe.getBareJidFromJid(stanza.getAttribute('from')); if (chat_utils_u.isSameBareJID(from_jid, shared_converse.bare_jid)) { return; } const chatbox = await core_api.chatboxes.get(from_jid); if (chatbox.get('type') === shared_converse.PRIVATE_CHAT_TYPE) { chatbox === null || chatbox === void 0 ? void 0 : chatbox.handleErrorMessageStanza(stanza); } } function autoJoinChats() { // Automatically join private chats, based on the // "auto_join_private_chats" configuration setting. core_api.settings.get('auto_join_private_chats').forEach(jid => { if (shared_converse.chatboxes.where({ 'jid': jid }).length) { return; } if (typeof jid === 'string') { core_api.chats.open(jid); } else { headless_log.error('Invalid jid criteria specified for "auto_join_private_chats"'); } }); /** * Triggered once any private chats have been automatically joined as * specified by the `auto_join_private_chats` setting. * See: https://conversejs.org/docs/html/configuration.html#auto-join-private-chats * @event _converse#privateChatsAutoJoined * @example _converse.api.listen.on('privateChatsAutoJoined', () => { ... }); * @example _converse.api.waitUntil('privateChatsAutoJoined').then(() => { ... }); */ core_api.trigger('privateChatsAutoJoined'); } function registerMessageHandlers() { shared_converse.connection.addHandler(stanza => { if (utils_sizzle(`message > result[xmlns="${utils_Strophe.NS.MAM}"]`, stanza).pop()) { // MAM messages are handled in converse-mam. // We shouldn't get MAM messages here because // they shouldn't have a `type` attribute. headless_log.warn(`Received a MAM message with type "chat".`); return true; } shared_converse.handleMessageStanza(stanza); return true; }, null, 'message', 'chat'); shared_converse.connection.addHandler(stanza => { // Message receipts are usually without the `type` attribute. See #1353 if (stanza.getAttribute('type') !== null) { // TODO: currently Strophe has no way to register a handler // for stanzas without a `type` attribute. // We could update it to accept null to mean no attribute, // but that would be a backward-incompatible change return true; // Gets handled above. } shared_converse.handleMessageStanza(stanza); return true; }, utils_Strophe.NS.RECEIPTS, 'message'); shared_converse.connection.addHandler(stanza => { handleErrorMessage(stanza); return true; }, null, 'message', 'error'); } /** * Handler method for all incoming single-user chat "message" stanzas. * @private * @param { MessageAttributes } attrs - The message attributes */ async function handleMessageStanza(stanza) { if (isServerMessage(stanza)) { // Prosody sends headline messages with type `chat`, so we need to filter them out here. const from = stanza.getAttribute('from'); return headless_log.info(`handleMessageStanza: Ignoring incoming server message from JID: ${from}`); } let attrs; try { attrs = await parseMessage(stanza, shared_converse); } catch (e) { return headless_log.error(e); } if (chat_utils_u.isErrorObject(attrs)) { attrs.stanza && headless_log.error(attrs.stanza); return headless_log.error(attrs.message); } // XXX: Need to take XEP-428 into consideration const has_body = !!(attrs.body || attrs.plaintext); const chatbox = await core_api.chats.get(attrs.contact_jid, { 'nickname': attrs.nick }, has_body); await (chatbox === null || chatbox === void 0 ? void 0 : chatbox.queueMessage(attrs)); /** * @typedef { Object } MessageData * An object containing the original message stanza, as well as the * parsed attributes. * @property { XMLElement } stanza * @property { MessageAttributes } stanza * @property { ChatBox } chatbox */ const data = { stanza, attrs, chatbox }; /** * Triggered when a message stanza is been received and processed. * @event _converse#message * @type { object } * @property { module:converse-chat~MessageData } data */ core_api.trigger('message', data); } ;// CONCATENATED MODULE: ./src/headless/plugins/chat/index.js /** * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.plugins.add('converse-chat', { /* 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. */ dependencies: ['converse-chatboxes', 'converse-disco'], initialize() { // Configuration values for this plugin // ==================================== // Refer to docs/source/configuration.rst for explanations of these // configuration settings. core_api.settings.extend({ 'allow_message_corrections': 'all', 'allow_message_retraction': 'all', 'allow_message_styling': true, 'auto_join_private_chats': [], 'clear_messages_on_reconnection': false, 'filter_by_resource': false, 'prune_messages_above': undefined, 'pruning_behavior': 'unscrolled', 'send_chat_markers': ["received", "displayed", "acknowledged"], 'send_chat_state_notifications': true }); shared_converse.Message = model_with_contact.extend(message); shared_converse.Messages = Collection.extend({ model: shared_converse.Message, comparator: 'time' }); Object.assign(shared_converse, { ChatBox: model, handleMessageStanza: handleMessageStanza }); Object.assign(core_api, chat_api); shared_converse.router.route('converse/chat?jid=:jid', openChat); core_api.listen.on('chatBoxesFetched', autoJoinChats); core_api.listen.on('presencesInitialized', registerMessageHandlers); core_api.listen.on('clearSession', onClearSession); } }); ;// CONCATENATED MODULE: ./src/headless/plugins/disco/entity.js const { Strophe: entity_Strophe } = core_converse.env; /** * @class * @namespace _converse.DiscoEntity * @memberOf _converse * * A Disco Entity is a JID addressable entity that can be queried for features. * * See XEP-0030: https://xmpp.org/extensions/xep-0030.html */ const DiscoEntity = Model.extend({ idAttribute: 'jid', async initialize(_, options) { this.waitUntilFeaturesDiscovered = getOpenPromise(); this.dataforms = new Collection(); let id = `converse.dataforms-${this.get('jid')}`; this.dataforms.browserStorage = shared_converse.createStore(id, 'session'); this.features = new Collection(); id = `converse.features-${this.get('jid')}`; this.features.browserStorage = shared_converse.createStore(id, 'session'); this.listenTo(this.features, 'add', this.onFeatureAdded); this.fields = new Collection(); id = `converse.fields-${this.get('jid')}`; this.fields.browserStorage = shared_converse.createStore(id, 'session'); this.listenTo(this.fields, 'add', this.onFieldAdded); this.items = new shared_converse.DiscoEntities(); id = `converse.disco-items-${this.get('jid')}`; this.items.browserStorage = shared_converse.createStore(id, 'session'); await new Promise(f => this.items.fetch({ 'success': f, 'error': f })); this.identities = new Collection(); id = `converse.identities-${this.get('jid')}`; this.identities.browserStorage = shared_converse.createStore(id, 'session'); this.fetchFeatures(options); }, /** * Returns a Promise which resolves with a map indicating * whether a given identity is provided by this entity. * @private * @method _converse.DiscoEntity#getIdentity * @param { String } category - The identity category * @param { String } type - The identity type */ async getIdentity(category, type) { await this.waitUntilFeaturesDiscovered; return this.identities.findWhere({ 'category': category, 'type': type }); }, /** * Returns a Promise which resolves with a map indicating * whether a given feature is supported. * @private * @method _converse.DiscoEntity#hasFeature * @param { String } feature - The feature that might be supported. */ async hasFeature(feature) { await this.waitUntilFeaturesDiscovered; if (this.features.findWhere({ 'var': feature })) { return this; } }, onFeatureAdded(feature) { feature.entity = this; /** * Triggered when Converse has learned of a service provided by the XMPP server. * See XEP-0030. * @event _converse#serviceDiscovered * @type { Model } * @example _converse.api.listen.on('featuresDiscovered', feature => { ... }); */ core_api.trigger('serviceDiscovered', feature); }, onFieldAdded(field) { field.entity = this; /** * Triggered when Converse has learned of a disco extension field. * See XEP-0030. * @event _converse#discoExtensionFieldDiscovered * @example _converse.api.listen.on('discoExtensionFieldDiscovered', () => { ... }); */ core_api.trigger('discoExtensionFieldDiscovered', field); }, async fetchFeatures(options) { if (options.ignore_cache) { this.queryInfo(); } else { const store_id = this.features.browserStorage.name; const result = await this.features.browserStorage.store.getItem(store_id); if (result && result.length === 0 || result === null) { this.queryInfo(); } else { this.features.fetch({ add: true, success: () => { this.waitUntilFeaturesDiscovered.resolve(this); this.trigger('featuresDiscovered'); } }); this.identities.fetch({ add: true }); } } }, async queryInfo() { let stanza; try { stanza = await core_api.disco.info(this.get('jid'), null); } catch (iq) { iq === null ? headless_log.error(`Timeout for disco#info query for ${this.get('jid')}`) : headless_log.error(iq); this.waitUntilFeaturesDiscovered.resolve(this); return; } this.onInfo(stanza); }, onDiscoItems(stanza) { sizzle_default()(`query[xmlns="${entity_Strophe.NS.DISCO_ITEMS}"] item`, stanza).forEach(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'); if (this.items.get(jid) === undefined) { const entities = shared_converse.disco_entities; const entity = entities.get(jid) || entities.create({ jid, name: item.getAttribute('name') }); this.items.add(entity); } }); }, async queryForItems() { if (this.identities.where({ 'category': 'server' }).length === 0) { // Don't fetch features and items if this is not a // server or a conference component. return; } const stanza = await core_api.disco.items(this.get('jid')); this.onDiscoItems(stanza); }, onInfo(stanza) { Array.from(stanza.querySelectorAll('identity')).forEach(identity => { this.identities.create({ 'category': identity.getAttribute('category'), 'type': identity.getAttribute('type'), 'name': identity.getAttribute('name') }); }); sizzle_default()(`x[type="result"][xmlns="${entity_Strophe.NS.XFORM}"]`, stanza).forEach(form => { const data = {}; sizzle_default()('field', form).forEach(field => { var _field$querySelector; data[field.getAttribute('var')] = { 'value': (_field$querySelector = field.querySelector('value')) === null || _field$querySelector === void 0 ? void 0 : _field$querySelector.textContent, 'type': field.getAttribute('type') }; }); this.dataforms.create(data); }); if (stanza.querySelector(`feature[var="${entity_Strophe.NS.DISCO_ITEMS}"]`)) { this.queryForItems(); } Array.from(stanza.querySelectorAll('feature')).forEach(feature => { this.features.create({ 'var': feature.getAttribute('var'), 'from': stanza.getAttribute('from') }); }); // XEP-0128 Service Discovery Extensions sizzle_default()('x[type="result"][xmlns="jabber:x:data"] field', stanza).forEach(field => { var _field$querySelector2; this.fields.create({ 'var': field.getAttribute('var'), 'value': (_field$querySelector2 = field.querySelector('value')) === null || _field$querySelector2 === void 0 ? void 0 : _field$querySelector2.textContent, 'from': stanza.getAttribute('from') }); }); this.waitUntilFeaturesDiscovered.resolve(this); this.trigger('featuresDiscovered'); } }); /* harmony default export */ const entity = (DiscoEntity); ;// CONCATENATED MODULE: ./src/headless/plugins/disco/entities.js const DiscoEntities = Collection.extend({ model: entity, fetchEntities() { return new Promise((resolve, reject) => { this.fetch({ add: true, success: resolve, error(_m, e) { headless_log.error(e); reject(new Error("Could not fetch disco entities")); } }); }); } }); /* harmony default export */ const entities = (DiscoEntities); ;// CONCATENATED MODULE: ./src/headless/plugins/disco/utils.js const { Strophe: disco_utils_Strophe, $iq: utils_$iq } = core_converse.env; function onDiscoInfoRequest(stanza) { const node = stanza.getElementsByTagName('query')[0].getAttribute('node'); const attrs = { xmlns: disco_utils_Strophe.NS.DISCO_INFO }; if (node) { attrs.node = node; } const iqresult = utils_$iq({ 'type': 'result', 'id': stanza.getAttribute('id') }); const from = stanza.getAttribute('from'); if (from !== null) { iqresult.attrs({ 'to': from }); } iqresult.c('query', attrs); shared_converse.disco._identities.forEach(identity => { const attrs = { 'category': identity.category, 'type': identity.type }; if (identity.name) { attrs.name = identity.name; } if (identity.lang) { attrs['xml:lang'] = identity.lang; } iqresult.c('identity', attrs).up(); }); shared_converse.disco._features.forEach(f => iqresult.c('feature', { 'var': f }).up()); core_api.send(iqresult.tree()); return true; } function addClientFeatures() { // See https://xmpp.org/registrar/disco-categories.html core_api.disco.own.identities.add('client', 'web', 'Converse'); core_api.disco.own.features.add(disco_utils_Strophe.NS.CHATSTATES); core_api.disco.own.features.add(disco_utils_Strophe.NS.DISCO_INFO); core_api.disco.own.features.add(disco_utils_Strophe.NS.ROSTERX); // Limited support if (core_api.settings.get("message_carbons")) { core_api.disco.own.features.add(disco_utils_Strophe.NS.CARBONS); } /** * Triggered in converse-disco once the core disco features of * Converse have been added. * @event _converse#addClientFeatures * @example _converse.api.listen.on('addClientFeatures', () => { ... }); */ core_api.trigger('addClientFeatures'); return this; } async function initializeDisco() { addClientFeatures(); shared_converse.connection.addHandler(stanza => onDiscoInfoRequest(stanza), disco_utils_Strophe.NS.DISCO_INFO, 'iq', 'get', null, null); shared_converse.disco_entities = new shared_converse.DiscoEntities(); const id = `converse.disco-entities-${shared_converse.bare_jid}`; shared_converse.disco_entities.browserStorage = shared_converse.createStore(id, 'session'); const collection = await shared_converse.disco_entities.fetchEntities(); if (collection.length === 0 || !collection.get(shared_converse.domain)) { // If we don't have an entity for our own XMPP server, create one. shared_converse.disco_entities.create({ 'jid': shared_converse.domain }); } /** * Triggered once the `converse-disco` plugin has been initialized and the * `_converse.disco_entities` collection will be available and populated with at * least the service discovery features of the user's own server. * @event _converse#discoInitialized * @example _converse.api.listen.on('discoInitialized', () => { ... }); */ core_api.trigger('discoInitialized'); } function initStreamFeatures() { // Initialize the stream_features collection, and if we're // re-attaching to a pre-existing BOSH session, we restore the // features from cache. // Otherwise the features will be created once we've received them // from the server (see populateStreamFeatures). if (!shared_converse.stream_features) { const bare_jid = disco_utils_Strophe.getBareJidFromJid(shared_converse.jid); const id = `converse.stream-features-${bare_jid}`; core_api.promises.add('streamFeaturesAdded'); shared_converse.stream_features = new Collection(); shared_converse.stream_features.browserStorage = shared_converse.createStore(id, "session"); } } function notifyStreamFeaturesAdded() { /** * Triggered as soon as the stream features are known. * If you want to check whether a stream feature is supported before proceeding, * then you'll first want to wait for this event. * @event _converse#streamFeaturesAdded * @example _converse.api.listen.on('streamFeaturesAdded', () => { ... }); */ core_api.trigger('streamFeaturesAdded'); } function populateStreamFeatures() { // Strophe.js sets the element on the // Strophe.Connection instance (_converse.connection). // // Once this is we populate the _converse.stream_features collection // and trigger streamFeaturesAdded. initStreamFeatures(); Array.from(shared_converse.connection.features.childNodes).forEach(feature => { shared_converse.stream_features.create({ 'name': feature.nodeName, 'xmlns': feature.getAttribute('xmlns') }); }); notifyStreamFeaturesAdded(); } function utils_clearSession() { var _converse$disco_entit, _converse$disco_entit2, _converse$disco_entit3, _converse$disco_entit4, _converse$disco_entit5; (_converse$disco_entit = shared_converse.disco_entities) === null || _converse$disco_entit === void 0 ? void 0 : _converse$disco_entit.forEach(e => e.features.clearStore()); (_converse$disco_entit2 = shared_converse.disco_entities) === null || _converse$disco_entit2 === void 0 ? void 0 : _converse$disco_entit2.forEach(e => e.identities.clearStore()); (_converse$disco_entit3 = shared_converse.disco_entities) === null || _converse$disco_entit3 === void 0 ? void 0 : _converse$disco_entit3.forEach(e => e.dataforms.clearStore()); (_converse$disco_entit4 = shared_converse.disco_entities) === null || _converse$disco_entit4 === void 0 ? void 0 : _converse$disco_entit4.forEach(e => e.fields.clearStore()); (_converse$disco_entit5 = shared_converse.disco_entities) === null || _converse$disco_entit5 === void 0 ? void 0 : _converse$disco_entit5.clearStore(); delete shared_converse.disco_entities; } ;// CONCATENATED MODULE: ./src/headless/plugins/disco/api.js const { Strophe: api_Strophe, $iq: api_$iq } = core_converse.env; /* harmony default export */ const disco_api = ({ /** * The XEP-0030 service discovery API * * This API lets you discover information about entities on the * XMPP network. * * @namespace api.disco * @memberOf api */ disco: { /** * @namespace api.disco.stream * @memberOf api.disco */ stream: { /** * @method api.disco.stream.getFeature * @param {String} name The feature name * @param {String} xmlns The XML namespace * @example _converse.api.disco.stream.getFeature('ver', 'urn:xmpp:features:rosterver') */ async getFeature(name, xmlns) { await core_api.waitUntil('streamFeaturesAdded'); if (!name || !xmlns) { throw new Error("name and xmlns need to be provided when calling disco.stream.getFeature"); } if (shared_converse.stream_features === undefined && !core_api.connection.connected()) { // Happens during tests when disco lookups happen asynchronously after teardown. const msg = `Tried to get feature ${name} ${xmlns} but _converse.stream_features has been torn down`; headless_log.warn(msg); return; } return shared_converse.stream_features.findWhere({ 'name': name, 'xmlns': xmlns }); } }, /** * @namespace api.disco.own * @memberOf api.disco */ own: { /** * @namespace api.disco.own.identities * @memberOf api.disco.own */ identities: { /** * Lets you add new identities for this client (i.e. instance of Converse) * @method api.disco.own.identities.add * * @param {String} category - server, client, gateway, directory, etc. * @param {String} type - phone, pc, web, etc. * @param {String} name - "Converse" * @param {String} lang - en, el, de, etc. * * @example _converse.api.disco.own.identities.clear(); */ add(category, type, name, lang) { for (var i = 0; i < shared_converse.disco._identities.length; i++) { if (shared_converse.disco._identities[i].category == category && shared_converse.disco._identities[i].type == type && shared_converse.disco._identities[i].name == name && shared_converse.disco._identities[i].lang == lang) { return false; } } shared_converse.disco._identities.push({ category: category, type: type, name: name, lang: lang }); }, /** * Clears all previously registered identities. * @method api.disco.own.identities.clear * @example _converse.api.disco.own.identities.clear(); */ clear() { shared_converse.disco._identities = []; }, /** * Returns all of the identities registered for this client * (i.e. instance of Converse). * @method api.disco.identities.get * @example const identities = api.disco.own.identities.get(); */ get() { return shared_converse.disco._identities; } }, /** * @namespace api.disco.own.features * @memberOf api.disco.own */ features: { /** * Lets you register new disco features for this client (i.e. instance of Converse) * @method api.disco.own.features.add * @param {String} name - e.g. http://jabber.org/protocol/caps * @example _converse.api.disco.own.features.add("http://jabber.org/protocol/caps"); */ add(name) { for (var i = 0; i < shared_converse.disco._features.length; i++) { if (shared_converse.disco._features[i] == name) { return false; } } shared_converse.disco._features.push(name); }, /** * Clears all previously registered features. * @method api.disco.own.features.clear * @example _converse.api.disco.own.features.clear(); */ clear() { shared_converse.disco._features = []; }, /** * Returns all of the features registered for this client (i.e. instance of Converse). * @method api.disco.own.features.get * @example const features = api.disco.own.features.get(); */ get() { return shared_converse.disco._features; } } }, /** * Query for information about an XMPP entity * * @method api.disco.info * @param {string} jid The Jabber ID of the entity to query * @param {string} [node] A specific node identifier associated with the JID * @returns {promise} Promise which resolves once we have a result from the server. */ info(jid, node) { const attrs = { xmlns: api_Strophe.NS.DISCO_INFO }; if (node) { attrs.node = node; } const info = api_$iq({ 'from': shared_converse.connection.jid, 'to': jid, 'type': 'get' }).c('query', attrs); return core_api.sendIQ(info); }, /** * Query for items associated with an XMPP entity * * @method api.disco.items * @param {string} jid The Jabber ID of the entity to query for items * @param {string} [node] A specific node identifier associated with the JID * @returns {promise} Promise which resolves once we have a result from the server. */ items(jid, node) { const attrs = { 'xmlns': api_Strophe.NS.DISCO_ITEMS }; if (node) { attrs.node = node; } return core_api.sendIQ(api_$iq({ 'from': shared_converse.connection.jid, 'to': jid, 'type': 'get' }).c('query', attrs)); }, /** * Namespace for methods associated with disco entities * * @namespace api.disco.entities * @memberOf api.disco */ entities: { /** * Get the corresponding `DiscoEntity` instance. * * @method api.disco.entities.get * @param {string} jid The Jabber ID of the entity * @param {boolean} [create] Whether the entity should be created if it doesn't exist. * @example _converse.api.disco.entities.get(jid); */ async get(jid) { let create = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false; await core_api.waitUntil('discoInitialized'); if (!jid) { return shared_converse.disco_entities; } if (shared_converse.disco_entities === undefined) { // Happens during tests when disco lookups happen asynchronously after teardown. const msg = `Tried to look up entity ${jid} but _converse.disco_entities has been torn down`; headless_log.warn(msg); return; } const entity = shared_converse.disco_entities.get(jid); if (entity || !create) { return entity; } return core_api.disco.entities.create(jid); }, /** * Create a new disco entity. It's identity and features * will automatically be fetched from cache or from the * XMPP server. * * Fetching from cache can be disabled by passing in * `ignore_cache: true` in the options parameter. * * @method api.disco.entities.create * @param {string} jid The Jabber ID of the entity * @param {object} [options] Additional options * @param {boolean} [options.ignore_cache] * If true, fetch all features from the XMPP server instead of restoring them from cache * @example _converse.api.disco.entities.create(jid, {'ignore_cache': true}); */ create(jid, options) { return shared_converse.disco_entities.create({ 'jid': jid }, options); } }, /** * @namespace api.disco.features * @memberOf api.disco */ features: { /** * Return a given feature of a disco entity * * @method api.disco.features.get * @param {string} feature The feature that might be * supported. In the XML stanza, this is the `var` * attribute of the `` element. For * example: `http://jabber.org/protocol/muc` * @param {string} jid The JID of the entity * (and its associated items) which should be queried * @returns {promise} A promise which resolves with a list containing * _converse.Entity instances representing the entity * itself or those items associated with the entity if * they support the given feature. * @example * api.disco.features.get(Strophe.NS.MAM, _converse.bare_jid); */ async get(feature, jid) { if (!jid) { throw new TypeError('You need to provide an entity JID'); } await core_api.waitUntil('discoInitialized'); let entity = await core_api.disco.entities.get(jid, true); if (shared_converse.disco_entities === undefined && !core_api.connection.connected()) { // Happens during tests when disco lookups happen asynchronously after teardown. const msg = `Tried to get feature ${feature} for ${jid} but _converse.disco_entities has been torn down`; headless_log.warn(msg); return; } entity = await entity.waitUntilFeaturesDiscovered; const promises = [...entity.items.map(i => i.hasFeature(feature)), entity.hasFeature(feature)]; const result = await Promise.all(promises); return result.filter(lodash_es_isObject); } }, /** * Used to determine whether an entity supports a given feature. * * @method api.disco.supports * @param {string} feature The feature that might be * supported. In the XML stanza, this is the `var` * attribute of the `` element. For * example: `http://jabber.org/protocol/muc` * @param {string} jid The JID of the entity * (and its associated items) which should be queried * @returns {promise} A promise which resolves with `true` or `false`. * @example * if (await api.disco.supports(Strophe.NS.MAM, _converse.bare_jid)) { * // The feature is supported * } else { * // The feature is not supported * } */ async supports(feature, jid) { const features = (await core_api.disco.features.get(feature, jid)) || []; return features.length > 0; }, /** * Refresh the features, fields and identities associated with a * disco entity by refetching them from the server * @method api.disco.refresh * @param {string} jid The JID of the entity whose features are refreshed. * @returns {promise} A promise which resolves once the features have been refreshed * @example * await api.disco.refresh('room@conference.example.org'); */ async refresh(jid) { if (!jid) { throw new TypeError('api.disco.refresh: You need to provide an entity JID'); } await core_api.waitUntil('discoInitialized'); let entity = await core_api.disco.entities.get(jid); if (entity) { entity.features.reset(); entity.fields.reset(); entity.identities.reset(); if (!entity.waitUntilFeaturesDiscovered.isPending) { entity.waitUntilFeaturesDiscovered = getOpenPromise(); } entity.queryInfo(); } else { // Create it if it doesn't exist entity = await core_api.disco.entities.create(jid, { 'ignore_cache': true }); } return entity.waitUntilFeaturesDiscovered; }, /** * @deprecated Use {@link api.disco.refresh} instead. * @method api.disco.refreshFeatures */ refreshFeatures(jid) { return core_api.refresh(jid); }, /** * Return all the features associated with a disco entity * * @method api.disco.getFeatures * @param {string} jid The JID of the entity whose features are returned. * @returns {promise} A promise which resolves with the returned features * @example * const features = await api.disco.getFeatures('room@conference.example.org'); */ async getFeatures(jid) { if (!jid) { throw new TypeError('api.disco.getFeatures: You need to provide an entity JID'); } await core_api.waitUntil('discoInitialized'); let entity = await core_api.disco.entities.get(jid, true); entity = await entity.waitUntilFeaturesDiscovered; return entity.features; }, /** * Return all the service discovery extensions fields * associated with an entity. * * See [XEP-0129: Service Discovery Extensions](https://xmpp.org/extensions/xep-0128.html) * * @method api.disco.getFields * @param {string} jid The JID of the entity whose fields are returned. * @example * const fields = await api.disco.getFields('room@conference.example.org'); */ async getFields(jid) { if (!jid) { throw new TypeError('api.disco.getFields: You need to provide an entity JID'); } await core_api.waitUntil('discoInitialized'); let entity = await core_api.disco.entities.get(jid, true); entity = await entity.waitUntilFeaturesDiscovered; return entity.fields; }, /** * Get the identity (with the given category and type) for a given disco entity. * * For example, when determining support for PEP (personal eventing protocol), you * want to know whether the user's own JID has an identity with * `category='pubsub'` and `type='pep'` as explained in this section of * XEP-0163: https://xmpp.org/extensions/xep-0163.html#support * * @method api.disco.getIdentity * @param {string} The identity category. * In the XML stanza, this is the `category` * attribute of the `` element. * For example: 'pubsub' * @param {string} type The identity type. * In the XML stanza, this is the `type` * attribute of the `` element. * For example: 'pep' * @param {string} jid The JID of the entity which might have the identity * @returns {promise} A promise which resolves with a map indicating * whether an identity with a given type is provided by the entity. * @example * api.disco.getIdentity('pubsub', 'pep', _converse.bare_jid).then( * function (identity) { * if (identity) { * // The entity DOES have this identity * } else { * // The entity DOES NOT have this identity * } * } * ).catch(e => log.error(e)); */ async getIdentity(category, type, jid) { const e = await core_api.disco.entities.get(jid, true); if (e === undefined && !core_api.connection.connected()) { // Happens during tests when disco lookups happen asynchronously after teardown. const msg = `Tried to look up category ${category} for ${jid} but _converse.disco_entities has been torn down`; headless_log.warn(msg); return; } return e.getIdentity(category, type); } } }); ;// CONCATENATED MODULE: ./src/headless/plugins/disco/index.js /** * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) * @description Converse plugin which add support for XEP-0030: Service Discovery */ const { Strophe: disco_Strophe } = core_converse.env; core_converse.plugins.add('converse-disco', { initialize() { Object.assign(core_api, disco_api); core_api.promises.add('discoInitialized'); core_api.promises.add('streamFeaturesAdded'); shared_converse.DiscoEntity = entity; shared_converse.DiscoEntities = entities; shared_converse.disco = { _identities: [], _features: [] }; core_api.listen.on('userSessionInitialized', async () => { initStreamFeatures(); if (shared_converse.connfeedback.get('connection_status') === disco_Strophe.Status.ATTACHED) { // When re-attaching to a BOSH session, we fetch the stream features from the cache. await new Promise((success, error) => shared_converse.stream_features.fetch({ success, error })); notifyStreamFeaturesAdded(); } }); core_api.listen.on('beforeResourceBinding', populateStreamFeatures); core_api.listen.on('reconnected', initializeDisco); core_api.listen.on('connected', initializeDisco); core_api.listen.on('beforeTearDown', async () => { core_api.promises.add('streamFeaturesAdded'); if (shared_converse.stream_features) { await shared_converse.stream_features.clearStore(); delete shared_converse.stream_features; } }); // All disco entities stored in sessionStorage and are refetched // upon login or reconnection and then stored with new ids, so to // avoid sessionStorage filling up, we remove them. core_api.listen.on('will-reconnect', utils_clearSession); core_api.listen.on('clearSession', utils_clearSession); } }); ;// CONCATENATED MODULE: ./src/headless/plugins/emoji/index.js /** * @module converse-emoji * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.emojis = { 'initialized': false, 'initialized_promise': getOpenPromise() }; core_converse.plugins.add('converse-emoji', { initialize() { /* The initialize function gets called as soon as the plugin is * loaded by converse.js's plugin machinery. */ const { ___ } = shared_converse; core_api.settings.extend({ 'emoji_image_path': 'https://twemoji.maxcdn.com/v/12.1.6/', 'emoji_categories': { "smileys": ":grinning:", "people": ":thumbsup:", "activity": ":soccer:", "travel": ":motorcycle:", "objects": ":bomb:", "nature": ":rainbow:", "food": ":hotdog:", "symbols": ":musical_note:", "flags": ":flag_ac:", "custom": null }, // We use the triple-underscore method which doesn't actually // translate but does signify to gettext that these strings should // go into the POT file. The translation then happens in the // template. We do this so that users can pass in their own // strings via converse.initialize, which is before __ is // available. 'emoji_category_labels': { "smileys": ___("Smileys and emotions"), "people": ___("People"), "activity": ___("Activities"), "travel": ___("Travel"), "objects": ___("Objects"), "nature": ___("Animals and nature"), "food": ___("Food and drink"), "symbols": ___("Symbols"), "flags": ___("Flags"), "custom": ___("Stickers") } }); /** * Model for storing data related to the Emoji picker widget * @class * @namespace _converse.EmojiPicker * @memberOf _converse */ shared_converse.EmojiPicker = Model.extend({ defaults: { 'current_category': 'smileys', 'current_skintone': '', 'scroll_position': 0 } }); // We extend the default converse.js API to add methods specific to MUC groupchats. Object.assign(core_api, { /** * @namespace api.emojis * @memberOf api */ emojis: { /** * Initializes Emoji support by downloading the emojis JSON (and any applicable images). * @method api.emojis.initialize * @returns {Promise} */ async initialize() { if (!core_converse.emojis.initialized) { core_converse.emojis.initialized = true; const { default: json } = await __webpack_require__.e(/* import() | emojis */ 4610).then(__webpack_require__.t.bind(__webpack_require__, 5175, 19)); core_converse.emojis.json = json; core_converse.emojis.by_sn = Object.keys(json).reduce((result, cat) => Object.assign(result, json[cat]), {}); core_converse.emojis.list = Object.values(core_converse.emojis.by_sn); core_converse.emojis.list.sort((a, b) => a.sn < b.sn ? -1 : a.sn > b.sn ? 1 : 0); core_converse.emojis.shortnames = core_converse.emojis.list.map(m => m.sn); const getShortNames = () => core_converse.emojis.shortnames.map(s => s.replace(/[+]/g, "\\$&")).join('|'); core_converse.emojis.shortnames_regex = new RegExp(getShortNames(), "gi"); core_converse.emojis.initialized_promise.resolve(); } return core_converse.emojis.initialized_promise; } } }); } }); ;// CONCATENATED MODULE: ./src/headless/plugins/muc/message.js /** * Mixing that turns a Message model into a ChatRoomMessage model. * @class * @namespace _converse.ChatRoomMessage * @memberOf _converse */ const ChatRoomMessageMixin = { initialize() { if (!this.checkValidity()) { return; } if (this.get('file')) { this.on('change:put', () => this.uploadFile()); } // If `type` changes from `error` to `groupchat`, we want to set the occupant. See #2733 this.on('change:type', () => this.setOccupant()); this.on('change:is_ephemeral', () => this.setTimerForEphemeralMessage()); this.setTimerForEphemeralMessage(); this.setOccupant(); /** * Triggered once a { @link _converse.ChatRoomMessage } has been created and initialized. * @event _converse#chatRoomMessageInitialized * @type { _converse.ChatRoomMessages} * @example _converse.api.listen.on('chatRoomMessageInitialized', model => { ... }); */ core_api.trigger('chatRoomMessageInitialized', this); }, getDisplayName() { var _this$occupant; return ((_this$occupant = this.occupant) === null || _this$occupant === void 0 ? void 0 : _this$occupant.getDisplayName()) || this.get('nick'); }, /** * Determines whether this messsage may be moderated, * based on configuration settings and server support. * @async * @private * @method _converse.ChatRoomMessages#mayBeModerated * @returns { Boolean } */ mayBeModerated() { if (typeof this.get('from_muc') === 'undefined') { // If from_muc is not defined, then this message hasn't been // reflected yet, which means we won't have a XEP-0359 stanza id. return; } return ['all', 'moderator'].includes(core_api.settings.get('allow_message_retraction')) && this.get(`stanza_id ${this.get('from_muc')}`) && this.collection.chatbox.canModerateMessages(); }, checkValidity() { const result = shared_converse.Message.prototype.checkValidity.call(this); !result && this.collection.chatbox.debouncedRejoin(); return result; }, onOccupantRemoved() { var _this$collection; this.stopListening(this.occupant); delete this.occupant; const chatbox = this === null || this === void 0 ? void 0 : (_this$collection = this.collection) === null || _this$collection === void 0 ? void 0 : _this$collection.chatbox; if (!chatbox) { return headless_log.error(`Could not get collection.chatbox for message: ${JSON.stringify(this.toJSON())}`); } this.listenTo(chatbox.occupants, 'add', this.onOccupantAdded); }, onOccupantAdded(occupant) { var _this$collection2; if (this.get('occupant_id')) { if (occupant.get('occupant_id') !== this.get('occupant_id')) { return; } } else if (occupant.get('nick') !== Strophe.getResourceFromJid(this.get('from'))) { return; } const chatbox = this === null || this === void 0 ? void 0 : (_this$collection2 = this.collection) === null || _this$collection2 === void 0 ? void 0 : _this$collection2.chatbox; if (!chatbox) { return headless_log.error(`Could not get collection.chatbox for message: ${JSON.stringify(this.toJSON())}`); } this.occupant = occupant; if (occupant.get('jid')) { this.save('from_real_jid', occupant.get('jid')); } this.trigger('occupantAdded'); this.listenTo(this.occupant, 'destroy', this.onOccupantRemoved); this.stopListening(chatbox.occupants, 'add', this.onOccupantAdded); }, setOccupant() { var _this$collection3; if (this.get('type') !== 'groupchat' || this.isEphemeral() || this.occupant) { return; } const chatbox = this === null || this === void 0 ? void 0 : (_this$collection3 = this.collection) === null || _this$collection3 === void 0 ? void 0 : _this$collection3.chatbox; if (!chatbox) { return headless_log.error(`Could not get collection.chatbox for message: ${JSON.stringify(this.toJSON())}`); } const nick = Strophe.getResourceFromJid(this.get('from')); const occupant_id = this.get('occupant_id'); this.occupant = chatbox.occupants.findOccupant({ nick, occupant_id }); if (!this.occupant && core_api.settings.get('muc_send_probes')) { this.occupant = chatbox.occupants.create({ nick, occupant_id, 'type': 'unavailable' }); const jid = `${chatbox.get('jid')}/${nick}`; core_api.user.presence.send('probe', jid); } if (this.occupant) { this.listenTo(this.occupant, 'destroy', this.onOccupantRemoved); } else { this.listenTo(chatbox.occupants, 'add', this.onOccupantAdded); } } }; /* harmony default export */ const muc_message = (ChatRoomMessageMixin); ;// CONCATENATED MODULE: ./src/headless/utils/parse-helpers.js /** * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) * @description Pure functions to help functionally parse messages. * @todo Other parsing helpers can be made more abstract and placed here. */ const helpers = {}; const escapeRegexChars = (string, char) => string.replace(RegExp('\\' + char, 'ig'), '\\' + char); helpers.escapeCharacters = characters => string => characters.split('').reduce(escapeRegexChars, string); helpers.escapeRegexString = helpers.escapeCharacters('[\\^$.?*+(){}|'); // `for` is ~25% faster than using `Array.find()` helpers.findFirstMatchInArray = array => text => { for (let i = 0; i < array.length; i++) { if (text.localeCompare(array[i], undefined, { sensitivity: 'base' }) === 0) { return array[i]; } } return null; }; const reduceReferences = (_ref, ref, index) => { let [text, refs] = _ref; let updated_text = text; let { begin, end } = ref; const { value } = ref; begin = begin - index; end = end - index - 1; // -1 to compensate for the removed @ updated_text = `${updated_text.slice(0, begin)}${value}${updated_text.slice(end + 1)}`; return [updated_text, [...refs, { ...ref, begin, end }]]; }; helpers.reduceTextFromReferences = (text, refs) => refs.reduce(reduceReferences, [text, []]); /* harmony default export */ const parse_helpers = (helpers); ;// CONCATENATED MODULE: ./src/headless/utils/form.js /** * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) * @description This is the form utilities module. */ /** * Takes an HTML DOM and turns it into an XForm field. * @private * @method u#webForm2xForm * @param { DOMElement } field - the field to convert */ utils_core.webForm2xForm = function (field) { const name = field.getAttribute('name'); if (!name) { return null; // See #1924 } let value; if (field.getAttribute('type') === 'checkbox') { value = field.checked && 1 || 0; } else if (field.tagName == "TEXTAREA") { value = field.value.split('\n').filter(s => s.trim()); } else if (field.tagName == "SELECT") { value = utils_core.getSelectValues(field); } else { value = field.value; } return utils_core.toStanza(` ${value.constructor === Array ? value.map(v => `${v}`) : `${value}`} `); }; /* harmony default export */ const utils_form = (utils_core); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseZipObject.js /** * This base implementation of `_.zipObject` which assigns values using `assignFunc`. * * @private * @param {Array} props The property identifiers. * @param {Array} values The property values. * @param {Function} assignFunc The function to assign values. * @returns {Object} Returns the new object. */ function baseZipObject(props, values, assignFunc) { var index = -1, length = props.length, valsLength = values.length, result = {}; while (++index < length) { var value = index < valsLength ? values[index] : undefined; assignFunc(result, props[index], value); } return result; } /* harmony default export */ const _baseZipObject = (baseZipObject); ;// CONCATENATED MODULE: ./node_modules/lodash-es/zipObject.js /** * This method is like `_.fromPairs` except that it accepts two arrays, * one of property identifiers and one of corresponding values. * * @static * @memberOf _ * @since 0.4.0 * @category Array * @param {Array} [props=[]] The property identifiers. * @param {Array} [values=[]] The property values. * @returns {Object} Returns the new object. * @example * * _.zipObject(['a', 'b'], [1, 2]); * // => { 'a': 1, 'b': 2 } */ function zipObject(props, values) { return _baseZipObject(props || [], values || [], _assignValue); } /* harmony default export */ const lodash_es_zipObject = (zipObject); ;// CONCATENATED MODULE: ./src/headless/plugins/muc/parsers.js const { Strophe: muc_parsers_Strophe, sizzle: muc_parsers_sizzle, u: parsers_u } = core_converse.env; const { NS: parsers_NS } = muc_parsers_Strophe; /** * Parses a message stanza for XEP-0317 MEP notification data * @param { XMLElement } stanza - The message stanza * @returns { Array } Returns an array of objects representing elements. */ function getMEPActivities(stanza) { const items_el = muc_parsers_sizzle(`items[node="${muc_parsers_Strophe.NS.CONFINFO}"]`, stanza).pop(); if (!items_el) { return null; } const from = stanza.getAttribute('from'); const msgid = stanza.getAttribute('id'); const selector = `item ` + `conference-info[xmlns="${muc_parsers_Strophe.NS.CONFINFO}"] ` + `activity[xmlns="${muc_parsers_Strophe.NS.ACTIVITY}"]`; return muc_parsers_sizzle(selector, items_el).map(el => { var _el$querySelector; const message = (_el$querySelector = el.querySelector('text')) === null || _el$querySelector === void 0 ? void 0 : _el$querySelector.textContent; if (message) { var _el$querySelector2; const references = getReferences(stanza); const reason = (_el$querySelector2 = el.querySelector('reason')) === null || _el$querySelector2 === void 0 ? void 0 : _el$querySelector2.textContent; return { from, msgid, message, reason, references, 'type': 'mep' }; } return {}; }); } /** * Given a MUC stanza, check whether it has extended message information that * includes the sender's real JID, as described here: * https://xmpp.org/extensions/xep-0313.html#business-storeret-muc-archives * * If so, parse and return that data and return the user's JID * * Note, this function doesn't check whether this is actually a MAM archived stanza. * * @private * @param { XMLElement } stanza - The message stanza * @returns { Object } */ function getJIDFromMUCUserData(stanza) { const item = muc_parsers_sizzle(`x[xmlns="${muc_parsers_Strophe.NS.MUC_USER}"] item`, stanza).pop(); return item === null || item === void 0 ? void 0 : item.getAttribute('jid'); } /** * @private * @param { XMLElement } stanza - The message stanza * @param { XMLElement } original_stanza - The original stanza, that contains the * message stanza, if it was contained, otherwise it's the message stanza itself. * @returns { Object } */ function getModerationAttributes(stanza) { const fastening = muc_parsers_sizzle(`apply-to[xmlns="${muc_parsers_Strophe.NS.FASTEN}"]`, stanza).pop(); if (fastening) { const applies_to_id = fastening.getAttribute('id'); const moderated = muc_parsers_sizzle(`moderated[xmlns="${muc_parsers_Strophe.NS.MODERATE}"]`, fastening).pop(); if (moderated) { const retracted = muc_parsers_sizzle(`retract[xmlns="${muc_parsers_Strophe.NS.RETRACT}"]`, moderated).pop(); if (retracted) { var _moderated$querySelec; return { 'editable': false, 'moderated': 'retracted', 'moderated_by': moderated.getAttribute('by'), 'moderated_id': applies_to_id, 'moderation_reason': (_moderated$querySelec = moderated.querySelector('reason')) === null || _moderated$querySelec === void 0 ? void 0 : _moderated$querySelec.textContent }; } } } else { const tombstone = muc_parsers_sizzle(`> moderated[xmlns="${muc_parsers_Strophe.NS.MODERATE}"]`, stanza).pop(); if (tombstone) { const retracted = muc_parsers_sizzle(`retracted[xmlns="${muc_parsers_Strophe.NS.RETRACT}"]`, tombstone).pop(); if (retracted) { var _tombstone$querySelec; return { 'editable': false, 'is_tombstone': true, 'moderated_by': tombstone.getAttribute('by'), 'retracted': tombstone.getAttribute('stamp'), 'moderation_reason': (_tombstone$querySelec = tombstone.querySelector('reason')) === null || _tombstone$querySelec === void 0 ? void 0 : _tombstone$querySelec.textContent }; } } } return {}; } function getOccupantID(stanza, chatbox) { if (chatbox.features.get(muc_parsers_Strophe.NS.OCCUPANTID)) { var _sizzle$pop; return (_sizzle$pop = muc_parsers_sizzle(`occupant-id[xmlns="${muc_parsers_Strophe.NS.OCCUPANTID}"]`, stanza).pop()) === null || _sizzle$pop === void 0 ? void 0 : _sizzle$pop.getAttribute('id'); } } /** * Parses a passed in message stanza and returns an object of attributes. * @param { XMLElement } stanza - The message stanza * @param { XMLElement } original_stanza - The original stanza, that contains the * message stanza, if it was contained, otherwise it's the message stanza itself. * @param { _converse.ChatRoom } chatbox * @param { _converse } _converse * @returns { Promise } */ async function parseMUCMessage(stanza, chatbox) { var _stanza$querySelector, _stanza$querySelector2, _stanza$querySelector3, _stanza$querySelector4, _chatbox$occupants$fi; throwErrorIfInvalidForward(stanza); const selector = `[xmlns="${parsers_NS.MAM}"] > forwarded[xmlns="${parsers_NS.FORWARD}"] > message`; const original_stanza = stanza; stanza = muc_parsers_sizzle(selector, stanza).pop() || stanza; if (muc_parsers_sizzle(`message > forwarded[xmlns="${muc_parsers_Strophe.NS.FORWARD}"]`, stanza).length) { return new StanzaParseError(`Invalid Stanza: Forged MAM groupchat message from ${stanza.getAttribute('from')}`, stanza); } const delay = muc_parsers_sizzle(`delay[xmlns="${muc_parsers_Strophe.NS.DELAY}"]`, original_stanza).pop(); const from = stanza.getAttribute('from'); const from_muc = muc_parsers_Strophe.getBareJidFromJid(from); const nick = muc_parsers_Strophe.unescapeNode(muc_parsers_Strophe.getResourceFromJid(from)); const marker = getChatMarker(stanza); const now = new Date().toISOString(); /** * @typedef { Object } MUCMessageAttributes * The object which {@link parseMUCMessage} returns * @property { ('me'|'them') } sender - Whether the message was sent by the current user or someone else * @property { Array } activities - A list of objects representing XEP-0316 MEP notification data * @property { Array } references - A list of objects representing XEP-0372 references * @property { Boolean } editable - Is this message editable via XEP-0308? * @property { Boolean } is_archived - Is this message from a XEP-0313 MAM archive? * @property { Boolean } is_carbon - Is this message a XEP-0280 Carbon? * @property { Boolean } is_delayed - Was delivery of this message was delayed as per XEP-0203? * @property { Boolean } is_encrypted - Is this message XEP-0384 encrypted? * @property { Boolean } is_error - Whether an error was received for this message * @property { Boolean } is_headline - Is this a "headline" message? * @property { Boolean } is_markable - Can this message be marked with a XEP-0333 chat marker? * @property { Boolean } is_marker - Is this message a XEP-0333 Chat Marker? * @property { Boolean } is_only_emojis - Does the message body contain only emojis? * @property { Boolean } is_spoiler - Is this a XEP-0382 spoiler message? * @property { Boolean } is_tombstone - Is this a XEP-0424 tombstone? * @property { Boolean } is_unstyled - Whether XEP-0393 styling hints should be ignored * @property { Boolean } is_valid_receipt_request - Does this message request a XEP-0184 receipt (and is not from us or a carbon or archived message) * @property { Object } encrypted - XEP-0384 encryption payload attributes * @property { String } body - The contents of the tag of the message stanza * @property { String } chat_state - The XEP-0085 chat state notification contained in this message * @property { String } edited - An ISO8601 string recording the time that the message was edited per XEP-0308 * @property { String } error_condition - The defined error condition * @property { String } error_text - The error text received from the server * @property { String } error_type - The type of error received from the server * @property { String } from - The sender JID (${muc_jid}/${nick}) * @property { String } from_muc - The JID of the MUC from which this message was sent * @property { String } from_real_jid - The real JID of the sender, if available * @property { String } fullname - The full name of the sender * @property { String } marker - The XEP-0333 Chat Marker value * @property { String } marker_id - The `id` attribute of a XEP-0333 chat marker * @property { String } moderated - The type of XEP-0425 moderation (if any) that was applied * @property { String } moderated_by - The JID of the user that moderated this message * @property { String } moderated_id - The XEP-0359 Stanza ID of the message that this one moderates * @property { String } moderation_reason - The reason provided why this message moderates another * @property { String } msgid - The root `id` attribute of the stanza * @property { String } nick - The MUC nickname of the sender * @property { String } occupant_id - The XEP-0421 occupant ID * @property { String } oob_desc - The description of the XEP-0066 out of band data * @property { String } oob_url - The URL of the XEP-0066 out of band data * @property { String } origin_id - The XEP-0359 Origin ID * @property { String } receipt_id - The `id` attribute of a XEP-0184 element * @property { String } received - An ISO8601 string recording the time that the message was received * @property { String } replace_id - The `id` attribute of a XEP-0308 element * @property { String } retracted - An ISO8601 string recording the time that the message was retracted * @property { String } retracted_id - The `id` attribute of a XEP-424 element * @property { String } spoiler_hint The XEP-0382 spoiler hint * @property { String } stanza_id - The XEP-0359 Stanza ID. Note: the key is actualy `stanza_id ${by_jid}` and there can be multiple. * @property { String } subject - The element value * @property { String } thread - The element value * @property { String } time - The time (in ISO8601 format), either given by the XEP-0203 element, or of receipt. * @property { String } to - The recipient JID * @property { String } type - The type of message */ let attrs = Object.assign({ from, from_muc, nick, 'is_forwarded': !!stanza.querySelector('forwarded'), 'activities': getMEPActivities(stanza), 'body': (_stanza$querySelector = stanza.querySelector('body')) === null || _stanza$querySelector === void 0 ? void 0 : (_stanza$querySelector2 = _stanza$querySelector.textContent) === null || _stanza$querySelector2 === void 0 ? void 0 : _stanza$querySelector2.trim(), 'chat_state': getChatState(stanza), 'is_archived': isArchived(original_stanza), 'is_carbon': isCarbon(original_stanza), 'is_delayed': !!delay, 'is_headline': isHeadline(stanza), 'is_markable': !!muc_parsers_sizzle(`markable[xmlns="${muc_parsers_Strophe.NS.MARKERS}"]`, stanza).length, 'is_marker': !!marker, 'is_unstyled': !!muc_parsers_sizzle(`unstyled[xmlns="${muc_parsers_Strophe.NS.STYLING}"]`, stanza).length, 'marker_id': marker && marker.getAttribute('id'), 'msgid': stanza.getAttribute('id') || original_stanza.getAttribute('id'), 'occupant_id': getOccupantID(stanza, chatbox), 'receipt_id': getReceiptId(stanza), 'received': new Date().toISOString(), 'references': getReferences(stanza), 'subject': (_stanza$querySelector3 = stanza.querySelector('subject')) === null || _stanza$querySelector3 === void 0 ? void 0 : _stanza$querySelector3.textContent, 'thread': (_stanza$querySelector4 = stanza.querySelector('thread')) === null || _stanza$querySelector4 === void 0 ? void 0 : _stanza$querySelector4.textContent, 'time': delay ? dayjs_min_default()(delay.getAttribute('stamp')).toISOString() : now, 'to': stanza.getAttribute('to'), 'type': stanza.getAttribute('type') }, getErrorAttributes(stanza), getOutOfBandAttributes(stanza), getSpoilerAttributes(stanza), getCorrectionAttributes(stanza, original_stanza), getStanzaIDs(stanza, original_stanza), getOpenGraphMetadata(stanza), getRetractionAttributes(stanza, original_stanza), getModerationAttributes(stanza), getEncryptionAttributes(stanza, shared_converse)); await core_api.emojis.initialize(); const from_real_jid = attrs.is_archived && getJIDFromMUCUserData(stanza, attrs) || ((_chatbox$occupants$fi = chatbox.occupants.findOccupant(attrs)) === null || _chatbox$occupants$fi === void 0 ? void 0 : _chatbox$occupants$fi.get('jid')); attrs = Object.assign({ from_real_jid, 'is_only_emojis': attrs.body ? parsers_u.isOnlyEmojis(attrs.body) : false, 'is_valid_receipt_request': isValidReceiptRequest(stanza, attrs), 'message': attrs.body || attrs.error, // TODO: Remove and use body and error attributes instead 'sender': attrs.nick === chatbox.get('nick') ? 'me' : 'them' }, attrs); if (attrs.is_archived && original_stanza.getAttribute('from') !== attrs.from_muc) { return new StanzaParseError(`Invalid Stanza: Forged MAM message from ${original_stanza.getAttribute('from')}`, stanza); } else if (attrs.is_archived && original_stanza.getAttribute('from') !== chatbox.get('jid')) { return new StanzaParseError(`Invalid Stanza: Forged MAM groupchat message from ${stanza.getAttribute('from')}`, stanza); } else if (attrs.is_carbon) { return new StanzaParseError('Invalid Stanza: MUC messages SHOULD NOT be XEP-0280 carbon copied', stanza); } // We prefer to use one of the XEP-0359 unique and stable stanza IDs as the Model id, to avoid duplicates. attrs['id'] = attrs['origin_id'] || attrs[`stanza_id ${attrs.from_muc || attrs.from}`] || parsers_u.getUniqueId(); /** * *Hook* which allows plugins to add additional parsing * @event _converse#parseMUCMessage */ attrs = await core_api.hook('parseMUCMessage', stanza, attrs); // We call this after the hook, to allow plugins to decrypt encrypted // messages, since we need to parse the message text to determine whether // there are media urls. return Object.assign(attrs, getMediaURLsMetadata(attrs.is_encrypted ? attrs.plaintext : attrs.body)); } /** * Given an IQ stanza with a member list, create an array of objects containing * known member data (e.g. jid, nick, role, affiliation). * @private * @method muc_utils#parseMemberListIQ * @returns { MemberListItem[] } */ function parseMemberListIQ(iq) { return muc_parsers_sizzle(`query[xmlns="${muc_parsers_Strophe.NS.MUC_ADMIN}"] item`, iq).map(item => { /** * @typedef {Object} MemberListItem * Either the JID or the nickname (or both) will be available. * @property {string} affiliation * @property {string} [role] * @property {string} [jid] * @property {string} [nick] */ const data = { 'affiliation': item.getAttribute('affiliation') }; const jid = item.getAttribute('jid'); if (parsers_u.isValidJID(jid)) { data['jid'] = jid; } else { // XXX: Prosody sends nick for the jid attribute value // Perhaps for anonymous room? data['nick'] = jid; } const nick = item.getAttribute('nick'); if (nick) { data['nick'] = nick; } const role = item.getAttribute('role'); if (role) { data['role'] = nick; } return data; }); } /** * Parses a passed in MUC presence stanza and returns an object of attributes. * @method parseMUCPresence * @param { XMLElement } stanza - The presence stanza * @param { _converse.ChatRoom } chatbox * @returns { MUCPresenceAttributes } */ function parseMUCPresence(stanza, chatbox) { /** * @typedef { Object } MUCPresenceAttributes * The object which {@link parseMUCPresence} returns * @property { ("offline|online") } show * @property { Array } hats - An array of XEP-0317 hats * @property { Array } states * @property { String } from - The sender JID (${muc_jid}/${nick}) * @property { String } nick - The nickname of the sender * @property { String } occupant_id - The XEP-0421 occupant ID * @property { String } type - The type of presence */ const from = stanza.getAttribute('from'); const type = stanza.getAttribute('type'); const data = { 'from': from, 'occupant_id': getOccupantID(stanza, chatbox), 'nick': muc_parsers_Strophe.getResourceFromJid(from), 'type': type, 'states': [], 'hats': [], 'show': type !== 'unavailable' ? 'online' : 'offline' }; Array.from(stanza.children).forEach(child => { if (child.matches('status')) { data.status = child.textContent || null; } else if (child.matches('show')) { data.show = child.textContent || 'online'; } else if (child.matches('x') && child.getAttribute('xmlns') === muc_parsers_Strophe.NS.MUC_USER) { Array.from(child.children).forEach(item => { if (item.nodeName === 'item') { data.affiliation = item.getAttribute('affiliation'); data.role = item.getAttribute('role'); data.jid = item.getAttribute('jid'); data.nick = item.getAttribute('nick') || data.nick; } else if (item.nodeName == 'status' && item.getAttribute('code')) { data.states.push(item.getAttribute('code')); } }); } else if (child.matches('x') && child.getAttribute('xmlns') === muc_parsers_Strophe.NS.VCARDUPDATE) { var _child$querySelector; data.image_hash = (_child$querySelector = child.querySelector('photo')) === null || _child$querySelector === void 0 ? void 0 : _child$querySelector.textContent; } else if (child.matches('hats') && child.getAttribute('xmlns') === muc_parsers_Strophe.NS.MUC_HATS) { /** * @typedef { Object } MUCHat * Object representing a XEP-0371 Hat * @property { String } title * @property { String } uri */ data['hats'] = Array.from(child.children).map(c => c.matches('hat') && { 'title': c.getAttribute('title'), 'uri': c.getAttribute('uri') }); } }); return data; } ;// CONCATENATED MODULE: ./src/headless/plugins/muc/affiliations/utils.js /** * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) */ const { Strophe: affiliations_utils_Strophe, $iq: affiliations_utils_$iq, u: affiliations_utils_u } = core_converse.env; /** * Sends an IQ stanza to the server, asking it for the relevant affiliation list . * Returns an array of {@link MemberListItem} objects, representing occupants * that have the given affiliation. * See: https://xmpp.org/extensions/xep-0045.html#modifymember * @param { ("admin"|"owner"|"member") } affiliation * @param { String } muc_jid - The JID of the MUC for which the affiliation list should be fetched * @returns { Promise } */ async function getAffiliationList(affiliation, muc_jid) { const { __ } = shared_converse; const iq = affiliations_utils_$iq({ 'to': muc_jid, 'type': 'get' }).c('query', { xmlns: affiliations_utils_Strophe.NS.MUC_ADMIN }).c('item', { 'affiliation': affiliation }); const result = await core_api.sendIQ(iq, null, false); if (result === null) { const err_msg = __('Error: timeout while fetching %1s list for MUC %2s', affiliation, muc_jid); const err = new Error(err_msg); headless_log.warn(err_msg); return err; } if (affiliations_utils_u.isErrorStanza(result)) { const err_msg = __('Error: not allowed to fetch %1s list for MUC %2s', affiliation, muc_jid); const err = new Error(err_msg); headless_log.warn(err_msg); headless_log.warn(result); return err; } return parseMemberListIQ(result).filter(p => p).sort((a, b) => a.nick < b.nick ? -1 : a.nick > b.nick ? 1 : 0); } /** * Given an occupant model, see which affiliations may be assigned to that user. * @param { Model } occupant * @returns { Array<('owner'|'admin'|'member'|'outcast'|'none')> } - An array of assignable affiliations */ function getAssignableAffiliations(occupant) { let disabled = core_api.settings.get('modtools_disable_assign'); if (!Array.isArray(disabled)) { disabled = disabled ? AFFILIATIONS : []; } if (occupant.get('affiliation') === 'owner') { return AFFILIATIONS.filter(a => !disabled.includes(a)); } else if (occupant.get('affiliation') === 'admin') { return AFFILIATIONS.filter(a => !['owner', 'admin', ...disabled].includes(a)); } else { return []; } } // Necessary for tests shared_converse.getAssignableAffiliations = getAssignableAffiliations; /** * Send IQ stanzas to the server to modify affiliations for users in this groupchat. * See: https://xmpp.org/extensions/xep-0045.html#modifymember * @param { Array } users * @param { string } users[].jid - The JID of the user whose affiliation will change * @param { Array } users[].affiliation - The new affiliation for this user * @param { string } [users[].reason] - An optional reason for the affiliation change * @returns { Promise } */ function setAffiliations(muc_jid, users) { const affiliations = [...new Set(users.map(u => u.affiliation))]; return Promise.all(affiliations.map(a => setAffiliation(a, muc_jid, users))); } /** * Send IQ stanzas to the server to set an affiliation for * the provided JIDs. * See: https://xmpp.org/extensions/xep-0045.html#modifymember * * 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://issues.prosody.im/345 * * @param { ('outcast'|'member'|'admin'|'owner') } affiliation - The affiliation to be set * @param { String|Array } jids - The JID(s) of the MUCs in which the * affiliations need to be set. * @param { 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 { Promise } A promise which resolves and fails depending on the XMPP server response. */ function setAffiliation(affiliation, muc_jids, members) { if (!Array.isArray(muc_jids)) { muc_jids = [muc_jids]; } members = members.filter(m => [undefined, affiliation].includes(m.affiliation)); return Promise.all(muc_jids.reduce((acc, jid) => [...acc, ...members.map(m => sendAffiliationIQ(affiliation, jid, m))], [])); } /** * Send an IQ stanza specifying an affiliation change. * @private * @param { String } affiliation: affiliation (could also be stored on the member object). * @param { String } muc_jid: The JID of the MUC in which the affiliation should be set. * @param { Object } member: Map containing the member's jid and optionally a reason and affiliation. */ function sendAffiliationIQ(affiliation, muc_jid, member) { const iq = affiliations_utils_$iq({ to: muc_jid, type: 'set' }).c('query', { xmlns: affiliations_utils_Strophe.NS.MUC_ADMIN }).c('item', { 'affiliation': member.affiliation || affiliation, 'nick': member.nick, 'jid': member.jid }); if (member.reason !== undefined) { iq.c('reason', member.reason); } return core_api.sendIQ(iq); } /** * 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. * @param { 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. * @param { 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'. * @param { array } new_list - Array containing the new affiliations * @param { array } old_list - Array containing the old affiliations * @returns { array } */ function computeAffiliationsDelta(exclude_existing, remove_absentees, new_list, old_list) { const new_jids = new_list.map(o => o.jid); const old_jids = old_list.map(o => o.jid); // Get the new affiliations let delta = lodash_es_difference(new_jids, old_jids).map(jid => new_list[lodash_es_indexOf(new_jids, jid)]); if (!exclude_existing) { // Get the changed affiliations delta = delta.concat(new_list.filter(item => { const idx = lodash_es_indexOf(old_jids, item.jid); return idx >= 0 ? item.affiliation !== old_list[idx].affiliation : false; })); } if (remove_absentees) { // Get the removed affiliations delta = delta.concat(lodash_es_difference(old_jids, new_jids).map(jid => ({ 'jid': jid, 'affiliation': 'none' }))); } return delta; } ;// CONCATENATED MODULE: ./src/headless/plugins/muc/muc.js const OWNER_COMMANDS = ['owner']; const ADMIN_COMMANDS = ['admin', 'ban', 'deop', 'destroy', 'member', 'op', 'revoke']; const MODERATOR_COMMANDS = ['kick', 'mute', 'voice', 'modtools']; const VISITOR_COMMANDS = ['nick']; const METADATA_ATTRIBUTES = ["og:article:author", "og:article:published_time", "og:description", "og:image", "og:image:height", "og:image:width", "og:site_name", "og:title", "og:type", "og:url", "og:video:height", "og:video:secure_url", "og:video:tag", "og:video:type", "og:video:url", "og:video:width"]; const ACTION_INFO_CODES = ['301', '303', '333', '307', '321', '322']; const MUCSession = Model.extend({ defaults() { return { 'connection_status': core_converse.ROOMSTATUS.DISCONNECTED }; } }); /** * Represents an open/ongoing groupchat conversation. * @mixin * @namespace _converse.ChatRoom * @memberOf _converse */ const ChatRoomMixin = { defaults() { return { 'bookmarked': false, 'chat_state': undefined, 'has_activity': false, // XEP-437 'hidden': isUniView() && !core_api.settings.get('singleton'), 'hidden_occupants': !!core_api.settings.get('hide_muc_participants'), 'message_type': 'groupchat', 'name': '', // 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, 'num_unread': 0, 'roomconfig': {}, 'time_opened': this.get('time_opened') || new Date().getTime(), 'time_sent': new Date(0).toISOString(), 'type': shared_converse.CHATROOMS_TYPE }; }, async initialize() { this.initialized = getOpenPromise(); this.debouncedRejoin = lodash_es_debounce(this.rejoin, 250); this.set('box_id', `box-${this.get('jid')}`); this.initNotifications(); this.initMessages(); this.initUI(); this.initOccupants(); this.initDiscoModels(); // sendChatState depends on this.features this.registerHandlers(); this.on('change:chat_state', this.sendChatState, this); this.on('change:hidden', this.onHiddenChange, this); this.on('destroy', this.removeHandlers, this); this.ui.on('change:scrolled', this.onScrolledChanged, this); await this.restoreSession(); this.session.on('change:connection_status', this.onConnectionStatusChanged, this); this.listenTo(this.occupants, 'add', this.onOccupantAdded); this.listenTo(this.occupants, 'remove', this.onOccupantRemoved); this.listenTo(this.occupants, 'change:show', this.onOccupantShowChanged); this.listenTo(this.occupants, 'change:affiliation', this.createAffiliationChangeMessage); this.listenTo(this.occupants, 'change:role', this.createRoleChangeMessage); const restored = await this.restoreFromCache(); if (!restored) { this.join(); } /** * Triggered once a {@link _converse.ChatRoom} has been created and initialized. * @event _converse#chatRoomInitialized * @type { _converse.ChatRoom } * @example _converse.api.listen.on('chatRoomInitialized', model => { ... }); */ await core_api.trigger('chatRoomInitialized', this, { 'Synchronous': true }); this.initialized.resolve(); }, isEntered() { return this.session.get('connection_status') === core_converse.ROOMSTATUS.ENTERED; }, /** * Checks whether we're still joined and if so, restores the MUC state from cache. * @private * @method _converse.ChatRoom#restoreFromCache * @returns { Boolean } Returns `true` if we're still joined, otherwise returns `false`. */ async restoreFromCache() { if (this.isEntered() && (await this.isJoined())) { // We've restored the room from cache and we're still joined. await new Promise(r => this.features.fetch({ 'success': r, 'error': r })); await new Promise(r => this.config.fetch({ 'success': r, 'error': r })); await this.fetchOccupants().catch(e => headless_log.error(e)); await this.fetchMessages().catch(e => headless_log.error(e)); return true; } else { this.session.save('connection_status', core_converse.ROOMSTATUS.DISCONNECTED); this.clearOccupantsCache(); return false; } }, /** * Join the MUC * @private * @method _converse.ChatRoom#join * @param { String } nick - The user's nickname * @param { String } [password] - Optional password, if required by the groupchat. * Will fall back to the `password` value stored in the room * model (if available). */ async join(nick, password) { if (this.isEntered()) { // We have restored a groupchat from session storage, // so we don't send out a presence stanza again. return this; } // Set this early, so we don't rejoin in onHiddenChange this.session.save('connection_status', core_converse.ROOMSTATUS.CONNECTING); await this.refreshDiscoInfo(); nick = await this.getAndPersistNickname(nick); if (!nick) { safeSave(this.session, { 'connection_status': core_converse.ROOMSTATUS.NICKNAME_REQUIRED }); if (core_api.settings.get('muc_show_logs_before_join')) { await this.fetchMessages(); } return this; } core_api.send(await this.constructJoinPresence(password)); return this; }, /** * Clear stale cache and re-join a MUC we've been in before. * @private * @method _converse.ChatRoom#rejoin */ rejoin() { this.session.save('connection_status', core_converse.ROOMSTATUS.DISCONNECTED); this.registerHandlers(); this.clearOccupantsCache(); return this.join(); }, async constructJoinPresence(password) { let stanza = $pres({ 'id': getUniqueId(), 'from': shared_converse.connection.jid, 'to': this.getRoomJIDAndNick() }).c('x', { 'xmlns': Strophe.NS.MUC }).c('history', { 'maxstanzas': this.features.get('mam_enabled') ? 0 : core_api.settings.get('muc_history_max_stanzas') }).up(); password = password || this.get('password'); if (password) { stanza.cnode(Strophe.xmlElement('password', [], password)); } stanza.up(); // Go one level up, out of the `x` element. /** * *Hook* which allows plugins to update an outgoing MUC join presence stanza * @event _converse#constructedMUCPresence * @param { _converse.ChatRoom } - The MUC from which this message stanza is being sent. * @param { XMLElement } stanza - The stanza which will be sent out */ stanza = await core_api.hook('constructedMUCPresence', this, stanza); return stanza; }, clearOccupantsCache() { if (this.occupants.length) { // Remove non-members when reconnecting this.occupants.filter(o => !o.isMember()).forEach(o => o.destroy()); } else { // Looks like we haven't restored occupants from cache, so we clear it. this.occupants.clearStore(); } }, /** * Given the passed in MUC message, send a XEP-0333 chat marker. * @param { _converse.MUCMessage } msg * @param { ('received'|'displayed'|'acknowledged') } [type='displayed'] * @param { Boolean } force - Whether a marker should be sent for the * message, even if it didn't include a `markable` element. */ sendMarkerForMessage(msg) { let type = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 'displayed'; let force = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false; if (!msg || !core_api.settings.get('send_chat_markers').includes(type) || (msg === null || msg === void 0 ? void 0 : msg.get('type')) !== 'groupchat') { return; } if (msg !== null && msg !== void 0 && msg.get('is_markable') || force) { const key = `stanza_id ${this.get('jid')}`; const id = msg.get(key); if (!id) { headless_log.error(`Can't send marker for message without stanza ID: ${key}`); return; } const from_jid = Strophe.getBareJidFromJid(msg.get('from')); sendMarker(from_jid, id, type, msg.get('type')); } }, /** * Ensures that the user is subscribed to XEP-0437 Room Activity Indicators * if `muc_subscribe_to_rai` is set to `true`. * Only affiliated users can subscribe to RAI, but this method doesn't * check whether the current user is affiliated because it's intended to be * called after the MUC has been left and we don't have that information * anymore. * @private * @method _converse.ChatRoom#enableRAI */ enableRAI() { if (core_api.settings.get('muc_subscribe_to_rai')) { const muc_domain = Strophe.getDomainFromJid(this.get('jid')); core_api.user.presence.send(null, muc_domain, null, $build('rai', { 'xmlns': Strophe.NS.RAI })); } }, /** * Handler that gets called when the 'hidden' flag is toggled. * @private * @method _converse.ChatRoom#onHiddenChange */ async onHiddenChange() { const roomstatus = core_converse.ROOMSTATUS; const conn_status = this.session.get('connection_status'); if (this.get('hidden')) { if (conn_status === roomstatus.ENTERED && core_api.settings.get('muc_subscribe_to_rai') && this.getOwnAffiliation() !== 'none') { this.sendMarkerForLastMessage('received', true); await this.leave(); this.enableRAI(); } } else { if (conn_status === roomstatus.DISCONNECTED) { this.rejoin(); } this.clearUnreadMsgCounter(); } }, onOccupantAdded(occupant) { if (shared_converse.isInfoVisible(core_converse.MUC_TRAFFIC_STATES.ENTERED) && this.session.get('connection_status') === core_converse.ROOMSTATUS.ENTERED && occupant.get('show') === 'online') { this.updateNotifications(occupant.get('nick'), core_converse.MUC_TRAFFIC_STATES.ENTERED); } }, onOccupantRemoved(occupant) { if (shared_converse.isInfoVisible(core_converse.MUC_TRAFFIC_STATES.EXITED) && this.isEntered() && occupant.get('show') === 'online') { this.updateNotifications(occupant.get('nick'), core_converse.MUC_TRAFFIC_STATES.EXITED); } }, onOccupantShowChanged(occupant) { if (occupant.get('states').includes('303')) { return; } if (occupant.get('show') === 'offline' && shared_converse.isInfoVisible(core_converse.MUC_TRAFFIC_STATES.EXITED)) { this.updateNotifications(occupant.get('nick'), core_converse.MUC_TRAFFIC_STATES.EXITED); } else if (occupant.get('show') === 'online' && shared_converse.isInfoVisible(core_converse.MUC_TRAFFIC_STATES.ENTERED)) { this.updateNotifications(occupant.get('nick'), core_converse.MUC_TRAFFIC_STATES.ENTERED); } }, async onRoomEntered() { await this.occupants.fetchMembers(); if (core_api.settings.get('clear_messages_on_reconnection')) { await this.clearMessages(); } else { await this.fetchMessages(); } /** * Triggered when the user has entered a new MUC * @event _converse#enteredNewRoom * @type { _converse.ChatRoom} * @example _converse.api.listen.on('enteredNewRoom', model => { ... }); */ core_api.trigger('enteredNewRoom', this); if (core_api.settings.get('auto_register_muc_nickname') && (await core_api.disco.supports(Strophe.NS.MUC_REGISTER, this.get('jid')))) { this.registerNickname(); } }, async onConnectionStatusChanged() { if (this.isEntered()) { if (this.get('hidden') && core_api.settings.get('muc_subscribe_to_rai') && this.getOwnAffiliation() !== 'none') { try { await this.leave(); } catch (e) { headless_log.error(e); } this.enableRAI(); } else { await this.onRoomEntered(); } } }, async onReconnection() { await this.rejoin(); this.announceReconnection(); }, getMessagesCollection() { return new shared_converse.ChatRoomMessages(); }, restoreSession() { const id = `muc.session-${shared_converse.bare_jid}-${this.get('jid')}`; this.session = new MUCSession({ id }); initStorage(this.session, id, 'session'); return new Promise(r => this.session.fetch({ 'success': r, 'error': r })); }, initDiscoModels() { let id = `converse.muc-features-${shared_converse.bare_jid}-${this.get('jid')}`; this.features = new Model(Object.assign({ id }, lodash_es_zipObject(core_converse.ROOM_FEATURES, core_converse.ROOM_FEATURES.map(() => false)))); this.features.browserStorage = shared_converse.createStore(id, 'session'); this.features.listenTo(shared_converse, 'beforeLogout', () => this.features.browserStorage.flush()); id = `converse.muc-config-${shared_converse.bare_jid}-${this.get('jid')}`; this.config = new Model({ id }); this.config.browserStorage = shared_converse.createStore(id, 'session'); this.config.listenTo(shared_converse, 'beforeLogout', () => this.config.browserStorage.flush()); }, initOccupants() { this.occupants = new shared_converse.ChatRoomOccupants(); const id = `converse.occupants-${shared_converse.bare_jid}${this.get('jid')}`; this.occupants.browserStorage = shared_converse.createStore(id, 'session'); this.occupants.chatroom = this; this.occupants.listenTo(shared_converse, 'beforeLogout', () => this.occupants.browserStorage.flush()); }, fetchOccupants() { this.occupants.fetched = new Promise(resolve => { this.occupants.fetch({ 'add': true, 'silent': true, 'success': resolve, 'error': resolve }); }); return this.occupants.fetched; }, handleAffiliationChangedMessage(stanza) { const item = sizzle_default()(`x[xmlns="${Strophe.NS.MUC_USER}"] item`, stanza).pop(); if (item) { const from = stanza.getAttribute('from'); const type = stanza.getAttribute('type'); const affiliation = item.getAttribute('affiliation'); const jid = item.getAttribute('jid'); const data = { from, type, affiliation, 'states': [], 'show': type == 'unavailable' ? 'offline' : 'online', 'role': item.getAttribute('role'), 'jid': Strophe.getBareJidFromJid(jid), 'resource': Strophe.getResourceFromJid(jid) }; const occupant = this.occupants.findOccupant({ 'jid': data.jid }); if (occupant) { occupant.save(data); } else { this.occupants.create(data); } } }, async handleErrorMessageStanza(stanza) { const { __ } = shared_converse; const attrs = await parseMUCMessage(stanza, this, shared_converse); if (!(await this.shouldShowErrorMessage(attrs))) { return; } const message = this.getMessageReferencedByError(attrs); if (message) { const new_attrs = { 'error': attrs.error, 'error_condition': attrs.error_condition, 'error_text': attrs.error_text, 'error_type': attrs.error_type, 'editable': false }; if (attrs.msgid === message.get('retraction_id')) { // The error message refers to a retraction new_attrs.retracted = undefined; new_attrs.retraction_id = undefined; new_attrs.retracted_id = undefined; if (!attrs.error) { if (attrs.error_condition === 'forbidden') { new_attrs.error = __("You're not allowed to retract your message."); } else if (attrs.error_condition === 'not-acceptable') { new_attrs.error = __("Your retraction was not delivered because you're not present in the groupchat."); } else { new_attrs.error = __('Sorry, an error occurred while trying to retract your message.'); } } } else if (!attrs.error) { if (attrs.error_condition === 'forbidden') { new_attrs.error = __("Your message was not delivered because you weren't allowed to send it."); } else if (attrs.error_condition === 'not-acceptable') { new_attrs.error = __("Your message was not delivered because you're not present in the groupchat."); } else { new_attrs.error = __('Sorry, an error occurred while trying to send your message.'); } } message.save(new_attrs); } else { this.createMessage(attrs); } }, /** * Handles incoming message stanzas from the service that hosts this MUC * @private * @method _converse.ChatRoom#handleMessageFromMUCHost * @param { XMLElement } stanza */ handleMessageFromMUCHost(stanza) { if (this.isEntered()) { // We're not interested in activity indicators when already joined to the room return; } const rai = sizzle_default()(`rai[xmlns="${Strophe.NS.RAI}"]`, stanza).pop(); const active_mucs = Array.from((rai === null || rai === void 0 ? void 0 : rai.querySelectorAll('activity')) || []).map(m => m.textContent); if (active_mucs.includes(this.get('jid'))) { this.save({ 'has_activity': true, 'num_unread_general': 0 // Either/or between activity and unreads }); } }, /** * Handles XEP-0452 MUC Mention Notification messages * @private * @method _converse.ChatRoom#handleForwardedMentions * @param { XMLElement } stanza */ handleForwardedMentions(stanza) { if (this.isEntered()) { // Avoid counting mentions twice return; } const msgs = sizzle_default()(`mentions[xmlns="${Strophe.NS.MENTIONS}"] forwarded[xmlns="${Strophe.NS.FORWARD}"] message[type="groupchat"]`, stanza); const muc_jid = this.get('jid'); const mentions = msgs.filter(m => Strophe.getBareJidFromJid(m.getAttribute('from')) === muc_jid); if (mentions.length) { this.save({ 'has_activity': true, 'num_unread': this.get('num_unread') + mentions.length }); mentions.forEach(async stanza => { const attrs = await parseMUCMessage(stanza, this, shared_converse); const data = { stanza, attrs, 'chatbox': this }; core_api.trigger('message', data); }); } }, /** * Parses an incoming message stanza and queues it for processing. * @private * @method _converse.ChatRoom#handleMessageStanza * @param { XMLElement } stanza */ async handleMessageStanza(stanza) { const type = stanza.getAttribute('type'); if (type === 'error') { return this.handleErrorMessageStanza(stanza); } if (type === 'groupchat') { if (isArchived(stanza)) { // MAM messages are handled in converse-mam. // We shouldn't get MAM messages here because // they shouldn't have a `type` attribute. return headless_log.warn(`Received a MAM message with type "groupchat"`); } this.createInfoMessages(stanza); this.fetchFeaturesIfConfigurationChanged(stanza); } else if (!type) { return this.handleForwardedMentions(stanza); } /** * @typedef { Object } MUCMessageData * An object containing the parsed {@link MUCMessageAttributes} and * current {@link ChatRoom}. * @property { MUCMessageAttributes } attrs * @property { ChatRoom } chatbox */ let attrs; try { attrs = await parseMUCMessage(stanza, this, shared_converse); } catch (e) { return headless_log.error(e); } const data = { stanza, attrs, 'chatbox': this }; /** * Triggered when a groupchat message stanza has been received and parsed. * @event _converse#message * @type { object } * @property { module:converse-muc~MUCMessageData } data */ core_api.trigger('message', data); return attrs && this.queueMessage(attrs); }, /** * Register presence and message handlers relevant to this groupchat * @private * @method _converse.ChatRoom#registerHandlers */ registerHandlers() { const muc_jid = this.get('jid'); const muc_domain = Strophe.getDomainFromJid(muc_jid); this.removeHandlers(); this.presence_handler = shared_converse.connection.addHandler(stanza => this.onPresence(stanza) || true, null, 'presence', null, null, muc_jid, { 'ignoreNamespaceFragment': true, 'matchBareFromJid': true }); this.domain_presence_handler = shared_converse.connection.addHandler(stanza => this.onPresenceFromMUCHost(stanza) || true, null, 'presence', null, null, muc_domain); this.message_handler = shared_converse.connection.addHandler(stanza => !!this.handleMessageStanza(stanza) || true, null, 'message', null, null, muc_jid, { 'matchBareFromJid': true }); this.domain_message_handler = shared_converse.connection.addHandler(stanza => this.handleMessageFromMUCHost(stanza) || true, null, 'message', null, null, muc_domain); this.affiliation_message_handler = shared_converse.connection.addHandler(stanza => this.handleAffiliationChangedMessage(stanza) || true, Strophe.NS.MUC_USER, 'message', null, null, muc_jid); }, removeHandlers() { // Remove the presence and message handlers that were // registered for this groupchat. if (this.message_handler) { shared_converse.connection && shared_converse.connection.deleteHandler(this.message_handler); delete this.message_handler; } if (this.domain_message_handler) { shared_converse.connection && shared_converse.connection.deleteHandler(this.domain_message_handler); delete this.domain_message_handler; } if (this.presence_handler) { shared_converse.connection && shared_converse.connection.deleteHandler(this.presence_handler); delete this.presence_handler; } if (this.domain_presence_handler) { shared_converse.connection && shared_converse.connection.deleteHandler(this.domain_presence_handler); delete this.domain_presence_handler; } if (this.affiliation_message_handler) { shared_converse.connection && shared_converse.connection.deleteHandler(this.affiliation_message_handler); delete this.affiliation_message_handler; } return this; }, invitesAllowed() { return core_api.settings.get('allow_muc_invitations') && (this.features.get('open') || this.getOwnAffiliation() === 'owner'); }, getDisplayName() { const name = this.get('name'); if (name) { return name; } else if (core_api.settings.get('locked_muc_domain') === 'hidden') { return Strophe.getNodeFromJid(this.get('jid')); } else { return this.get('jid'); } }, /** * Sends a message stanza to the XMPP server and expects a reflection * or error message within a specific timeout period. * @private * @method _converse.ChatRoom#sendTimedMessage * @param { _converse.Message|XMLElement } message * @returns { Promise|Promise<_converse.TimeoutError> } Returns a promise * which resolves with the reflected message stanza or with an error stanza or {@link _converse.TimeoutError}. */ sendTimedMessage(el) { if (typeof el.tree === 'function') { el = el.tree(); } let id = el.getAttribute('id'); if (!id) { // inject id if not found id = this.getUniqueId('sendIQ'); el.setAttribute('id', id); } const promise = getOpenPromise(); const timeoutHandler = shared_converse.connection.addTimedHandler(shared_converse.STANZA_TIMEOUT, () => { shared_converse.connection.deleteHandler(handler); const err = new shared_converse.TimeoutError('Timeout Error: No response from server'); promise.resolve(err); return false; }); const handler = shared_converse.connection.addHandler(stanza => { timeoutHandler && shared_converse.connection.deleteTimedHandler(timeoutHandler); promise.resolve(stanza); }, null, 'message', ['error', 'groupchat'], id); core_api.send(el); return promise; }, /** * Retract one of your messages in this groupchat * @private * @method _converse.ChatRoom#retractOwnMessage * @param { _converse.Message } message - The message which we're retracting. */ async retractOwnMessage(message) { const __ = shared_converse.__; const origin_id = message.get('origin_id'); if (!origin_id) { throw new Error("Can't retract message without a XEP-0359 Origin ID"); } const editable = message.get('editable'); const stanza = $msg({ 'id': getUniqueId(), 'to': this.get('jid'), 'type': 'groupchat' }).c('store', { xmlns: Strophe.NS.HINTS }).up().c('apply-to', { 'id': origin_id, 'xmlns': Strophe.NS.FASTEN }).c('retract', { xmlns: Strophe.NS.RETRACT }); // Optimistic save message.set({ 'retracted': new Date().toISOString(), 'retracted_id': origin_id, 'retraction_id': stanza.nodeTree.getAttribute('id'), 'editable': false }); const result = await this.sendTimedMessage(stanza); if (utils_form.isErrorStanza(result)) { headless_log.error(result); } else if (result instanceof shared_converse.TimeoutError) { headless_log.error(result); message.save({ editable, 'error_type': 'timeout', 'error': __('A timeout happened while while trying to retract your message.'), 'retracted': undefined, 'retracted_id': undefined, 'retraction_id': undefined }); } }, /** * Retract someone else's message in this groupchat. * @private * @method _converse.ChatRoom#retractOtherMessage * @param { _converse.Message } message - The message which we're retracting. * @param { string } [reason] - The reason for retracting the message. * @example * const room = await api.rooms.get(jid); * const message = room.messages.findWhere({'body': 'Get rich quick!'}); * room.retractOtherMessage(message, 'spam'); */ async retractOtherMessage(message, reason) { const editable = message.get('editable'); // Optimistic save message.save({ 'moderated': 'retracted', 'moderated_by': shared_converse.bare_jid, 'moderated_id': message.get('msgid'), 'moderation_reason': reason, 'editable': false }); const result = await this.sendRetractionIQ(message, reason); if (result === null || utils_form.isErrorStanza(result)) { // Undo the save if something went wrong message.save({ editable, 'moderated': undefined, 'moderated_by': undefined, 'moderated_id': undefined, 'moderation_reason': undefined }); } return result; }, /** * Sends an IQ stanza to the XMPP server to retract a message in this groupchat. * @private * @method _converse.ChatRoom#sendRetractionIQ * @param { _converse.Message } message - The message which we're retracting. * @param { string } [reason] - The reason for retracting the message. */ sendRetractionIQ(message, reason) { const iq = $iq({ 'to': this.get('jid'), 'type': 'set' }).c('apply-to', { 'id': message.get(`stanza_id ${this.get('jid')}`), 'xmlns': Strophe.NS.FASTEN }).c('moderate', { xmlns: Strophe.NS.MODERATE }).c('retract', { xmlns: Strophe.NS.RETRACT }).up().c('reason').t(reason || ''); return core_api.sendIQ(iq, null, false); }, /** * Sends an IQ stanza to the XMPP server to destroy this groupchat. Not * to be confused with the {@link _converse.ChatRoom#destroy} * method, which simply removes the room from the local browser storage cache. * @private * @method _converse.ChatRoom#sendDestroyIQ * @param { string } [reason] - The reason for destroying the groupchat. * @param { string } [new_jid] - The JID of the new groupchat which replaces this one. */ sendDestroyIQ(reason, new_jid) { const destroy = $build('destroy'); if (new_jid) { destroy.attrs({ 'jid': new_jid }); } const iq = $iq({ 'to': this.get('jid'), 'type': 'set' }).c('query', { 'xmlns': Strophe.NS.MUC_OWNER }).cnode(destroy.node); if (reason && reason.length > 0) { iq.c('reason', reason); } return core_api.sendIQ(iq); }, /** * Leave the groupchat. * @private * @method _converse.ChatRoom#leave * @param { string } [exit_msg] - Message to indicate your reason for leaving */ async leave(exit_msg) { var _converse$disco_entit; core_api.connection.connected() && core_api.user.presence.send('unavailable', this.getRoomJIDAndNick(), exit_msg); // Delete the features model if (this.features) { await new Promise(resolve => this.features.destroy({ 'success': resolve, 'error': (_, e) => { headless_log.error(e); resolve(); } })); } // Delete disco entity const disco_entity = (_converse$disco_entit = shared_converse.disco_entities) === null || _converse$disco_entit === void 0 ? void 0 : _converse$disco_entit.get(this.get('jid')); if (disco_entity) { await new Promise(resolve => disco_entity.destroy({ 'success': resolve, 'error': (_, e) => { headless_log.error(e); resolve(); } })); } safeSave(this.session, { 'connection_status': core_converse.ROOMSTATUS.DISCONNECTED }); }, async close(ev) { safeSave(this.session, { 'connection_status': core_converse.ROOMSTATUS.CLOSING }); this.sendMarkerForLastMessage('received', true); await this.unregisterNickname(); await this.leave(); this.occupants.clearStore(); if ((ev === null || ev === void 0 ? void 0 : ev.name) !== 'closeAllChatBoxes' && core_api.settings.get('muc_clear_messages_on_leave')) { this.clearMessages(); } // Delete the session model await new Promise(resolve => this.session.destroy({ 'success': resolve, 'error': (_, e) => { headless_log.error(e); resolve(); } })); return shared_converse.ChatBox.prototype.close.call(this); }, canModerateMessages() { const self = this.getOwnOccupant(); return self && self.isModerator() && core_api.disco.supports(Strophe.NS.MODERATE, this.get('jid')); }, /** * Return an array of unique nicknames based on all occupants and messages in this MUC. * @private * @method _converse.ChatRoom#getAllKnownNicknames * @returns { String[] } */ getAllKnownNicknames() { return [...new Set([...this.occupants.map(o => o.get('nick')), ...this.messages.map(m => m.get('nick'))])].filter(n => n); }, getAllKnownNicknamesRegex() { const longNickString = this.getAllKnownNicknames().map(n => parse_helpers.escapeRegexString(n)).join('|'); return RegExp(`(?:\\p{P}|\\p{Z}|^)@(${longNickString})(?![\\w@-])`, 'uig'); }, getOccupantByJID(jid) { return this.occupants.findOccupant({ jid }); }, getOccupantByNickname(nick) { return this.occupants.findOccupant({ nick }); }, /** * Given a text message, look for `@` mentions and turn them into * XEP-0372 references * @param { String } text */ parseTextForReferences(text) { const mentions_regex = /(\p{P}|\p{Z}|^)([@][\w_-]+(?:\.\w+)*)/giu; if (!text || !mentions_regex.test(text)) { return [text, []]; } const getMatchingNickname = parse_helpers.findFirstMatchInArray(this.getAllKnownNicknames()); const uriFromNickname = nickname => { const jid = this.get('jid'); const occupant = this.getOccupant(nickname) || this.getOccupant(jid); const uri = this.features.get('nonanonymous') && (occupant === null || occupant === void 0 ? void 0 : occupant.get('jid')) || `${jid}/${nickname}`; return encodeURI(`xmpp:${uri}`); }; const matchToReference = match => { let at_sign_index = match[0].indexOf('@'); if (match[0][at_sign_index + 1] === '@') { // edge-case at_sign_index += 1; } const begin = match.index + at_sign_index; const end = begin + match[0].length - at_sign_index; const value = getMatchingNickname(match[1]); const type = 'mention'; const uri = uriFromNickname(value); return { begin, end, value, type, uri }; }; const regex = this.getAllKnownNicknamesRegex(); const mentions = [...text.matchAll(regex)].filter(m => !m[0].startsWith('/')); const references = mentions.map(matchToReference); const [updated_message, updated_references] = parse_helpers.reduceTextFromReferences(text, references); return [updated_message, updated_references]; }, async getOutgoingMessageAttributes(attrs) { var _attrs; const is_spoiler = this.get('composing_spoiler'); let text = '', references; if ((_attrs = attrs) !== null && _attrs !== void 0 && _attrs.body) { [text, references] = this.parseTextForReferences(attrs.body); } const origin_id = getUniqueId(); const body = text ? utils_form.httpToGeoUri(utils_form.shortnamesToUnicode(text), shared_converse) : undefined; attrs = Object.assign({}, attrs, { body, is_spoiler, origin_id, references, 'id': origin_id, 'msgid': origin_id, 'from': `${this.get('jid')}/${this.get('nick')}`, 'fullname': this.get('nick'), 'is_only_emojis': text ? utils_form.isOnlyEmojis(text) : false, 'message': body, 'nick': this.get('nick'), 'sender': 'me', 'type': 'groupchat' }, getMediaURLsMetadata(text)); /** * *Hook* which allows plugins to update the attributes of an outgoing * message. * @event _converse#getOutgoingMessageAttributes */ attrs = await core_api.hook('getOutgoingMessageAttributes', this, attrs); return attrs; }, /** * Utility method to construct the JID for the current user as occupant of the groupchat. * @private * @method _converse.ChatRoom#getRoomJIDAndNick * @returns {string} - The groupchat JID with the user's nickname added at the end. * @example groupchat@conference.example.org/nickname */ getRoomJIDAndNick() { const nick = this.get('nick'); const jid = Strophe.getBareJidFromJid(this.get('jid')); return jid + (nick !== null ? `/${nick}` : ''); }, /** * Sends a message with the current XEP-0085 chat state of the user * as taken from the `chat_state` attribute of the {@link _converse.ChatRoom}. * @private * @method _converse.ChatRoom#sendChatState */ sendChatState() { if (!core_api.settings.get('send_chat_state_notifications') || !this.get('chat_state') || !this.isEntered() || this.features.get('moderated') && this.getOwnRole() === 'visitor') { return; } const allowed = core_api.settings.get('send_chat_state_notifications'); if (Array.isArray(allowed) && !allowed.includes(this.get('chat_state'))) { return; } const chat_state = this.get('chat_state'); if (chat_state === shared_converse.GONE) { // is not applicable within MUC context return; } core_api.send($msg({ 'to': this.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 })); }, /** * Send a direct invitation as per XEP-0249 * @private * @method _converse.ChatRoom#directInvite * @param { String } recipient - JID of the person being invited * @param { String } [reason] - Reason for the invitation */ directInvite(recipient, reason) { if (this.features.get('membersonly')) { // When inviting to a members-only groupchat, we first add // the person to the member list by giving them an // affiliation of 'member' otherwise they won't be able to join. this.updateMemberLists([{ 'jid': recipient, 'affiliation': 'member', 'reason': reason }]); } const attrs = { 'xmlns': 'jabber:x:conference', 'jid': this.get('jid') }; if (reason !== null) { attrs.reason = reason; } if (this.get('password')) { attrs.password = this.get('password'); } const invitation = $msg({ 'from': shared_converse.connection.jid, 'to': recipient, 'id': getUniqueId() }).c('x', attrs); core_api.send(invitation); /** * After the user has sent out a direct invitation (as per XEP-0249), * to a roster contact, asking them to join a room. * @event _converse#chatBoxMaximized * @type {object} * @property {_converse.ChatRoom} room * @property {string} recipient - The JID of the person being invited * @property {string} reason - The original reason for the invitation * @example _converse.api.listen.on('chatBoxMaximized', view => { ... }); */ core_api.trigger('roomInviteSent', { 'room': this, 'recipient': recipient, 'reason': reason }); }, /** * Refresh the disco identity, features and fields for this {@link _converse.ChatRoom}. * *features* are stored on the features {@link Model} attribute on this {@link _converse.ChatRoom}. * *fields* are stored on the config {@link Model} attribute on this {@link _converse.ChatRoom}. * @private * @returns {Promise} */ refreshDiscoInfo() { return core_api.disco.refresh(this.get('jid')).then(() => this.getDiscoInfo()).catch(e => headless_log.error(e)); }, /** * Fetch the *extended* MUC info from the server and cache it locally * https://xmpp.org/extensions/xep-0045.html#disco-roominfo * @private * @method _converse.ChatRoom#getDiscoInfo * @returns {Promise} */ getDiscoInfo() { return core_api.disco.getIdentity('conference', 'text', this.get('jid')).then(identity => this.save({ 'name': identity === null || identity === void 0 ? void 0 : identity.get('name') })).then(() => this.getDiscoInfoFields()).then(() => this.getDiscoInfoFeatures()).catch(e => headless_log.error(e)); }, /** * Fetch the *extended* MUC info fields from the server and store them locally * in the `config` {@link Model} attribute. * See: https://xmpp.org/extensions/xep-0045.html#disco-roominfo * @private * @method _converse.ChatRoom#getDiscoInfoFields * @returns {Promise} */ async getDiscoInfoFields() { const fields = await core_api.disco.getFields(this.get('jid')); const config = fields.reduce((config, f) => { const name = f.get('var'); if (name !== null && name !== void 0 && name.startsWith('muc#roominfo_')) { config[name.replace('muc#roominfo_', '')] = f.get('value'); } return config; }, {}); this.config.save(config); }, /** * Use converse-disco to populate the features {@link Model} which * is stored as an attibute on this {@link _converse.ChatRoom}. * The results may be cached. If you want to force fetching the features from the * server, call {@link _converse.ChatRoom#refreshDiscoInfo} instead. * @private * @returns {Promise} */ async getDiscoInfoFeatures() { const features = await core_api.disco.getFeatures(this.get('jid')); const attrs = Object.assign(lodash_es_zipObject(core_converse.ROOM_FEATURES, core_converse.ROOM_FEATURES.map(() => false)), { 'fetched': new Date().toISOString() }); features.each(feature => { const fieldname = feature.get('var'); if (!fieldname.startsWith('muc_')) { if (fieldname === Strophe.NS.MAM) { attrs.mam_enabled = true; } else { attrs[fieldname] = true; } return; } attrs[fieldname.replace('muc_', '')] = true; }); this.features.save(attrs); }, /** * Given a element, return a copy with a child if * we can find a value for it in this rooms config. * @private * @method _converse.ChatRoom#addFieldValue * @returns { Element } */ addFieldValue(field) { const type = field.getAttribute('type'); if (type === 'fixed') { return field; } const fieldname = field.getAttribute('var').replace('muc#roomconfig_', ''); const config = this.get('roomconfig'); if (fieldname in config) { let values; switch (type) { case 'boolean': values = [config[fieldname] ? 1 : 0]; break; case 'list-multi': values = config[fieldname]; break; default: values = [config[fieldname]]; } field.innerHTML = values.map(v => $build('value').t(v)).join(''); } return field; }, /** * Automatically configure the groupchat based on this model's * 'roomconfig' data. * @private * @method _converse.ChatRoom#autoConfigureChatRoom * @returns { Promise } * Returns a promise which resolves once a response IQ has * been received. */ async autoConfigureChatRoom() { const stanza = await this.fetchRoomConfiguration(); const fields = sizzle_default()('field', stanza); const configArray = fields.map(f => this.addFieldValue(f)); if (configArray.length) { return this.sendConfiguration(configArray); } }, /** * Send an IQ stanza to fetch the groupchat configuration data. * Returns a promise which resolves once the response IQ * has been received. * @private * @method _converse.ChatRoom#fetchRoomConfiguration * @returns { Promise } */ fetchRoomConfiguration() { return core_api.sendIQ($iq({ 'to': this.get('jid'), 'type': 'get' }).c('query', { xmlns: Strophe.NS.MUC_OWNER })); }, /** * Sends an IQ stanza with the groupchat configuration. * @private * @method _converse.ChatRoom#sendConfiguration * @param { Array } config - The groupchat configuration * @returns { Promise } - A promise which resolves with * the `result` stanza received from the XMPP server. */ sendConfiguration() { let config = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : []; const iq = $iq({ to: this.get('jid'), type: 'set' }).c('query', { xmlns: Strophe.NS.MUC_OWNER }).c('x', { xmlns: Strophe.NS.XFORM, type: 'submit' }); config.forEach(node => iq.cnode(node).up()); return core_api.sendIQ(iq); }, onCommandError(err) { const { __ } = shared_converse; headless_log.fatal(err); const message = __('Sorry, an error happened while running the command.') + ' ' + __("Check your browser's developer console for details."); this.createMessage({ message, 'type': 'error' }); }, getNickOrJIDFromCommandArgs(args) { const { __ } = shared_converse; if (utils_form.isValidJID(args.trim())) { return args.trim(); } if (!args.startsWith('@')) { args = '@' + args; } const [text, references] = this.parseTextForReferences(args); // eslint-disable-line no-unused-vars if (!references.length) { const message = __("Error: couldn't find a groupchat participant based on your arguments"); this.createMessage({ message, 'type': 'error' }); return; } if (references.length > 1) { const message = __('Error: found multiple groupchat participant based on your arguments'); this.createMessage({ message, 'type': 'error' }); return; } const nick_or_jid = references.pop().value; const reason = args.split(nick_or_jid, 2)[1]; if (reason && !reason.startsWith(' ')) { const message = __("Error: couldn't find a groupchat participant based on your arguments"); this.createMessage({ message, 'type': 'error' }); return; } return nick_or_jid; }, validateRoleOrAffiliationChangeArgs(command, args) { const { __ } = shared_converse; if (!args) { const message = __('Error: the "%1$s" command takes two arguments, the user\'s nickname and optionally a reason.', command); this.createMessage({ message, 'type': 'error' }); return false; } return true; }, getAllowedCommands() { let allowed_commands = ['clear', 'help', 'me', 'nick', 'register']; if (this.config.get('changesubject') || ['owner', 'admin'].includes(this.getOwnAffiliation())) { allowed_commands = [...allowed_commands, ...['subject', 'topic']]; } const occupant = this.occupants.findWhere({ 'jid': shared_converse.bare_jid }); if (this.verifyAffiliations(['owner'], occupant, false)) { allowed_commands = allowed_commands.concat(OWNER_COMMANDS).concat(ADMIN_COMMANDS); } else if (this.verifyAffiliations(['admin'], occupant, false)) { allowed_commands = allowed_commands.concat(ADMIN_COMMANDS); } if (this.verifyRoles(['moderator'], occupant, false)) { allowed_commands = allowed_commands.concat(MODERATOR_COMMANDS).concat(VISITOR_COMMANDS); } else if (!this.verifyRoles(['visitor', 'participant', 'moderator'], occupant, false)) { allowed_commands = allowed_commands.concat(VISITOR_COMMANDS); } allowed_commands.sort(); if (Array.isArray(core_api.settings.get('muc_disable_slash_commands'))) { return allowed_commands.filter(c => !core_api.settings.get('muc_disable_slash_commands').includes(c)); } else { return allowed_commands; } }, verifyAffiliations(affiliations, occupant) { let show_error = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : true; const { __ } = shared_converse; if (!Array.isArray(affiliations)) { throw new TypeError('affiliations must be an Array'); } if (!affiliations.length) { return true; } occupant = occupant || this.occupants.findWhere({ 'jid': shared_converse.bare_jid }); if (occupant) { const a = occupant.get('affiliation'); if (affiliations.includes(a)) { return true; } } if (show_error) { const message = __('Forbidden: you do not have the necessary affiliation in order to do that.'); this.createMessage({ message, 'type': 'error' }); } return false; }, verifyRoles(roles, occupant) { let show_error = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : true; const { __ } = shared_converse; if (!Array.isArray(roles)) { throw new TypeError('roles must be an Array'); } if (!roles.length) { return true; } occupant = occupant || this.occupants.findWhere({ 'jid': shared_converse.bare_jid }); if (occupant) { const role = occupant.get('role'); if (roles.includes(role)) { return true; } } if (show_error) { const message = __('Forbidden: you do not have the necessary role in order to do that.'); this.createMessage({ message, 'type': 'error', 'is_ephemeral': 20000 }); } return false; }, /** * Returns the `role` which the current user has in this MUC * @private * @method _converse.ChatRoom#getOwnRole * @returns { ('none'|'visitor'|'participant'|'moderator') } */ getOwnRole() { var _this$getOwnOccupant, _this$getOwnOccupant$; return (_this$getOwnOccupant = this.getOwnOccupant()) === null || _this$getOwnOccupant === void 0 ? void 0 : (_this$getOwnOccupant$ = _this$getOwnOccupant.attributes) === null || _this$getOwnOccupant$ === void 0 ? void 0 : _this$getOwnOccupant$.role; }, /** * Returns the `affiliation` which the current user has in this MUC * @private * @method _converse.ChatRoom#getOwnAffiliation * @returns { ('none'|'outcast'|'member'|'admin'|'owner') } */ getOwnAffiliation() { var _this$getOwnOccupant2, _this$getOwnOccupant3; return ((_this$getOwnOccupant2 = this.getOwnOccupant()) === null || _this$getOwnOccupant2 === void 0 ? void 0 : (_this$getOwnOccupant3 = _this$getOwnOccupant2.attributes) === null || _this$getOwnOccupant3 === void 0 ? void 0 : _this$getOwnOccupant3.affiliation) || 'none'; }, /** * Get the {@link _converse.ChatRoomOccupant} instance which * represents the current user. * @method _converse.ChatRoom#getOwnOccupant * @returns { _converse.ChatRoomOccupant } */ getOwnOccupant() { return this.occupants.getOwnOccupant(); }, /** * Send a presence stanza to update the user's nickname in this MUC. * @param { String } nick */ async setNickname(nick) { if (core_api.settings.get('auto_register_muc_nickname') && (await core_api.disco.supports(Strophe.NS.MUC_REGISTER, this.get('jid')))) { const old_nick = this.get('nick'); this.set({ nick }); try { await this.registerNickname(); } catch (e) { const { __ } = shared_converse; headless_log.error(e); const message = __("Error: couldn't register new nickname in members only room"); this.createMessage({ message, 'type': 'error', 'is_ephemeral': true }); this.set({ 'nick': old_nick }); return; } } const jid = Strophe.getBareJidFromJid(this.get('jid')); core_api.send($pres({ 'from': shared_converse.connection.jid, 'to': `${jid}/${nick}`, 'id': getUniqueId() }).tree()); }, /** * Send an IQ stanza to modify an occupant's role * @private * @method _converse.ChatRoom#setRole * @param { _converse.ChatRoomOccupant } occupant * @param { String } role * @param { String } reason * @param { function } onSuccess - callback for a succesful response * @param { function } onError - callback for an error response */ setRole(occupant, role, reason, onSuccess, onError) { const item = $build('item', { 'nick': occupant.get('nick'), role }); const iq = $iq({ 'to': this.get('jid'), 'type': 'set' }).c('query', { xmlns: Strophe.NS.MUC_ADMIN }).cnode(item.node); if (reason !== null) { iq.c('reason', reason); } return core_api.sendIQ(iq).then(onSuccess).catch(onError); }, /** * @private * @method _converse.ChatRoom#getOccupant * @param { String } nickname_or_jid - The nickname or JID of the occupant to be returned * @returns { _converse.ChatRoomOccupant } */ getOccupant(nickname_or_jid) { return utils_form.isValidJID(nickname_or_jid) ? this.getOccupantByJID(nickname_or_jid) : this.getOccupantByNickname(nickname_or_jid); }, /** * Return an array of occupant models that have the required role * @private * @method _converse.ChatRoom#getOccupantsWithRole * @param { String } role * @returns { _converse.ChatRoomOccupant[] } */ getOccupantsWithRole(role) { return this.getOccupantsSortedBy('nick').filter(o => o.get('role') === role).map(item => { return { 'jid': item.get('jid'), 'nick': item.get('nick'), 'role': item.get('role') }; }); }, /** * Return an array of occupant models that have the required affiliation * @private * @method _converse.ChatRoom#getOccupantsWithAffiliation * @param { String } affiliation * @returns { _converse.ChatRoomOccupant[] } */ getOccupantsWithAffiliation(affiliation) { return this.getOccupantsSortedBy('nick').filter(o => o.get('affiliation') === affiliation).map(item => { return { 'jid': item.get('jid'), 'nick': item.get('nick'), 'affiliation': item.get('affiliation') }; }); }, /** * Return an array of occupant models, sorted according to the passed-in attribute. * @private * @method _converse.ChatRoom#getOccupantsSortedBy * @param { String } attr - The attribute to sort the returned array by * @returns { _converse.ChatRoomOccupant[] } */ getOccupantsSortedBy(attr) { return Array.from(this.occupants.models).sort((a, b) => a.get(attr) < b.get(attr) ? -1 : a.get(attr) > b.get(attr) ? 1 : 0); }, /** * 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. * @private * @method _converse.ChatRoom#updateMemberLists * @param { object } members - Map of member jids and affiliations. * @returns { Promise } * 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. */ async updateMemberLists(members) { const muc_jid = this.get('jid'); const all_affiliations = ['member', 'admin', 'owner']; const aff_lists = await Promise.all(all_affiliations.map(a => getAffiliationList(a, muc_jid))); const old_members = aff_lists.reduce((acc, val) => utils_form.isErrorObject(val) ? acc : [...val, ...acc], []); await setAffiliations(muc_jid, computeAffiliationsDelta(true, false, members, old_members)); await this.occupants.fetchMembers(); }, /** * Given a nick name, save it to the model state, otherwise, look * for a server-side reserved nickname or default configured * nickname and if found, persist that to the model state. * @private * @method _converse.ChatRoom#getAndPersistNickname * @returns { Promise } A promise which resolves with the nickname */ async getAndPersistNickname(nick) { nick = nick || this.get('nick') || (await this.getReservedNick()) || shared_converse.getDefaultMUCNickname(); if (nick) safeSave(this, { nick }, { 'silent': true }); return nick; }, /** * Use service-discovery to ask the XMPP server whether * this user has a reserved nickname for this groupchat. * If so, we'll use that, otherwise we render the nickname form. * @private * @method _converse.ChatRoom#getReservedNick * @returns { Promise } A promise which resolves with the reserved nick or null */ async getReservedNick() { const stanza = $iq({ 'to': this.get('jid'), 'from': shared_converse.connection.jid, 'type': 'get' }).c('query', { 'xmlns': Strophe.NS.DISCO_INFO, 'node': 'x-roomuser-item' }); const result = await core_api.sendIQ(stanza, null, false); if (utils_form.isErrorObject(result)) { throw result; } // Result might be undefined due to a timeout const identity_el = result === null || result === void 0 ? void 0 : result.querySelector('query[node="x-roomuser-item"] identity'); return identity_el ? identity_el.getAttribute('name') : null; }, /** * Send an IQ stanza to the MUC to register this user's nickname. * This sets the user's affiliation to 'member' (if they weren't affiliated * before) and reserves the nickname for this user, thereby preventing other * users from using it in this MUC. * See https://xmpp.org/extensions/xep-0045.html#register * @private * @method _converse.ChatRoom#registerNickname */ async registerNickname() { const { __ } = shared_converse; const nick = this.get('nick'); const jid = this.get('jid'); let iq, err_msg; try { iq = await core_api.sendIQ($iq({ 'to': jid, 'type': 'get' }).c('query', { 'xmlns': Strophe.NS.MUC_REGISTER })); } catch (e) { if (sizzle_default()(`not-allowed[xmlns="${Strophe.NS.STANZAS}"]`, e).length) { err_msg = __("You're not allowed to register yourself in this groupchat."); } else if (sizzle_default()(`registration-required[xmlns="${Strophe.NS.STANZAS}"]`, e).length) { err_msg = __("You're not allowed to register in this groupchat because it's members-only."); } headless_log.error(e); return err_msg; } const required_fields = sizzle_default()('field required', iq).map(f => f.parentElement); if (required_fields.length > 1 && required_fields[0].getAttribute('var') !== 'muc#register_roomnick') { return headless_log.error(`Can't register the user register in the groupchat ${jid} due to the required fields`); } try { await core_api.sendIQ($iq({ 'to': jid, 'type': 'set' }).c('query', { 'xmlns': Strophe.NS.MUC_REGISTER }).c('x', { 'xmlns': Strophe.NS.XFORM, 'type': 'submit' }).c('field', { 'var': 'FORM_TYPE' }).c('value').t('http://jabber.org/protocol/muc#register').up().up().c('field', { 'var': 'muc#register_roomnick' }).c('value').t(nick)); } catch (e) { if (sizzle_default()(`service-unavailable[xmlns="${Strophe.NS.STANZAS}"]`, e).length) { err_msg = __("Can't register your nickname in this groupchat, it doesn't support registration."); } else if (sizzle_default()(`bad-request[xmlns="${Strophe.NS.STANZAS}"]`, e).length) { err_msg = __("Can't register your nickname in this groupchat, invalid data form supplied."); } headless_log.error(err_msg); headless_log.error(e); return err_msg; } }, /** * Check whether we should unregister the user from this MUC, and if so, * call { @link _converse.ChatRoom#sendUnregistrationIQ } * @method _converse.ChatRoom#unregisterNickname */ async unregisterNickname() { if (core_api.settings.get('auto_register_muc_nickname') === 'unregister') { try { if (await core_api.disco.supports(Strophe.NS.MUC_REGISTER, this.get('jid'))) { await this.sendUnregistrationIQ(); } } catch (e) { headless_log.error(e); } } }, /** * Send an IQ stanza to the MUC to unregister this user's nickname. * If the user had a 'member' affiliation, it'll be removed and their * nickname will no longer be reserved and can instead be used (and * registered) by other users. * @method _converse.ChatRoom#sendUnregistrationIQ */ sendUnregistrationIQ() { const iq = $iq({ 'to': this.get('jid'), 'type': 'set' }).c('query', { 'xmlns': Strophe.NS.MUC_REGISTER }).c('remove'); return core_api.sendIQ(iq).catch(e => headless_log.error(e)); }, /** * Given a presence stanza, update the occupant model based on its contents. * @private * @method _converse.ChatRoom#updateOccupantsOnPresence * @param { XMLElement } pres - The presence stanza */ updateOccupantsOnPresence(pres) { var _occupant$attributes, _occupant$attributes2; const data = parseMUCPresence(pres, this); if (data.type === 'error' || !data.jid && !data.nick && !data.occupant_id) { return true; } const occupant = this.occupants.findOccupant(data); // Destroy an unavailable occupant if this isn't a nick change operation and if they're not affiliated if (data.type === 'unavailable' && occupant && !data.states.includes(core_converse.MUC_NICK_CHANGED_CODE) && !['admin', 'owner', 'member'].includes(data['affiliation'])) { // Before destroying we set the new data, so that we can show the disconnection message occupant.set(data); occupant.destroy(); return; } const jid = data.jid || ''; const attributes = { ...data, 'jid': Strophe.getBareJidFromJid(jid) || (occupant === null || occupant === void 0 ? void 0 : (_occupant$attributes = occupant.attributes) === null || _occupant$attributes === void 0 ? void 0 : _occupant$attributes.jid), 'resource': Strophe.getResourceFromJid(jid) || (occupant === null || occupant === void 0 ? void 0 : (_occupant$attributes2 = occupant.attributes) === null || _occupant$attributes2 === void 0 ? void 0 : _occupant$attributes2.resource) }; if (occupant) { occupant.save(attributes); } else { this.occupants.create(attributes); } }, fetchFeaturesIfConfigurationChanged(stanza) { // 104: configuration change // 170: logging enabled // 171: logging disabled // 172: room no longer anonymous // 173: room now semi-anonymous // 174: room now fully anonymous const codes = ['104', '170', '171', '172', '173', '174']; if (sizzle_default()('status', stanza).filter(e => codes.includes(e.getAttribute('status'))).length) { this.refreshDiscoInfo(); } }, /** * Given two JIDs, which can be either user JIDs or MUC occupant JIDs, * determine whether they belong to the same user. * @private * @method _converse.ChatRoom#isSameUser * @param { String } jid1 * @param { String } jid2 * @returns { Boolean } */ isSameUser(jid1, jid2) { const bare_jid1 = Strophe.getBareJidFromJid(jid1); const bare_jid2 = Strophe.getBareJidFromJid(jid2); const resource1 = Strophe.getResourceFromJid(jid1); const resource2 = Strophe.getResourceFromJid(jid2); if (utils_form.isSameBareJID(jid1, jid2)) { if (bare_jid1 === this.get('jid')) { // MUC JIDs return resource1 === resource2; } else { return true; } } else { const occupant1 = bare_jid1 === this.get('jid') ? this.occupants.findOccupant({ 'nick': resource1 }) : this.occupants.findOccupant({ 'jid': bare_jid1 }); const occupant2 = bare_jid2 === this.get('jid') ? this.occupants.findOccupant({ 'nick': resource2 }) : this.occupants.findOccupant({ 'jid': bare_jid2 }); return occupant1 === occupant2; } }, async isSubjectHidden() { const jids = await core_api.user.settings.get('mucs_with_hidden_subject', []); return jids.includes(this.get('jid')); }, async toggleSubjectHiddenState() { const muc_jid = this.get('jid'); const jids = await core_api.user.settings.get('mucs_with_hidden_subject', []); if (jids.includes(this.get('jid'))) { core_api.user.settings.set('mucs_with_hidden_subject', jids.filter(jid => jid !== muc_jid)); } else { core_api.user.settings.set('mucs_with_hidden_subject', [...jids, muc_jid]); } }, /** * Handle a possible subject change and return `true` if so. * @private * @method _converse.ChatRoom#handleSubjectChange * @param { object } attrs - Attributes representing a received * message, as returned by {@link parseMUCMessage} */ async handleSubjectChange(attrs) { const __ = shared_converse.__; if (typeof attrs.subject === 'string' && !attrs.thread && !attrs.message) { // https://xmpp.org/extensions/xep-0045.html#subject-mod // ----------------------------------------------------- // The subject is changed by sending a message of type "groupchat" to the , // where the MUST contain a element that specifies the new subject but // MUST NOT contain a element (or a element). const subject = attrs.subject; const author = attrs.nick; safeSave(this, { 'subject': { author, 'text': attrs.subject || '' } }); if (!attrs.is_delayed && author) { const message = subject ? __('Topic set by %1$s', author) : __('Topic cleared by %1$s', author); const prev_msg = this.messages.last(); if ((prev_msg === null || prev_msg === void 0 ? void 0 : prev_msg.get('nick')) !== attrs.nick || (prev_msg === null || prev_msg === void 0 ? void 0 : prev_msg.get('type')) !== 'info' || (prev_msg === null || prev_msg === void 0 ? void 0 : prev_msg.get('message')) !== message) { this.createMessage({ message, 'nick': attrs.nick, 'type': 'info', 'is_ephemeral': true }); } if (await this.isSubjectHidden()) { this.toggleSubjectHiddenState(); } } return true; } return false; }, /** * Set the subject for this {@link _converse.ChatRoom} * @private * @method _converse.ChatRoom#setSubject * @param { String } value */ setSubject() { let value = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : ''; core_api.send($msg({ to: this.get('jid'), from: shared_converse.connection.jid, type: 'groupchat' }).c('subject', { xmlns: 'jabber:client' }).t(value).tree()); }, /** * Is this a chat state notification that can be ignored, * because it's old or because it's from us. * @private * @method _converse.ChatRoom#ignorableCSN * @param { Object } attrs - The message attributes */ ignorableCSN(attrs) { return attrs.chat_state && !attrs.body && (attrs.is_delayed || this.isOwnMessage(attrs)); }, /** * Determines whether the message is from ourselves by checking * the `from` attribute. Doesn't check the `type` attribute. * @private * @method _converse.ChatRoom#isOwnMessage * @param { Object|XMLElement|_converse.Message } msg * @returns { boolean } */ isOwnMessage(msg) { let from; if (lodash_es_isElement(msg)) { from = msg.getAttribute('from'); } else if (msg instanceof shared_converse.Message) { from = msg.get('from'); } else { from = msg.from; } return Strophe.getResourceFromJid(from) == this.get('nick'); }, getUpdatedMessageAttributes(message, attrs) { const new_attrs = shared_converse.ChatBox.prototype.getUpdatedMessageAttributes.call(this, message, attrs); new_attrs['from_muc'] = attrs['from_muc']; if (this.isOwnMessage(attrs)) { const stanza_id_keys = Object.keys(attrs).filter(k => k.startsWith('stanza_id')); Object.assign(new_attrs, lodash_es_pick(attrs, stanza_id_keys)); if (!message.get('received')) { new_attrs.received = new Date().toISOString(); } } return new_attrs; }, /** * Send a MUC-0410 MUC Self-Ping stanza to room to determine * whether we're still joined. * @async * @private * @method _converse.ChatRoom#isJoined * @returns {Promise} */ async isJoined() { const jid = this.get('jid'); const ping = $iq({ 'to': `${jid}/${this.get('nick')}`, 'type': 'get' }).c('ping', { 'xmlns': Strophe.NS.PING }); try { await core_api.sendIQ(ping); } catch (e) { if (e === null) { headless_log.warn(`isJoined: Timeout error while checking whether we're joined to MUC: ${jid}`); } else { headless_log.warn(`isJoined: Apparently we're no longer connected to MUC: ${jid}`); } return false; } return true; }, /** * Sends a status update presence (i.e. based on the `` element) * @method _converse.ChatRoom#sendStatusPresence * @param { String } type * @param { String } [status] - An optional status message * @param { Element[]|Strophe.Builder[]|Element|Strophe.Builder } [child_nodes] * Nodes(s) to be added as child nodes of the `presence` XML element. */ async sendStatusPresence(type, status, child_nodes) { if (this.session.get('connection_status') === core_converse.ROOMSTATUS.ENTERED) { const presence = await shared_converse.xmppstatus.constructPresence(type, this.getRoomJIDAndNick(), status); child_nodes === null || child_nodes === void 0 ? void 0 : child_nodes.map(c => (c === null || c === void 0 ? void 0 : c.tree()) ?? c).forEach(c => presence.cnode(c).up()); core_api.send(presence); } }, /** * Check whether we're still joined and re-join if not * @async * @method _converse.ChatRoom#rejoinIfNecessary */ async rejoinIfNecessary() { if (!(await this.isJoined())) { this.rejoin(); return true; } }, /** * @private * @method _converse.ChatRoom#shouldShowErrorMessage * @returns {Promise} */ async shouldShowErrorMessage(attrs) { if (attrs.error_type === 'Decryption') { if (attrs.error_message === "Message key not found. The counter was repeated or the key was not filled.") { // OMEMO message which we already decrypted before return false; } else if (attrs.error_condition === 'not-encrypted-for-this-device') { return false; } } else if (attrs.error_condition === 'not-acceptable' && (await this.rejoinIfNecessary())) { return false; } return shared_converse.ChatBox.prototype.shouldShowErrorMessage.call(this, attrs); }, /** * Looks whether we already have a moderation message for this * incoming message. If so, it's considered "dangling" because * it probably hasn't been applied to anything yet, given that * the relevant message is only coming in now. * @private * @method _converse.ChatRoom#findDanglingModeration * @param { object } attrs - Attributes representing a received * message, as returned by {@link parseMUCMessage} * @returns { _converse.ChatRoomMessage } */ findDanglingModeration(attrs) { if (!this.messages.length) { return null; } // Only look for dangling moderation if there are newer // messages than this one, since moderation come after. if (this.messages.last().get('time') > attrs.time) { // Search from latest backwards const messages = Array.from(this.messages.models); const stanza_id = attrs[`stanza_id ${this.get('jid')}`]; if (!stanza_id) { return null; } messages.reverse(); return messages.find(_ref => { let { attributes } = _ref; return attributes.moderated === 'retracted' && attributes.moderated_id === stanza_id && attributes.moderated_by; }); } }, /** * Handles message moderation based on the passed in attributes. * @private * @method _converse.ChatRoom#handleModeration * @param { object } attrs - Attributes representing a received * message, as returned by {@link parseMUCMessage} * @returns { Boolean } Returns `true` or `false` depending on * whether a message was moderated or not. */ async handleModeration(attrs) { const MODERATION_ATTRIBUTES = ['editable', 'moderated', 'moderated_by', 'moderated_id', 'moderation_reason']; if (attrs.moderated === 'retracted') { const query = {}; const key = `stanza_id ${this.get('jid')}`; query[key] = attrs.moderated_id; const message = this.messages.findWhere(query); if (!message) { attrs['dangling_moderation'] = true; await this.createMessage(attrs); return true; } message.save(lodash_es_pick(attrs, MODERATION_ATTRIBUTES)); return true; } else { // Check if we have dangling moderation message const message = this.findDanglingModeration(attrs); if (message) { const moderation_attrs = lodash_es_pick(message.attributes, MODERATION_ATTRIBUTES); const new_attrs = Object.assign({ 'dangling_moderation': false }, attrs, moderation_attrs); delete new_attrs['id']; // Delete id, otherwise a new cache entry gets created message.save(new_attrs); return true; } } return false; }, getNotificationsText() { const { __ } = shared_converse; const actors_per_state = this.notifications.toJSON(); const role_changes = core_api.settings.get('muc_show_info_messages').filter(role_change => core_converse.MUC_ROLE_CHANGES_LIST.includes(role_change)); const join_leave_events = core_api.settings.get('muc_show_info_messages').filter(join_leave_event => core_converse.MUC_TRAFFIC_STATES_LIST.includes(join_leave_event)); const states = [...core_converse.CHAT_STATES, ...join_leave_events, ...role_changes]; return states.reduce((result, state) => { const existing_actors = actors_per_state[state]; if (!(existing_actors !== null && existing_actors !== void 0 && existing_actors.length)) { return result; } const actors = existing_actors.map(a => { var _this$getOccupant; return ((_this$getOccupant = this.getOccupant(a)) === null || _this$getOccupant === void 0 ? void 0 : _this$getOccupant.getDisplayName()) || a; }); if (actors.length === 1) { if (state === 'composing') { return `${result}${__('%1$s is typing', actors[0])}\n`; } else if (state === 'paused') { return `${result}${__('%1$s has stopped typing', actors[0])}\n`; } else if (state === shared_converse.GONE) { return `${result}${__('%1$s has gone away', actors[0])}\n`; } else if (state === 'entered') { return `${result}${__('%1$s has entered the groupchat', actors[0])}\n`; } else if (state === 'exited') { return `${result}${__('%1$s has left the groupchat', actors[0])}\n`; } else if (state === 'op') { return `${result}${__('%1$s is now a moderator', actors[0])}\n`; } else if (state === 'deop') { return `${result}${__('%1$s is no longer a moderator', actors[0])}\n`; } else if (state === 'voice') { return `${result}${__('%1$s has been given a voice', actors[0])}\n`; } else if (state === 'mute') { return `${result}${__('%1$s has been muted', actors[0])}\n`; } } else if (actors.length > 1) { let actors_str; if (actors.length > 3) { actors_str = `${Array.from(actors).slice(0, 2).join(', ')} and others`; } else { const last_actor = actors.pop(); actors_str = __('%1$s and %2$s', actors.join(', '), last_actor); } if (state === 'composing') { return `${result}${__('%1$s are typing', actors_str)}\n`; } else if (state === 'paused') { return `${result}${__('%1$s have stopped typing', actors_str)}\n`; } else if (state === shared_converse.GONE) { return `${result}${__('%1$s have gone away', actors_str)}\n`; } else if (state === 'entered') { return `${result}${__('%1$s have entered the groupchat', actors_str)}\n`; } else if (state === 'exited') { return `${result}${__('%1$s have left the groupchat', actors_str)}\n`; } else if (state === 'op') { return `${result}${__('%1$s are now moderators', actors[0])}\n`; } else if (state === 'deop') { return `${result}${__('%1$s are no longer moderators', actors[0])}\n`; } else if (state === 'voice') { return `${result}${__('%1$s have been given voices', actors[0])}\n`; } else if (state === 'mute') { return `${result}${__('%1$s have been muted', actors[0])}\n`; } } return result; }, ''); }, /** * @param {String} actor - The nickname of the actor that caused the notification * @param {String|Array} states - The state or states representing the type of notificcation */ removeNotification(actor, states) { const actors_per_state = this.notifications.toJSON(); states = Array.isArray(states) ? states : [states]; states.forEach(state => { const existing_actors = Array.from(actors_per_state[state] || []); if (existing_actors.includes(actor)) { const idx = existing_actors.indexOf(actor); existing_actors.splice(idx, 1); this.notifications.set(state, Array.from(existing_actors)); } }); }, /** * Update the notifications model by adding the passed in nickname * to the array of nicknames that all match a particular state. * * Removes the nickname from any other states it might be associated with. * * The state can be a XEP-0085 Chat State or a XEP-0045 join/leave * state. * @param {String} actor - The nickname of the actor that causes the notification * @param {String} state - The state representing the type of notificcation */ updateNotifications(actor, state) { const actors_per_state = this.notifications.toJSON(); const existing_actors = actors_per_state[state] || []; if (existing_actors.includes(actor)) { return; } const reducer = (out, s) => { if (s === state) { out[s] = [...existing_actors, actor]; } else { out[s] = (actors_per_state[s] || []).filter(a => a !== actor); } return out; }; const actors_per_chat_state = core_converse.CHAT_STATES.reduce(reducer, {}); const actors_per_traffic_state = core_converse.MUC_TRAFFIC_STATES_LIST.reduce(reducer, {}); const actors_per_role_change = core_converse.MUC_ROLE_CHANGES_LIST.reduce(reducer, {}); this.notifications.set(Object.assign(actors_per_chat_state, actors_per_traffic_state, actors_per_role_change)); window.setTimeout(() => this.removeNotification(actor, state), 10000); }, handleMetadataFastening(attrs) { if (attrs.ogp_for_id) { if (attrs.from !== this.get('jid')) { // For now we only allow metadata from the MUC itself and not // from individual users who are deemed less trustworthy. return false; } const message = this.messages.findWhere({ 'origin_id': attrs.ogp_for_id }); if (message) { const old_list = message.get('ogp_metadata') || []; if (old_list.filter(m => m['og:url'] === attrs['og:url']).length) { // Don't add metadata for the same URL again return false; } const list = [...old_list, lodash_es_pick(attrs, METADATA_ATTRIBUTES)]; message.save('ogp_metadata', list); return true; } } return false; }, /** * Given {@link MessageAttributes} look for XEP-0316 Room Notifications and create info * messages for them. * @param { XMLElement } stanza */ handleMEPNotification(attrs) { var _attrs$activities; if (attrs.from !== this.get('jid') || !attrs.activities) { return false; } (_attrs$activities = attrs.activities) === null || _attrs$activities === void 0 ? void 0 : _attrs$activities.forEach(activity_attrs => { const data = Object.assign(attrs, activity_attrs); this.createMessage(data); // Trigger so that notifications are shown core_api.trigger('message', { 'attrs': data, 'chatbox': this }); }); return !!attrs.activities.length; }, /** * Returns an already cached message (if it exists) based on the * passed in attributes map. * @method _converse.ChatRoom#getDuplicateMessage * @param { object } attrs - Attributes representing a received * message, as returned by {@link parseMUCMessage} * @returns {Promise<_converse.Message>} */ getDuplicateMessage(attrs) { var _attrs$activities2; if ((_attrs$activities2 = attrs.activities) !== null && _attrs$activities2 !== void 0 && _attrs$activities2.length) { return this.messages.findWhere({ 'type': 'mep', 'msgid': attrs.msgid }); } else { return shared_converse.ChatBox.prototype.getDuplicateMessage.call(this, attrs); } }, /** * Handler for all MUC messages sent to this groupchat. This method * shouldn't be called directly, instead {@link _converse.ChatRoom#queueMessage} * should be called. * @method _converse.ChatRoom#onMessage * @param { MessageAttributes } attrs - A promise which resolves to the message attributes. */ async onMessage(attrs) { attrs = await attrs; if (utils_form.isErrorObject(attrs)) { attrs.stanza && headless_log.error(attrs.stanza); return headless_log.error(attrs.message); } else if (attrs.type === 'error' && !(await this.shouldShowErrorMessage(attrs))) { return; } const message = this.getDuplicateMessage(attrs); if (message) { message.get('type') === 'groupchat' && this.updateMessage(message, attrs); return; } else if (attrs.receipt_id || attrs.is_marker || this.ignorableCSN(attrs)) { return; } if (this.handleMetadataFastening(attrs) || this.handleMEPNotification(attrs) || (await this.handleRetraction(attrs)) || (await this.handleModeration(attrs)) || (await this.handleSubjectChange(attrs))) { attrs.nick && this.removeNotification(attrs.nick, ['composing', 'paused']); return; } this.setEditable(attrs, attrs.time); if (attrs['chat_state']) { this.updateNotifications(attrs.nick, attrs.chat_state); } if (utils_form.shouldCreateGroupchatMessage(attrs)) { const msg = this.handleCorrection(attrs) || (await this.createMessage(attrs)); this.removeNotification(attrs.nick, ['composing', 'paused']); this.handleUnreadMessage(msg); } }, handleModifyError(pres) { var _pres$querySelector; const text = (_pres$querySelector = pres.querySelector('error text')) === null || _pres$querySelector === void 0 ? void 0 : _pres$querySelector.textContent; if (text) { if (this.session.get('connection_status') === core_converse.ROOMSTATUS.CONNECTING) { this.setDisconnectionState(text); } else { const attrs = { 'type': 'error', 'message': text, 'is_ephemeral': true }; this.createMessage(attrs); } } }, /** * Handle a presence stanza that disconnects the user from the MUC * @param { XMLElement } stanza */ handleDisconnection(stanza) { var _item$querySelector; const is_self = stanza.querySelector("status[code='110']") !== null; const x = sizzle_default()(`x[xmlns="${Strophe.NS.MUC_USER}"]`, stanza).pop(); if (!x) { return; } const disconnection_codes = Object.keys(shared_converse.muc.disconnect_messages); const codes = sizzle_default()('status', x).map(s => s.getAttribute('code')).filter(c => disconnection_codes.includes(c)); const disconnected = is_self && codes.length > 0; if (!disconnected) { return; } // By using querySelector we assume here there is // one per // element. This appears to be a safe assumption, since // each element pertains to a single user. const item = x.querySelector('item'); const reason = item ? (_item$querySelector = item.querySelector('reason')) === null || _item$querySelector === void 0 ? void 0 : _item$querySelector.textContent : undefined; const actor = item ? lodash_es_invoke(item.querySelector('actor'), 'getAttribute', 'nick') : undefined; const message = shared_converse.muc.disconnect_messages[codes[0]]; const status = codes.includes('301') ? core_converse.ROOMSTATUS.BANNED : core_converse.ROOMSTATUS.DISCONNECTED; this.setDisconnectionState(message, reason, actor, status); }, getActionInfoMessage(code, nick, actor) { const __ = shared_converse.__; if (code === '301') { return actor ? __('%1$s has been banned by %2$s', nick, actor) : __('%1$s has been banned', nick); } else if (code === '303') { return __("%1$s's nickname has changed", nick); } else if (code === '307') { return actor ? __('%1$s has been kicked out by %2$s', nick, actor) : __('%1$s has been kicked out', nick); } else if (code === '321') { return __('%1$s has been removed because of an affiliation change', nick); } else if (code === '322') { return __('%1$s has been removed for not being a member', nick); } }, createAffiliationChangeMessage(occupant) { const __ = shared_converse.__; const previous_affiliation = occupant._previousAttributes.affiliation; if (!previous_affiliation) { // If no previous affiliation was set, then we don't // interpret this as an affiliation change. // For example, if muc_send_probes is true, then occupants // are created based on incoming messages, in which case // we don't yet know the affiliation return; } const current_affiliation = occupant.get('affiliation'); if (previous_affiliation === 'admin' && shared_converse.isInfoVisible(core_converse.AFFILIATION_CHANGES.EXADMIN)) { this.createMessage({ 'type': 'info', 'message': __('%1$s is no longer an admin of this groupchat', occupant.get('nick')) }); } else if (previous_affiliation === 'owner' && shared_converse.isInfoVisible(core_converse.AFFILIATION_CHANGES.EXOWNER)) { this.createMessage({ 'type': 'info', 'message': __('%1$s is no longer an owner of this groupchat', occupant.get('nick')) }); } else if (previous_affiliation === 'outcast' && shared_converse.isInfoVisible(core_converse.AFFILIATION_CHANGES.EXOUTCAST)) { this.createMessage({ 'type': 'info', 'message': __('%1$s is no longer banned from this groupchat', occupant.get('nick')) }); } if (current_affiliation === 'none' && previous_affiliation === 'member' && shared_converse.isInfoVisible(core_converse.AFFILIATION_CHANGES.EXMEMBER)) { this.createMessage({ 'type': 'info', 'message': __('%1$s is no longer a member of this groupchat', occupant.get('nick')) }); } if (current_affiliation === 'member' && shared_converse.isInfoVisible(core_converse.AFFILIATION_CHANGES.MEMBER)) { this.createMessage({ 'type': 'info', 'message': __('%1$s is now a member of this groupchat', occupant.get('nick')) }); } else if (current_affiliation === 'admin' && shared_converse.isInfoVisible(core_converse.AFFILIATION_CHANGES.ADMIN) || current_affiliation == 'owner' && shared_converse.isInfoVisible(core_converse.AFFILIATION_CHANGES.OWNER)) { // For example: AppleJack is now an (admin|owner) of this groupchat this.createMessage({ 'type': 'info', 'message': __('%1$s is now an %2$s of this groupchat', occupant.get('nick'), current_affiliation) }); } }, createRoleChangeMessage(occupant, changed) { if (changed === 'none' || occupant.changed.affiliation) { // We don't inform of role changes if they accompany affiliation changes. return; } const previous_role = occupant._previousAttributes.role; if (previous_role === 'moderator' && shared_converse.isInfoVisible(core_converse.MUC_ROLE_CHANGES.DEOP)) { this.updateNotifications(occupant.get('nick'), core_converse.MUC_ROLE_CHANGES.DEOP); } else if (previous_role === 'visitor' && shared_converse.isInfoVisible(core_converse.MUC_ROLE_CHANGES.VOICE)) { this.updateNotifications(occupant.get('nick'), core_converse.MUC_ROLE_CHANGES.VOICE); } if (occupant.get('role') === 'visitor' && shared_converse.isInfoVisible(core_converse.MUC_ROLE_CHANGES.MUTE)) { this.updateNotifications(occupant.get('nick'), core_converse.MUC_ROLE_CHANGES.MUTE); } else if (occupant.get('role') === 'moderator') { if (!['owner', 'admin'].includes(occupant.get('affiliation')) && shared_converse.isInfoVisible(core_converse.MUC_ROLE_CHANGES.OP)) { // Oly show this message if the user isn't already // an admin or owner, otherwise this isn't new information. this.updateNotifications(occupant.get('nick'), core_converse.MUC_ROLE_CHANGES.OP); } } }, /** * Create an info message based on a received MUC status code * @private * @method _converse.ChatRoom#createInfoMessage * @param { string } code - The MUC status code * @param { XMLElement } stanza - The original stanza that contains the code * @param { Boolean } is_self - Whether this stanza refers to our own presence */ createInfoMessage(code, stanza, is_self) { const __ = shared_converse.__; const data = { 'type': 'info', 'is_ephemeral': true }; if (!shared_converse.isInfoVisible(code)) { return; } if (code === '110' || code === '100' && !is_self) { return; } else if (code in shared_converse.muc.info_messages) { data.message = shared_converse.muc.info_messages[code]; } else if (!is_self && ACTION_INFO_CODES.includes(code)) { var _item$querySelector2, _item$querySelector3; const nick = Strophe.getResourceFromJid(stanza.getAttribute('from')); const item = sizzle_default()(`x[xmlns="${Strophe.NS.MUC_USER}"] item`, stanza).pop(); data.actor = item ? (_item$querySelector2 = item.querySelector('actor')) === null || _item$querySelector2 === void 0 ? void 0 : _item$querySelector2.getAttribute('nick') : undefined; data.reason = item ? (_item$querySelector3 = item.querySelector('reason')) === null || _item$querySelector3 === void 0 ? void 0 : _item$querySelector3.textContent : undefined; data.message = this.getActionInfoMessage(code, nick, data.actor); } else if (is_self && code in shared_converse.muc.new_nickname_messages) { // XXX: Side-effect of setting the nick. Should ideally be refactored out of this method let nick; if (is_self && code === '210') { nick = Strophe.getResourceFromJid(stanza.getAttribute('from')); } else if (is_self && code === '303') { nick = sizzle_default()(`x[xmlns="${Strophe.NS.MUC_USER}"] item`, stanza).pop().getAttribute('nick'); } this.save('nick', nick); data.message = __(shared_converse.muc.new_nickname_messages[code], nick); } if (data.message) { if (code === '201' && this.messages.findWhere(data)) { return; } this.createMessage(data); } }, /** * Create info messages based on a received presence or message stanza * @private * @method _converse.ChatRoom#createInfoMessages * @param { XMLElement } stanza */ createInfoMessages(stanza) { const codes = sizzle_default()(`x[xmlns="${Strophe.NS.MUC_USER}"] status`, stanza).map(s => s.getAttribute('code')); if (codes.includes('333') && codes.includes('307')) { // See: https://github.com/xsf/xeps/pull/969/files#diff-ac5113766e59219806793c1f7d967f1bR4966 codes.splice(codes.indexOf('307'), 1); } const is_self = codes.includes('110'); codes.forEach(code => this.createInfoMessage(code, stanza, is_self)); }, /** * Set parameters regarding disconnection from this room. This helps to * communicate to the user why they were disconnected. * @param { String } message - The disconnection message, as received from (or * implied by) the server. * @param { String } reason - The reason provided for the disconnection * @param { String } actor - The person (if any) responsible for this disconnection * @param { Integer } status - The status code (see `converse.ROOMSTATUS`) */ setDisconnectionState(message, reason, actor) { let status = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : core_converse.ROOMSTATUS.DISCONNECTED; this.session.save({ 'connection_status': status, 'disconnection_actor': actor, 'disconnection_message': message, 'disconnection_reason': reason }); }, onNicknameClash(presence) { const __ = shared_converse.__; if (core_api.settings.get('muc_nickname_from_jid')) { const nick = presence.getAttribute('from').split('/')[1]; if (nick === shared_converse.getDefaultMUCNickname()) { this.join(nick + '-2'); } else { const del = nick.lastIndexOf('-'); const num = nick.substring(del + 1, nick.length); this.join(nick.substring(0, del + 1) + String(Number(num) + 1)); } } else { this.save({ 'nickname_validation_message': __('The nickname you chose is reserved or ' + 'currently in use, please choose a different one.') }); this.session.save({ 'connection_status': core_converse.ROOMSTATUS.NICKNAME_REQUIRED }); } }, /** * Parses a stanza with type "error" and sets the proper * `connection_status` value for this {@link _converse.ChatRoom} as * well as any additional output that can be shown to the user. * @private * @param { XMLElement } stanza - The presence stanza */ onErrorPresence(stanza) { var _sizzle$pop; const __ = shared_converse.__; const error = stanza.querySelector('error'); const error_type = error.getAttribute('type'); const reason = (_sizzle$pop = sizzle_default()(`text[xmlns="${Strophe.NS.STANZAS}"]`, error).pop()) === null || _sizzle$pop === void 0 ? void 0 : _sizzle$pop.textContent; if (error_type === 'modify') { this.handleModifyError(stanza); } else if (error_type === 'auth') { if (sizzle_default()(`not-authorized[xmlns="${Strophe.NS.STANZAS}"]`, error).length) { this.save({ 'password_validation_message': reason || __('Password incorrect') }); this.session.save({ 'connection_status': core_converse.ROOMSTATUS.PASSWORD_REQUIRED }); } if (error.querySelector('registration-required')) { const message = __('You are not on the member list of this groupchat.'); this.setDisconnectionState(message, reason); } else if (error.querySelector('forbidden')) { this.setDisconnectionState(shared_converse.muc.disconnect_messages[301], reason, null, core_converse.ROOMSTATUS.BANNED); } } else if (error_type === 'cancel') { if (error.querySelector('not-allowed')) { const message = __('You are not allowed to create new groupchats.'); this.setDisconnectionState(message, reason); } else if (error.querySelector('not-acceptable')) { const message = __("Your nickname doesn't conform to this groupchat's policies."); this.setDisconnectionState(message, reason); } else if (sizzle_default()(`gone[xmlns="${Strophe.NS.STANZAS}"]`, error).length) { var _sizzle$pop2; const moved_jid = (_sizzle$pop2 = sizzle_default()(`gone[xmlns="${Strophe.NS.STANZAS}"]`, error).pop()) === null || _sizzle$pop2 === void 0 ? void 0 : _sizzle$pop2.textContent.replace(/^xmpp:/, '').replace(/\?join$/, ''); this.save({ moved_jid, 'destroyed_reason': reason }); this.session.save({ 'connection_status': core_converse.ROOMSTATUS.DESTROYED }); } else if (error.querySelector('conflict')) { this.onNicknameClash(stanza); } else if (error.querySelector('item-not-found')) { const message = __('This groupchat does not (yet) exist.'); this.setDisconnectionState(message, reason); } else if (error.querySelector('service-unavailable')) { const message = __('This groupchat has reached its maximum number of participants.'); this.setDisconnectionState(message, reason); } else if (error.querySelector('remote-server-not-found')) { const message = __('Remote server not found'); this.setDisconnectionState(message, reason); } else if (error.querySelector('forbidden')) { const message = __("You're not allowed to enter this groupchat"); this.setDisconnectionState(message, reason); } else { const message = __("An error happened while trying to enter this groupchat"); this.setDisconnectionState(message, reason); } } }, /** * Listens for incoming presence stanzas from the service that hosts this MUC * @private * @method _converse.ChatRoom#onPresenceFromMUCHost * @param { XMLElement } stanza - The presence stanza */ onPresenceFromMUCHost(stanza) { if (stanza.getAttribute('type') === 'error') { const error = stanza.querySelector('error'); if ((error === null || error === void 0 ? void 0 : error.getAttribute('type')) === 'wait' && error !== null && error !== void 0 && error.querySelector('resource-constraint')) { // If we get a error, we assume it's in context of XEP-0437 RAI. // We remove this MUC's host from the list of enabled domains and rejoin the MUC. if (this.session.get('connection_status') === core_converse.ROOMSTATUS.DISCONNECTED) { this.rejoin(); } } } }, /** * Handles incoming presence stanzas coming from the MUC * @private * @method _converse.ChatRoom#onPresence * @param { XMLElement } stanza */ onPresence(stanza) { if (stanza.getAttribute('type') === 'error') { return this.onErrorPresence(stanza); } this.createInfoMessages(stanza); if (stanza.querySelector("status[code='110']")) { this.onOwnPresence(stanza); if (this.getOwnRole() !== 'none' && this.session.get('connection_status') === core_converse.ROOMSTATUS.CONNECTING) { this.session.save('connection_status', core_converse.ROOMSTATUS.CONNECTED); } } else { this.updateOccupantsOnPresence(stanza); } }, /** * Handles a received presence relating to the current user. * * For locked groupchats (which are by definition "new"), the * groupchat will either be auto-configured or created instantly * (with default config) or a configuration groupchat will be * rendered. * * If the groupchat is not locked, then the groupchat will be * auto-configured only if applicable and if the current * user is the groupchat's owner. * @private * @method _converse.ChatRoom#onOwnPresence * @param { XMLElement } pres - The stanza */ async onOwnPresence(stanza) { await this.occupants.fetched; const old_status = this.session.get('connection_status'); if (stanza.getAttribute('type') !== 'unavailable' && old_status !== core_converse.ROOMSTATUS.ENTERED && old_status !== core_converse.ROOMSTATUS.CLOSING) { // Set connection_status before creating the occupant, but // only trigger afterwards, so that plugins can access the // occupant in their event handlers. this.session.save('connection_status', core_converse.ROOMSTATUS.ENTERED, { 'silent': true }); this.updateOccupantsOnPresence(stanza); this.session.trigger('change:connection_status', this.session, old_status); } else { this.updateOccupantsOnPresence(stanza); } if (stanza.getAttribute('type') === 'unavailable') { this.handleDisconnection(stanza); return; } else { const locked_room = stanza.querySelector("status[code='201']"); if (locked_room) { if (this.get('auto_configure')) { this.autoConfigureChatRoom().then(() => this.refreshDiscoInfo()); } else if (core_api.settings.get('muc_instant_rooms')) { // Accept default configuration this.sendConfiguration().then(() => this.refreshDiscoInfo()); } else { this.session.save({ 'view': core_converse.MUC.VIEWS.CONFIG }); return; } } else if (!this.features.get('fetched')) { // The features for this groupchat weren't fetched. // That must mean it's a new groupchat without locking // (in which case Prosody doesn't send a 201 status), // otherwise the features would have been fetched in // the "initialize" method already. if (this.getOwnAffiliation() === 'owner' && this.get('auto_configure')) { this.autoConfigureChatRoom().then(() => this.refreshDiscoInfo()); } else { this.getDiscoInfo(); } } } }, /** * Returns a boolean to indicate whether the current user * was mentioned in a message. * @private * @method _converse.ChatRoom#isUserMentioned * @param { String } - The text message */ isUserMentioned(message) { const nick = this.get('nick'); if (message.get('references').length) { const mentions = message.get('references').filter(ref => ref.type === 'mention').map(ref => ref.value); return mentions.includes(nick); } else { return new RegExp(`\\b${nick}\\b`).test(message.get('body')); } }, incrementUnreadMsgsCounter(message) { const settings = { 'num_unread_general': this.get('num_unread_general') + 1 }; if (this.get('num_unread_general') === 0) { settings['first_unread_id'] = message.get('id'); } if (this.isUserMentioned(message)) { settings.num_unread = this.get('num_unread') + 1; } this.save(settings); }, clearUnreadMsgCounter() { if (this.get('num_unread_general') > 0 || this.get('num_unread') > 0 || this.get('has_activity')) { this.sendMarkerForMessage(this.messages.last()); } safeSave(this, { 'has_activity': false, 'num_unread': 0, 'num_unread_general': 0 }); } }; /* harmony default export */ const muc = (ChatRoomMixin); ;// CONCATENATED MODULE: ./src/headless/plugins/muc/occupant.js /** * Represents a participant in a MUC * @class * @namespace _converse.ChatRoomOccupant * @memberOf _converse */ class ChatRoomOccupant extends Model { defaults() { // eslint-disable-line class-methods-use-this return { 'hats': [], 'show': 'offline', 'states': [] }; } save(key, val, options) { let attrs; if (key == null) { // eslint-disable-line no-eq-null return super.save(key, val, options); } else if (typeof key === 'object') { attrs = key; options = val; } else { (attrs = {})[key] = val; } if (attrs.occupant_id) { attrs.id = attrs.occupant_id; } return super.save(attrs, options); } getDisplayName() { return this.get('nick') || this.get('jid'); } isMember() { return ['admin', 'owner', 'member'].includes(this.get('affiliation')); } isModerator() { return ['admin', 'owner'].includes(this.get('affiliation')) || this.get('role') === 'moderator'; } isSelf() { return this.get('states').includes('110'); } } /* harmony default export */ const occupant = (ChatRoomOccupant); ;// CONCATENATED MODULE: ./src/headless/plugins/muc/constants.js const MUC_ROLE_WEIGHTS = { 'moderator': 1, 'participant': 2, 'visitor': 3, 'none': 2 }; ;// CONCATENATED MODULE: ./src/headless/plugins/muc/utils.js const { Strophe: muc_utils_Strophe, sizzle: muc_utils_sizzle, u: muc_utils_u } = core_converse.env; function getAutoFetchedAffiliationLists() { const affs = core_api.settings.get('muc_fetch_members'); return Array.isArray(affs) ? affs : affs ? ['member', 'admin', 'owner'] : []; } /** * Given an occupant model, see which roles may be assigned to that user. * @param { Model } occupant * @returns { Array<('moderator'|'participant'|'visitor')> } - An array of assignable roles */ function getAssignableRoles(occupant) { let disabled = core_api.settings.get('modtools_disable_assign'); if (!Array.isArray(disabled)) { disabled = disabled ? ROLES : []; } if (occupant.get('role') === 'moderator') { return ROLES.filter(r => !disabled.includes(r)); } else { return []; } } function registerDirectInvitationHandler() { shared_converse.connection.addHandler(message => { shared_converse.onDirectMUCInvitation(message); return true; }, 'jabber:x:conference', 'message'); } function disconnectChatRooms() { /* When disconnecting, mark all groupchats as * disconnected, so that they will be properly entered again * when fetched from session storage. */ return shared_converse.chatboxes.filter(m => m.get('type') === shared_converse.CHATROOMS_TYPE).forEach(m => m.session.save({ 'connection_status': core_converse.ROOMSTATUS.DISCONNECTED })); } async function onWindowStateChanged(data) { if (data.state === 'visible' && core_api.connection.connected()) { const rooms = await core_api.rooms.get(); rooms.forEach(room => room.rejoinIfNecessary()); } } async function routeToRoom(jid) { if (!muc_utils_u.isValidMUCJID(jid)) { return headless_log.warn(`invalid jid "${jid}" provided in url fragment`); } await core_api.waitUntil('roomsAutoJoined'); if (core_api.settings.get('allow_bookmarks')) { await core_api.waitUntil('bookmarksInitialized'); } core_api.rooms.open(jid); } /* Opens a groupchat, making sure that certain attributes * are correct, for example that the "type" is set to * "chatroom". */ async function openChatRoom(jid, settings) { settings.type = shared_converse.CHATROOMS_TYPE; settings.id = jid; const chatbox = await core_api.rooms.get(jid, settings, true); chatbox.maybeShow(true); return chatbox; } /** * A direct MUC invitation to join a groupchat has been received * See XEP-0249: Direct MUC invitations. * @private * @method _converse.ChatRoom#onDirectMUCInvitation * @param { XMLElement } message - The message stanza containing the invitation. */ async function onDirectMUCInvitation(message) { const { __ } = shared_converse; const x_el = muc_utils_sizzle('x[xmlns="jabber:x:conference"]', message).pop(), from = muc_utils_Strophe.getBareJidFromJid(message.getAttribute('from')), room_jid = x_el.getAttribute('jid'), reason = x_el.getAttribute('reason'); let result; if (core_api.settings.get('auto_join_on_invite')) { result = true; } else { // Invite request might come from someone not your roster list let contact = shared_converse.roster.get(from); contact = contact ? contact.getDisplayName() : from; if (!reason) { result = confirm(__('%1$s has invited you to join a groupchat: %2$s', contact, room_jid)); } else { result = confirm(__('%1$s has invited you to join a groupchat: %2$s, and left the following reason: "%3$s"', contact, room_jid, reason)); } } if (result === true) { const chatroom = await openChatRoom(room_jid, { 'password': x_el.getAttribute('password') }); if (chatroom.session.get('connection_status') === core_converse.ROOMSTATUS.DISCONNECTED) { shared_converse.chatboxes.get(room_jid).rejoin(); } } } function getDefaultMUCNickname() { // XXX: if anything changes here, update the docs for the // locked_muc_nickname setting. if (!shared_converse.xmppstatus) { throw new Error("Can't call _converse.getDefaultMUCNickname before the statusInitialized has been fired."); } const nick = shared_converse.xmppstatus.getNickname(); if (nick) { return nick; } else if (core_api.settings.get('muc_nickname_from_jid')) { return muc_utils_Strophe.unescapeNode(muc_utils_Strophe.getNodeFromJid(shared_converse.bare_jid)); } } /** * Determines info message visibility based on * muc_show_info_messages configuration setting * @param {*} code * @memberOf _converse */ function isInfoVisible(code) { const info_messages = core_api.settings.get('muc_show_info_messages'); if (info_messages.includes(code)) { return true; } return false; } /** * Automatically join groupchats, based on the * "auto_join_rooms" configuration setting, which is an array * of strings (groupchat JIDs) or objects (with groupchat JID and other settings). */ async function autoJoinRooms() { await Promise.all(core_api.settings.get('auto_join_rooms').map(muc => { if (typeof muc === 'string') { if (shared_converse.chatboxes.where({ 'jid': muc }).length) { return Promise.resolve(); } return core_api.rooms.open(muc); } else if (lodash_es_isObject(muc)) { return core_api.rooms.open(muc.jid, { ...muc }); } else { headless_log.error('Invalid muc criteria specified for "auto_join_rooms"'); return Promise.resolve(); } })); /** * Triggered once any rooms that have been configured to be automatically joined, * specified via the _`auto_join_rooms` setting, have been entered. * @event _converse#roomsAutoJoined * @example _converse.api.listen.on('roomsAutoJoined', () => { ... }); * @example _converse.api.waitUntil('roomsAutoJoined').then(() => { ... }); */ core_api.trigger('roomsAutoJoined'); } function onAddClientFeatures() { core_api.disco.own.features.add(muc_utils_Strophe.NS.MUC); if (core_api.settings.get('allow_muc_invitations')) { core_api.disco.own.features.add('jabber:x:conference'); // Invites } } function onBeforeTearDown() { shared_converse.chatboxes.where({ 'type': shared_converse.CHATROOMS_TYPE }).forEach(muc => safeSave(muc.session, { 'connection_status': core_converse.ROOMSTATUS.DISCONNECTED })); } function onStatusInitialized() { window.addEventListener(shared_converse.unloadevent, () => { const using_websocket = core_api.connection.isType('websocket'); if (using_websocket && (!core_api.settings.get('enable_smacks') || !shared_converse.session.get('smacks_stream_id'))) { // For non-SMACKS websocket connections, or non-resumeable // connections, we disconnect all chatrooms when the page unloads. // See issue #1111 disconnectChatRooms(); } }); } function onBeforeResourceBinding() { shared_converse.connection.addHandler(stanza => { const muc_jid = muc_utils_Strophe.getBareJidFromJid(stanza.getAttribute('from')); if (!shared_converse.chatboxes.get(muc_jid)) { core_api.waitUntil('chatBoxesFetched').then(async () => { const muc = shared_converse.chatboxes.get(muc_jid); if (muc) { await muc.initialized; muc.message_handler.run(stanza); } }); } return true; }, null, 'message', 'groupchat'); } Object.assign(shared_converse, { getAssignableRoles }); ;// CONCATENATED MODULE: ./src/headless/plugins/muc/occupants.js function occupants_defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } /** * A list of {@link _converse.ChatRoomOccupant} instances, representing participants in a MUC. * @class * @namespace _converse.ChatRoomOccupants * @memberOf _converse */ class ChatRoomOccupants extends Collection { constructor() { super(...arguments); occupants_defineProperty(this, "model", occupant); } comparator(occupant1, occupant2) { // eslint-disable-line class-methods-use-this const role1 = occupant1.get('role') || 'none'; const role2 = occupant2.get('role') || 'none'; if (MUC_ROLE_WEIGHTS[role1] === MUC_ROLE_WEIGHTS[role2]) { const nick1 = occupant1.getDisplayName().toLowerCase(); const nick2 = occupant2.getDisplayName().toLowerCase(); return nick1 < nick2 ? -1 : nick1 > nick2 ? 1 : 0; } else { return MUC_ROLE_WEIGHTS[role1] < MUC_ROLE_WEIGHTS[role2] ? -1 : 1; } } create(attrs, options) { if (attrs.id || attrs instanceof Model) { return super.create(attrs, options); } attrs.id = attrs.occupant_id || getUniqueId(); return super.create(attrs, options); } /** * Get the {@link _converse.ChatRoomOccupant} instance which * represents the current user. * @method _converse.ChatRoomOccupants#getOwnOccupant * @returns { _converse.ChatRoomOccupant } */ getOwnOccupant() { return this.findWhere({ 'jid': shared_converse.bare_jid }); } async fetchMembers() { var _this$getOwnOccupant; if (!['member', 'admin', 'owner'].includes((_this$getOwnOccupant = this.getOwnOccupant()) === null || _this$getOwnOccupant === void 0 ? void 0 : _this$getOwnOccupant.get('affiliation'))) { // https://xmpp.org/extensions/xep-0045.html#affil-priv return; } const affiliations = getAutoFetchedAffiliationLists(); if (affiliations.length === 0) { return; } const muc_jid = this.chatroom.get('jid'); const aff_lists = await Promise.all(affiliations.map(a => getAffiliationList(a, muc_jid))); const new_members = aff_lists.reduce((acc, val) => utils_form.isErrorObject(val) ? acc : [...val, ...acc], []); const known_affiliations = affiliations.filter(a => !utils_form.isErrorObject(aff_lists[affiliations.indexOf(a)])); const new_jids = new_members.map(m => m.jid).filter(m => m !== undefined); const new_nicks = new_members.map(m => !m.jid && m.nick || undefined).filter(m => m !== undefined); const removed_members = this.filter(m => { return known_affiliations.includes(m.get('affiliation')) && !new_nicks.includes(m.get('nick')) && !new_jids.includes(m.get('jid')); }); removed_members.forEach(occupant => { if (occupant.get('jid') === shared_converse.bare_jid) { return; } else if (occupant.get('show') === 'offline') { occupant.destroy(); } else { occupant.save('affiliation', null); } }); new_members.forEach(attrs => { const occupant = this.findOccupant(attrs); occupant ? occupant.save(attrs) : this.create(attrs); }); /** * Triggered once the member lists for this MUC have been fetched and processed. * @event _converse#membersFetched * @example _converse.api.listen.on('membersFetched', () => { ... }); */ core_api.trigger('membersFetched'); } /** * @typedef { Object} OccupantData * @property { String } [jid] * @property { String } [nick] * @property { String } [occupant_id] */ /** * Try to find an existing occupant based on the passed in * data object. * * Fetching the user by occupant_id is the quickest, O(1), * since it's a dictionary lookup. * * Fetching by jid or nick is O(n), since it requires traversing an array. * * Lookup by occupant_id is done first, then jid, and then nick. * * @method _converse.ChatRoomOccupants#findOccupant * @param { OccupantData } data */ findOccupant(data) { if (data.occupant_id && this.get(data.occupant_id)) { return this.get(data.occupant_id); } const jid = data.jid && Strophe.getBareJidFromJid(data.jid); return jid && this.findWhere({ jid }) || data.nick && this.findWhere({ 'nick': data.nick }); } } /* harmony default export */ const occupants = (ChatRoomOccupants); ;// CONCATENATED MODULE: ./src/headless/plugins/muc/affiliations/api.js /* harmony default export */ const affiliations_api = ({ /** * The "affiliations" namespace groups methods relevant to setting and * getting MUC affiliations. * * @namespace api.rooms.affiliations * @memberOf api.rooms */ affiliations: { /** * Set the given affliation for the given JIDs in the specified MUCs * * @param { String|Array } muc_jids - The JIDs of the MUCs in * which the affiliation should be set. * @param { Object[] } users - An array of objects representing users * for whom the affiliation is to be set. * @param { String } users[].jid - The JID of the user whose affiliation will change * @param { ('outcast'|'member'|'admin'|'owner') } users[].affiliation - The new affiliation for this user * @param { String } [users[].reason] - An optional reason for the affiliation change * @returns { Promise } * * @example * api.rooms.affiliations.set( * [ * 'muc1@muc.example.org', * 'muc2@muc.example.org' * ], [ * { * 'jid': 'user@example.org', * 'affiliation': 'member', * 'reason': "You're one of us now!" * } * ] * ) */ set(muc_jids, users) { users = !Array.isArray(users) ? [users] : users; muc_jids = !Array.isArray(muc_jids) ? [muc_jids] : muc_jids; return setAffiliations(muc_jids, users); } } }); ;// CONCATENATED MODULE: ./src/headless/plugins/muc/api.js /* harmony default export */ const muc_api = ({ /** * The "rooms" namespace groups methods relevant to chatrooms * (aka groupchats). * * @namespace api.rooms * @memberOf api */ rooms: { /** * Creates a new MUC chatroom (aka groupchat) * * Similar to {@link api.rooms.open}, but creates * the chatroom in the background (i.e. doesn't cause a view to open). * * @method api.rooms.create * @param {(string[]|string)} jid|jids The JID or array of * JIDs of the chatroom(s) to create * @param {object} [attrs] attrs The room attributes * @returns {Promise} Promise which resolves with the Model representing the chat. */ create(jids) { let attrs = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; attrs = typeof attrs === 'string' ? { 'nick': attrs } : attrs || {}; if (!attrs.nick && core_api.settings.get('muc_nickname_from_jid')) { attrs.nick = Strophe.getNodeFromJid(shared_converse.bare_jid); } if (jids === undefined) { throw new TypeError('rooms.create: You need to provide at least one JID'); } else if (typeof jids === 'string') { return core_api.rooms.get(utils_form.getJIDFromURI(jids), attrs, true); } return jids.map(jid => core_api.rooms.get(utils_form.getJIDFromURI(jid), attrs, true)); }, /** * Opens a MUC chatroom (aka groupchat) * * Similar to {@link api.chats.open}, but for groupchats. * * @method api.rooms.open * @param {string} jid The room JID or JIDs (if not specified, all * currently open rooms will be returned). * @param {string} attrs A map containing any extra room attributes. * @param {string} [attrs.nick] The current user's nickname for the MUC * @param {boolean} [attrs.auto_configure] A boolean, indicating * whether the room should be configured automatically or not. * If set to `true`, then it makes sense to pass in configuration settings. * @param {object} [attrs.roomconfig] A map of configuration settings to be used when the room gets * configured automatically. Currently it doesn't make sense to specify * `roomconfig` values if `auto_configure` is set to `false`. * For a list of configuration values that can be passed in, refer to these values * in the [XEP-0045 MUC specification](https://xmpp.org/extensions/xep-0045.html#registrar-formtype-owner). * The values should be named without the `muc#roomconfig_` prefix. * @param {boolean} [attrs.minimized] A boolean, indicating whether the room should be opened minimized or not. * @param {boolean} [attrs.bring_to_foreground] A boolean indicating whether the room should be * brought to the foreground and therefore replace the currently shown chat. * If there is no chat currently open, then this option is ineffective. * @param {Boolean} [force=false] - By default, a minimized * room won't be maximized (in `overlayed` view mode) and in * `fullscreen` view mode a newly opened room won't replace * another chat already in the foreground. * Set `force` to `true` if you want to force the room to be * maximized or shown. * @returns {Promise} Promise which resolves with the Model representing the chat. * * @example * api.rooms.open('group@muc.example.com') * * @example * // To return an array of rooms, provide an array of room JIDs: * api.rooms.open(['group1@muc.example.com', 'group2@muc.example.com']) * * @example * // To setup a custom nickname when joining the room, provide the optional nick argument: * api.rooms.open('group@muc.example.com', {'nick': 'mycustomnick'}) * * @example * // For example, opening a room with a specific default configuration: * api.rooms.open( * 'myroom@conference.example.org', * { 'nick': 'coolguy69', * 'auto_configure': true, * 'roomconfig': { * 'changesubject': false, * 'membersonly': true, * 'persistentroom': true, * 'publicroom': true, * 'roomdesc': 'Comfy room for hanging out', * 'whois': 'anyone' * } * } * ); */ async open(jids) { let attrs = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; let force = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false; await core_api.waitUntil('chatBoxesFetched'); if (jids === undefined) { const err_msg = 'rooms.open: You need to provide at least one JID'; headless_log.error(err_msg); throw new TypeError(err_msg); } else if (typeof jids === 'string') { const room = await core_api.rooms.get(jids, attrs, true); !attrs.hidden && (room === null || room === void 0 ? void 0 : room.maybeShow(force)); return room; } else { const rooms = await Promise.all(jids.map(jid => core_api.rooms.get(jid, attrs, true))); rooms.forEach(r => !attrs.hidden && r.maybeShow(force)); return rooms; } }, /** * Fetches the object representing a MUC chatroom (aka groupchat) * * @method api.rooms.get * @param { String } [jid] The room JID (if not specified, all rooms will be returned). * @param { Object } [attrs] A map containing any extra room attributes * to be set if `create` is set to `true` * @param { String } [attrs.nick] Specify the nickname * @param { String } [attrs.password ] Specify a password if needed to enter a new room * @param { Boolean } create A boolean indicating whether the room should be created * if not found (default: `false`) * @returns { Promise<_converse.ChatRoom> } * @example * api.waitUntil('roomsAutoJoined').then(() => { * const create_if_not_found = true; * api.rooms.get( * 'group@muc.example.com', * {'nick': 'dread-pirate-roberts', 'password': 'secret'}, * create_if_not_found * ) * }); */ async get(jids) { let attrs = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; let create = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false; await core_api.waitUntil('chatBoxesFetched'); async function _get(jid) { jid = utils_form.getJIDFromURI(jid); let model = await core_api.chatboxes.get(jid); if (!model && create) { model = await core_api.chatboxes.create(jid, attrs, shared_converse.ChatRoom); } else { model = model && model.get('type') === shared_converse.CHATROOMS_TYPE ? model : null; if (model && Object.keys(attrs).length) { model.save(attrs); } } return model; } if (jids === undefined) { const chats = await core_api.chatboxes.get(); return chats.filter(c => c.get('type') === shared_converse.CHATROOMS_TYPE); } else if (typeof jids === 'string') { return _get(jids); } return Promise.all(jids.map(jid => _get(jid))); } } }); ;// CONCATENATED MODULE: ./src/headless/plugins/muc/index.js /** * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) * @description Implements the non-view logic for XEP-0045 Multi-User Chat */ const ROLES = ['moderator', 'participant', 'visitor']; const AFFILIATIONS = ['owner', 'admin', 'member', 'outcast', 'none']; core_converse.AFFILIATION_CHANGES = { OWNER: 'owner', ADMIN: 'admin', MEMBER: 'member', EXADMIN: 'exadmin', EXOWNER: 'exowner', EXOUTCAST: 'exoutcast', EXMEMBER: 'exmember' }; core_converse.AFFILIATION_CHANGES_LIST = Object.values(core_converse.AFFILIATION_CHANGES); core_converse.MUC_TRAFFIC_STATES = { ENTERED: 'entered', EXITED: 'exited' }; core_converse.MUC_TRAFFIC_STATES_LIST = Object.values(core_converse.MUC_TRAFFIC_STATES); core_converse.MUC_ROLE_CHANGES = { OP: 'op', DEOP: 'deop', VOICE: 'voice', MUTE: 'mute' }; core_converse.MUC_ROLE_CHANGES_LIST = Object.values(core_converse.MUC_ROLE_CHANGES); core_converse.MUC = {}; core_converse.MUC.INFO_CODES = { 'visibility_changes': ['100', '102', '103', '172', '173', '174'], 'self': ['110'], 'non_privacy_changes': ['104', '201'], 'muc_logging_changes': ['170', '171'], 'nickname_changes': ['210', '303'], 'disconnected': ['301', '307', '321', '322', '332', '333'], 'affiliation_changes': [...core_converse.AFFILIATION_CHANGES_LIST], 'join_leave_events': [...core_converse.MUC_TRAFFIC_STATES_LIST], 'role_changes': [...core_converse.MUC_ROLE_CHANGES_LIST] }; const { Strophe: muc_Strophe } = core_converse.env; // Add Strophe Namespaces muc_Strophe.addNamespace('MUC_ADMIN', muc_Strophe.NS.MUC + '#admin'); muc_Strophe.addNamespace('MUC_OWNER', muc_Strophe.NS.MUC + '#owner'); muc_Strophe.addNamespace('MUC_REGISTER', 'jabber:iq:register'); muc_Strophe.addNamespace('MUC_ROOMCONF', muc_Strophe.NS.MUC + '#roomconfig'); muc_Strophe.addNamespace('MUC_USER', muc_Strophe.NS.MUC + '#user'); muc_Strophe.addNamespace('MUC_HATS', 'xmpp:prosody.im/protocol/hats:1'); muc_Strophe.addNamespace('CONFINFO', 'urn:ietf:params:xml:ns:conference-info'); core_converse.MUC_NICK_CHANGED_CODE = '303'; core_converse.ROOM_FEATURES = ['passwordprotected', 'unsecured', 'hidden', 'publicroom', 'membersonly', 'open', 'persistent', 'temporary', 'nonanonymous', 'semianonymous', 'moderated', 'unmoderated', 'mam_enabled']; // No longer used in code, but useful as reference. // // const ROOM_FEATURES_MAP = { // 'passwordprotected': 'unsecured', // 'unsecured': 'passwordprotected', // 'hidden': 'publicroom', // 'publicroom': 'hidden', // 'membersonly': 'open', // 'open': 'membersonly', // 'persistent': 'temporary', // 'temporary': 'persistent', // 'nonanonymous': 'semianonymous', // 'semianonymous': 'nonanonymous', // 'moderated': 'unmoderated', // 'unmoderated': 'moderated' // }; core_converse.ROOMSTATUS = { CONNECTED: 0, CONNECTING: 1, NICKNAME_REQUIRED: 2, PASSWORD_REQUIRED: 3, DISCONNECTED: 4, ENTERED: 5, DESTROYED: 6, BANNED: 7, CLOSING: 8 }; core_converse.plugins.add('converse-muc', { /* 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. */ dependencies: ['converse-chatboxes', 'converse-chat', 'converse-disco'], overrides: { ChatBoxes: { model(attrs, options) { const { _converse } = this.__super__; if (attrs && attrs.type == _converse.CHATROOMS_TYPE) { return new _converse.ChatRoom(attrs, options); } else { return this.__super__.model.apply(this, arguments); } } } }, initialize() { /* The initialize function gets called as soon as the plugin is * loaded by converse.js's plugin machinery. */ const { __, ___ } = shared_converse; // Configuration values for this plugin // ==================================== // Refer to docs/source/configuration.rst for explanations of these // configuration settings. core_api.settings.extend({ 'allow_muc_invitations': true, 'auto_join_on_invite': false, 'auto_join_rooms': [], 'auto_register_muc_nickname': false, 'hide_muc_participants': false, 'locked_muc_domain': false, 'modtools_disable_assign': false, 'muc_clear_messages_on_leave': true, 'muc_domain': undefined, 'muc_fetch_members': true, 'muc_history_max_stanzas': undefined, 'muc_instant_rooms': true, 'muc_nickname_from_jid': false, 'muc_send_probes': false, 'muc_show_info_messages': [...core_converse.MUC.INFO_CODES.visibility_changes, ...core_converse.MUC.INFO_CODES.self, ...core_converse.MUC.INFO_CODES.non_privacy_changes, ...core_converse.MUC.INFO_CODES.muc_logging_changes, ...core_converse.MUC.INFO_CODES.nickname_changes, ...core_converse.MUC.INFO_CODES.disconnected, ...core_converse.MUC.INFO_CODES.affiliation_changes, ...core_converse.MUC.INFO_CODES.join_leave_events, ...core_converse.MUC.INFO_CODES.role_changes], 'muc_show_logs_before_join': false, 'muc_subscribe_to_rai': false }); core_api.promises.add(['roomsAutoJoined']); if (core_api.settings.get('locked_muc_domain') && typeof core_api.settings.get('muc_domain') !== 'string') { throw new Error('Config Error: it makes no sense to set locked_muc_domain ' + 'to true when muc_domain is not set'); } // This is for tests (at least until we can import modules inside tests) core_converse.env.muc_utils = { computeAffiliationsDelta: computeAffiliationsDelta }; Object.assign(core_api, muc_api); Object.assign(core_api.rooms, affiliations_api); /* https://xmpp.org/extensions/xep-0045.html * ---------------------------------------- * 100 message Entering a groupchat 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 groupchat * 102 message Configuration change Inform occupants that groupchat now shows unavailable members * 103 message Configuration change Inform occupants that groupchat now does not show unavailable members * 104 message Configuration change Inform occupants that a non-privacy-related groupchat configuration change has occurred * 110 presence Any groupchat presence Inform user that presence refers to one of its own groupchat occupants * 170 message or initial presence Configuration change Inform occupants that groupchat logging is now enabled * 171 message Configuration change Inform occupants that groupchat logging is now disabled * 172 message Configuration change Inform occupants that the groupchat is now non-anonymous * 173 message Configuration change Inform occupants that the groupchat is now semi-anonymous * 174 message Configuration change Inform occupants that the groupchat is now fully-anonymous * 201 presence Entering a groupchat Inform user that a new groupchat has been created * 210 presence Entering a groupchat Inform user that the service has assigned or modified the occupant's roomnick * 301 presence Removal from groupchat Inform user that he or she has been banned from the groupchat * 303 presence Exiting a groupchat Inform all occupants of new groupchat nickname * 307 presence Removal from groupchat Inform user that he or she has been kicked from the groupchat * 321 presence Removal from groupchat Inform user that he or she is being removed from the groupchat because of an affiliation change * 322 presence Removal from groupchat Inform user that he or she is being removed from the groupchat because the groupchat has been changed to members-only and the user is not a member * 332 presence Removal from groupchat Inform user that he or she is being removed from the groupchat because of a system shutdown */ shared_converse.muc = { info_messages: { 100: __('This groupchat is not anonymous'), 102: __('This groupchat now shows unavailable members'), 103: __('This groupchat does not show unavailable members'), 104: __('The groupchat configuration has changed'), 170: __('Groupchat logging is now enabled'), 171: __('Groupchat logging is now disabled'), 172: __('This groupchat is now no longer anonymous'), 173: __('This groupchat is now semi-anonymous'), 174: __('This groupchat is now fully-anonymous'), 201: __('A new groupchat has been created') }, new_nickname_messages: { // XXX: Note the triple underscore function and not double underscore. 210: ___('Your nickname has been automatically set to %1$s'), 303: ___('Your nickname has been changed to %1$s') }, disconnect_messages: { 301: __('You have been banned from this groupchat'), 333: __('You have exited this groupchat due to a technical problem'), 307: __('You have been kicked from this groupchat'), 321: __('You have been removed from this groupchat because of an affiliation change'), 322: __("You have been removed from this groupchat because the groupchat has changed to members-only and you're not a member"), 332: __('You have been removed from this groupchat because the service hosting it is being shut down') } }; shared_converse.router.route('converse/room?jid=:jid', routeToRoom); shared_converse.ChatRoom = shared_converse.ChatBox.extend(muc); shared_converse.ChatRoomMessage = shared_converse.Message.extend(muc_message); shared_converse.ChatRoomOccupants = occupants; shared_converse.ChatRoomOccupant = occupant; /** * Collection which stores MUC messages * @class * @namespace _converse.ChatRoomMessages * @memberOf _converse */ shared_converse.ChatRoomMessages = Collection.extend({ model: shared_converse.ChatRoomMessage, comparator: 'time' }); Object.assign(shared_converse, { getDefaultMUCNickname: getDefaultMUCNickname, isInfoVisible: isInfoVisible, onDirectMUCInvitation: onDirectMUCInvitation }); /************************ BEGIN Event Handlers ************************/ if (core_api.settings.get('allow_muc_invitations')) { core_api.listen.on('connected', registerDirectInvitationHandler); core_api.listen.on('reconnected', registerDirectInvitationHandler); } core_api.listen.on('addClientFeatures', () => core_api.disco.own.features.add(`${muc_Strophe.NS.CONFINFO}+notify`)); core_api.listen.on('addClientFeatures', onAddClientFeatures); core_api.listen.on('beforeResourceBinding', onBeforeResourceBinding); core_api.listen.on('beforeTearDown', onBeforeTearDown); core_api.listen.on('chatBoxesFetched', autoJoinRooms); core_api.listen.on('disconnected', disconnectChatRooms); core_api.listen.on('statusInitialized', onStatusInitialized); core_api.listen.on('windowStateChanged', onWindowStateChanged); } }); ;// CONCATENATED MODULE: ./src/headless/plugins/bookmarks/model.js const { Strophe: bookmarks_model_Strophe } = core_converse.env; const Bookmark = Model.extend({ idAttribute: 'jid', getDisplayName() { return bookmarks_model_Strophe.xmlunescape(this.get('name')); } }); /* harmony default export */ const bookmarks_model = (Bookmark); ;// CONCATENATED MODULE: ./src/headless/plugins/bookmarks/collection.js const { Strophe: collection_Strophe, $iq: collection_$iq, sizzle: collection_sizzle } = core_converse.env; const Bookmarks = { model: bookmarks_model, comparator: item => item.get('name').toLowerCase(), async initialize() { this.on('add', bm => this.openBookmarkedRoom(bm).then(bm => this.markRoomAsBookmarked(bm)).catch(e => headless_log.fatal(e))); this.on('remove', this.markRoomAsUnbookmarked, this); this.on('remove', this.sendBookmarkStanza, this); const cache_key = `converse.room-bookmarks${shared_converse.bare_jid}`; this.fetched_flag = cache_key + 'fetched'; initStorage(this, cache_key); await this.fetchBookmarks(); /** * Triggered once the _converse.Bookmarks collection * has been created and cached bookmarks have been fetched. * @event _converse#bookmarksInitialized * @type { _converse.Bookmarks } * @example _converse.api.listen.on('bookmarksInitialized', (bookmarks) => { ... }); */ core_api.trigger('bookmarksInitialized', this); }, async openBookmarkedRoom(bookmark) { if (core_api.settings.get('muc_respect_autojoin') && bookmark.get('autojoin')) { const groupchat = await core_api.rooms.create(bookmark.get('jid'), { 'nick': bookmark.get('nick') }); groupchat.maybeShow(); } return bookmark; }, fetchBookmarks() { const deferred = getOpenPromise(); if (window.sessionStorage.getItem(this.fetched_flag)) { this.fetch({ 'success': () => deferred.resolve(), 'error': () => deferred.resolve() }); } else { this.fetchBookmarksFromServer(deferred); } return deferred; }, createBookmark(options) { this.create(options); this.sendBookmarkStanza().catch(iq => this.onBookmarkError(iq, options)); }, sendBookmarkStanza() { const stanza = collection_$iq({ 'type': 'set', 'from': shared_converse.connection.jid }).c('pubsub', { 'xmlns': collection_Strophe.NS.PUBSUB }).c('publish', { 'node': collection_Strophe.NS.BOOKMARKS }).c('item', { 'id': 'current' }).c('storage', { 'xmlns': collection_Strophe.NS.BOOKMARKS }); this.forEach(model => { stanza.c('conference', { 'name': model.get('name'), 'autojoin': model.get('autojoin'), 'jid': model.get('jid') }).c('nick').t(model.get('nick')).up().up(); }); stanza.up().up().up(); stanza.c('publish-options').c('x', { 'xmlns': collection_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'); return core_api.sendIQ(stanza); }, onBookmarkError(iq, options) { var _this$get; const { __ } = shared_converse; headless_log.error("Error while trying to add bookmark"); headless_log.error(iq); core_api.alert('error', __('Error'), [__("Sorry, something went wrong while trying to save your bookmark.")]); (_this$get = this.get(options.jid)) === null || _this$get === void 0 ? void 0 : _this$get.destroy(); }, fetchBookmarksFromServer(deferred) { const stanza = collection_$iq({ 'from': shared_converse.connection.jid, 'type': 'get' }).c('pubsub', { 'xmlns': collection_Strophe.NS.PUBSUB }).c('items', { 'node': collection_Strophe.NS.BOOKMARKS }); core_api.sendIQ(stanza).then(iq => this.onBookmarksReceived(deferred, iq)).catch(iq => this.onBookmarksReceivedError(deferred, iq)); }, markRoomAsBookmarked(bookmark) { const groupchat = shared_converse.chatboxes.get(bookmark.get('jid')); groupchat === null || groupchat === void 0 ? void 0 : groupchat.save('bookmarked', true); }, markRoomAsUnbookmarked(bookmark) { const groupchat = shared_converse.chatboxes.get(bookmark.get('jid')); groupchat === null || groupchat === void 0 ? void 0 : groupchat.save('bookmarked', false); }, createBookmarksFromStanza(stanza) { const xmlns = collection_Strophe.NS.BOOKMARKS; const sel = `items[node="${xmlns}"] item storage[xmlns="${xmlns}"] conference`; collection_sizzle(sel, stanza).forEach(el => { var _el$querySelector; const jid = el.getAttribute('jid'); const bookmark = this.get(jid); const attrs = { 'jid': jid, 'name': el.getAttribute('name') || jid, 'autojoin': el.getAttribute('autojoin') === 'true', 'nick': ((_el$querySelector = el.querySelector('nick')) === null || _el$querySelector === void 0 ? void 0 : _el$querySelector.textContent) || '' }; bookmark ? bookmark.save(attrs) : this.create(attrs); }); }, onBookmarksReceived(deferred, iq) { this.createBookmarksFromStanza(iq); window.sessionStorage.setItem(this.fetched_flag, true); if (deferred !== undefined) { return deferred.resolve(); } }, onBookmarksReceivedError(deferred, iq) { const { __ } = shared_converse; if (iq === null) { headless_log.error('Error: timeout while fetching bookmarks'); core_api.alert('error', __('Timeout Error'), [__("The server did not return your bookmarks within the allowed time. " + "You can reload the page to request them again.")]); } else if (deferred) { if (iq.querySelector('error[type="cancel"] item-not-found')) { // Not an exception, the user simply doesn't have any bookmarks. window.sessionStorage.setItem(this.fetched_flag, true); return deferred.resolve(); } else { headless_log.error('Error while fetching bookmarks'); headless_log.error(iq); return deferred.reject(new Error("Could not fetch bookmarks")); } } else { headless_log.error('Error while fetching bookmarks'); headless_log.error(iq); } }, async getUnopenedBookmarks() { await core_api.waitUntil('bookmarksInitialized'); await core_api.waitUntil('chatBoxesFetched'); return this.filter(b => !shared_converse.chatboxes.get(b.get('jid'))); } }; /* harmony default export */ const collection = (Bookmarks); ;// CONCATENATED MODULE: ./src/headless/plugins/bookmarks/utils.js const { Strophe: bookmarks_utils_Strophe, sizzle: bookmarks_utils_sizzle } = core_converse.env; async function checkBookmarksSupport() { const identity = await core_api.disco.getIdentity('pubsub', 'pep', shared_converse.bare_jid); if (core_api.settings.get('allow_public_bookmarks')) { return !!identity; } else { return core_api.disco.supports(bookmarks_utils_Strophe.NS.PUBSUB + '#publish-options', shared_converse.bare_jid); } } async function initBookmarks() { if (!core_api.settings.get('allow_bookmarks')) { return; } if (await checkBookmarksSupport()) { shared_converse.bookmarks = new shared_converse.Bookmarks(); } } function getNicknameFromBookmark(jid) { var _converse$bookmarks, _converse$bookmarks$g; if (!core_api.settings.get('allow_bookmarks')) { return null; } return (_converse$bookmarks = shared_converse.bookmarks) === null || _converse$bookmarks === void 0 ? void 0 : (_converse$bookmarks$g = _converse$bookmarks.get(jid)) === null || _converse$bookmarks$g === void 0 ? void 0 : _converse$bookmarks$g.get('nick'); } function handleBookmarksPush(message) { if (bookmarks_utils_sizzle(`event[xmlns="${bookmarks_utils_Strophe.NS.PUBSUB}#event"] items[node="${bookmarks_utils_Strophe.NS.BOOKMARKS}"]`, message).length) { core_api.waitUntil('bookmarksInitialized').then(() => shared_converse.bookmarks.createBookmarksFromStanza(message)).catch(e => headless_log.fatal(e)); } return true; } ;// CONCATENATED MODULE: ./src/headless/plugins/bookmarks/index.js /** * @description * Converse.js plugin which adds views for bookmarks specified in XEP-0048. * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ const { Strophe: bookmarks_Strophe } = core_converse.env; bookmarks_Strophe.addNamespace('BOOKMARKS', 'storage:bookmarks'); core_converse.plugins.add('converse-bookmarks', { dependencies: ["converse-chatboxes", "converse-muc"], 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. ChatRoom: { getDisplayName() { var _converse$bookmarks; const { _converse, getDisplayName } = this.__super__; const bookmark = this.get('bookmarked') ? (_converse$bookmarks = _converse.bookmarks) === null || _converse$bookmarks === void 0 ? void 0 : _converse$bookmarks.get(this.get('jid')) : null; return (bookmark === null || bookmark === void 0 ? void 0 : bookmark.get('name')) || getDisplayName.apply(this, arguments); }, getAndPersistNickname(nick) { nick = nick || getNicknameFromBookmark(this.get('jid')); return this.__super__.getAndPersistNickname.call(this, nick); } } }, initialize() { // Configuration values for this plugin // ==================================== // Refer to docs/source/configuration.rst for explanations of these // configuration settings. core_api.settings.extend({ allow_bookmarks: true, allow_public_bookmarks: false, muc_respect_autojoin: true }); core_api.promises.add('bookmarksInitialized'); shared_converse.Bookmark = bookmarks_model; shared_converse.Bookmarks = Collection.extend(collection); shared_converse.BookmarksList = Model.extend({ defaults: { "toggle-state": shared_converse.OPENED } }); core_api.listen.on('addClientFeatures', () => { if (core_api.settings.get('allow_bookmarks')) { core_api.disco.own.features.add(bookmarks_Strophe.NS.BOOKMARKS + '+notify'); } }); core_api.listen.on('clearSession', () => { if (shared_converse.bookmarks) { shared_converse.bookmarks.clearStore({ 'silent': true }); window.sessionStorage.removeItem(shared_converse.bookmarks.fetched_flag); delete shared_converse.bookmarks; } }); core_api.listen.on('connected', async () => { // Add a handler for bookmarks pushed from other connected clients const { connection } = shared_converse; connection.addHandler(handleBookmarksPush, null, 'message', 'headline', null, shared_converse.bare_jid); await Promise.all([core_api.waitUntil('chatBoxesFetched')]); initBookmarks(); }); } }); ;// CONCATENATED MODULE: ./src/headless/plugins/bosh.js /** * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) * @description Converse.js plugin which add support for XEP-0206: XMPP Over BOSH */ const { Strophe: bosh_Strophe } = core_converse.env; const BOSH_SESSION_ID = 'converse.bosh-session'; core_converse.plugins.add('converse-bosh', { enabled() { return !shared_converse.api.settings.get("blacklisted_plugins").includes('converse-bosh'); }, initialize() { core_api.settings.extend({ bosh_service_url: undefined, prebind_url: null }); async function initBOSHSession() { const id = BOSH_SESSION_ID; if (!shared_converse.bosh_session) { shared_converse.bosh_session = new Model({ id }); shared_converse.bosh_session.browserStorage = shared_converse.createStore(id, "session"); await new Promise(resolve => shared_converse.bosh_session.fetch({ 'success': resolve, 'error': resolve })); } if (shared_converse.jid) { if (shared_converse.bosh_session.get('jid') !== shared_converse.jid) { const jid = await setUserJID(shared_converse.jid); shared_converse.bosh_session.clear({ 'silent': true }); shared_converse.bosh_session.save({ jid }); } } else { // Keepalive const jid = shared_converse.bosh_session.get('jid'); jid && (await setUserJID(jid)); } return shared_converse.bosh_session; } shared_converse.startNewPreboundBOSHSession = function () { if (!core_api.settings.get('prebind_url')) { throw new Error("startNewPreboundBOSHSession: If you use prebind then you MUST supply a prebind_url"); } const xhr = new XMLHttpRequest(); xhr.open('GET', core_api.settings.get('prebind_url'), true); xhr.setRequestHeader('Accept', 'application/json, text/javascript'); xhr.onload = async function () { if (xhr.status >= 200 && xhr.status < 400) { const data = JSON.parse(xhr.responseText); const jid = await setUserJID(data.jid); shared_converse.connection.attach(jid, data.sid, data.rid, shared_converse.connection.onConnectStatusChanged, BOSH_WAIT); } else { xhr.onerror(); } }; xhr.onerror = function () { delete shared_converse.connection; /** * Triggered when fetching prebind tokens failed * @event _converse#noResumeableBOSHSession * @type { _converse } * @example _converse.api.listen.on('noResumeableBOSHSession', _converse => { ... }); */ core_api.trigger('noResumeableBOSHSession', shared_converse); }; xhr.send(); }; shared_converse.restoreBOSHSession = async function () { const jid = (await initBOSHSession()).get('jid'); if (jid && shared_converse.connection._proto instanceof bosh_Strophe.Bosh) { try { shared_converse.connection.restore(jid, shared_converse.connection.onConnectStatusChanged); return true; } catch (e) { !shared_converse.isTestEnv() && headless_log.warn("Could not restore session for jid: " + jid + " Error message: " + e.message); return false; } } return false; }; /************************ BEGIN Event Handlers ************************/ core_api.listen.on('clearSession', () => { if (shared_converse.bosh_session === undefined) { // Remove manually, even if we don't have the corresponding // model, to avoid trying to reconnect to a stale BOSH session const id = BOSH_SESSION_ID; sessionStorage.removeItem(id); sessionStorage.removeItem(`${id}-${id}`); } else { shared_converse.bosh_session.destroy(); delete shared_converse.bosh_session; } }); core_api.listen.on('setUserJID', () => { if (shared_converse.bosh_session !== undefined) { shared_converse.bosh_session.save({ 'jid': shared_converse.jid }); } }); core_api.listen.on('addClientFeatures', () => core_api.disco.own.features.add(bosh_Strophe.NS.BOSH)); /************************ END Event Handlers ************************/ /************************ BEGIN API ************************/ Object.assign(core_api, { /** * This namespace lets you access the BOSH tokens * * @namespace api.tokens * @memberOf api */ tokens: { /** * @method api.tokens.get * @param {string} [id] The type of token to return ('rid' or 'sid'). * @returns 'string' A token, either the RID or SID token depending on what's asked for. * @example _converse.api.tokens.get('rid'); */ get(id) { if (shared_converse.connection === undefined) { return null; } if (id.toLowerCase() === 'rid') { return shared_converse.connection.rid || shared_converse.connection._proto.rid; } else if (id.toLowerCase() === 'sid') { return shared_converse.connection.sid || shared_converse.connection._proto.sid; } } } }); /************************ end api ************************/ } }); ;// CONCATENATED MODULE: ./src/headless/plugins/caps/utils.js const { Strophe: caps_utils_Strophe, $build: utils_$build } = core_converse.env; function propertySort(array, property) { return array.sort((a, b) => { return a[property] > b[property] ? -1 : 1; }); } function generateVerificationString() { const identities = shared_converse.api.disco.own.identities.get(); const features = shared_converse.api.disco.own.features.get(); if (identities.length > 1) { propertySort(identities, "category"); propertySort(identities, "type"); propertySort(identities, "lang"); } let S = identities.reduce((result, id) => `${result}${id.category}/${id.type}/${(id === null || id === void 0 ? void 0 : id.lang) ?? ''}/${id.name}<`, ""); features.sort(); S = features.reduce((result, feature) => `${result}${feature}<`, S); return SHA1.b64_sha1(S); } function createCapsNode() { return utils_$build("c", { 'xmlns': caps_utils_Strophe.NS.CAPS, 'hash': "sha-1", 'node': "https://conversejs.org", 'ver': generateVerificationString() }).nodeTree; } ;// CONCATENATED MODULE: ./src/headless/plugins/caps/index.js /** * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ const { Strophe: caps_Strophe } = core_converse.env; caps_Strophe.addNamespace('CAPS', "http://jabber.org/protocol/caps"); core_converse.plugins.add('converse-caps', { dependencies: ['converse-status'], initialize() { core_api.listen.on('constructedPresence', (_, p) => p.root().cnode(createCapsNode()).up() && p); core_api.listen.on('constructedMUCPresence', (_, p) => p.root().cnode(createCapsNode()).up() && p); } }); ;// CONCATENATED MODULE: ./src/headless/plugins/carbons.js /** * @module converse-carbons * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) * @description Implements support for XEP-0280 Message Carbons */ const { u: carbons_u } = core_converse.env; /** * Ask the XMPP server to enable Message Carbons * See [XEP-0280](https://xmpp.org/extensions/xep-0280.html#enabling) */ async function enableCarbons(reconnecting) { var _converse$session; if (reconnecting && shared_converse.session.get('carbons_enabled')) { if (shared_converse.session.get('smacks_enabled')) { // No need to re-enable carbons when resuming a XEP-0198 stream return; } shared_converse.session.set({ 'carbons_enabled': false }); } if (!core_api.settings.get("message_carbons") || (_converse$session = shared_converse.session) !== null && _converse$session !== void 0 && _converse$session.get('carbons_enabled')) { return; } const iq = new Strophe.Builder('iq', { 'from': shared_converse.connection.jid, 'type': 'set' }).c('enable', { xmlns: Strophe.NS.CARBONS }); const result = await core_api.sendIQ(iq, null, false); if (result === null) { headless_log.warn(`A timeout occurred while trying to enable carbons`); } else if (carbons_u.isErrorStanza(result)) { headless_log.warn('An error occurred while trying to enable message carbons.'); headless_log.error(result); } else { shared_converse.session.set({ 'carbons_enabled': true }); headless_log.debug('Message carbons have been enabled.'); } shared_converse.session.save(); // Gather multiple sets into one save } core_converse.plugins.add('converse-carbons', { initialize() { core_api.settings.extend({ message_carbons: true }); core_api.listen.on('connected', () => enableCarbons()); core_api.listen.on('reconnected', () => enableCarbons(true)); } }); ;// CONCATENATED MODULE: ./src/headless/plugins/chatboxes/chatboxes.js const ChatBoxes = Collection.extend({ comparator: 'time_opened', model(attrs, options) { return new shared_converse.ChatBox(attrs, options); }, onChatBoxesFetched(collection) { collection.filter(c => !c.isValid()).forEach(c => c.destroy()); /** * Triggered once all chat boxes have been recreated from the browser cache * @event _converse#chatBoxesFetched * @type { object } * @property { _converse.ChatBox | _converse.ChatRoom } chatbox * @property { XMLElement } stanza * @example _converse.api.listen.on('chatBoxesFetched', obj => { ... }); * @example _converse.api.waitUntil('chatBoxesFetched').then(() => { ... }); */ core_api.trigger('chatBoxesFetched'); }, onConnected(reconnecting) { if (reconnecting) { return; } initStorage(this, `converse.chatboxes-${shared_converse.bare_jid}`); this.fetch({ 'add': true, 'success': c => this.onChatBoxesFetched(c) }); } }); /* harmony default export */ const chatboxes = (ChatBoxes); ;// CONCATENATED MODULE: ./src/headless/plugins/chatboxes/utils.js const { Strophe: chatboxes_utils_Strophe } = core_converse.env; async function createChatBox(jid, attrs, Model) { jid = chatboxes_utils_Strophe.getBareJidFromJid(jid.toLowerCase()); Object.assign(attrs, { 'jid': jid, 'id': jid }); let chatbox; try { chatbox = new Model(attrs, { 'collection': shared_converse.chatboxes }); } catch (e) { headless_log.error(e); return null; } await chatbox.initialized; if (!chatbox.isValid()) { chatbox.destroy(); return null; } shared_converse.chatboxes.add(chatbox); return chatbox; } ;// CONCATENATED MODULE: ./src/headless/plugins/chatboxes/api.js /** * The "chatboxes" namespace. * * @namespace api.chatboxes * @memberOf api */ /* harmony default export */ const chatboxes_api = ({ /** * @method api.chats.create * @param { String|String[] } jids - A JID or array of JIDs * @param { Object } [attrs] An object containing configuration attributes * @param { Model } model - The type of chatbox that should be created */ async create() { let jids = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : []; let attrs = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; let model = arguments.length > 2 ? arguments[2] : undefined; await core_api.waitUntil('chatBoxesFetched'); if (typeof jids === 'string') { return createChatBox(jids, attrs, model); } else { return Promise.all(jids.map(jid => createChatBox(jid, attrs, model))); } }, /** * @method api.chats.get * @param { String|String[] } jids - A JID or array of JIDs */ async get(jids) { await core_api.waitUntil('chatBoxesFetched'); if (jids === undefined) { return shared_converse.chatboxes.models; } else if (typeof jids === 'string') { return shared_converse.chatboxes.get(jids.toLowerCase()); } else { jids = jids.map(j => j.toLowerCase()); return shared_converse.chatboxes.models.filter(m => jids.includes(m.get('jid'))); } } }); ;// CONCATENATED MODULE: ./src/headless/plugins/chatboxes/index.js /** * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ const { Strophe: chatboxes_Strophe } = core_converse.env; core_converse.plugins.add('converse-chatboxes', { dependencies: ["converse-emoji", "converse-roster", "converse-vcard"], initialize() { core_api.promises.add(['chatBoxesFetched', 'chatBoxesInitialized', 'privateChatsAutoJoined']); Object.assign(core_api, { 'chatboxes': chatboxes_api }); shared_converse.ChatBoxes = chatboxes; core_api.listen.on('addClientFeatures', () => { core_api.disco.own.features.add(chatboxes_Strophe.NS.MESSAGE_CORRECT); core_api.disco.own.features.add(chatboxes_Strophe.NS.HTTPUPLOAD); core_api.disco.own.features.add(chatboxes_Strophe.NS.OUTOFBAND); }); core_api.listen.on('pluginsInitialized', () => { shared_converse.chatboxes = new shared_converse.ChatBoxes(); /** * Triggered once the _converse.ChatBoxes collection has been initialized. * @event _converse#chatBoxesInitialized * @example _converse.api.listen.on('chatBoxesInitialized', () => { ... }); * @example _converse.api.waitUntil('chatBoxesInitialized').then(() => { ... }); */ core_api.trigger('chatBoxesInitialized'); }); core_api.listen.on('presencesInitialized', reconnecting => shared_converse.chatboxes.onConnected(reconnecting)); core_api.listen.on('reconnected', () => shared_converse.chatboxes.forEach(m => m.onReconnection())); } }); ;// CONCATENATED MODULE: ./src/headless/plugins/headlines.js /** * @module converse-headlines * @copyright 2022, the Converse.js contributors * @description XEP-0045 Multi-User Chat Views */ core_converse.plugins.add('converse-headlines', { /* Plugin dependencies are other plugins which might be * overridden or relied upon, and therefore need to be loaded before * this plugin. * * If the setting "strict_plugin_dependencies" is set to true, * an error will be raised if the plugin is not found. By default it's * false, which means these plugins are only loaded opportunistically. * * NB: These plugins need to have already been loaded via require.js. */ dependencies: ["converse-chat"], 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. ChatBoxes: { model(attrs, options) { const { _converse } = this.__super__; if (attrs.type == _converse.HEADLINES_TYPE) { return new _converse.HeadlinesBox(attrs, options); } else { return this.__super__.model.apply(this, arguments); } } } }, initialize() { /* The initialize function gets called as soon as the plugin is * loaded by converse.js's plugin machinery. */ /** * Shows headline messages * @class * @namespace _converse.HeadlinesBox * @memberOf _converse */ shared_converse.HeadlinesBox = shared_converse.ChatBox.extend({ defaults() { return { 'bookmarked': false, 'hidden': ['mobile', 'fullscreen'].includes(core_api.settings.get("view_mode")), 'message_type': 'headline', 'num_unread': 0, 'time_opened': this.get('time_opened') || new Date().getTime(), 'type': shared_converse.HEADLINES_TYPE }; }, async initialize() { this.set({ 'box_id': `box-${this.get('jid')}` }); this.initUI(); this.initMessages(); await this.fetchMessages(); /** * Triggered once a {@link _converse.HeadlinesBox} has been created and initialized. * @event _converse#headlinesBoxInitialized * @type { _converse.HeadlinesBox } * @example _converse.api.listen.on('headlinesBoxInitialized', model => { ... }); */ core_api.trigger('headlinesBoxInitialized', this); } }); async function onHeadlineMessage(stanza) { // Handler method for all incoming messages of type "headline". if (isHeadline(stanza) || isServerMessage(stanza)) { const from_jid = stanza.getAttribute('from'); await core_api.waitUntil('rosterInitialized'); if (from_jid.includes('@') && !shared_converse.roster.get(from_jid) && !core_api.settings.get("allow_non_roster_messaging")) { return; } if (stanza.querySelector('body') === null) { // Avoid creating a chat box if we have nothing to show inside it. return; } const chatbox = shared_converse.chatboxes.create({ 'id': from_jid, 'jid': from_jid, 'type': shared_converse.HEADLINES_TYPE, 'from': from_jid }); const attrs = await parseMessage(stanza, shared_converse); await chatbox.createMessage(attrs); core_api.trigger('message', { chatbox, stanza, attrs }); } } /************************ BEGIN Event Handlers ************************/ function registerHeadlineHandler() { shared_converse.connection.addHandler(message => onHeadlineMessage(message) || true, null, 'message'); } core_api.listen.on('connected', registerHeadlineHandler); core_api.listen.on('reconnected', registerHeadlineHandler); /************************ END Event Handlers ************************/ /************************ BEGIN API ************************/ Object.assign(core_api, { /** * The "headlines" namespace, which is used for headline-channels * which are read-only channels containing messages of type * "headline". * * @namespace api.headlines * @memberOf api */ headlines: { /** * Retrieves a headline-channel or all headline-channels. * * @method api.headlines.get * @param {String|String[]} jids - e.g. 'buddy@example.com' or ['buddy1@example.com', 'buddy2@example.com'] * @param {Object} [attrs] - Attributes to be set on the _converse.ChatBox model. * @param {Boolean} [create=false] - Whether the chat should be created if it's not found. * @returns { Promise<_converse.HeadlinesBox> } */ async get(jids) { let attrs = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; let create = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false; async function _get(jid) { let model = await core_api.chatboxes.get(jid); if (!model && create) { model = await core_api.chatboxes.create(jid, attrs, shared_converse.HeadlinesBox); } else { model = model && model.get('type') === shared_converse.HEADLINES_TYPE ? model : null; if (model && Object.keys(attrs).length) { model.save(attrs); } } return model; } if (jids === undefined) { const chats = await core_api.chatboxes.get(); return chats.filter(c => c.get('type') === shared_converse.HEADLINES_TYPE); } else if (typeof jids === 'string') { return _get(jids); } return Promise.all(jids.map(jid => _get(jid))); } } }); /************************ END API ************************/ } }); ;// CONCATENATED MODULE: ./src/headless/plugins/mam/placeholder.js const placeholder_u = core_converse.env.utils; class MAMPlaceholderMessage extends Model { defaults() { // eslint-disable-line class-methods-use-this return { 'msgid': placeholder_u.getUniqueId(), 'is_ephemeral': false }; } } ;// CONCATENATED MODULE: ./src/headless/shared/rsm.js /** * @module converse-rsm * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) * @description XEP-0059 Result Set Management * Some code taken from the Strophe RSM plugin, licensed under the MIT License * Copyright 2006-2017 Strophe (https://github.com/strophe/strophejs) */ const { Strophe: rsm_Strophe, $build: rsm_$build } = core_converse.env; rsm_Strophe.addNamespace('RSM', 'http://jabber.org/protocol/rsm'); /** * @typedef { Object } RSMQueryParameters * [XEP-0059 RSM](https://xmpp.org/extensions/xep-0059.html) Attributes that can be used to filter query results * @property { String } [after] - The XEP-0359 stanza ID of a message after which messages should be returned. Implies forward paging. * @property { String } [before] - The XEP-0359 stanza ID of a message before which messages should be returned. Implies backward paging. * @property { Integer } [index=0] - The index of the results page to return. * @property { Integer } [max] - The maximum number of items to return. */ const RSM_QUERY_PARAMETERS = ['after', 'before', 'index', 'max']; const rsm_toNumber = v => Number(v); const rsm_toString = v => v.toString(); const RSM_TYPES = { 'after': rsm_toString, 'before': rsm_toString, 'count': rsm_toNumber, 'first': rsm_toString, 'index': rsm_toNumber, 'last': rsm_toString, 'max': rsm_toNumber }; const isUndefined = x => typeof x === 'undefined'; // This array contains both query attributes and response attributes const RSM_ATTRIBUTES = Object.keys(RSM_TYPES); /** * Instances of this class are used to page through query results according to XEP-0059 Result Set Management * @class RSM */ class RSM { static getQueryParameters() { let options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; return lodash_es_pick(options, RSM_QUERY_PARAMETERS); } static parseXMLResult(set) { const result = {}; for (var i = 0; i < RSM_ATTRIBUTES.length; i++) { const attr = RSM_ATTRIBUTES[i]; const elem = set.getElementsByTagName(attr)[0]; if (!isUndefined(elem) && elem !== null) { result[attr] = RSM_TYPES[attr](rsm_Strophe.getText(elem)); if (attr == 'first') { result.index = RSM_TYPES['index'](elem.getAttribute('index')); } } } return result; } /** * Create a new RSM instance * @param { Object } options - Configuration options * @constructor */ constructor() { let options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; this.query = RSM.getQueryParameters(options); this.result = options.xml ? RSM.parseXMLResult(options.xml) : {}; } /** * Returns a `` XML element that confirms to XEP-0059 Result Set Management. * The element is constructed based on the {@link module:converse-rsm~RSMQueryParameters} * that are set on this RSM instance. * @returns { XMLElement } */ toXML() { const xml = rsm_$build('set', { xmlns: rsm_Strophe.NS.RSM }); const reducer = (xml, a) => !isUndefined(this.query[a]) ? xml.c(a).t((this.query[a] || '').toString()).up() : xml; return RSM_QUERY_PARAMETERS.reduce(reducer, xml).tree(); } next(max, before) { const options = Object.assign({}, this.query, { after: this.result.last, before, max }); return new RSM(options); } previous(max, after) { const options = Object.assign({}, this.query, { after, before: this.result.first, max }); return new RSM(options); } } shared_converse.RSM_ATTRIBUTES = RSM_ATTRIBUTES; shared_converse.RSM = RSM; ;// CONCATENATED MODULE: ./src/headless/plugins/mam/api.js const { Strophe: mam_api_Strophe, $iq: mam_api_$iq, dayjs } = core_converse.env; const { NS: api_NS } = mam_api_Strophe; const api_u = core_converse.env.utils; /* harmony default export */ const mam_api = ({ /** * The [XEP-0313](https://xmpp.org/extensions/xep-0313.html) Message Archive Management API * * Enables you to query an XMPP server for archived messages. * * See also the [message-archiving](/docs/html/configuration.html#message-archiving) * option in the configuration settings section, which you'll * usually want to use in conjunction with this API. * * @namespace _converse.api.archive * @memberOf _converse.api */ archive: { /** * @typedef { module:converse-rsm~RSMQueryParameters } MAMFilterParameters * Filter parameters which can be used to filter a MAM XEP-0313 archive * @property { String } [end] - A date string in ISO-8601 format, before which messages should be returned. Implies backward paging. * @property { String } [start] - A date string in ISO-8601 format, after which messages should be returned. Implies forward paging. * @property { String } [with] - A JID against which to match messages, according to either their `to` or `from` attributes. * An item in a MUC archive matches if the publisher of the item matches the JID. * If `with` is omitted, all messages that match the rest of the query will be returned, regardless of to/from * addresses of each message. */ /** * The options that can be passed in to the {@link _converse.api.archive.query } method * @typedef { module:converse-mam~MAMFilterParameters } ArchiveQueryOptions * @property { Boolean } [groupchat=false] - Whether the MAM archive is for a groupchat. */ /** * Query for archived messages. * * The options parameter can also be an instance of * RSM to enable easy querying between results pages. * * @method _converse.api.archive.query * @param { module:converse-mam~ArchiveQueryOptions } options - An object containing query parameters * @throws {Error} An error is thrown if the XMPP server responds with an error. * @returns { Promise } A promise which resolves * to a {@link module:converse-mam~MAMQueryResult } object. * * @example * // Requesting all archived messages * // ================================ * // * // The simplest query that can be made is to simply not pass in any parameters. * // Such a query will return all archived messages for the current user. * * let result; * try { * result = await api.archive.query(); * } catch (e) { * // The query was not successful, perhaps inform the user? * // The IQ stanza returned by the XMPP server is passed in, so that you * // may inspect it and determine what the problem was. * } * // Do something with the messages, like showing them in your webpage. * result.messages.forEach(m => this.showMessage(m)); * * @example * // Requesting all archived messages for a particular contact or room * // ================================================================= * // * // To query for messages sent between the current user and another user or room, * // the query options need to contain the the JID (Jabber ID) of the user or * // room under the `with` key. * * // For a particular user * let result; * try { * result = await api.archive.query({'with': 'john@doe.net'}); * } catch (e) { * // The query was not successful * } * * // For a particular room * let result; * try { * result = await api.archive.query({'with': 'discuss@conference.doglovers.net', 'groupchat': true}); * } catch (e) { * // The query was not successful * } * * @example * // Requesting all archived messages before or after a certain date * // =============================================================== * // * // The `start` and `end` parameters are used to query for messages * // within a certain timeframe. The passed in date values may either be ISO8601 * // formatted date strings, or JavaScript Date objects. * * const options = { * 'with': 'john@doe.net', * 'start': '2010-06-07T00:00:00Z', * 'end': '2010-07-07T13:23:54Z' * }; * let result; * try { * result = await api.archive.query(options); * } catch (e) { * // The query was not successful * } * * @example * // Limiting the amount of messages returned * // ======================================== * // * // The amount of returned messages may be limited with the `max` parameter. * // By default, the messages are returned from oldest to newest. * * // Return maximum 10 archived messages * let result; * try { * result = await api.archive.query({'with': 'john@doe.net', 'max':10}); * } catch (e) { * // The query was not successful * } * * @example * // Paging forwards through a set of archived messages * // ================================================== * // * // When limiting the amount of messages returned per query, you might want to * // repeatedly make a further query to fetch the next batch of messages. * // * // To simplify this usecase for you, the callback method receives not only an array * // with the returned archived messages, but also a special RSM (*Result Set Management*) * // object which contains the query parameters you passed in, as well * // as two utility methods `next`, and `previous`. * // * // When you call one of these utility methods on the returned RSM object, and then * // pass the result into a new query, you'll receive the next or previous batch of * // archived messages. Please note, when calling these methods, pass in an integer * // to limit your results. * * const options = {'with': 'john@doe.net', 'max':10}; * let result; * try { * result = await api.archive.query(options); * } catch (e) { * // The query was not successful * } * // Do something with the messages, like showing them in your webpage. * result.messages.forEach(m => this.showMessage(m)); * * while (!result.complete) { * try { * result = await api.archive.query(Object.assign(options, rsm.next(10).query)); * } catch (e) { * // The query was not successful * } * // Do something with the messages, like showing them in your webpage. * result.messages.forEach(m => this.showMessage(m)); * } * * @example * // Paging backwards through a set of archived messages * // =================================================== * // * // To page backwards through the archive, you need to know the UID of the message * // which you'd like to page backwards from and then pass that as value for the * // `before` parameter. If you simply want to page backwards from the most recent * // message, pass in the `before` parameter with an empty string value `''`. * * let result; * const options = {'before': '', 'max':5}; * try { * result = await api.archive.query(options); * } catch (e) { * // The query was not successful * } * // Do something with the messages, like showing them in your webpage. * result.messages.forEach(m => this.showMessage(m)); * * // Now we query again, to get the previous batch. * try { * result = await api.archive.query(Object.assign(options, rsm.previous(5).query)); * } catch (e) { * // The query was not successful * } * // Do something with the messages, like showing them in your webpage. * result.messages.forEach(m => this.showMessage(m)); * */ async query(options) { if (!core_api.connection.connected()) { throw new Error('Can\'t call `api.archive.query` before having established an XMPP session'); } const attrs = { 'type': 'set' }; if (options && options.groupchat) { if (!options['with']) { throw new Error('You need to specify a "with" value containing ' + 'the chat room JID, when querying groupchat messages.'); } attrs.to = options['with']; } const jid = attrs.to || shared_converse.bare_jid; const supported = await core_api.disco.supports(api_NS.MAM, jid); if (!supported) { headless_log.warn(`Did not fetch MAM archive for ${jid} because it doesn't support ${api_NS.MAM}`); return { 'messages': [] }; } const queryid = api_u.getUniqueId(); const stanza = mam_api_$iq(attrs).c('query', { 'xmlns': api_NS.MAM, 'queryid': queryid }); if (options) { stanza.c('x', { 'xmlns': api_NS.XFORM, 'type': 'submit' }).c('field', { 'var': 'FORM_TYPE', 'type': 'hidden' }).c('value').t(api_NS.MAM).up().up(); if (options['with'] && !options.groupchat) { stanza.c('field', { 'var': 'with' }).c('value').t(options['with']).up().up(); } ['start', 'end'].forEach(t => { if (options[t]) { const date = dayjs(options[t]); if (date.isValid()) { stanza.c('field', { 'var': t }).c('value').t(date.toISOString()).up().up(); } else { throw new TypeError(`archive.query: invalid date provided for: ${t}`); } } }); stanza.up(); const rsm = new RSM(options); if (Object.keys(rsm.query).length) { stanza.cnode(rsm.toXML()); } } const messages = []; const message_handler = shared_converse.connection.addHandler(stanza => { const result = sizzle_default()(`message > result[xmlns="${api_NS.MAM}"]`, stanza).pop(); if (result === undefined || result.getAttribute('queryid') !== queryid) { return true; } const from = stanza.getAttribute('from') || shared_converse.bare_jid; if (options.groupchat) { if (from !== options['with']) { headless_log.warn(`Ignoring alleged groupchat MAM message from ${stanza.getAttribute('from')}`); return true; } } else if (from !== shared_converse.bare_jid) { headless_log.warn(`Ignoring alleged MAM message from ${stanza.getAttribute('from')}`); return true; } messages.push(stanza); return true; }, api_NS.MAM); let error; const timeout = core_api.settings.get('message_archiving_timeout'); const iq_result = await core_api.sendIQ(stanza, timeout, false); if (iq_result === null) { const { __ } = shared_converse; const err_msg = __("Timeout while trying to fetch archived messages."); headless_log.error(err_msg); error = new shared_converse.TimeoutError(err_msg); return { messages, error }; } else if (api_u.isErrorStanza(iq_result)) { const { __ } = shared_converse; const err_msg = __('An error occurred while querying for archived messages.'); headless_log.error(err_msg); headless_log.error(iq_result); error = new Error(err_msg); return { messages, error }; } shared_converse.connection.deleteHandler(message_handler); let rsm; const fin = iq_result && sizzle_default()(`fin[xmlns="${api_NS.MAM}"]`, iq_result).pop(); const complete = (fin === null || fin === void 0 ? void 0 : fin.getAttribute('complete')) === 'true'; const set = sizzle_default()(`set[xmlns="${api_NS.RSM}"]`, fin).pop(); if (set) { rsm = new RSM({ ...options, 'xml': set }); } /** * @typedef { Object } MAMQueryResult * @property { Array } messages * @property { RSM } [rsm] - An instance of {@link RSM}. * You can call `next()` or `previous()` on this instance, * to get the RSM query parameters for the next or previous * page in the result set. * @property { Boolean } complete * @property { Error } [error] */ return { messages, rsm, complete }; } } }); ;// CONCATENATED MODULE: ./src/headless/plugins/mam/utils.js const { Strophe: mam_utils_Strophe, $iq: mam_utils_$iq } = core_converse.env; const { NS: utils_NS } = mam_utils_Strophe; const mam_utils_u = core_converse.env.utils; function onMAMError(iq) { if (iq !== null && iq !== void 0 && iq.querySelectorAll('feature-not-implemented').length) { headless_log.warn(`Message Archive Management (XEP-0313) not supported by ${iq.getAttribute('from')}`); } else { headless_log.error(`Error while trying to set archiving preferences for ${iq.getAttribute('from')}.`); headless_log.error(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. */ function onMAMPreferences(iq, feature) { const preference = sizzle_default()(`prefs[xmlns="${utils_NS.MAM}"]`, iq).pop(); const default_pref = preference.getAttribute('default'); if (default_pref !== core_api.settings.get('message_archiving')) { const stanza = mam_utils_$iq({ 'type': 'set' }).c('prefs', { 'xmlns': utils_NS.MAM, 'default': core_api.settings.get('message_archiving') }); Array.from(preference.children).forEach(child => stanza.cnode(child).up()); // 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. core_api.sendIQ(stanza).then(() => feature.save({ 'preferences': { 'default': core_api.settings.get('message_archiving') } })).catch(shared_converse.onMAMError); } else { feature.save({ 'preferences': { 'default': core_api.settings.get('message_archiving') } }); } } function getMAMPrefsFromFeature(feature) { const prefs = feature.get('preferences') || {}; if (feature.get('var') !== utils_NS.MAM || core_api.settings.get('message_archiving') === undefined) { return; } if (prefs['default'] !== core_api.settings.get('message_archiving')) { core_api.sendIQ(mam_utils_$iq({ 'type': 'get' }).c('prefs', { 'xmlns': utils_NS.MAM })).then(iq => shared_converse.onMAMPreferences(iq, feature)).catch(shared_converse.onMAMError); } } function preMUCJoinMAMFetch(muc) { if (!core_api.settings.get('muc_show_logs_before_join') || !muc.features.get('mam_enabled') || muc.get('prejoin_mam_fetched')) { return; } fetchNewestMessages(muc); muc.save({ 'prejoin_mam_fetched': true }); } async function handleMAMResult(model, result, query, options, should_page) { await core_api.emojis.initialize(); const is_muc = model.get('type') === shared_converse.CHATROOMS_TYPE; const doParseMessage = s => is_muc ? parseMUCMessage(s, model) : parseMessage(s); const messages = await Promise.all(result.messages.map(doParseMessage)); result.messages = messages; /** * Synchronous event which allows listeners to first do some * work based on the MAM result before calling the handlers here. * @event _converse#MAMResult */ const data = { query, 'chatbox': model, messages }; await core_api.trigger('MAMResult', data, { 'synchronous': true }); messages.forEach(m => model.queueMessage(m)); if (result.error) { const event_id = result.error.retry_event_id = mam_utils_u.getUniqueId(); core_api.listen.once(event_id, () => fetchArchivedMessages(model, options, should_page)); model.createMessageFromError(result.error); } } /** * @typedef { Object } MAMOptions * A map of MAM related options that may be passed to fetchArchivedMessages * @param { integer } [options.max] - The maximum number of items to return. * Defaults to "archived_messages_page_size" * @param { string } [options.after] - The XEP-0359 stanza ID of a message * after which messages should be returned. Implies forward paging. * @param { string } [options.before] - The XEP-0359 stanza ID of a message * before which messages should be returned. Implies backward paging. * @param { string } [options.end] - A date string in ISO-8601 format, * before which messages should be returned. Implies backward paging. * @param { string } [options.start] - A date string in ISO-8601 format, * after which messages should be returned. Implies forward paging. * @param { string } [options.with] - The JID of the entity with * which messages were exchanged. * @param { boolean } [options.groupchat] - True if archive in groupchat. */ /** * Fetch XEP-0313 archived messages based on the passed in criteria. * @param { _converse.ChatBox | _converse.ChatRoom } model * @param { MAMOptions } [options] * @param { ('forwards'|'backwards'|null)} [should_page=null] - Determines whether * this function should recursively page through the entire result set if a limited * number of results were returned. */ async function fetchArchivedMessages(model) { let options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; let should_page = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : null; if (model.disable_mam) { return; } const is_muc = model.get('type') === shared_converse.CHATROOMS_TYPE; const mam_jid = is_muc ? model.get('jid') : shared_converse.bare_jid; if (!(await core_api.disco.supports(utils_NS.MAM, mam_jid))) { return; } const max = core_api.settings.get('archived_messages_page_size'); const query = Object.assign({ 'groupchat': is_muc, 'max': max, 'with': model.get('jid') }, options); const result = await core_api.archive.query(query); await handleMAMResult(model, result, query, options, should_page); if (result.rsm && !result.complete) { if (should_page) { if (should_page === 'forwards') { options = result.rsm.next(max, options.before).query; } else if (should_page === 'backwards') { options = result.rsm.previous(max, options.after).query; } return fetchArchivedMessages(model, options, should_page); } else { createPlaceholder(model, options, result); } } } /** * Create a placeholder message which is used to indicate gaps in the history. * @param { _converse.ChatBox | _converse.ChatRoom } model * @param { MAMOptions } options * @param { object } result - The RSM result object */ async function createPlaceholder(model, options, result) { if (options.before == '' && (model.messages.length === 0 || !options.start)) { // Fetching the latest MAM messages with an empty local cache return; } if (options.before && !options.start) { // Infinite scrolling upward return; } if (options.before == null) { // eslint-disable-line no-eq-null // Adding placeholders when paging forwards is not supported yet, // since currently with standard Converse, we only page forwards // when fetching the entire history (i.e. no gaps should arise). return; } const msgs = await Promise.all(result.messages); const { rsm } = result; const key = `stanza_id ${model.get('jid')}`; const adjacent_message = msgs.find(m => m[key] === rsm.result.first); const msg_data = { 'template_hook': 'getMessageTemplate', 'time': new Date(new Date(adjacent_message['time']) - 1).toISOString(), 'before': rsm.result.first, 'start': options.start }; model.messages.add(new MAMPlaceholderMessage(msg_data)); } /** * Fetches messages that might have been archived *after* * the last archived message in our local cache. * @param { _converse.ChatBox | _converse.ChatRoom } */ function fetchNewestMessages(model) { if (model.disable_mam) { return; } const most_recent_msg = model.getMostRecentMessage(); // if clear_messages_on_reconnection is true, than any recent messages // must have been received *after* connection and we instead must query // for earlier messages if (most_recent_msg && !core_api.settings.get('clear_messages_on_reconnection')) { const should_page = core_api.settings.get('mam_request_all_pages'); if (should_page) { const stanza_id = most_recent_msg.get(`stanza_id ${model.get('jid')}`); if (stanza_id) { fetchArchivedMessages(model, { 'after': stanza_id }, 'forwards'); } else { fetchArchivedMessages(model, { 'start': most_recent_msg.get('time') }, 'forwards'); } } else { fetchArchivedMessages(model, { 'before': '', 'start': most_recent_msg.get('time') }); } } else { fetchArchivedMessages(model, { 'before': '' }); } } ;// CONCATENATED MODULE: ./src/headless/plugins/mam/index.js /** * @description XEP-0313 Message Archive Management * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ const { Strophe: mam_Strophe } = core_converse.env; const { NS: mam_NS } = mam_Strophe; core_converse.plugins.add('converse-mam', { dependencies: ['converse-disco', 'converse-muc'], initialize() { core_api.settings.extend({ archived_messages_page_size: '50', mam_request_all_pages: true, message_archiving: undefined, // Supported values are 'always', 'never', 'roster' (https://xmpp.org/extensions/xep-0313.html#prefs) message_archiving_timeout: 20000 // Time (in milliseconds) to wait before aborting MAM request }); Object.assign(core_api, mam_api); // This is mainly done to aid with tests Object.assign(shared_converse, { onMAMError: onMAMError, onMAMPreferences: onMAMPreferences, handleMAMResult: handleMAMResult, MAMPlaceholderMessage: MAMPlaceholderMessage }); /************************ Event Handlers ************************/ core_api.listen.on('addClientFeatures', () => core_api.disco.own.features.add(mam_NS.MAM)); core_api.listen.on('serviceDiscovered', getMAMPrefsFromFeature); core_api.listen.on('chatRoomViewInitialized', view => { if (core_api.settings.get('muc_show_logs_before_join')) { preMUCJoinMAMFetch(view.model); // If we want to show MAM logs before entering the MUC, we need // to be informed once it's clear that this MUC supports MAM. view.model.features.on('change:mam_enabled', () => preMUCJoinMAMFetch(view.model)); } }); core_api.listen.on('enteredNewRoom', muc => muc.features.get('mam_enabled') && fetchNewestMessages(muc)); core_api.listen.on('chatReconnected', chat => { // XXX: For MUCs, we listen to enteredNewRoom instead if (chat.get('type') === shared_converse.PRIVATE_CHAT_TYPE) { fetchNewestMessages(chat); } }); core_api.listen.on('afterMessagesFetched', chat => { // XXX: We don't want to query MAM every time this is triggered // since it's not necessary when the chat is restored from cache. // (given that BOSH or SMACKS will ensure that you get messages // sent during the reload). // With MUCs we can listen for `enteredNewRoom`. if (chat.get('type') === shared_converse.PRIVATE_CHAT_TYPE && !shared_converse.connection.restored) { fetchNewestMessages(chat); } }); } }); ;// CONCATENATED MODULE: ./src/headless/plugins/ping/utils.js const { Strophe: ping_utils_Strophe, $iq: ping_utils_$iq } = core_converse.env; let lastStanzaDate; function utils_onWindowStateChanged(data) { if (data.state === 'visible' && core_api.connection.connected()) { core_api.ping(null, 5000); } } function setLastStanzaDate(date) { lastStanzaDate = date; } function pong(ping) { lastStanzaDate = new Date(); const from = ping.getAttribute('from'); const id = ping.getAttribute('id'); const iq = ping_utils_$iq({ type: 'result', to: from, id: id }); shared_converse.connection.sendIQ(iq); return true; } function registerPongHandler() { const { connection } = shared_converse; if (connection.disco) { core_api.disco.own.features.add(ping_utils_Strophe.NS.PING); } return connection.addHandler(pong, ping_utils_Strophe.NS.PING, "iq", "get"); } function registerPingHandler() { shared_converse.connection.addHandler(() => { if (core_api.settings.get('ping_interval') > 0) { // Handler on each stanza, saves the received date // in order to ping only when needed. lastStanzaDate = new Date(); return true; } }); } function onConnected() { // Wrapper so that we can spy on registerPingHandler in tests registerPongHandler(); registerPingHandler(); } function onEverySecond() { if (shared_converse.isTestEnv() || !core_api.connection.connected()) { return; } const ping_interval = core_api.settings.get('ping_interval'); if (ping_interval > 0) { const now = new Date(); if (!lastStanzaDate) { lastStanzaDate = now; } if ((now - lastStanzaDate) / 1000 > ping_interval) { core_api.ping(); } } } ;// CONCATENATED MODULE: ./src/headless/plugins/ping/api.js const { Strophe: ping_api_Strophe, $iq: ping_api_$iq, u: ping_api_u } = core_converse.env; /* harmony default export */ const ping_api = ({ /** * Pings the service represented by the passed in JID by sending an IQ stanza. * @private * @method api.ping * @param { String } [jid] - The JID of the service to ping * @param { Integer } [timeout] - The amount of time in * milliseconds to wait for a response. The default is 10000; */ async ping(jid, timeout) { // XXX: We could first check here if the server advertised that it supports PING. // However, some servers don't advertise while still responding to pings // // const feature = _converse.disco_entities[_converse.domain].features.findWhere({'var': Strophe.NS.PING}); setLastStanzaDate(new Date()); jid = jid || ping_api_Strophe.getDomainFromJid(shared_converse.bare_jid); if (shared_converse.connection) { const iq = ping_api_$iq({ 'type': 'get', 'to': jid, 'id': ping_api_u.getUniqueId('ping') }).c('ping', { 'xmlns': ping_api_Strophe.NS.PING }); const result = await core_api.sendIQ(iq, timeout || 10000, false); if (result === null) { headless_log.warn(`Timeout while pinging ${jid}`); if (jid === ping_api_Strophe.getDomainFromJid(shared_converse.bare_jid)) { core_api.connection.reconnect(); } } else if (ping_api_u.isErrorStanza(result)) { headless_log.error(`Error while pinging ${jid}`); headless_log.error(result); } return true; } return false; } }); ;// CONCATENATED MODULE: ./src/headless/plugins/ping/index.js /** * @description * Converse.js plugin which add support for application-level pings * as specified in XEP-0199 XMPP Ping. * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ const { Strophe: ping_Strophe } = core_converse.env; ping_Strophe.addNamespace('PING', "urn:xmpp:ping"); core_converse.plugins.add('converse-ping', { initialize() { core_api.settings.extend({ ping_interval: 60 //in seconds }); Object.assign(core_api, ping_api); setInterval(onEverySecond, 1000); core_api.listen.on('connected', onConnected); core_api.listen.on('reconnected', onConnected); core_api.listen.on('windowStateChanged', utils_onWindowStateChanged); } }); ;// CONCATENATED MODULE: ./src/headless/plugins/pubsub.js /** * @module converse-pubsub * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) */ const { Strophe: pubsub_Strophe, $iq: pubsub_$iq } = core_converse.env; pubsub_Strophe.addNamespace('PUBSUB_ERROR', pubsub_Strophe.NS.PUBSUB + "#errors"); core_converse.plugins.add('converse-pubsub', { dependencies: ["converse-disco"], initialize() { /************************ BEGIN API ************************/ // We extend the default converse.js API to add methods specific to MUC groupchats. Object.assign(shared_converse.api, { /** * The "pubsub" namespace groups methods relevant to PubSub * * @namespace _converse.api.pubsub * @memberOf _converse.api */ 'pubsub': { /** * Publshes an item to a PubSub node * * @method _converse.api.pubsub.publish * @param {string} jid The JID of the pubsub service where the node resides. * @param {string} node The node being published to * @param {Strophe.Builder} item The Strophe.Builder representation of the XML element being published * @param {object} options An object representing the publisher options * (see https://xmpp.org/extensions/xep-0060.html#publisher-publish-options) * @param {boolean} strict_options Indicates whether the publisher * options are a strict requirement or not. If they're NOT * strict, then Converse will publish to the node even if * the publish options precondication cannot be met. */ async 'publish'(jid, node, item, options) { let strict_options = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : true; const stanza = pubsub_$iq({ 'from': shared_converse.bare_jid, 'type': 'set', 'to': jid }).c('pubsub', { 'xmlns': pubsub_Strophe.NS.PUBSUB }).c('publish', { 'node': node }).cnode(item.tree()).up().up(); if (options) { jid = jid || shared_converse.bare_jid; if (await core_api.disco.supports(pubsub_Strophe.NS.PUBSUB + '#publish-options', jid)) { stanza.c('publish-options').c('x', { 'xmlns': pubsub_Strophe.NS.XFORM, 'type': 'submit' }).c('field', { 'var': 'FORM_TYPE', 'type': 'hidden' }).c('value').t(`${pubsub_Strophe.NS.PUBSUB}#publish-options`).up().up(); Object.keys(options).forEach(k => stanza.c('field', { 'var': k }).c('value').t(options[k]).up().up()); } else { headless_log.warn(`_converse.api.publish: ${jid} does not support #publish-options, ` + `so we didn't set them even though they were provided.`); } } try { await core_api.sendIQ(stanza); } catch (iq) { if (iq instanceof Element && strict_options && iq.querySelector(`precondition-not-met[xmlns="${pubsub_Strophe.NS.PUBSUB_ERROR}"]`)) { // The publish-options precondition couldn't be // met. We re-publish but without publish-options. const el = stanza.nodeTree; el.querySelector('publish-options').outerHTML = ''; headless_log.warn(`PubSub: Republishing without publish options. ${el.outerHTML}`); await core_api.sendIQ(el); } else { throw iq; } } } } }); /************************ END API ************************/ } }); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isNumber.js /** `Object#toString` result references. */ var isNumber_numberTag = '[object Number]'; /** * Checks if `value` is classified as a `Number` primitive or object. * * **Note:** To exclude `Infinity`, `-Infinity`, and `NaN`, which are * classified as numbers, use the `_.isFinite` method. * * @static * @memberOf _ * @since 0.1.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a number, else `false`. * @example * * _.isNumber(3); * // => true * * _.isNumber(Number.MIN_VALUE); * // => true * * _.isNumber(Infinity); * // => true * * _.isNumber('3'); * // => false */ function isNumber(value) { return typeof value == 'number' || (lodash_es_isObjectLike(value) && _baseGetTag(value) == isNumber_numberTag); } /* harmony default export */ const lodash_es_isNumber = (isNumber); ;// CONCATENATED MODULE: ./node_modules/lodash-es/isNaN.js /** * Checks if `value` is `NaN`. * * **Note:** This method is based on * [`Number.isNaN`](https://mdn.io/Number/isNaN) and is not the same as * global [`isNaN`](https://mdn.io/isNaN) which returns `true` for * `undefined` and other non-number values. * * @static * @memberOf _ * @since 0.1.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is `NaN`, else `false`. * @example * * _.isNaN(NaN); * // => true * * _.isNaN(new Number(NaN)); * // => true * * isNaN(undefined); * // => true * * _.isNaN(undefined); * // => false */ function isNaN_isNaN(value) { // An `NaN` primitive is the only value that is not equal to itself. // Perform the `toStringTag` check first to avoid errors with some // ActiveX objects in IE. return lodash_es_isNumber(value) && value != +value; } /* harmony default export */ const lodash_es_isNaN = (isNaN_isNaN); ;// CONCATENATED MODULE: ./src/headless/plugins/status/status.js const { Strophe: status_Strophe, $pres: status_$pres } = core_converse.env; const XMPPStatus = Model.extend({ defaults() { return { "status": core_api.settings.get("default_state") }; }, initialize() { this.on('change', item => { if (!lodash_es_isObject(item.changed)) { return; } if ('status' in item.changed || 'status_message' in item.changed) { core_api.user.presence.send(this.get('status'), null, this.get('status_message')); } }); }, getNickname() { return core_api.settings.get('nickname'); }, getFullname() { // Gets overridden in converse-vcard return ''; }, async constructPresence(type) { let to = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : null; let status_message = arguments.length > 2 ? arguments[2] : undefined; type = typeof type === 'string' ? type : this.get('status') || core_api.settings.get("default_state"); status_message = typeof status_message === 'string' ? status_message : this.get('status_message'); let presence; const attrs = { to }; if (type === 'unavailable' || type === 'probe' || type === 'error' || type === 'unsubscribe' || type === 'unsubscribed' || type === 'subscribe' || type === 'subscribed') { attrs['type'] = type; presence = status_$pres(attrs); } else if (type === 'offline') { attrs['type'] = 'unavailable'; presence = status_$pres(attrs); } else if (type === 'online') { presence = status_$pres(attrs); } else { presence = status_$pres(attrs).c('show').t(type).up(); } if (status_message) { presence.c('status').t(status_message).up(); } const priority = core_api.settings.get("priority"); presence.c('priority').t(lodash_es_isNaN(Number(priority)) ? 0 : priority).up(); if (shared_converse.idle) { const idle_since = new Date(); idle_since.setSeconds(idle_since.getSeconds() - shared_converse.idle_seconds); presence.c('idle', { xmlns: status_Strophe.NS.IDLE, since: idle_since.toISOString() }); } presence = await core_api.hook('constructedPresence', null, presence); return presence; } }); /* harmony default export */ const status_status = (XMPPStatus); ;// CONCATENATED MODULE: ./src/headless/plugins/status/api.js /* harmony default export */ const status_api = ({ /** * @namespace _converse.api.user.presence * @memberOf _converse.api.user */ presence: { /** * Send out a presence stanza * @method _converse.api.user.presence.send * @param { String } type * @param { String } to * @param { String } [status] - An optional status message * @param { Element[]|Strophe.Builder[]|Element|Strophe.Builder } [child_nodes] * Nodes(s) to be added as child nodes of the `presence` XML element. */ async send(type, to, status, child_nodes) { var _child_nodes; await core_api.waitUntil('statusInitialized'); if (child_nodes && !Array.isArray(child_nodes)) { child_nodes = [child_nodes]; } const model = shared_converse.xmppstatus; const presence = await model.constructPresence(type, to, status); (_child_nodes = child_nodes) === null || _child_nodes === void 0 ? void 0 : _child_nodes.map(c => (c === null || c === void 0 ? void 0 : c.tree()) ?? c).forEach(c => presence.cnode(c).up()); core_api.send(presence); if (['away', 'chat', 'dnd', 'online', 'xa', undefined].includes(type)) { const mucs = await core_api.rooms.get(); mucs.forEach(muc => muc.sendStatusPresence(type, status, child_nodes)); } } }, /** * Set and get the user's chat status, also called their *availability*. * @namespace _converse.api.user.status * @memberOf _converse.api.user */ status: { /** * Return the current user's availability status. * @async * @method _converse.api.user.status.get * @example _converse.api.user.status.get(); */ async get() { await core_api.waitUntil('statusInitialized'); return shared_converse.xmppstatus.get('status'); }, /** * The user's status can be set to one of the following values: * * @async * @method _converse.api.user.status.set * @param {string} value The user's chat status (e.g. 'away', 'dnd', 'offline', 'online', 'unavailable' or 'xa') * @param {string} [message] A custom status message * * @example _converse.api.user.status.set('dnd'); * @example _converse.api.user.status.set('dnd', 'In a meeting'); */ async set(value, message) { const data = { 'status': value }; if (!Object.keys(shared_converse.STATUS_WEIGHTS).includes(value)) { throw new Error('Invalid availability value. See https://xmpp.org/rfcs/rfc3921.html#rfc.section.2.2.2.1'); } if (typeof message === 'string') { data.status_message = message; } await core_api.waitUntil('statusInitialized'); shared_converse.xmppstatus.save(data); }, /** * Set and retrieve the user's custom status message. * * @namespace _converse.api.user.status.message * @memberOf _converse.api.user.status */ message: { /** * @async * @method _converse.api.user.status.message.get * @returns {string} The status message * @example const message = _converse.api.user.status.message.get() */ async get() { await core_api.waitUntil('statusInitialized'); return shared_converse.xmppstatus.get('status_message'); }, /** * @async * @method _converse.api.user.status.message.set * @param {string} status The status message * @example _converse.api.user.status.message.set('In a meeting'); */ async set(status) { await core_api.waitUntil('statusInitialized'); shared_converse.xmppstatus.save({ status_message: status }); } } } }); ;// CONCATENATED MODULE: ./src/headless/plugins/status/utils.js const { Strophe: status_utils_Strophe, $build: status_utils_$build } = core_converse.env; function utils_onStatusInitialized(reconnecting) { /** * Triggered when the user's own chat status has been initialized. * @event _converse#statusInitialized * @example _converse.api.listen.on('statusInitialized', status => { ... }); * @example _converse.api.waitUntil('statusInitialized').then(() => { ... }); */ core_api.trigger('statusInitialized', reconnecting); } function initStatus(reconnecting) { // If there's no xmppstatus obj, then we were never connected to // begin with, so we set reconnecting to false. reconnecting = shared_converse.xmppstatus === undefined ? false : reconnecting; if (reconnecting) { utils_onStatusInitialized(reconnecting); } else { const id = `converse.xmppstatus-${shared_converse.bare_jid}`; shared_converse.xmppstatus = new shared_converse.XMPPStatus({ id }); initStorage(shared_converse.xmppstatus, id, 'session'); shared_converse.xmppstatus.fetch({ 'success': () => utils_onStatusInitialized(reconnecting), 'error': () => utils_onStatusInitialized(reconnecting), 'silent': true }); } } function onUserActivity() { var _converse$connection; /* Resets counters and flags relating to CSI and auto_away/auto_xa */ if (shared_converse.idle_seconds > 0) { shared_converse.idle_seconds = 0; } if (!((_converse$connection = shared_converse.connection) !== null && _converse$connection !== void 0 && _converse$connection.authenticated)) { // We can't send out any stanzas when there's no authenticated connection. // This can happen when the connection reconnects. return; } if (shared_converse.inactive) { shared_converse.sendCSI(shared_converse.ACTIVE); } if (shared_converse.idle) { shared_converse.idle = false; core_api.user.presence.send(); } if (shared_converse.auto_changed_status === true) { shared_converse.auto_changed_status = false; // XXX: we should really remember the original state here, and // then set it back to that... shared_converse.xmppstatus.set('status', core_api.settings.get("default_state")); } } function utils_onEverySecond() { var _converse$connection2; /* An interval handler running every second. * Used for CSI and the auto_away and auto_xa features. */ if (!((_converse$connection2 = shared_converse.connection) !== null && _converse$connection2 !== void 0 && _converse$connection2.authenticated)) { // We can't send out any stanzas when there's no authenticated connection. // This can happen when the connection reconnects. return; } const stat = shared_converse.xmppstatus.get('status'); shared_converse.idle_seconds++; if (core_api.settings.get("csi_waiting_time") > 0 && shared_converse.idle_seconds > core_api.settings.get("csi_waiting_time") && !shared_converse.inactive) { shared_converse.sendCSI(shared_converse.INACTIVE); } if (core_api.settings.get("idle_presence_timeout") > 0 && shared_converse.idle_seconds > core_api.settings.get("idle_presence_timeout") && !shared_converse.idle) { shared_converse.idle = true; core_api.user.presence.send(); } if (core_api.settings.get("auto_away") > 0 && shared_converse.idle_seconds > core_api.settings.get("auto_away") && stat !== 'away' && stat !== 'xa' && stat !== 'dnd') { shared_converse.auto_changed_status = true; shared_converse.xmppstatus.set('status', 'away'); } else if (core_api.settings.get("auto_xa") > 0 && shared_converse.idle_seconds > core_api.settings.get("auto_xa") && stat !== 'xa' && stat !== 'dnd') { shared_converse.auto_changed_status = true; shared_converse.xmppstatus.set('status', 'xa'); } } /** * Send out a Client State Indication (XEP-0352) * @function sendCSI * @param { String } stat - The user's chat status */ function sendCSI(stat) { core_api.send(status_utils_$build(stat, { xmlns: status_utils_Strophe.NS.CSI })); shared_converse.inactive = stat === shared_converse.INACTIVE ? true : false; } function registerIntervalHandler() { /* 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 (core_api.settings.get("auto_away") < 1 && core_api.settings.get("auto_xa") < 1 && core_api.settings.get("csi_waiting_time") < 1 && core_api.settings.get("idle_presence_timeout") < 1) { // Waiting time of less then one second means features aren't used. return; } shared_converse.idle_seconds = 0; shared_converse.auto_changed_status = false; // Was the user's status changed by Converse? const { unloadevent } = shared_converse; window.addEventListener('click', shared_converse.onUserActivity); window.addEventListener('focus', shared_converse.onUserActivity); window.addEventListener('keypress', shared_converse.onUserActivity); window.addEventListener('mousemove', shared_converse.onUserActivity); window.addEventListener(unloadevent, shared_converse.onUserActivity, { 'once': true, 'passive': true }); window.addEventListener(unloadevent, () => { var _converse$session; return (_converse$session = shared_converse.session) === null || _converse$session === void 0 ? void 0 : _converse$session.save('active', false); }); shared_converse.everySecondTrigger = window.setInterval(shared_converse.onEverySecond, 1000); } function addStatusToMUCJoinPresence(_, stanza) { const { xmppstatus } = shared_converse; const status = xmppstatus.get('status'); if (['away', 'chat', 'dnd', 'xa'].includes(status)) { stanza.c('show').t(status).up(); } const status_message = xmppstatus.get('status_message'); if (status_message) { stanza.c('status').t(status_message).up(); } return stanza; } ;// CONCATENATED MODULE: ./src/headless/plugins/status/index.js /** * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) */ const { Strophe: plugins_status_Strophe } = core_converse.env; plugins_status_Strophe.addNamespace('IDLE', 'urn:xmpp:idle:1'); core_converse.plugins.add('converse-status', { initialize() { core_api.settings.extend({ auto_away: 0, // Seconds after which user status is set to 'away' auto_xa: 0, // Seconds after which user status is set to 'xa' csi_waiting_time: 0, // Support for XEP-0352. Seconds before client is considered idle and CSI is sent out. default_state: 'online', idle_presence_timeout: 300, // Seconds after which an idle presence is sent priority: 0 }); core_api.promises.add(['statusInitialized']); shared_converse.XMPPStatus = status_status; shared_converse.onUserActivity = onUserActivity; shared_converse.onEverySecond = utils_onEverySecond; shared_converse.sendCSI = sendCSI; shared_converse.registerIntervalHandler = registerIntervalHandler; Object.assign(shared_converse.api.user, status_api); if (core_api.settings.get("idle_presence_timeout") > 0) { core_api.listen.on('addClientFeatures', () => core_api.disco.own.features.add(plugins_status_Strophe.NS.IDLE)); } core_api.listen.on('presencesInitialized', reconnecting => { if (!reconnecting) { shared_converse.registerIntervalHandler(); } }); core_api.listen.on('clearSession', () => { if (shared_converse.shouldClearCache() && shared_converse.xmppstatus) { shared_converse.xmppstatus.destroy(); delete shared_converse.xmppstatus; core_api.promises.add(['statusInitialized']); } }); core_api.listen.on('connected', () => initStatus(false)); core_api.listen.on('reconnected', () => initStatus(true)); core_api.listen.on('constructedMUCPresence', addStatusToMUCJoinPresence); } }); ;// CONCATENATED MODULE: ./src/headless/plugins/roster/filter.js const RosterFilter = Model.extend({ initialize() { this.set({ 'filter_text': '', 'filter_type': 'contacts', 'chat_state': 'online' }); } }); ;// CONCATENATED MODULE: ./src/headless/plugins/roster/utils.js const { $pres: utils_$pres } = core_converse.env; function initRoster() { // Initialize the collections that represent the roster contacts and groups const roster = shared_converse.roster = new shared_converse.RosterContacts(); let id = `converse.contacts-${shared_converse.bare_jid}`; initStorage(roster, id); const filter = shared_converse.roster_filter = new RosterFilter(); filter.id = `_converse.rosterfilter-${shared_converse.bare_jid}`; initStorage(filter, filter.id); filter.fetch(); id = `converse-roster-model-${shared_converse.bare_jid}`; roster.data = new Model(); roster.data.id = id; initStorage(roster.data, id); roster.data.fetch(); /** * Triggered once the `_converse.RosterContacts` * been created, but not yet populated with data. * This event is useful when you want to create views for these collections. * @event _converse#chatBoxMaximized * @example _converse.api.listen.on('rosterInitialized', () => { ... }); * @example _converse.api.waitUntil('rosterInitialized').then(() => { ... }); */ core_api.trigger('rosterInitialized'); } /** * Fetch all the roster groups, and then the roster contacts. * Emit an event after fetching is done in each case. * @private * @param { Bool } ignore_cache - If set to to true, the local cache * will be ignored it's guaranteed that the XMPP server * will be queried for the roster. */ async function populateRoster() { let ignore_cache = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : false; if (ignore_cache) { shared_converse.send_initial_presence = true; } try { await shared_converse.roster.fetchRosterContacts(); core_api.trigger('rosterContactsFetched'); } catch (reason) { headless_log.error(reason); } finally { shared_converse.send_initial_presence && core_api.user.presence.send(); } } function updateUnreadCounter(chatbox) { var _converse$roster; const contact = (_converse$roster = shared_converse.roster) === null || _converse$roster === void 0 ? void 0 : _converse$roster.get(chatbox.get('jid')); contact === null || contact === void 0 ? void 0 : contact.save({ 'num_unread': chatbox.get('num_unread') }); } function registerPresenceHandler() { unregisterPresenceHandler(); shared_converse.presence_ref = shared_converse.connection.addHandler(presence => { shared_converse.roster.presenceHandler(presence); return true; }, null, 'presence', null); } function unregisterPresenceHandler() { if (shared_converse.presence_ref !== undefined) { shared_converse.connection.deleteHandler(shared_converse.presence_ref); delete shared_converse.presence_ref; } } async function clearPresences() { var _converse$presences; await ((_converse$presences = shared_converse.presences) === null || _converse$presences === void 0 ? void 0 : _converse$presences.clearStore()); } /** * Roster specific event handler for the clearSession event */ async function utils_onClearSession() { await clearPresences(); if (shared_converse.shouldClearCache()) { if (shared_converse.rostergroups) { await shared_converse.rostergroups.clearStore(); delete shared_converse.rostergroups; } if (shared_converse.roster) { var _converse$roster$data; (_converse$roster$data = shared_converse.roster.data) === null || _converse$roster$data === void 0 ? void 0 : _converse$roster$data.destroy(); await shared_converse.roster.clearStore(); delete shared_converse.roster; } } } /** * Roster specific event handler for the presencesInitialized event * @param { Boolean } reconnecting */ function onPresencesInitialized(reconnecting) { if (reconnecting) { /** * Similar to `rosterInitialized`, but instead pertaining to reconnection. * This event indicates that the roster and its groups are now again * available after Converse.js has reconnected. * @event _converse#rosterReadyAfterReconnection * @example _converse.api.listen.on('rosterReadyAfterReconnection', () => { ... }); */ core_api.trigger('rosterReadyAfterReconnection'); } else { initRoster(); } shared_converse.roster.onConnected(); registerPresenceHandler(); populateRoster(!shared_converse.connection.restored); } /** * Roster specific event handler for the statusInitialized event * @param { Boolean } reconnecting */ async function roster_utils_onStatusInitialized(reconnecting) { if (reconnecting) { // When reconnecting and not resuming a previous session, // we clear all cached presence data, since it might be stale // and we'll receive new presence updates !shared_converse.connection.hasResumed() && (await clearPresences()); } else { shared_converse.presences = new shared_converse.Presences(); const id = `converse.presences-${shared_converse.bare_jid}`; initStorage(shared_converse.presences, id, 'session'); // We might be continuing an existing session, so we fetch // cached presence data. shared_converse.presences.fetch(); } /** * Triggered once the _converse.Presences collection has been * initialized and its cached data fetched. * Returns a boolean indicating whether this event has fired due to * Converse having reconnected. * @event _converse#presencesInitialized * @type { bool } * @example _converse.api.listen.on('presencesInitialized', reconnecting => { ... }); */ core_api.trigger('presencesInitialized', reconnecting); } /** * Roster specific event handler for the chatBoxesInitialized event */ function onChatBoxesInitialized() { shared_converse.chatboxes.on('change:num_unread', updateUnreadCounter); shared_converse.chatboxes.on('add', chatbox => { if (chatbox.get('type') === shared_converse.PRIVATE_CHAT_TYPE) { chatbox.setRosterContact(chatbox.get('jid')); } }); } /** * Roster specific handler for the rosterContactsFetched promise */ function onRosterContactsFetched() { shared_converse.roster.on('add', contact => { // When a new contact is added, check if we already have a // chatbox open for it, and if so attach it to the chatbox. const chatbox = shared_converse.chatboxes.findWhere({ 'jid': contact.get('jid') }); chatbox === null || chatbox === void 0 ? void 0 : chatbox.setRosterContact(contact.get('jid')); }); } /** * Reject or cancel another user's subscription to our presence updates. * @function rejectPresenceSubscription * @param { String } jid - The Jabber ID of the user whose subscription is being canceled * @param { String } message - An optional message to the user */ function rejectPresenceSubscription(jid, message) { const pres = utils_$pres({ to: jid, type: "unsubscribed" }); if (message && message !== "") { pres.c("status").t(message); } core_api.send(pres); } function contactsComparator(contact1, contact2) { const status1 = contact1.presence.get('show') || 'offline'; const status2 = contact2.presence.get('show') || 'offline'; if (shared_converse.STATUS_WEIGHTS[status1] === shared_converse.STATUS_WEIGHTS[status2]) { const name1 = contact1.getDisplayName().toLowerCase(); const name2 = contact2.getDisplayName().toLowerCase(); return name1 < name2 ? -1 : name1 > name2 ? 1 : 0; } else { return shared_converse.STATUS_WEIGHTS[status1] < shared_converse.STATUS_WEIGHTS[status2] ? -1 : 1; } } function groupsComparator(a, b) { const HEADER_WEIGHTS = {}; HEADER_WEIGHTS[shared_converse.HEADER_UNREAD] = 0; HEADER_WEIGHTS[shared_converse.HEADER_REQUESTING_CONTACTS] = 1; HEADER_WEIGHTS[shared_converse.HEADER_CURRENT_CONTACTS] = 2; HEADER_WEIGHTS[shared_converse.HEADER_UNGROUPED] = 3; HEADER_WEIGHTS[shared_converse.HEADER_PENDING_CONTACTS] = 4; const WEIGHTS = HEADER_WEIGHTS; const special_groups = Object.keys(HEADER_WEIGHTS); const a_is_special = special_groups.includes(a); const b_is_special = special_groups.includes(b); if (!a_is_special && !b_is_special) { return a.toLowerCase() < b.toLowerCase() ? -1 : a.toLowerCase() > b.toLowerCase() ? 1 : 0; } else if (a_is_special && b_is_special) { return WEIGHTS[a] < WEIGHTS[b] ? -1 : WEIGHTS[a] > WEIGHTS[b] ? 1 : 0; } else if (!a_is_special && b_is_special) { const a_header = shared_converse.HEADER_CURRENT_CONTACTS; return WEIGHTS[a_header] < WEIGHTS[b] ? -1 : WEIGHTS[a_header] > WEIGHTS[b] ? 1 : 0; } else if (a_is_special && !b_is_special) { const b_header = shared_converse.HEADER_CURRENT_CONTACTS; return WEIGHTS[a] < WEIGHTS[b_header] ? -1 : WEIGHTS[a] > WEIGHTS[b_header] ? 1 : 0; } } ;// CONCATENATED MODULE: ./src/headless/plugins/roster/contact.js const { Strophe: contact_Strophe, $iq: contact_$iq, $pres: contact_$pres } = core_converse.env; /** * @class * @namespace RosterContact */ const RosterContact = Model.extend({ idAttribute: 'jid', defaults: { 'chat_state': undefined, 'groups': [], 'image': shared_converse.DEFAULT_IMAGE, 'image_type': shared_converse.DEFAULT_IMAGE_TYPE, 'num_unread': 0, 'status': undefined }, async initialize(attributes) { this.initialized = getOpenPromise(); this.setPresence(); const { jid } = attributes; this.set({ ...attributes, ...{ 'jid': contact_Strophe.getBareJidFromJid(jid).toLowerCase(), 'user_id': contact_Strophe.getNodeFromJid(jid) } }); /** * When a contact's presence status has changed. * The presence status is either `online`, `offline`, `dnd`, `away` or `xa`. * @event _converse#contactPresenceChanged * @type { _converse.RosterContact } * @example _converse.api.listen.on('contactPresenceChanged', contact => { ... }); */ this.listenTo(this.presence, 'change:show', () => core_api.trigger('contactPresenceChanged', this)); this.listenTo(this.presence, 'change:show', () => this.trigger('presenceChanged')); /** * Synchronous event which provides a hook for further initializing a RosterContact * @event _converse#rosterContactInitialized * @param { _converse.RosterContact } contact */ await core_api.trigger('rosterContactInitialized', this, { 'Synchronous': true }); this.initialized.resolve(); }, setPresence() { const jid = this.get('jid'); this.presence = shared_converse.presences.findWhere(jid) || shared_converse.presences.create({ jid }); }, openChat() { const attrs = this.attributes; core_api.chats.open(attrs.jid, attrs, true); }, /** * Return a string of tab-separated values that are to be used when * matching against filter text. * * The goal is to be able to filter against the VCard fullname, * roster nickname and JID. * @returns { String } Lower-cased, tab-separated values */ getFilterCriteria() { const nick = this.get('nickname'); const jid = this.get('jid'); let criteria = this.getDisplayName(); criteria = !criteria.includes(jid) ? criteria.concat(` ${jid}`) : criteria; criteria = !criteria.includes(nick) ? criteria.concat(` ${nick}`) : criteria; return criteria.toLowerCase(); }, getDisplayName() { // Gets overridden in converse-vcard where the fullname is may be returned if (this.get('nickname')) { return this.get('nickname'); } else { return this.get('jid'); } }, getFullname() { // Gets overridden in converse-vcard where the fullname may be returned return this.get('jid'); }, /** * Send a presence subscription request to this roster contact * @private * @method _converse.RosterContacts#subscribe * @param { String } message - An optional message to explain the * reason for the subscription request. */ subscribe(message) { const pres = contact_$pres({ to: this.get('jid'), type: "subscribe" }); if (message && message !== "") { pres.c("status").t(message).up(); } const nick = shared_converse.xmppstatus.getNickname() || shared_converse.xmppstatus.getFullname(); if (nick) { pres.c('nick', { 'xmlns': contact_Strophe.NS.NICK }).t(nick).up(); } core_api.send(pres); this.save('ask', "subscribe"); // ask === 'subscribe' Means we have asked to subscribe to them. return this; }, /** * 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 * @private * @method _converse.RosterContacts#ackSubscribe */ ackSubscribe() { core_api.send(contact_$pres({ 'type': 'subscribe', 'to': this.get('jid') })); }, /** * 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. * @private * @method _converse.RosterContacts#ackUnsubscribe * @param { String } jid - The Jabber ID of the user who is unsubscribing */ ackUnsubscribe() { core_api.send(contact_$pres({ 'type': 'unsubscribe', 'to': this.get('jid') })); this.removeFromRoster(); this.destroy(); }, /** * Unauthorize this contact's presence subscription * @private * @method _converse.RosterContacts#unauthorize * @param { String } message - Optional message to send to the person being unauthorized */ unauthorize(message) { rejectPresenceSubscription(this.get('jid'), message); return this; }, /** * Authorize presence subscription * @private * @method _converse.RosterContacts#authorize * @param { String } message - Optional message to send to the person being authorized */ authorize(message) { const pres = contact_$pres({ 'to': this.get('jid'), 'type': "subscribed" }); if (message && message !== "") { pres.c("status").t(message); } core_api.send(pres); return this; }, /** * Instruct the XMPP server to remove this contact from our roster * @private * @method _converse.RosterContacts# * @returns { Promise } */ removeFromRoster() { const iq = contact_$iq({ type: 'set' }).c('query', { xmlns: contact_Strophe.NS.ROSTER }).c('item', { jid: this.get('jid'), subscription: "remove" }); return core_api.sendIQ(iq); } }); /* harmony default export */ const contact = (RosterContact); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseSum.js /** * The base implementation of `_.sum` and `_.sumBy` without support for * iteratee shorthands. * * @private * @param {Array} array The array to iterate over. * @param {Function} iteratee The function invoked per iteration. * @returns {number} Returns the sum. */ function baseSum(array, iteratee) { var result, index = -1, length = array.length; while (++index < length) { var current = iteratee(array[index]); if (current !== undefined) { result = result === undefined ? current : (result + current); } } return result; } /* harmony default export */ const _baseSum = (baseSum); ;// CONCATENATED MODULE: ./node_modules/lodash-es/sum.js /** * Computes the sum of the values in `array`. * * @static * @memberOf _ * @since 3.4.0 * @category Math * @param {Array} array The array to iterate over. * @returns {number} Returns the sum. * @example * * _.sum([4, 2, 8, 6]); * // => 20 */ function sum(array) { return (array && array.length) ? _baseSum(array, lodash_es_identity) : 0; } /* harmony default export */ const lodash_es_sum = (sum); ;// CONCATENATED MODULE: ./src/headless/plugins/roster/contacts.js const { Strophe: contacts_Strophe, $iq: contacts_$iq, sizzle: contacts_sizzle, u: contacts_u } = core_converse.env; const RosterContacts = Collection.extend({ model: contact, initialize() { const id = `roster.state-${shared_converse.bare_jid}-${this.get('jid')}`; this.state = new Model({ id, 'collapsed_groups': [] }); initStorage(this.state, id); this.state.fetch(); }, 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() { // Register a handler for roster IQ "set" stanzas, which update // roster contacts. shared_converse.connection.addHandler(iq => { shared_converse.roster.onRosterPush(iq); return true; }, contacts_Strophe.NS.ROSTER, 'iq', "set"); }, registerRosterXHandler() { // Register a handler for RosterX message stanzas, which are // used to suggest roster contacts to a user. let t = 0; shared_converse.connection.addHandler(function (msg) { window.setTimeout(function () { shared_converse.connection.flush(); shared_converse.roster.subscribeToSuggestedItems.bind(shared_converse.roster)(msg); }, t); t += msg.querySelectorAll('item').length * 250; return true; }, contacts_Strophe.NS.ROSTERX, 'message', null); }, /** * Fetches the roster contacts, first by trying the browser cache, * and if that's empty, then by querying the XMPP server. * @private * @returns {promise} Promise which resolves once the contacts have been fetched. */ async fetchRosterContacts() { const result = await new Promise((resolve, reject) => { this.fetch({ 'add': true, 'silent': true, 'success': resolve, 'error': (_, e) => reject(e) }); }); if (contacts_u.isErrorObject(result)) { headless_log.error(result); // Force a full roster refresh shared_converse.session.save('roster_cached', false); this.data.save('version', undefined); } if (shared_converse.session.get('roster_cached')) { /** * The contacts roster has been retrieved from the local cache (`sessionStorage`). * @event _converse#cachedRoster * @type { _converse.RosterContacts } * @example _converse.api.listen.on('cachedRoster', (items) => { ... }); * @example _converse.api.waitUntil('cachedRoster').then(items => { ... }); */ core_api.trigger('cachedRoster', result); } else { shared_converse.send_initial_presence = true; return shared_converse.roster.fetchFromServer(); } }, subscribeToSuggestedItems(msg) { Array.from(msg.querySelectorAll('item')).forEach(item => { if (item.getAttribute('action') === 'add') { shared_converse.roster.addAndSubscribe(item.getAttribute('jid'), shared_converse.xmppstatus.getNickname() || shared_converse.xmppstatus.getFullname()); } }); return true; }, isSelf(jid) { return contacts_u.isSameBareJID(jid, shared_converse.connection.jid); }, /** * Add a roster contact and then once we have confirmation from * the XMPP server we subscribe to that contact's presence updates. * @private * @method _converse.RosterContacts#addAndSubscribe * @param { String } jid - The Jabber ID of the user being added and subscribed to. * @param { String } name - The name of that user * @param { Array.String } groups - Any roster groups the user might belong to * @param { String } message - An optional message to explain the reason for the subscription request. * @param { Object } attributes - Any additional attributes to be stored on the user's model. */ async addAndSubscribe(jid, name, groups, message, attributes) { const contact = await this.addContactToRoster(jid, name, groups, attributes); if (contact instanceof shared_converse.RosterContact) { contact.subscribe(message); } }, /** * Send an IQ stanza to the XMPP server to add a new roster contact. * @private * @method _converse.RosterContacts#sendContactAddIQ * @param { String } jid - The Jabber ID of the user being added * @param { String } name - The name of that user * @param { Array.String } groups - Any roster groups the user might belong to * @param { Function } callback - A function to call once the IQ is returned * @param { Function } errback - A function to call if an error occurred */ sendContactAddIQ(jid, name, groups) { name = name ? name : null; const iq = contacts_$iq({ 'type': 'set' }).c('query', { 'xmlns': contacts_Strophe.NS.ROSTER }).c('item', { jid, name }); groups.forEach(g => iq.c('group').t(g).up()); return core_api.sendIQ(iq); }, /** * 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. * @private * @method _converse.RosterContacts#addContactToRoster * @param { String } jid - The Jabber ID of the user being added and subscribed to. * @param { String } name - The name of that user * @param { Array.String } groups - Any roster groups the user might belong to * @param { Object } attributes - Any additional attributes to be stored on the user's model. */ async addContactToRoster(jid, name, groups, attributes) { await core_api.waitUntil('rosterContactsFetched'); groups = groups || []; try { await this.sendContactAddIQ(jid, name, groups); } catch (e) { const { __ } = shared_converse; headless_log.error(e); alert(__('Sorry, there was an error while trying to add %1$s as a contact.', name || jid)); return e; } return this.create(Object.assign({ 'ask': undefined, 'nickname': name, groups, jid, 'requesting': false, 'subscription': 'none' }, attributes), { 'sort': false }); }, async subscribeBack(bare_jid, presence) { const contact = this.get(bare_jid); if (contact instanceof shared_converse.RosterContact) { contact.authorize().subscribe(); } else { var _sizzle$pop; // Can happen when a subscription is retried or roster was deleted const nickname = ((_sizzle$pop = contacts_sizzle(`nick[xmlns="${contacts_Strophe.NS.NICK}"]`, presence).pop()) === null || _sizzle$pop === void 0 ? void 0 : _sizzle$pop.textContent) || null; const contact = await this.addContactToRoster(bare_jid, nickname, [], { 'subscription': 'from' }); if (contact instanceof shared_converse.RosterContact) { contact.authorize().subscribe(); } } }, getNumOnlineContacts() { const ignored = ['offline', 'unavailable']; return lodash_es_sum(this.models.filter(m => !ignored.includes(m.presence.get('show')))); }, /** * Handle roster updates from the XMPP server. * See: https://xmpp.org/rfcs/rfc6121.html#roster-syntax-actions-push * @private * @method _converse.RosterContacts#onRosterPush * @param { XMLElement } IQ - The IQ stanza received from the XMPP server. */ onRosterPush(iq) { const id = iq.getAttribute('id'); const from = iq.getAttribute('from'); if (from && from !== shared_converse.bare_jid) { // https://tools.ietf.org/html/rfc6121#page-15 // // A receiving client MUST ignore the stanza unless it has no 'from' // attribute (i.e., implicitly from the bare JID of the user's // account) or it has a 'from' attribute whose value matches the // user's bare JID . headless_log.warn(`Ignoring roster illegitimate roster push message from ${iq.getAttribute('from')}`); return; } core_api.send(contacts_$iq({ type: 'result', id, from: shared_converse.connection.jid })); const query = contacts_sizzle(`query[xmlns="${contacts_Strophe.NS.ROSTER}"]`, iq).pop(); this.data.save('version', query.getAttribute('ver')); const items = contacts_sizzle(`item`, query); if (items.length > 1) { headless_log.error(iq); throw new Error('Roster push query may not contain more than one "item" element.'); } if (items.length === 0) { headless_log.warn(iq); headless_log.warn('Received a roster push stanza without an "item" element.'); return; } this.updateContact(items.pop()); /** * When the roster receives a push event from server (i.e. new entry in your contacts roster). * @event _converse#rosterPush * @type { XMLElement } * @example _converse.api.listen.on('rosterPush', iq => { ... }); */ core_api.trigger('rosterPush', iq); return; }, rosterVersioningSupported() { return core_api.disco.stream.getFeature('ver', 'urn:xmpp:features:rosterver') && this.data.get('version'); }, /** * Fetch the roster from the XMPP server * @private * @emits _converse#roster * @returns {promise} */ async fetchFromServer() { const stanza = contacts_$iq({ 'type': 'get', 'id': contacts_u.getUniqueId('roster') }).c('query', { xmlns: contacts_Strophe.NS.ROSTER }); if (this.rosterVersioningSupported()) { stanza.attrs({ 'ver': this.data.get('version') }); } const iq = await core_api.sendIQ(stanza, null, false); if (iq.getAttribute('type') === 'result') { const query = contacts_sizzle(`query[xmlns="${contacts_Strophe.NS.ROSTER}"]`, iq).pop(); if (query) { const items = contacts_sizzle(`item`, query); if (!this.data.get('version')) { // We're getting the full roster, so remove all cached // contacts that aren't included in it. const jids = items.map(item => item.getAttribute('jid')); this.models.forEach(m => !m.get('requesting') && !jids.includes(m.get('jid')) && m.destroy()); } items.forEach(item => this.updateContact(item)); this.data.save('version', query.getAttribute('ver')); } } else if (!contacts_u.isServiceUnavailableError(iq)) { // Some unknown error happened, so we will try to fetch again if the page reloads. headless_log.error(iq); headless_log.error("Error while trying to fetch roster from the server"); return; } shared_converse.session.save('roster_cached', true); /** * When the roster has been received from the XMPP server. * See also the `cachedRoster` event further up, which gets called instead of * `roster` if its already in `sessionStorage`. * @event _converse#roster * @type { XMLElement } * @example _converse.api.listen.on('roster', iq => { ... }); * @example _converse.api.waitUntil('roster').then(iq => { ... }); */ core_api.trigger('roster', iq); }, /* Update or create RosterContact models based on the given `item` XML * node received in the resulting IQ stanza from the server. * @private * @param { XMLElement } item */ updateContact(item) { const jid = item.getAttribute('jid'); const contact = this.get(jid); const subscription = item.getAttribute("subscription"); const ask = item.getAttribute("ask"); const groups = [...new Set(contacts_sizzle('group', item).map(e => e.textContent))]; if (!contact) { if (subscription === "none" && ask === null || subscription === "remove") { return; // We're lazy when adding contacts. } this.create({ 'ask': ask, 'nickname': item.getAttribute("name"), 'groups': groups, 'jid': jid, 'subscription': subscription }, { sort: false }); } else { if (subscription === "remove") { return contact.destroy(); } // 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, 'nickname': item.getAttribute("name"), 'requesting': null, 'groups': groups }); } }, createRequestingContact(presence) { var _sizzle$pop2; const bare_jid = contacts_Strophe.getBareJidFromJid(presence.getAttribute('from')); const nickname = ((_sizzle$pop2 = contacts_sizzle(`nick[xmlns="${contacts_Strophe.NS.NICK}"]`, presence).pop()) === null || _sizzle$pop2 === void 0 ? void 0 : _sizzle$pop2.textContent) || null; const user_data = { 'jid': bare_jid, 'subscription': 'none', 'ask': null, 'requesting': true, 'nickname': nickname }; /** * Triggered when someone has requested to subscribe to your presence (i.e. to be your contact). * @event _converse#contactRequest * @type { _converse.RosterContact } * @example _converse.api.listen.on('contactRequest', contact => { ... }); */ core_api.trigger('contactRequest', this.create(user_data)); }, handleIncomingSubscription(presence) { const jid = presence.getAttribute('from'), bare_jid = contacts_Strophe.getBareJidFromJid(jid), contact = this.get(bare_jid); if (!core_api.settings.get('allow_contact_requests')) { const { __ } = shared_converse; rejectPresenceSubscription(jid, __("This client does not allow presence subscriptions")); } if (core_api.settings.get('auto_subscribe')) { if (!contact || contact.get('subscription') !== 'to') { this.subscribeBack(bare_jid, presence); } else { contact.authorize(); } } else { if (contact) { if (contact.get('subscription') !== 'none') { contact.authorize(); } else if (contact.get('ask') === "subscribe") { contact.authorize(); } } else { this.createRequestingContact(presence); } } }, handleOwnPresence(presence) { const jid = presence.getAttribute('from'), resource = contacts_Strophe.getResourceFromJid(jid), presence_type = presence.getAttribute('type'); if (shared_converse.connection.jid !== jid && presence_type !== 'unavailable' && (core_api.settings.get('synchronize_availability') === true || core_api.settings.get('synchronize_availability') === resource)) { var _presence$querySelect, _presence$querySelect2; // Another resource has changed its status and // synchronize_availability option set to update, // we'll update ours as well. const show = ((_presence$querySelect = presence.querySelector('show')) === null || _presence$querySelect === void 0 ? void 0 : _presence$querySelect.textContent) || 'online'; shared_converse.xmppstatus.save({ 'status': show }, { 'silent': true }); const status_message = (_presence$querySelect2 = presence.querySelector('status')) === null || _presence$querySelect2 === void 0 ? void 0 : _presence$querySelect2.textContent; if (status_message) { shared_converse.xmppstatus.save({ 'status_message': status_message }); } } if (shared_converse.jid === jid && presence_type === 'unavailable') { // XXX: We've received an "unavailable" presence from our // own resource. Apparently this happens due to a // Prosody bug, whereby we send an IQ stanza to remove // a roster contact, and Prosody then sends // "unavailable" globally, instead of directed to the // particular user that's removed. // // Here is the bug report: https://prosody.im/issues/1121 // // I'm not sure whether this might legitimately happen // in other cases. // // As a workaround for now we simply send our presence again, // otherwise we're treated as offline. core_api.user.presence.send(); } }, presenceHandler(presence) { var _presence$querySelect3; const presence_type = presence.getAttribute('type'); if (presence_type === 'error') { return true; } const jid = presence.getAttribute('from'), bare_jid = contacts_Strophe.getBareJidFromJid(jid); if (this.isSelf(bare_jid)) { return this.handleOwnPresence(presence); } else if (contacts_sizzle(`query[xmlns="${contacts_Strophe.NS.MUC}"]`, presence).length) { return; // Ignore MUC } const status_message = (_presence$querySelect3 = presence.querySelector('status')) === null || _presence$querySelect3 === void 0 ? void 0 : _presence$querySelect3.textContent; const contact = this.get(bare_jid); if (contact && status_message !== contact.get('status')) { contact.save({ 'status': status_message }); } 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) { const resource = contacts_Strophe.getResourceFromJid(jid); contact.presence.removeResource(resource); } else if (contact) { // presence_type is undefined contact.presence.addResource(presence); } } }); /* harmony default export */ const contacts = (RosterContacts); ;// CONCATENATED MODULE: ./src/headless/plugins/roster/api.js const { Strophe: roster_api_Strophe } = core_converse.env; /* harmony default export */ const roster_api = ({ /** * @namespace _converse.api.contacts * @memberOf _converse.api */ contacts: { /** * This method is used to retrieve roster contacts. * * @method _converse.api.contacts.get * @params {(string[]|string)} jid|jids The JID or JIDs of * the contacts to be returned. * @returns {promise} Promise which resolves with the * _converse.RosterContact (or an array of them) representing the contact. * * @example * // Fetch a single contact * _converse.api.listen.on('rosterContactsFetched', function () { * const contact = await _converse.api.contacts.get('buddy@example.com') * // ... * }); * * @example * // To get multiple contacts, pass in an array of JIDs: * _converse.api.listen.on('rosterContactsFetched', function () { * const contacts = await _converse.api.contacts.get( * ['buddy1@example.com', 'buddy2@example.com'] * ) * // ... * }); * * @example * // To return all contacts, simply call ``get`` without any parameters: * _converse.api.listen.on('rosterContactsFetched', function () { * const contacts = await _converse.api.contacts.get(); * // ... * }); */ async get(jids) { await core_api.waitUntil('rosterContactsFetched'); const _getter = jid => shared_converse.roster.get(roster_api_Strophe.getBareJidFromJid(jid)); if (jids === undefined) { jids = shared_converse.roster.pluck('jid'); } else if (typeof jids === 'string') { return _getter(jids); } return jids.map(_getter); }, /** * Add a contact. * * @method _converse.api.contacts.add * @param {string} jid The JID of the contact to be added * @param {string} [name] A custom name to show the user by in the roster * @example * _converse.api.contacts.add('buddy@example.com') * @example * _converse.api.contacts.add('buddy@example.com', 'Buddy') */ async add(jid, name) { await core_api.waitUntil('rosterContactsFetched'); if (typeof jid !== 'string' || !jid.includes('@')) { throw new TypeError('contacts.add: invalid jid'); } return shared_converse.roster.addAndSubscribe(jid, name); } } }); ;// CONCATENATED MODULE: ./src/headless/plugins/roster/presence.js const { Strophe: presence_Strophe, dayjs: presence_dayjs, sizzle: presence_sizzle } = core_converse.env; const Resource = Model.extend({ 'idAttribute': 'name' }); const Resources = Collection.extend({ 'model': Resource }); const Presence = Model.extend({ idAttribute: 'jid', defaults: { 'show': 'offline' }, initialize() { this.resources = new Resources(); const id = `converse.identities-${this.get('jid')}`; initStorage(this.resources, id, 'session'); this.listenTo(this.resources, 'update', this.onResourcesChanged); this.listenTo(this.resources, 'change', this.onResourcesChanged); }, onResourcesChanged() { var _hpr$attributes; const hpr = this.getHighestPriorityResource(); const show = (hpr === null || hpr === void 0 ? void 0 : (_hpr$attributes = hpr.attributes) === null || _hpr$attributes === void 0 ? void 0 : _hpr$attributes.show) || 'offline'; if (this.get('show') !== show) { this.save({ 'show': show }); } }, /** * Return the resource with the highest priority. * If multiple resources have the same priority, take the latest one. * @private */ getHighestPriorityResource() { return this.resources.sortBy(r => `${r.get('priority')}-${r.get('timestamp')}`).reverse()[0]; }, /** * Adds a new resource and it's associated attributes as taken * from the passed in presence stanza. * Also updates the presence if the resource has higher priority (and is newer). * @private * @param { XMLElement } presence: The presence stanza */ addResource(presence) { var _presence$querySelect, _presence$querySelect2; const jid = presence.getAttribute('from'), name = presence_Strophe.getResourceFromJid(jid), delay = presence_sizzle(`delay[xmlns="${presence_Strophe.NS.DELAY}"]`, presence).pop(), priority = ((_presence$querySelect = presence.querySelector('priority')) === null || _presence$querySelect === void 0 ? void 0 : _presence$querySelect.textContent) ?? 0, resource = this.resources.get(name), settings = { 'name': name, 'priority': lodash_es_isNaN(parseInt(priority, 10)) ? 0 : parseInt(priority, 10), 'show': ((_presence$querySelect2 = presence.querySelector('show')) === null || _presence$querySelect2 === void 0 ? void 0 : _presence$querySelect2.textContent) ?? 'online', 'timestamp': delay ? presence_dayjs(delay.getAttribute('stamp')).toISOString() : new Date().toISOString() }; if (resource) { resource.save(settings); } else { this.resources.create(settings); } }, /** * Remove the passed in resource from the resources map. * Also redetermines the presence given that there's one less * resource. * @private * @param { string } name: The resource name */ removeResource(name) { const resource = this.resources.get(name); if (resource) { resource.destroy(); } } }); const Presences = Collection.extend({ 'model': Presence }); ;// CONCATENATED MODULE: ./src/headless/plugins/roster/index.js /** * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.plugins.add('converse-roster', { dependencies: ['converse-status'], initialize() { core_api.settings.extend({ 'allow_contact_requests': true, 'auto_subscribe': false, 'synchronize_availability': true }); core_api.promises.add(['cachedRoster', 'roster', 'rosterContactsFetched', 'rosterInitialized']); // API methods only available to plugins Object.assign(shared_converse.api, roster_api); const { __ } = shared_converse; shared_converse.HEADER_CURRENT_CONTACTS = __('My contacts'); shared_converse.HEADER_PENDING_CONTACTS = __('Pending contacts'); shared_converse.HEADER_REQUESTING_CONTACTS = __('Contact requests'); shared_converse.HEADER_UNGROUPED = __('Ungrouped'); shared_converse.HEADER_UNREAD = __('New messages'); shared_converse.Presence = Presence; shared_converse.Presences = Presences; shared_converse.RosterContact = contact; shared_converse.RosterContacts = contacts; core_api.listen.on('beforeTearDown', () => unregisterPresenceHandler()); core_api.listen.on('chatBoxesInitialized', onChatBoxesInitialized); core_api.listen.on('clearSession', utils_onClearSession); core_api.listen.on('presencesInitialized', onPresencesInitialized); core_api.listen.on('statusInitialized', roster_utils_onStatusInitialized); core_api.listen.on('streamResumptionFailed', () => shared_converse.session.set('roster_cached', false)); core_api.waitUntil('rosterContactsFetched').then(onRosterContactsFetched); } }); ;// CONCATENATED MODULE: ./src/headless/plugins/smacks/utils.js const { Strophe: smacks_utils_Strophe } = core_converse.env; const smacks_utils_u = core_converse.env.utils; function isStreamManagementSupported() { if (core_api.connection.isType('bosh') && !shared_converse.isTestEnv()) { return false; } return core_api.disco.stream.getFeature('sm', smacks_utils_Strophe.NS.SM); } function handleAck(el) { if (!shared_converse.session.get('smacks_enabled')) { return true; } const handled = parseInt(el.getAttribute('h'), 10); const last_known_handled = shared_converse.session.get('num_stanzas_handled_by_server'); const delta = handled - last_known_handled; if (delta < 0) { const err_msg = `New reported stanza count lower than previous. ` + `New: ${handled} - Previous: ${last_known_handled}`; headless_log.error(err_msg); } const unacked_stanzas = shared_converse.session.get('unacked_stanzas'); if (delta > unacked_stanzas.length) { const err_msg = `Higher reported acknowledge count than unacknowledged stanzas. ` + `Reported Acknowledged Count: ${delta} -` + `Unacknowledged Stanza Count: ${unacked_stanzas.length} -` + `New: ${handled} - Previous: ${last_known_handled}`; headless_log.error(err_msg); } shared_converse.session.save({ 'num_stanzas_handled_by_server': handled, 'num_stanzas_since_last_ack': 0, 'unacked_stanzas': unacked_stanzas.slice(delta) }); return true; } function sendAck() { if (shared_converse.session.get('smacks_enabled')) { const h = shared_converse.session.get('num_stanzas_handled'); const stanza = smacks_utils_u.toStanza(``); core_api.send(stanza); } return true; } function stanzaHandler(el) { if (shared_converse.session.get('smacks_enabled')) { if (smacks_utils_u.isTagEqual(el, 'iq') || smacks_utils_u.isTagEqual(el, 'presence') || smacks_utils_u.isTagEqual(el, 'message')) { const h = shared_converse.session.get('num_stanzas_handled'); shared_converse.session.save('num_stanzas_handled', h + 1); } } return true; } function initSessionData() { shared_converse.session.save({ 'smacks_enabled': shared_converse.session.get('smacks_enabled') || false, 'num_stanzas_handled': shared_converse.session.get('num_stanzas_handled') || 0, 'num_stanzas_handled_by_server': shared_converse.session.get('num_stanzas_handled_by_server') || 0, 'num_stanzas_since_last_ack': shared_converse.session.get('num_stanzas_since_last_ack') || 0, 'unacked_stanzas': shared_converse.session.get('unacked_stanzas') || [] }); } function resetSessionData() { var _converse$session; (_converse$session = shared_converse.session) === null || _converse$session === void 0 ? void 0 : _converse$session.save({ 'smacks_enabled': false, 'num_stanzas_handled': 0, 'num_stanzas_handled_by_server': 0, 'num_stanzas_since_last_ack': 0, 'unacked_stanzas': [] }); } function saveSessionData(el) { const data = { 'smacks_enabled': true }; if (['1', 'true'].includes(el.getAttribute('resume'))) { data['smacks_stream_id'] = el.getAttribute('id'); } shared_converse.session.save(data); return true; } function onFailedStanza(el) { if (el.querySelector('item-not-found')) { // Stream resumption must happen before resource binding but // enabling a new stream must happen after resource binding. // Since resumption failed, we simply continue. // // After resource binding, sendEnableStanza will be called // based on the afterResourceBinding event. headless_log.warn('Could not resume previous SMACKS session, session id not found. ' + 'A new session will be established.'); } else { headless_log.error('Failed to enable stream management'); headless_log.error(el.outerHTML); } resetSessionData(); /** * Triggered when the XEP-0198 stream could not be resumed. * @event _converse#streamResumptionFailed */ core_api.trigger('streamResumptionFailed'); return true; } function resendUnackedStanzas() { const stanzas = shared_converse.session.get('unacked_stanzas'); // We clear the unacked_stanzas array because it'll get populated // again in `onStanzaSent` shared_converse.session.save('unacked_stanzas', []); // XXX: Currently we're resending *all* unacked stanzas, including // IQ[type="get"] stanzas that longer have handlers (because the // page reloaded or we reconnected, causing removal of handlers). // // *Side-note:* Is it necessary to clear handlers upon reconnection? // // I've considered not resending those stanzas, but then keeping // track of what's been sent and ack'd and their order gets // prohibitively complex. // // It's unclear how much of a problem this poses. // // Two possible solutions are running @converse/headless as a // service worker or handling IQ[type="result"] stanzas // differently, more like push stanzas, so that they don't need // explicit handlers. stanzas.forEach(s => core_api.send(s)); } function onResumedStanza(el) { saveSessionData(el); handleAck(el); resendUnackedStanzas(); shared_converse.connection.do_bind = false; // No need to bind our resource anymore shared_converse.connection.authenticated = true; shared_converse.connection.restored = true; shared_converse.connection._changeConnectStatus(smacks_utils_Strophe.Status.CONNECTED, null); } async function sendResumeStanza() { const promise = getOpenPromise(); shared_converse.connection._addSysHandler(el => promise.resolve(onResumedStanza(el)), smacks_utils_Strophe.NS.SM, 'resumed'); shared_converse.connection._addSysHandler(el => promise.resolve(onFailedStanza(el)), smacks_utils_Strophe.NS.SM, 'failed'); const previous_id = shared_converse.session.get('smacks_stream_id'); const h = shared_converse.session.get('num_stanzas_handled'); const stanza = smacks_utils_u.toStanza(``); core_api.send(stanza); shared_converse.connection.flush(); await promise; } async function sendEnableStanza() { if (!core_api.settings.get('enable_smacks') || shared_converse.session.get('smacks_enabled')) { return; } if (await isStreamManagementSupported()) { const promise = getOpenPromise(); shared_converse.connection._addSysHandler(el => promise.resolve(saveSessionData(el)), smacks_utils_Strophe.NS.SM, 'enabled'); shared_converse.connection._addSysHandler(el => promise.resolve(onFailedStanza(el)), smacks_utils_Strophe.NS.SM, 'failed'); const resume = core_api.connection.isType('websocket') || shared_converse.isTestEnv(); const stanza = smacks_utils_u.toStanza(``); core_api.send(stanza); shared_converse.connection.flush(); await promise; } } const smacks_handlers = []; async function enableStreamManagement() { var _converse$session2; if (!core_api.settings.get('enable_smacks')) { return; } if (!(await isStreamManagementSupported())) { return; } const conn = shared_converse.connection; while (smacks_handlers.length) { conn.deleteHandler(smacks_handlers.pop()); } smacks_handlers.push(conn.addHandler(stanzaHandler)); smacks_handlers.push(conn.addHandler(sendAck, smacks_utils_Strophe.NS.SM, 'r')); smacks_handlers.push(conn.addHandler(handleAck, smacks_utils_Strophe.NS.SM, 'a')); if ((_converse$session2 = shared_converse.session) !== null && _converse$session2 !== void 0 && _converse$session2.get('smacks_stream_id')) { await sendResumeStanza(); } else { resetSessionData(); } } function onStanzaSent(stanza) { if (!shared_converse.session) { headless_log.warn('No _converse.session!'); return; } if (!shared_converse.session.get('smacks_enabled')) { return; } if (smacks_utils_u.isTagEqual(stanza, 'iq') || smacks_utils_u.isTagEqual(stanza, 'presence') || smacks_utils_u.isTagEqual(stanza, 'message')) { const stanza_string = smacks_utils_Strophe.serialize(stanza); shared_converse.session.save('unacked_stanzas', (shared_converse.session.get('unacked_stanzas') || []).concat([stanza_string])); const max_unacked = core_api.settings.get('smacks_max_unacked_stanzas'); if (max_unacked > 0) { const num = shared_converse.session.get('num_stanzas_since_last_ack') + 1; if (num % max_unacked === 0) { // Request confirmation of sent stanzas core_api.send(smacks_utils_u.toStanza(``)); } shared_converse.session.save({ 'num_stanzas_since_last_ack': num }); } } } ;// CONCATENATED MODULE: ./src/headless/plugins/smacks/index.js /** * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) * @description Converse.js plugin which adds support for XEP-0198: Stream Management */ const { Strophe: smacks_Strophe } = core_converse.env; smacks_Strophe.addNamespace('SM', 'urn:xmpp:sm:3'); core_converse.plugins.add('converse-smacks', { initialize() { // Configuration values for this plugin // ==================================== // Refer to docs/source/configuration.rst for explanations of these // configuration settings. core_api.settings.extend({ 'enable_smacks': true, 'smacks_max_unacked_stanzas': 5 }); core_api.listen.on('afterResourceBinding', sendEnableStanza); core_api.listen.on('beforeResourceBinding', enableStreamManagement); core_api.listen.on('send', onStanzaSent); core_api.listen.on('userSessionInitialized', initSessionData); } }); ;// CONCATENATED MODULE: ./src/headless/plugins/vcard/vcard.js /** * Represents a VCard * @class * @namespace _converse.VCard * @memberOf _converse */ const VCard = Model.extend({ idAttribute: 'jid', defaults: { 'image': shared_converse.DEFAULT_IMAGE, 'image_type': shared_converse.DEFAULT_IMAGE_TYPE }, set(key, val, options) { // Override Model.prototype.set to make sure that the // default `image` and `image_type` values are maintained. let attrs; if (typeof key === 'object') { attrs = key; options = val; } else { (attrs = {})[key] = val; } if ('image' in attrs && !attrs['image']) { attrs['image'] = shared_converse.DEFAULT_IMAGE; attrs['image_type'] = shared_converse.DEFAULT_IMAGE_TYPE; return Model.prototype.set.call(this, attrs, options); } else { return Model.prototype.set.apply(this, arguments); } }, getDisplayName() { return this.get('nickname') || this.get('fullname') || this.get('jid'); } }); /* harmony default export */ const vcard = (VCard); ;// CONCATENATED MODULE: ./src/headless/plugins/vcard/utils.js const { Strophe: vcard_utils_Strophe, $iq: vcard_utils_$iq, u: vcard_utils_u } = core_converse.env; async function onVCardData(jid, iq) { const vcard = iq.querySelector('vCard'); let result = {}; if (vcard !== null) { var _vcard$querySelector, _vcard$querySelector2, _vcard$querySelector3, _vcard$querySelector4, _vcard$querySelector5, _vcard$querySelector6, _vcard$querySelector7; result = { 'stanza': iq, 'fullname': (_vcard$querySelector = vcard.querySelector('FN')) === null || _vcard$querySelector === void 0 ? void 0 : _vcard$querySelector.textContent, 'nickname': (_vcard$querySelector2 = vcard.querySelector('NICKNAME')) === null || _vcard$querySelector2 === void 0 ? void 0 : _vcard$querySelector2.textContent, 'image': (_vcard$querySelector3 = vcard.querySelector('PHOTO BINVAL')) === null || _vcard$querySelector3 === void 0 ? void 0 : _vcard$querySelector3.textContent, 'image_type': (_vcard$querySelector4 = vcard.querySelector('PHOTO TYPE')) === null || _vcard$querySelector4 === void 0 ? void 0 : _vcard$querySelector4.textContent, 'url': (_vcard$querySelector5 = vcard.querySelector('URL')) === null || _vcard$querySelector5 === void 0 ? void 0 : _vcard$querySelector5.textContent, 'role': (_vcard$querySelector6 = vcard.querySelector('ROLE')) === null || _vcard$querySelector6 === void 0 ? void 0 : _vcard$querySelector6.textContent, 'email': (_vcard$querySelector7 = vcard.querySelector('EMAIL USERID')) === null || _vcard$querySelector7 === void 0 ? void 0 : _vcard$querySelector7.textContent, 'vcard_updated': new Date().toISOString(), 'vcard_error': undefined }; } if (result.image) { const buffer = vcard_utils_u.base64ToArrayBuffer(result['image']); const ab = await crypto.subtle.digest('SHA-1', buffer); result['image_hash'] = vcard_utils_u.arrayBufferToHex(ab); } return result; } function createStanza(type, jid, vcard_el) { const iq = vcard_utils_$iq(jid ? { 'type': type, 'to': jid } : { 'type': type }); if (!vcard_el) { iq.c("vCard", { 'xmlns': vcard_utils_Strophe.NS.VCARD }); } else { iq.cnode(vcard_el); } return iq; } function onOccupantAvatarChanged(occupant) { const hash = occupant.get('image_hash'); const vcards = []; if (occupant.get('jid')) { vcards.push(shared_converse.vcards.get(occupant.get('jid'))); } vcards.push(shared_converse.vcards.get(occupant.get('from'))); vcards.forEach(v => hash && (v === null || v === void 0 ? void 0 : v.get('image_hash')) !== hash && core_api.vcard.update(v, true)); } async function setVCardOnModel(model) { let jid; if (model instanceof shared_converse.Message) { if (['error', 'info'].includes(model.get('type'))) { return; } jid = model.get('from'); } else { jid = model.get('jid'); } if (!jid) { headless_log.warn(`Could not set VCard on model because no JID found!`); return; } await core_api.waitUntil('VCardsInitialized'); model.vcard = shared_converse.vcards.get(jid) || shared_converse.vcards.create({ jid }); model.vcard.on('change', () => model.trigger('vcard:change')); model.trigger('vcard:add'); } function getVCardForOccupant(occupant) { var _occupant$collection; const muc = occupant === null || occupant === void 0 ? void 0 : (_occupant$collection = occupant.collection) === null || _occupant$collection === void 0 ? void 0 : _occupant$collection.chatroom; const nick = occupant.get('nick'); if (nick && (muc === null || muc === void 0 ? void 0 : muc.get('nick')) === nick) { return shared_converse.xmppstatus.vcard; } else { const jid = occupant.get('jid') || occupant.get('from'); if (jid) { return shared_converse.vcards.get(jid) || shared_converse.vcards.create({ jid }); } else { headless_log.warn(`Could not get VCard for occupant because no JID found!`); return; } } } async function setVCardOnOccupant(occupant) { await core_api.waitUntil('VCardsInitialized'); occupant.vcard = getVCardForOccupant(occupant); if (occupant.vcard) { occupant.vcard.on('change', () => occupant.trigger('vcard:change')); occupant.trigger('vcard:add'); } } function getVCardForMUCMessage(message) { var _message$collection; const muc = message === null || message === void 0 ? void 0 : (_message$collection = message.collection) === null || _message$collection === void 0 ? void 0 : _message$collection.chatbox; const nick = vcard_utils_Strophe.getResourceFromJid(message.get('from')); if (nick && (muc === null || muc === void 0 ? void 0 : muc.get('nick')) === nick) { return shared_converse.xmppstatus.vcard; } else { var _message$occupant; const jid = ((_message$occupant = message.occupant) === null || _message$occupant === void 0 ? void 0 : _message$occupant.get('jid')) || message.get('from'); if (jid) { return shared_converse.vcards.get(jid) || shared_converse.vcards.create({ jid }); } else { headless_log.warn(`Could not get VCard for message because no JID found! msgid: ${message.get('msgid')}`); return; } } } async function setVCardOnMUCMessage(message) { if (['error', 'info'].includes(message.get('type'))) { return; } else { await core_api.waitUntil('VCardsInitialized'); message.vcard = getVCardForMUCMessage(message); if (message.vcard) { message.vcard.on('change', () => message.trigger('vcard:change')); message.trigger('vcard:add'); } } } async function initVCardCollection() { shared_converse.vcards = new shared_converse.VCards(); const id = `${shared_converse.bare_jid}-converse.vcards`; initStorage(shared_converse.vcards, id); await new Promise(resolve => { shared_converse.vcards.fetch({ 'success': resolve, 'error': resolve }, { 'silent': true }); }); const vcards = shared_converse.vcards; if (shared_converse.session) { const jid = shared_converse.session.get('bare_jid'); const status = shared_converse.xmppstatus; status.vcard = vcards.get(jid) || vcards.create({ 'jid': jid }); if (status.vcard) { status.vcard.on('change', () => status.trigger('vcard:change')); status.trigger('vcard:add'); } } /** * Triggered as soon as the `_converse.vcards` collection has been initialized and populated from cache. * @event _converse#VCardsInitialized */ core_api.trigger('VCardsInitialized'); } function clearVCardsSession() { if (shared_converse.shouldClearCache()) { core_api.promises.add('VCardsInitialized'); if (shared_converse.vcards) { shared_converse.vcards.clearStore(); delete shared_converse.vcards; } } } async function getVCard(jid) { const to = vcard_utils_Strophe.getBareJidFromJid(jid) === shared_converse.bare_jid ? null : jid; let iq; try { iq = await core_api.sendIQ(createStanza("get", to)); } catch (iq) { return { jid, 'stanza': iq, 'vcard_error': new Date().toISOString() }; } return onVCardData(jid, iq); } ;// CONCATENATED MODULE: ./src/headless/plugins/vcard/api.js const { dayjs: api_dayjs, u: vcard_api_u } = core_converse.env; /* harmony default export */ const vcard_api = ({ /** * The XEP-0054 VCard API * * This API lets you access and update user VCards * * @namespace _converse.api.vcard * @memberOf _converse.api */ vcard: { /** * Enables setting new values for a VCard. * * Sends out an IQ stanza to set the user's VCard and if * successful, it updates the {@link _converse.VCard} * for the passed in JID. * * @method _converse.api.vcard.set * @param {string} jid The JID for which the VCard should be set * @param {object} data A map of VCard keys and values * @example * let jid = _converse.bare_jid; * _converse.api.vcard.set( jid, { * 'fn': 'John Doe', * 'nickname': 'jdoe' * }).then(() => { * // Succes * }).catch((e) => { * // Failure, e is your error object * }). */ async set(jid, data) { if (!jid) { throw Error("No jid provided for the VCard data"); } const div = document.createElement('div'); const vcard_el = vcard_api_u.toStanza(` ${data.fn} ${data.nickname} ${data.url} ${data.role} ${data.email} ${data.image_type} ${data.image} `, div); let result; try { result = await core_api.sendIQ(createStanza("set", jid, vcard_el)); } catch (e) { throw e; } await core_api.vcard.update(jid, true); return result; }, /** * @method _converse.api.vcard.get * @param {Model|string} model Either a `Model` instance, or a string JID. * If a `Model` instance is passed in, then it must have either a `jid` * attribute or a `muc_jid` attribute. * @param {boolean} [force] A boolean indicating whether the vcard should be * fetched from the server even if it's been fetched before. * @returns {promise} A Promise which resolves with the VCard data for a particular JID or for * a `Model` instance which represents an entity with a JID (such as a roster contact, * chat or chatroom occupant). * * @example * const { api } = _converse; * api.waitUntil('rosterContactsFetched').then(() => { * api.vcard.get('someone@example.org').then( * (vcard) => { * // Do something with the vcard... * } * ); * }); */ get(model, force) { if (typeof model === 'string') { return getVCard(model); } const error_date = model.get('vcard_error'); const already_tried_today = error_date && api_dayjs(error_date).isSame(new Date(), "day"); if (force || !model.get('vcard_updated') && !already_tried_today) { const jid = model.get('jid'); if (!jid) { headless_log.error("No JID to get vcard for"); } return getVCard(jid); } else { return Promise.resolve({}); } }, /** * Fetches the VCard associated with a particular `Model` instance * (by using its `jid` or `muc_jid` attribute) and then updates the model with the * returned VCard data. * * @method _converse.api.vcard.update * @param {Model} model A `Model` instance * @param {boolean} [force] A boolean indicating whether the vcard should be * fetched again even if it's been fetched before. * @returns {promise} A promise which resolves once the update has completed. * @example * const { api } = _converse; * api.waitUntil('rosterContactsFetched').then(async () => { * const chatbox = await api.chats.get('someone@example.org'); * api.vcard.update(chatbox); * }); */ async update(model, force) { const data = await this.get(model, force); model = typeof model === 'string' ? shared_converse.vcards.get(model) : model; if (!model) { headless_log.error(`Could not find a VCard model for ${model}`); return; } if (Object.keys(data).length) { delete data['stanza']; model.save(data); } } } }); ;// CONCATENATED MODULE: ./src/headless/plugins/vcard/index.js /** * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) */ const { Strophe: vcard_Strophe } = core_converse.env; core_converse.plugins.add('converse-vcard', { dependencies: ["converse-status", "converse-roster"], overrides: { XMPPStatus: { getNickname() { const { _converse } = this.__super__; const nick = this.__super__.getNickname.apply(this); if (!nick && _converse.xmppstatus.vcard) { return _converse.xmppstatus.vcard.get('nickname'); } else { return nick; } }, getFullname() { const { _converse } = this.__super__; const fullname = this.__super__.getFullname.apply(this); if (!fullname && _converse.xmppstatus.vcard) { return _converse.xmppstatus.vcard.get('fullname'); } else { return fullname; } } }, RosterContact: { getDisplayName() { if (!this.get('nickname') && this.vcard) { return this.vcard.getDisplayName(); } else { return this.__super__.getDisplayName.apply(this); } }, getFullname() { if (this.vcard) { return this.vcard.get('fullname'); } else { return this.__super__.getFullname.apply(this); } } } }, initialize() { core_api.promises.add('VCardsInitialized'); shared_converse.VCard = vcard; shared_converse.VCards = Collection.extend({ model: shared_converse.VCard, initialize() { this.on('add', v => v.get('jid') && core_api.vcard.update(v)); } }); core_api.listen.on('chatRoomInitialized', m => { setVCardOnModel(m); m.occupants.forEach(setVCardOnOccupant); m.listenTo(m.occupants, 'add', setVCardOnOccupant); m.listenTo(m.occupants, 'change:image_hash', o => onOccupantAvatarChanged(o)); }); core_api.listen.on('chatBoxInitialized', m => setVCardOnModel(m)); core_api.listen.on('chatRoomMessageInitialized', m => setVCardOnMUCMessage(m)); core_api.listen.on('addClientFeatures', () => core_api.disco.own.features.add(vcard_Strophe.NS.VCARD)); core_api.listen.on('clearSession', () => clearVCardsSession()); core_api.listen.on('messageInitialized', m => setVCardOnModel(m)); core_api.listen.on('rosterContactInitialized', m => setVCardOnModel(m)); core_api.listen.on('statusInitialized', initVCardCollection); Object.assign(shared_converse.api, vcard_api); } }); ;// CONCATENATED MODULE: ./src/headless/headless.js /* START: Removable components * -------------------- * Any of the following components may be removed if they're not needed. */ // XEP-0050 Ad Hoc Commands // XEP-0199 XMPP Ping // XEP-0206 BOSH // XEP-0115 Entity Capabilities // XEP-0280 Message Carbons // RFC-6121 Instant messaging // XEP-0030 Service discovery // Support for headline messages // XEP-0313 Message Archive Management // XEP-0045 Multi-user chat // XEP-0199 XMPP Ping // XEP-0060 Pubsub // RFC-6121 Contacts Roster // XEP-0198 Stream Management // XEP-0054 VCard-temp /* END: Removable components */ /* harmony default export */ const headless = ((/* unused pure expression or super */ null && (converse))); // EXTERNAL MODULE: ./node_modules/jed/jed.js var jed = __webpack_require__(2353); var jed_default = /*#__PURE__*/__webpack_require__.n(jed); ;// CONCATENATED MODULE: ./src/i18n/index.js /** * @module i18n * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) * @description This is the internationalization module */ const { dayjs: i18n_dayjs } = core_converse.env; function detectLocale(library_check) { /* Determine which locale is supported by the user's system as well * as by the relevant library (e.g. converse.js or dayjs). * @param { Function } library_check - Returns a boolean indicating whether * the locale is supported. */ let locale; if (window.navigator.userLanguage) { locale = isLocaleAvailable(window.navigator.userLanguage, library_check); } if (window.navigator.languages && !locale) { for (let i = 0; i < window.navigator.languages.length && !locale; i++) { locale = isLocaleAvailable(window.navigator.languages[i], library_check); } } if (window.navigator.browserLanguage && !locale) { locale = isLocaleAvailable(window.navigator.browserLanguage, library_check); } if (window.navigator.language && !locale) { locale = isLocaleAvailable(window.navigator.language, library_check); } if (window.navigator.systemLanguage && !locale) { locale = isLocaleAvailable(window.navigator.systemLanguage, library_check); } return locale || 'en'; } function isConverseLocale(locale, supported_locales) { return typeof locale === 'string' && supported_locales.includes(locale); } function getLocale(preferred_locale, isSupportedByLibrary) { if (typeof preferred_locale === 'string') { if (preferred_locale === 'en' || isSupportedByLibrary(preferred_locale)) { return preferred_locale; } } return detectLocale(isSupportedByLibrary) || 'en'; } /* Check whether the locale or sub locale (e.g. en-US, en) is supported. * @param { String } locale - The locale to check for * @param { Function } available - Returns a boolean indicating whether the locale is supported */ function isLocaleAvailable(locale, available) { if (available(locale)) { return locale; } else { var sublocale = locale.split("-")[0]; if (sublocale !== locale && available(sublocale)) { return sublocale; } } } /* Fetch the translations for the given local at the given URL. * @private * @method i18n#fetchTranslations * @param { _converse } */ async function fetchTranslations(_converse) { const { api, locale } = _converse; const dayjs_locale = locale.toLowerCase().replace('_', '-'); if (!isConverseLocale(locale, api.settings.get("locales")) || locale === 'en') { return; } const { default: data } = await __webpack_require__(7521)(`./${locale}/LC_MESSAGES/converse.po`); await __webpack_require__(9434)(`./${dayjs_locale}.js`); i18n_dayjs.locale(getLocale(dayjs_locale, l => i18n_dayjs.locale(l))); jed_instance = new (jed_default())(data); } let jed_instance; /** * @namespace i18n */ Object.assign(i18n, { getLocale(preferred_locale, available_locales) { return getLocale(preferred_locale, preferred => isConverseLocale(preferred, available_locales)); }, translate(str) { if (!jed_instance) { return jed_default().sprintf.apply((jed_default()), arguments); } const t = jed_instance.translate(str); if (arguments.length > 1) { return t.fetch.apply(t, [].slice.call(arguments, 1)); } else { return t.fetch(); } }, async initialize() { if (shared_converse.isTestEnv()) { shared_converse.locale = 'en'; } else { try { const preferred_locale = core_api.settings.get('i18n'); shared_converse.locale = i18n.getLocale(preferred_locale, core_api.settings.get("locales")); await fetchTranslations(shared_converse); } catch (e) { headless_log.fatal(e.message); shared_converse.locale = 'en'; } } }, __() { return i18n.translate(...arguments); } }); const __ = i18n.__; ;// CONCATENATED MODULE: ./src/shared/registry.js const registry = {}; function registry_define(name, constructor) { this.registry[name] = constructor; } function register() { Object.keys(registry).forEach(name => { if (!customElements.get(name)) { customElements.define(name, registry[name]); } }); } core_api.elements = { registry, define: registry_define, register }; ;// CONCATENATED MODULE: ./src/shared/components/element.js class CustomElement extends lit_element_s { createRenderRoot() { // Render without the shadow DOM return this; } connectedCallback() { var _this$initialize; super.connectedCallback(); (_this$initialize = this.initialize) === null || _this$initialize === void 0 ? void 0 : _this$initialize.call(this); } disconnectedCallback() { super.disconnectedCallback(); this.stopListening(); } } Object.assign(CustomElement.prototype, Events); ;// CONCATENATED MODULE: ./src/shared/constants.js // These are all the view-layer plugins. // // For the full Converse build, this list serves // as a whitelist (see src/converse.js) in addition to the // CORE_PLUGINS list in src/headless/consts.js. const VIEW_PLUGINS = ['converse-bookmark-views', 'converse-chatboxviews', 'converse-chatview', 'converse-controlbox', 'converse-dragresize', 'converse-fullscreen', 'converse-headlines-view', 'converse-mam-views', 'converse-minimize', 'converse-modal', 'converse-muc-views', 'converse-notification', 'converse-omemo', 'converse-profile', 'converse-push', 'converse-register', 'converse-roomslist', 'converse-rootview', 'converse-rosterview', 'converse-singleton']; // EXTERNAL MODULE: ./node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js var injectStylesIntoStyleTag = __webpack_require__(3379); var injectStylesIntoStyleTag_default = /*#__PURE__*/__webpack_require__.n(injectStylesIntoStyleTag); // EXTERNAL MODULE: ./node_modules/style-loader/dist/runtime/styleDomAPI.js var styleDomAPI = __webpack_require__(7795); var styleDomAPI_default = /*#__PURE__*/__webpack_require__.n(styleDomAPI); // EXTERNAL MODULE: ./node_modules/style-loader/dist/runtime/insertBySelector.js var insertBySelector = __webpack_require__(569); var insertBySelector_default = /*#__PURE__*/__webpack_require__.n(insertBySelector); // EXTERNAL MODULE: ./node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js var setAttributesWithoutAttributes = __webpack_require__(3565); var setAttributesWithoutAttributes_default = /*#__PURE__*/__webpack_require__.n(setAttributesWithoutAttributes); // EXTERNAL MODULE: ./node_modules/style-loader/dist/runtime/insertStyleElement.js var insertStyleElement = __webpack_require__(9216); var insertStyleElement_default = /*#__PURE__*/__webpack_require__.n(insertStyleElement); // EXTERNAL MODULE: ./node_modules/style-loader/dist/runtime/styleTagTransform.js var styleTagTransform = __webpack_require__(4589); var styleTagTransform_default = /*#__PURE__*/__webpack_require__.n(styleTagTransform); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/shared/styles/index.scss var styles = __webpack_require__(9537); ;// CONCATENATED MODULE: ./src/shared/styles/index.scss var options = {}; options.styleTagTransform = (styleTagTransform_default()); options.setAttributes = (setAttributesWithoutAttributes_default()); options.insert = insertBySelector_default().bind(null, "head"); options.domAPI = (styleDomAPI_default()); options.insertStyleElement = (insertStyleElement_default()); var update = injectStylesIntoStyleTag_default()(styles/* default */.Z, options); /* harmony default export */ const shared_styles = (styles/* default */.Z && styles/* default.locals */.Z.locals ? styles/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/bookmark-views/templates/form.js /* harmony default export */ const templates_form = (o => { var _o$bookmark, _o$bookmark2; const name = ((_o$bookmark = o.bookmark) === null || _o$bookmark === void 0 ? void 0 : _o$bookmark.get('name')) ?? o.name; const nick = ((_o$bookmark2 = o.bookmark) === null || _o$bookmark2 === void 0 ? void 0 : _o$bookmark2.get('nick')) ?? o.nick; const i18n_heading = __('Bookmark for "%1$s"', name); const i18n_autojoin = __('Would you like this groupchat to be automatically joined upon startup?'); const i18n_remove = __('Remove'); const i18n_name = __('The name for this bookmark:'); const i18n_nick = __('What should your nickname for this groupchat be?'); const i18n_submit = o.bookmark ? __('Update') : __('Save'); return $`
${i18n_heading}
${o.bookmark ? $`` : ''}
`; }); ;// CONCATENATED MODULE: ./src/plugins/bookmark-views/form.js class MUCBookmarkForm extends CustomElement { static get properties() { return { 'jid': { type: String } }; } connectedCallback() { super.connectedCallback(); this.model = shared_converse.chatboxes.get(this.jid); this.bookmark = shared_converse.bookmarks.get(this.model.get('jid')); } render() { return templates_form(Object.assign(this.model.toJSON(), { 'bookmark': this.bookmark, 'onCancel': ev => this.removeBookmark(ev), 'onSubmit': ev => this.onBookmarkFormSubmitted(ev) })); } onBookmarkFormSubmitted(ev) { var _ev$target$querySelec, _ev$target$querySelec2, _ev$target$querySelec3; ev.preventDefault(); shared_converse.bookmarks.createBookmark({ 'jid': this.model.get('jid'), 'autojoin': ((_ev$target$querySelec = ev.target.querySelector('input[name="autojoin"]')) === null || _ev$target$querySelec === void 0 ? void 0 : _ev$target$querySelec.checked) || false, 'name': (_ev$target$querySelec2 = ev.target.querySelector('input[name=name]')) === null || _ev$target$querySelec2 === void 0 ? void 0 : _ev$target$querySelec2.value, 'nick': (_ev$target$querySelec3 = ev.target.querySelector('input[name=nick]')) === null || _ev$target$querySelec3 === void 0 ? void 0 : _ev$target$querySelec3.value }); this.closeBookmarkForm(ev); } removeBookmark(ev) { var _this$bookmark; (_this$bookmark = this.bookmark) === null || _this$bookmark === void 0 ? void 0 : _this$bookmark.destroy(); this.closeBookmarkForm(ev); } closeBookmarkForm(ev) { ev.preventDefault(); const evt = document.createEvent('Event'); evt.initEvent('hide.bs.modal', true, true); this.dispatchEvent(evt); } } core_api.elements.define('converse-muc-bookmark-form', MUCBookmarkForm); /* harmony default export */ const bookmark_views_form = (MUCBookmarkForm); // EXTERNAL MODULE: ./node_modules/bootstrap.native/dist/bootstrap-native.js var bootstrap_native = __webpack_require__(6151); var bootstrap_native_default = /*#__PURE__*/__webpack_require__.n(bootstrap_native); ;// CONCATENATED MODULE: ./src/templates/alert.js /* harmony default export */ const templates_alert = (o => $``); ;// CONCATENATED MODULE: ./node_modules/@converse/skeletor/src/view.js // Backbone.js 1.4.0 // (c) 2010-2019 Jeremy Ashkenas and DocumentCloud // Backbone may be freely distributed under the MIT license. // View // ---- // Views are almost more convention than they are actual code. A View // is simply a JavaScript object that represents a logical chunk of UI in the // DOM. This might be a single item, an entire list, a sidebar or panel, or // even the surrounding frame which wraps your whole app. Defining a chunk of // UI as a **View** allows you to define your DOM events declaratively, without // having to worry about render order ... and makes it easy for the view to // react to specific changes in the state of your models. const paddedLt = /^\s* this.onHide(), false); }, onHide() { base_u.removeClass('selected', this.trigger_el); !this.persistent && core_api.modal.remove(this); }, insertIntoDOM() { const container_el = document.querySelector("#converse-modals"); container_el.insertAdjacentElement('beforeEnd', this.el); }, switchTab(ev) { ev.stopPropagation(); ev.preventDefault(); base_sizzle('.nav-link.active', this.el).forEach(el => { base_u.removeClass('active', this.el.querySelector(el.getAttribute('href'))); base_u.removeClass('active', el); }); base_u.addClass('active', ev.target); base_u.addClass('active', this.el.querySelector(ev.target.getAttribute('href'))); }, alert(message) { let type = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 'primary'; const body = this.el.querySelector('.modal-alert'); if (body === null) { headless_log.error("Could not find a .modal-alert element in the modal to show an alert message in!"); return; } // FIXME: Instead of adding the alert imperatively, we should // find a way to let the modal rerender with an alert message x(templates_alert({ 'type': `alert-${type}`, 'message': message }), body); const el = body.firstElementChild; setTimeout(() => { base_u.addClass('fade-out', el); setTimeout(() => base_u.removeElement(el), 600); }, 5000); }, show(ev) { if (ev) { ev.preventDefault(); this.trigger_el = ev.target; !base_u.hasClass('chat-image', this.trigger_el) && base_u.addClass('selected', this.trigger_el); } this.modal.show(); } }); /* harmony default export */ const base = (BaseModal); ;// CONCATENATED MODULE: ./src/plugins/modal/templates/buttons.js const modal_close_button = $``; const modal_header_close_button = $``; ;// CONCATENATED MODULE: ./src/plugins/bookmark-views/templates/modal.js /* harmony default export */ const modal = (o => { const i18n_moderator_tools = __('Bookmark'); return $` `; }); ;// CONCATENATED MODULE: ./src/plugins/bookmark-views/modal.js const MUCBookmarkFormModal = base.extend({ id: "converse-bookmark-modal", initialize(attrs) { this.jid = attrs.jid; this.affiliation = attrs.affiliation; base.prototype.initialize.apply(this, arguments); }, toHTML() { return modal(this); } }); /* harmony default export */ const bookmark_views_modal = (MUCBookmarkFormModal); ;// CONCATENATED MODULE: ./node_modules/lodash-es/invokeMap.js /** * Invokes the method at `path` of each element in `collection`, returning * an array of the results of each invoked method. Any additional arguments * are provided to each invoked method. If `path` is a function, it's invoked * for, and `this` bound to, each element in `collection`. * * @static * @memberOf _ * @since 4.0.0 * @category Collection * @param {Array|Object} collection The collection to iterate over. * @param {Array|Function|string} path The path of the method to invoke or * the function invoked per iteration. * @param {...*} [args] The arguments to invoke each method with. * @returns {Array} Returns the array of results. * @example * * _.invokeMap([[5, 1, 7], [3, 2, 1]], 'sort'); * // => [[1, 5, 7], [1, 2, 3]] * * _.invokeMap([123, 456], String.prototype.split, ''); * // => [['1', '2', '3'], ['4', '5', '6']] */ var invokeMap = _baseRest(function(collection, path, args) { var index = -1, isFunc = typeof path == 'function', result = lodash_es_isArrayLike(collection) ? Array(collection.length) : []; _baseEach(collection, function(value) { result[++index] = isFunc ? _apply(path, value, args) : _baseInvoke(value, path, args); }); return result; }); /* harmony default export */ const lodash_es_invokeMap = (invokeMap); ;// CONCATENATED MODULE: ./src/plugins/bookmark-views/utils.js function getHeadingButtons(view, buttons) { if (core_api.settings.get('allow_bookmarks') && view.model.get('type') === shared_converse.CHATROOMS_TYPE) { const bookmarked = view.model.get('bookmarked'); const data = { 'i18n_title': bookmarked ? __('Unbookmark this groupchat') : __('Bookmark this groupchat'), 'i18n_text': bookmarked ? __('Unbookmark') : __('Bookmark'), 'handler': ev => view.showBookmarkModal(ev), 'a_class': 'toggle-bookmark', 'icon_class': 'fa-bookmark', 'name': 'bookmark' }; const names = buttons.map(t => t.name); const idx = names.indexOf('details'); const data_promise = checkBookmarksSupport().then(s => s ? data : null); return idx > -1 ? [...buttons.slice(0, idx), data_promise, ...buttons.slice(idx)] : [data_promise, ...buttons]; } return buttons; } function removeBookmarkViaEvent(ev) { ev.preventDefault(); const name = ev.target.getAttribute('data-bookmark-name'); const jid = ev.target.getAttribute('data-room-jid'); if (confirm(__('Are you sure you want to remove the bookmark "%1$s"?', name))) { lodash_es_invokeMap(shared_converse.bookmarks.where({ jid }), Model.prototype.destroy); } } function addBookmarkViaEvent(ev) { ev.preventDefault(); const jid = ev.target.getAttribute('data-room-jid'); core_api.modal.show(bookmark_views_modal, { jid }, ev); } function openRoomViaEvent(ev) { ev.preventDefault(); const { Strophe } = core_converse.env; const name = ev.target.textContent; const jid = ev.target.getAttribute('data-room-jid'); const data = { 'name': name || Strophe.unescapeNode(Strophe.getNodeFromJid(jid)) || jid }; core_api.rooms.open(jid, data, true); } ;// CONCATENATED MODULE: ./src/plugins/bookmark-views/templates/item.js /* harmony default export */ const item = (bm => { const jid = bm.get('jid'); const is_hidden = !!(core_api.settings.get('hide_open_bookmarks') && shared_converse.chatboxes.get(jid)); const info_remove_bookmark = __('Unbookmark this groupchat'); const open_title = __('Click to open this groupchat'); return $`
`; }); ;// CONCATENATED MODULE: ./node_modules/lit-html/directive.js /** * @license * Copyright 2017 Google LLC * SPDX-License-Identifier: BSD-3-Clause */ const directive_t = { ATTRIBUTE: 1, CHILD: 2, PROPERTY: 3, BOOLEAN_ATTRIBUTE: 4, EVENT: 5, ELEMENT: 6 }, directive_e = t => function () { for (var _len = arguments.length, e = new Array(_len), _key = 0; _key < _len; _key++) { e[_key] = arguments[_key]; } return { _$litDirective$: t, values: e }; }; class directive_i { constructor(t) {} get _$AU() { return this._$AM._$AU; } _$AT(t, e, i) { this._$Ct = t, this._$AM = e, this._$Ci = i; } _$AS(t, e) { return this.update(t, e); } update(t, e) { return this.render(...e); } } ;// CONCATENATED MODULE: ./node_modules/lit-html/directive-helpers.js /** * @license * Copyright 2020 Google LLC * SPDX-License-Identifier: BSD-3-Clause */ const { H: directive_helpers_i } = R, directive_helpers_t = o => null === o || "object" != typeof o && "function" != typeof o, directive_helpers_n = { HTML: 1, SVG: 2 }, directive_helpers_v = (o, i) => { var t, n; return void 0 === i ? void 0 !== (null === (t = o) || void 0 === t ? void 0 : t._$litType$) : (null === (n = o) || void 0 === n ? void 0 : n._$litType$) === i; }, directive_helpers_l = o => { var i; return void 0 !== (null === (i = o) || void 0 === i ? void 0 : i._$litDirective$); }, directive_helpers_d = o => { var i; return null === (i = o) || void 0 === i ? void 0 : i._$litDirective$; }, directive_helpers_r = o => void 0 === o.strings, directive_helpers_e = () => document.createComment(""), directive_helpers_u = (o, t, n) => { var v; const l = o._$AA.parentNode, d = void 0 === t ? o._$AB : t._$AA; if (void 0 === n) { const t = l.insertBefore(directive_helpers_e(), d), v = l.insertBefore(directive_helpers_e(), d); n = new directive_helpers_i(t, v, o, o.options); } else { const i = n._$AB.nextSibling, t = n._$AM, r = t !== o; if (r) { let i; null === (v = n._$AQ) || void 0 === v || v.call(n, o), n._$AM = o, void 0 !== n._$AP && (i = o._$AU) !== t._$AU && n._$AP(i); } if (i !== d || r) { let o = n._$AA; for (; o !== i;) { const i = o.nextSibling; l.insertBefore(o, d), o = i; } } } return n; }, directive_helpers_c = function (o, i) { let t = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : o; return o._$AI(i, t), o; }, directive_helpers_f = {}, directive_helpers_s = function (o) { let i = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : directive_helpers_f; return o._$AH = i; }, directive_helpers_a = o => o._$AH, directive_helpers_m = o => { var i; null === (i = o._$AP) || void 0 === i || i.call(o, !1, !0); let t = o._$AA; const n = o._$AB.nextSibling; for (; t !== n;) { const o = t.nextSibling; t.remove(), t = o; } }, directive_helpers_p = o => { o._$AR(); }; ;// CONCATENATED MODULE: ./node_modules/lit-html/async-directive.js /** * @license * Copyright 2017 Google LLC * SPDX-License-Identifier: BSD-3-Clause */ const async_directive_e = (i, t) => { var s, o; const n = i._$AN; if (void 0 === n) return !1; for (const i of n) null === (o = (s = i)._$AO) || void 0 === o || o.call(s, t, !1), async_directive_e(i, t); return !0; }, async_directive_o = i => { let t, s; do { if (void 0 === (t = i._$AM)) break; s = t._$AN, s.delete(i), i = t; } while (0 === (null == s ? void 0 : s.size)); }, async_directive_n = i => { for (let t; t = i._$AM; i = t) { let s = t._$AN; if (void 0 === s) t._$AN = s = new Set();else if (s.has(i)) break; s.add(i), async_directive_l(t); } }; function async_directive_r(i) { void 0 !== this._$AN ? (async_directive_o(this), this._$AM = i, async_directive_n(this)) : this._$AM = i; } function async_directive_h(i) { let t = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : !1; let s = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 0; const n = this._$AH, r = this._$AN; if (void 0 !== r && 0 !== r.size) if (t) { if (Array.isArray(n)) for (let i = s; i < n.length; i++) async_directive_e(n[i], !1), async_directive_o(n[i]);else null != n && (async_directive_e(n, !1), async_directive_o(n)); } else async_directive_e(this, i); } const async_directive_l = i => { var t, e, o, n; i.type == directive_t.CHILD && (null !== (t = (o = i)._$AP) && void 0 !== t || (o._$AP = async_directive_h), null !== (e = (n = i)._$AQ) && void 0 !== e || (n._$AQ = async_directive_r)); }; class async_directive_d extends directive_i { constructor() { super(...arguments), this._$AN = void 0; } _$AT(i, t, s) { super._$AT(i, t, s), async_directive_n(this), this.isConnected = i._$AU; } _$AO(i) { let t = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : !0; var s, n; i !== this.isConnected && (this.isConnected = i, i ? null === (s = this.reconnected) || void 0 === s || s.call(this) : null === (n = this.disconnected) || void 0 === n || n.call(this)), t && (async_directive_e(this, i), async_directive_o(this)); } setValue(t) { if (directive_helpers_r(this._$Ct)) this._$Ct._$AI(t, this);else { const i = [...this._$Ct._$AH]; i[this._$Ci] = t, this._$Ct._$AI(i, this, 0); } } disconnected() {} reconnected() {} } ;// CONCATENATED MODULE: ./node_modules/lit-html/directives/private-async-helpers.js /** * @license * Copyright 2021 Google LLC * SPDX-License-Identifier: BSD-3-Clause */ const private_async_helpers_t = async (t, s) => { for await (const i of t) if (!1 === (await s(i))) return; }; class private_async_helpers_s { constructor(t) { this.U = t; } disconnect() { this.U = void 0; } reconnect(t) { this.U = t; } deref() { return this.U; } } class private_async_helpers_i { constructor() { this.Y = void 0, this.q = void 0; } get() { return this.Y; } pause() { var t; null !== (t = this.Y) && void 0 !== t || (this.Y = new Promise(t => this.q = t)); } resume() { var t; null === (t = this.q) || void 0 === t || t.call(this), this.Y = this.q = void 0; } } ;// CONCATENATED MODULE: ./node_modules/lit-html/directives/until.js /** * @license * Copyright 2017 Google LLC * SPDX-License-Identifier: BSD-3-Clause */ const until_n = t => !directive_helpers_t(t) && "function" == typeof t.then; class until_h extends async_directive_d { constructor() { super(...arguments), this._$Cwt = 1073741823, this._$Cyt = [], this._$CG = new private_async_helpers_s(this), this._$CK = new private_async_helpers_i(); } render() { var i; for (var _len = arguments.length, s = new Array(_len), _key = 0; _key < _len; _key++) { s[_key] = arguments[_key]; } return null !== (i = s.find(t => !until_n(t))) && void 0 !== i ? i : b; } update(s, i) { const r = this._$Cyt; let e = r.length; this._$Cyt = i; const o = this._$CG, h = this._$CK; this.isConnected || this.disconnected(); for (let t = 0; t < i.length && !(t > this._$Cwt); t++) { const s = i[t]; if (!until_n(s)) return this._$Cwt = t, s; t < e && s === r[t] || (this._$Cwt = 1073741823, e = 0, Promise.resolve(s).then(async t => { for (; h.get();) await h.get(); const i = o.deref(); if (void 0 !== i) { const r = i._$Cyt.indexOf(s); r > -1 && r < i._$Cwt && (i._$Cwt = r, i.setValue(t)); } })); } return b; } disconnected() { this._$CG.disconnect(), this._$CK.pause(); } reconnected() { this._$CG.reconnect(this), this._$CK.resume(); } } const until_c = directive_e(until_h); ;// CONCATENATED MODULE: ./node_modules/lit/directives/until.js //# sourceMappingURL=until.js.map ;// CONCATENATED MODULE: ./src/plugins/bookmark-views/templates/list.js const list = (el, bookmarks) => { const desc_bookmarks = __('Click to toggle the bookmarks list'); const label_bookmarks = __('Bookmarks'); const toggle_state = el.model.get('toggle-state'); return $`
el.toggleBookmarksList()}> ${label_bookmarks}
${shared_converse.bookmarks.map(bm => item(bm))}
`; }; /* harmony default export */ const templates_list = (el => { const bookmarks = shared_converse.bookmarks.getUnopenedBookmarks(); return until_c(bookmarks.then(bookmarks => list(el, bookmarks)), ''); }); ;// CONCATENATED MODULE: ./src/plugins/bookmark-views/bookmarks-list.js class BookmarksView extends CustomElement { async initialize() { await core_api.waitUntil('bookmarksInitialized'); const { bookmarks, chatboxes } = shared_converse; this.listenTo(bookmarks, 'add', () => this.requestUpdate()); this.listenTo(bookmarks, 'remove', () => this.requestUpdate()); this.listenTo(chatboxes, 'add', () => this.requestUpdate()); this.listenTo(chatboxes, 'remove', () => this.requestUpdate()); const id = `converse.bookmarks-list-model-${shared_converse.bare_jid}`; this.model = new shared_converse.BookmarksList({ id }); initStorage(this.model, id); this.listenTo(this.model, 'change', () => this.requestUpdate()); this.model.fetch({ 'success': () => this.requestUpdate(), 'error': () => this.requestUpdate() }); } render() { return shared_converse.bookmarks && this.model ? templates_list(this) : ''; } toggleBookmarksList(ev) { var _ev$preventDefault; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); const { CLOSED, OPENED } = shared_converse; this.model.save({ 'toggle-state': this.model.get('toggle-state') === CLOSED ? OPENED : CLOSED }); } } core_api.elements.define('converse-bookmarks', BookmarksView); ;// CONCATENATED MODULE: ./src/plugins/bookmark-views/mixins.js const { u: mixins_u } = core_converse.env; const bookmarkableChatRoomView = { /** * Set whether the groupchat is bookmarked or not. * @private */ setBookmarkState() { if (shared_converse.bookmarks !== undefined) { const models = shared_converse.bookmarks.where({ 'jid': this.model.get('jid') }); if (!models.length) { this.model.save('bookmarked', false); } else { this.model.save('bookmarked', true); } } }, renderBookmarkForm() { if (!this.bookmark_form) { this.bookmark_form = new shared_converse.MUCBookmarkForm({ 'model': this.model, 'chatroomview': this }); const container_el = this.querySelector('.chatroom-body'); container_el.insertAdjacentElement('beforeend', this.bookmark_form.el); } mixins_u.showElement(this.bookmark_form.el); }, showBookmarkModal(ev) { ev === null || ev === void 0 ? void 0 : ev.preventDefault(); const jid = this.model.get('jid'); core_api.modal.show(bookmark_views_modal, { jid }, ev); } }; // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/bookmark-views/styles/bookmarks.scss var bookmarks = __webpack_require__(5251); ;// CONCATENATED MODULE: ./src/plugins/bookmark-views/styles/bookmarks.scss var bookmarks_options = {}; bookmarks_options.styleTagTransform = (styleTagTransform_default()); bookmarks_options.setAttributes = (setAttributesWithoutAttributes_default()); bookmarks_options.insert = insertBySelector_default().bind(null, "head"); bookmarks_options.domAPI = (styleDomAPI_default()); bookmarks_options.insertStyleElement = (insertStyleElement_default()); var bookmarks_update = injectStylesIntoStyleTag_default()(bookmarks/* default */.Z, bookmarks_options); /* harmony default export */ const styles_bookmarks = (bookmarks/* default */.Z && bookmarks/* default.locals */.Z.locals ? bookmarks/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/bookmark-views/index.js /** * @description Converse.js plugin which adds views for XEP-0048 bookmarks * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.plugins.add('converse-bookmark-views', { /* Plugin dependencies are other plugins which might be * overridden or relied upon, and therefore need to be loaded before * this plugin. * * If the setting "strict_plugin_dependencies" is set to true, * an error will be raised if the plugin is not found. By default it's * false, which means these plugins are only loaded opportunistically. */ dependencies: ['converse-chatboxes', 'converse-muc', 'converse-muc-views'], initialize() { // Configuration values for this plugin // ==================================== // Refer to docs/source/configuration.rst for explanations of these // configuration settings. core_api.settings.extend({ hide_open_bookmarks: true }); shared_converse.removeBookmarkViaEvent = removeBookmarkViaEvent; shared_converse.addBookmarkViaEvent = addBookmarkViaEvent; Object.assign(shared_converse.ChatRoomView.prototype, bookmarkableChatRoomView); shared_converse.MUCBookmarkForm = bookmark_views_form; shared_converse.BookmarksView = BookmarksView; core_api.listen.on('getHeadingButtons', getHeadingButtons); core_api.listen.on('chatRoomViewInitialized', view => view.setBookmarkState()); } }); ;// CONCATENATED MODULE: ./src/templates/background_logo.js /* harmony default export */ const background_logo = (() => $`
Logo Converse converse.js
${core_api.settings.get('view_mode') === 'overlayed' ? $`
` : ''}
`); ;// CONCATENATED MODULE: ./node_modules/lit-html/directives/repeat.js /** * @license * Copyright 2017 Google LLC * SPDX-License-Identifier: BSD-3-Clause */ const repeat_u = (e, s, t) => { const r = new Map(); for (let l = s; l <= t; l++) r.set(e[l], l); return r; }, repeat_c = directive_e(class extends directive_i { constructor(e) { if (super(e), e.type !== directive_t.CHILD) throw Error("repeat() can only be used in text expressions"); } dt(e, s, t) { let r; void 0 === t ? t = s : void 0 !== s && (r = s); const l = [], o = []; let i = 0; for (const s of e) l[i] = r ? r(s, i) : i, o[i] = t(s, i), i++; return { values: o, keys: l }; } render(e, s, t) { return this.dt(e, s, t).values; } update(s, _ref) { let [t, r, c] = _ref; var d; const a = directive_helpers_a(s), { values: p, keys: v } = this.dt(t, r, c); if (!Array.isArray(a)) return this.ut = v, p; const h = null !== (d = this.ut) && void 0 !== d ? d : this.ut = [], m = []; let y, x, j = 0, k = a.length - 1, w = 0, A = p.length - 1; for (; j <= k && w <= A;) if (null === a[j]) j++;else if (null === a[k]) k--;else if (h[j] === v[w]) m[w] = directive_helpers_c(a[j], p[w]), j++, w++;else if (h[k] === v[A]) m[A] = directive_helpers_c(a[k], p[A]), k--, A--;else if (h[j] === v[A]) m[A] = directive_helpers_c(a[j], p[A]), directive_helpers_u(s, m[A + 1], a[j]), j++, A--;else if (h[k] === v[w]) m[w] = directive_helpers_c(a[k], p[w]), directive_helpers_u(s, a[j], a[k]), k--, w++;else if (void 0 === y && (y = repeat_u(v, w, A), x = repeat_u(h, j, k)), y.has(h[j])) { if (y.has(h[k])) { const e = x.get(v[w]), t = void 0 !== e ? a[e] : null; if (null === t) { const e = directive_helpers_u(s, a[j]); directive_helpers_c(e, p[w]), m[w] = e; } else m[w] = directive_helpers_c(t, p[w]), directive_helpers_u(s, a[j], t), a[e] = null; w++; } else directive_helpers_m(a[k]), k--; } else directive_helpers_m(a[j]), j++; for (; w <= A;) { const e = directive_helpers_u(s, m[A + 1]); directive_helpers_c(e, p[w]), m[w++] = e; } for (; j <= k;) { const e = a[j++]; null !== e && directive_helpers_m(e); } return this.ut = v, directive_helpers_s(s, m), b; } }); ;// CONCATENATED MODULE: ./node_modules/lit/directives/repeat.js //# sourceMappingURL=repeat.js.map ;// CONCATENATED MODULE: ./src/plugins/chatboxviews/templates/chats.js function shouldShowChat(c) { const { CONTROLBOX_TYPE } = shared_converse; const is_minimized = core_api.settings.get('view_mode') === 'overlayed' && c.get('minimized'); return c.get('type') === CONTROLBOX_TYPE || !(c.get('hidden') || is_minimized); } /* harmony default export */ const chats = (() => { const { chatboxes, CONTROLBOX_TYPE, CHATROOMS_TYPE, HEADLINES_TYPE } = shared_converse; const view_mode = core_api.settings.get('view_mode'); const connection = shared_converse === null || shared_converse === void 0 ? void 0 : shared_converse.connection; const logged_out = !(connection !== null && connection !== void 0 && connection.connected) || !(connection !== null && connection !== void 0 && connection.authenticated) || (connection === null || connection === void 0 ? void 0 : connection.disconnecting); return $` ${!logged_out && view_mode === 'overlayed' ? $`` : ''} ${repeat_c(chatboxes.filter(shouldShowChat), m => m.get('jid'), m => { if (m.get('type') === CONTROLBOX_TYPE) { return $` ${view_mode === 'overlayed' ? $`` : ''} `; } else if (m.get('type') === CHATROOMS_TYPE) { return $` `; } else if (m.get('type') === HEADLINES_TYPE) { return $` `; } else { return $` `; } })} `; }); ;// CONCATENATED MODULE: ./src/plugins/chatboxviews/view.js class ConverseChats extends CustomElement { initialize() { this.model = shared_converse.chatboxes; this.listenTo(this.model, 'add', () => this.requestUpdate()); this.listenTo(this.model, 'change:closed', () => this.requestUpdate()); this.listenTo(this.model, 'change:hidden', () => this.requestUpdate()); this.listenTo(this.model, 'change:jid', () => this.requestUpdate()); this.listenTo(this.model, 'change:minimized', () => this.requestUpdate()); this.listenTo(this.model, 'destroy', () => this.requestUpdate()); // Use listenTo instead of api.listen.to so that event handlers // automatically get deregistered when the component is dismounted this.listenTo(shared_converse, 'connected', () => this.requestUpdate()); this.listenTo(shared_converse, 'reconnected', () => this.requestUpdate()); this.listenTo(shared_converse, 'disconnected', () => this.requestUpdate()); const settings = getAppSettings(); this.listenTo(settings, 'change:view_mode', () => this.requestUpdate()); this.listenTo(settings, 'change:singleton', () => this.requestUpdate()); const bg = document.getElementById('conversejs-bg'); if (bg && !bg.innerHTML.trim()) { x(background_logo(), bg); } const body = document.querySelector('body'); body.classList.add(`converse-${core_api.settings.get('view_mode')}`); /** * Triggered once the _converse.ChatBoxViews view-colleciton has been initialized * @event _converse#chatBoxViewsInitialized * @example _converse.api.listen.on('chatBoxViewsInitialized', () => { ... }); */ core_api.trigger('chatBoxViewsInitialized'); } render() { // eslint-disable-line class-methods-use-this return chats(); } } core_api.elements.define('converse-chats', ConverseChats); ;// CONCATENATED MODULE: ./src/plugins/chatboxviews/container.js class ChatBoxViews { constructor() { this.views = {}; } add(key, val) { this.views[key] = val; } get(key) { return this.views[key]; } xget(id) { return this.keys().filter(k => k !== id).reduce((acc, k) => { acc[k] = this.views[k]; return acc; }, {}); } getAll() { return Object.values(this.views); } keys() { return Object.keys(this.views); } remove(key) { delete this.views[key]; } map(f) { return Object.values(this.views).map(f); } forEach(f) { return Object.values(this.views).forEach(f); } filter(f) { return Object.values(this.views).filter(f); } closeAllChatBoxes() { return Promise.all(Object.values(this.views).map(v => v.close({ 'name': 'closeAllChatBoxes' }))); } } /* harmony default export */ const container = (ChatBoxViews); ;// CONCATENATED MODULE: ./src/plugins/chatboxviews/utils.js function calculateViewportHeightUnit() { const vh = window.innerHeight * 0.01; document.documentElement.style.setProperty('--vh', `${vh}px`); } // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/chatboxviews/styles/chats.scss var styles_chats = __webpack_require__(6931); ;// CONCATENATED MODULE: ./src/plugins/chatboxviews/styles/chats.scss var chats_options = {}; chats_options.styleTagTransform = (styleTagTransform_default()); chats_options.setAttributes = (setAttributesWithoutAttributes_default()); chats_options.insert = insertBySelector_default().bind(null, "head"); chats_options.domAPI = (styleDomAPI_default()); chats_options.insertStyleElement = (insertStyleElement_default()); var chats_update = injectStylesIntoStyleTag_default()(styles_chats/* default */.Z, chats_options); /* harmony default export */ const chatboxviews_styles_chats = (styles_chats/* default */.Z && styles_chats/* default.locals */.Z.locals ? styles_chats/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/chatboxviews/index.js /** * @module converse-chatboxviews * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.plugins.add('converse-chatboxviews', { dependencies: ['converse-chatboxes', 'converse-vcard'], initialize() { core_api.promises.add(['chatBoxViewsInitialized']); // Configuration values for this plugin // ==================================== // Refer to docs/source/configuration.rst for explanations of these // configuration settings. core_api.settings.extend({ 'animate': true }); shared_converse.chatboxviews = new container(); /************************ BEGIN Event Handlers ************************/ core_api.listen.on('chatBoxesInitialized', () => { shared_converse.chatboxes.on('destroy', m => shared_converse.chatboxviews.remove(m.get('jid'))); }); core_api.listen.on('cleanup', () => delete shared_converse.chatboxviews); core_api.listen.on('clearSession', () => shared_converse.chatboxviews.closeAllChatBoxes()); core_api.listen.on('chatBoxViewsInitialized', calculateViewportHeightUnit); window.addEventListener('resize', calculateViewportHeightUnit); /************************ END Event Handlers ************************/ Object.assign(core_converse, { /** * Public API method which will ensure that the #conversejs element * is inserted into a container element. * * This method is useful when the #conversejs element has been * detached from the DOM somehow. * @async * @memberOf converse * @method insertInto * @example * converse.insertInto(document.querySelector('#converse-container')); */ insertInto(container) { var _converse$chatboxview; const el = (_converse$chatboxview = shared_converse.chatboxviews) === null || _converse$chatboxview === void 0 ? void 0 : _converse$chatboxview.el; if (el && !container.contains(el)) { container.insertAdjacentElement('afterBegin', el); } else if (!el) { throw new Error('Cannot insert non-existing #conversejs element into the DOM'); } } }); } }); ;// CONCATENATED MODULE: ./src/plugins/modal/templates/alert.js /* harmony default export */ const modal_templates_alert = (o => $` `); ;// CONCATENATED MODULE: ./src/plugins/modal/alert.js const Alert = base.extend({ id: 'alert-modal', initialize() { base.prototype.initialize.apply(this, arguments); this.listenTo(this.model, 'change', this.render); }, toHTML() { return modal_templates_alert(Object.assign({ __: __ }, this.model.toJSON())); } }); /* harmony default export */ const modal_alert = (Alert); ;// CONCATENATED MODULE: ./src/plugins/modal/templates/prompt.js const tpl_field = f => $`
`; /* harmony default export */ const templates_prompt = (o => $` `); ;// CONCATENATED MODULE: ./src/plugins/modal/confirm.js const Confirm = base.extend({ id: 'confirm-modal', events: { 'submit .confirm': 'onConfimation' }, initialize() { this.confirmation = getOpenPromise(); base.prototype.initialize.apply(this, arguments); this.listenTo(this.model, 'change', this.render); this.el.addEventListener('closed.bs.modal', () => this.confirmation.reject(), false); }, toHTML() { return templates_prompt(this.model.toJSON()); }, afterRender() { if (!this.close_handler_registered) { this.el.addEventListener('closed.bs.modal', () => { if (!this.confirmation.isResolved) { this.confirmation.reject(); } }, false); this.close_handler_registered = true; } }, onConfimation(ev) { ev.preventDefault(); const form_data = new FormData(ev.target); const fields = (this.model.get('fields') || []).map(field => { const value = form_data.get(field.name).trim(); field.value = value; if (field.challenge) { field.challenge_failed = value !== field.challenge; } return field; }); if (fields.filter(c => c.challenge_failed).length) { this.model.set('fields', fields); // Setting an array doesn't trigger a change event this.model.trigger('change'); return; } this.confirmation.resolve(fields); this.modal.hide(); } }); /* harmony default export */ const modal_confirm = (Confirm); ;// CONCATENATED MODULE: ./src/plugins/modal/api.js let modals = []; const modal_api = { /** * API namespace for methods relating to modals * @namespace _converse.api.modal * @memberOf _converse.api */ modal: { /** * Shows a modal of type `ModalClass` to the user. * Will create a new instance of that class if an existing one isn't * found. * @param { Class } ModalClass * @param { Object } [properties] - Optional properties that will be * set on a newly created modal instance (if no pre-existing modal was * found). * @param { Event } [event] - The DOM event that causes the modal to be shown. */ show(ModalClass, properties, ev) { const modal = this.get(ModalClass.id) || this.create(ModalClass, properties); modal.show(ev); return modal; }, /** * Return a modal with the passed-in identifier, if it exists. * @param { String } id */ get(id) { return modals.filter(m => m.id == id).pop(); }, /** * Create a modal of the passed-in type. * @param { Class } ModalClass * @param { Object } [properties] - Optional properties that will be * set on the modal instance. */ create(ModalClass, properties) { const modal = new ModalClass(properties); modals.push(modal); return modal; }, /** * Remove a particular modal * @param { View } modal */ remove(modal) { modals = modals.filter(m => m !== modal); modal.remove(); }, /** * Remove all modals */ removeAll() { modals.forEach(m => m.remove()); modals = []; } }, /** * Show a confirm modal to the user. * @method _converse.api.confirm * @param { String } title - The header text for the confirmation dialog * @param { (Array|String) } messages - The text to show to the user * @param { Array } fields - An object representing a fields presented to the user. * @property { String } Field.label - The form label for the input field. * @property { String } Field.name - The name for the input field. * @property { String } [Field.challenge] - A challenge value that must be provided by the user. * @property { String } [Field.placeholder] - The placeholder for the input field. * @property { Boolean} [Field.required] - Whether the field is required or not * @returns { Promise } A promise which resolves with an array of * filled in fields or `false` if the confirm dialog was closed or canceled. */ async confirm(title) { let messages = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : []; let fields = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : []; if (typeof messages === 'string') { messages = [messages]; } const model = new Model({ title, messages, fields, 'type': 'confirm' }); const confirm = new modal_confirm({ model }); confirm.show(); let result; try { result = await confirm.confirmation; } catch (e) { result = false; } confirm.remove(); return result; }, /** * Show a prompt modal to the user. * @method _converse.api.prompt * @param { String } title - The header text for the prompt * @param { (Array|String) } messages - The prompt text to show to the user * @param { String } placeholder - The placeholder text for the prompt input * @returns { Promise } A promise which resolves with the text provided by the * user or `false` if the user canceled the prompt. */ async prompt(title) { let messages = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : []; let placeholder = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : ''; if (typeof messages === 'string') { messages = [messages]; } const model = new Model({ title, messages, 'fields': [{ 'name': 'reason', 'placeholder': placeholder }], 'type': 'prompt' }); const prompt = new modal_confirm({ model }); prompt.show(); let result; try { var _await$prompt$confirm; result = (_await$prompt$confirm = (await prompt.confirmation).pop()) === null || _await$prompt$confirm === void 0 ? void 0 : _await$prompt$confirm.value; } catch (e) { result = false; } prompt.remove(); return result; }, /** * Show an alert modal to the user. * @method _converse.api.alert * @param { ('info'|'warn'|'error') } type - The type of alert. * @param { String } title - The header text for the alert. * @param { (Array|String) } messages - The alert text to show to the user. */ alert(type, title, messages) { if (typeof messages === 'string') { messages = [messages]; } let level; if (type === 'error') { level = 'alert-danger'; } else if (type === 'info') { level = 'alert-info'; } else if (type === 'warn') { level = 'alert-warning'; } const model = new Model({ 'title': title, 'messages': messages, 'level': level, 'type': 'alert' }); modal_api.modal.show(modal_alert, { model }); } }; /* harmony default export */ const plugins_modal_api = (modal_api); ;// CONCATENATED MODULE: ./src/plugins/modal/index.js /** * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.env.BootstrapModal = base; // expose to plugins core_converse.plugins.add('converse-modal', { initialize() { core_api.listen.on('disconnect', () => { const container = document.querySelector("#converse-modals"); if (container) { container.innerHTML = ''; } }); core_api.listen.on('clearSession', () => core_api.modal.removeAll()); Object.assign(shared_converse.api, plugins_modal_api); } }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/shared/chat/styles/message-actions.scss var message_actions = __webpack_require__(4166); ;// CONCATENATED MODULE: ./src/shared/chat/styles/message-actions.scss var message_actions_options = {}; message_actions_options.styleTagTransform = (styleTagTransform_default()); message_actions_options.setAttributes = (setAttributesWithoutAttributes_default()); message_actions_options.insert = insertBySelector_default().bind(null, "head"); message_actions_options.domAPI = (styleDomAPI_default()); message_actions_options.insertStyleElement = (insertStyleElement_default()); var message_actions_update = injectStylesIntoStyleTag_default()(message_actions/* default */.Z, message_actions_options); /* harmony default export */ const styles_message_actions = (message_actions/* default */.Z && message_actions/* default.locals */.Z.locals ? message_actions/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/shared/chat/message-actions.js const { Strophe: message_actions_Strophe, u: message_actions_u } = core_converse.env; class MessageActions extends CustomElement { static get properties() { return { is_retracted: { type: Boolean }, model: { type: Object } }; } initialize() { const settings = getAppSettings(); this.listenTo(settings, 'change:allowed_audio_domains', () => this.requestUpdate()); this.listenTo(settings, 'change:allowed_image_domains', () => this.requestUpdate()); this.listenTo(settings, 'change:allowed_video_domains', () => this.requestUpdate()); this.listenTo(settings, 'change:render_media', () => this.requestUpdate()); this.listenTo(this.model, 'change', () => this.requestUpdate()); } render() { return $`${until_c(this.renderActions(), '')}`; } async renderActions() { // We want to let the message actions menu drop upwards if we're at the // bottom of the message history, and down otherwise. This is to avoid // the menu disappearing behind the bottom panel (toolbar, textarea etc). // That's difficult to know from state, so we're making an approximation here. const should_drop_up = this.model.collection.length > 2 && this.model === this.model.collection.last(); const buttons = await this.getActionButtons(); const items = buttons.map(b => MessageActions.getActionsDropdownItem(b)); if (items.length) { return $``; } else { return ''; } } static getActionsDropdownItem(o) { return $` `; } onMessageEditButtonClicked(ev) { var _u$ancestor, _u$ancestor$querySele; ev.preventDefault(); const currently_correcting = this.model.collection.findWhere('correcting'); // TODO: Use state intead of DOM querying // Then this code can also be put on the model const unsent_text = (_u$ancestor = message_actions_u.ancestor(this, '.chatbox')) === null || _u$ancestor === void 0 ? void 0 : (_u$ancestor$querySele = _u$ancestor.querySelector('.chat-textarea')) === null || _u$ancestor$querySele === void 0 ? void 0 : _u$ancestor$querySele.value; if (unsent_text && (!currently_correcting || currently_correcting.getMessageText() !== unsent_text)) { if (!confirm(__('You have an unsent message which will be lost if you continue. Are you sure?'))) { return; } } if (currently_correcting !== this.model) { currently_correcting === null || currently_correcting === void 0 ? void 0 : currently_correcting.save('correcting', false); this.model.save('correcting', true); } else { this.model.save('correcting', false); } } async onDirectMessageRetractButtonClicked() { if (this.model.get('sender') !== 'me') { return headless_log.error("onMessageRetractButtonClicked called for someone else's message!"); } const retraction_warning = __('Be aware that other XMPP/Jabber clients (and servers) may ' + 'not yet support retractions and that this message may not ' + 'be removed everywhere.'); const messages = [__('Are you sure you want to retract this message?')]; if (core_api.settings.get('show_retraction_warning')) { messages[1] = retraction_warning; } const result = await core_api.confirm(__('Confirm'), messages); if (result) { const chatbox = this.model.collection.chatbox; chatbox.retractOwnMessage(this.model); } } /** * Retract someone else's message in this groupchat. * @private * @param { _converse.Message } message - The message which we're retracting. * @param { string } [reason] - The reason for retracting the message. */ async retractOtherMessage(reason) { const chatbox = this.model.collection.chatbox; const result = await chatbox.retractOtherMessage(this.model, reason); if (result === null) { const err_msg = __(`A timeout occurred while trying to retract the message`); core_api.alert('error', __('Error'), err_msg); headless_log(err_msg, message_actions_Strophe.LogLevel.WARN); } else if (message_actions_u.isErrorStanza(result)) { const err_msg = __(`Sorry, you're not allowed to retract this message.`); core_api.alert('error', __('Error'), err_msg); headless_log(err_msg, message_actions_Strophe.LogLevel.WARN); headless_log(result, message_actions_Strophe.LogLevel.WARN); } } async onMUCMessageRetractButtonClicked() { const retraction_warning = __('Be aware that other XMPP/Jabber clients (and servers) may ' + 'not yet support retractions and that this message may not ' + 'be removed everywhere.'); if (this.model.mayBeRetracted()) { const messages = [__('Are you sure you want to retract this message?')]; if (core_api.settings.get('show_retraction_warning')) { messages[1] = retraction_warning; } if (await core_api.confirm(__('Confirm'), messages)) { const chatbox = this.model.collection.chatbox; chatbox.retractOwnMessage(this.model); } } else if (await this.model.mayBeModerated()) { if (this.model.get('sender') === 'me') { let messages = [__('Are you sure you want to retract this message?')]; if (core_api.settings.get('show_retraction_warning')) { messages = [messages[0], retraction_warning, messages[1]]; } !!(await core_api.confirm(__('Confirm'), messages)) && this.retractOtherMessage(); } else { let messages = [__('You are about to retract this message.'), __('You may optionally include a message, explaining the reason for the retraction.')]; if (core_api.settings.get('show_retraction_warning')) { messages = [messages[0], retraction_warning, messages[1]]; } const reason = await core_api.prompt(__('Message Retraction'), messages, __('Optional reason')); reason !== false && this.retractOtherMessage(reason); } } else { const err_msg = __(`Sorry, you're not allowed to retract this message`); core_api.alert('error', __('Error'), err_msg); } } onMessageRetractButtonClicked(ev) { var _ev$preventDefault; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); const chatbox = this.model.collection.chatbox; if (chatbox.get('type') === shared_converse.CHATROOMS_TYPE) { this.onMUCMessageRetractButtonClicked(); } else { this.onDirectMessageRetractButtonClicked(); } } onMediaToggleClicked(ev) { var _ev$preventDefault2; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault2 = ev.preventDefault) === null || _ev$preventDefault2 === void 0 ? void 0 : _ev$preventDefault2.call(ev); if (this.hasHiddenMedia(this.getMediaURLs())) { this.model.save({ 'hide_url_previews': false, 'url_preview_transition': 'fade-in' }); } else { const ogp_metadata = this.model.get('ogp_metadata') || []; if (ogp_metadata.length) { this.model.set('url_preview_transition', 'fade-out'); } else { this.model.save({ 'hide_url_previews': true, 'url_preview_transition': 'fade-in' }); } } } /** * Check whether media is hidden or shown, which is used to determine the toggle text. * * If `render_media` is an array, check if there are media URLs outside * of that array, in which case we consider message media on the whole to be hidden (since * those excluded by the whitelist will be, even if the render_media whitelisted URLs are shown). * @param { Array } media_urls * @returns { Boolean } */ hasHiddenMedia(media_urls) { if (typeof this.model.get('hide_url_previews') === 'boolean') { return this.model.get('hide_url_previews'); } const render_media = core_api.settings.get('render_media'); if (Array.isArray(render_media)) { return media_urls.reduce((acc, url) => acc || !isDomainWhitelisted(render_media, url), false); } else { return !render_media; } } getMediaURLs() { const unfurls_to_show = (this.model.get('ogp_metadata') || []).map(o => ({ 'url': o['og:image'], 'is_image': true })).filter(o => isMediaURLDomainAllowed(o)); const media_urls = getMediaURLs(this.model.get('media_urls') || [], this.model.get('body')).filter(o => isMediaURLDomainAllowed(o)); return [...new Set([...media_urls.map(o => o.url), ...unfurls_to_show.map(o => o.url)])]; } /** * Adds a media rendering toggle to this message's action buttons if necessary. * * The toggle is only added if the message contains media URLs and if the * user is allowed to show or hide media for those URLs. * * Whether a user is allowed to show or hide domains depends on the config settings: * * allowed_audio_domains * * allowed_video_domains * * allowed_image_domains * * Whether media is currently shown or hidden is determined by the { @link hasHiddenMedia } method. * * @param { Array } buttons - An array of objects representing action buttons */ addMediaRenderingToggle(buttons) { const urls = this.getMediaURLs(); if (urls.length) { const hidden = this.hasHiddenMedia(urls); buttons.push({ 'i18n_text': hidden ? __('Show media') : __('Hide media'), 'handler': ev => this.onMediaToggleClicked(ev), 'button_class': 'chat-msg__action-hide-previews', 'icon_class': hidden ? 'fas fa-eye' : 'fas fa-eye-slash', 'name': 'hide' }); } } async getActionButtons() { const buttons = []; if (this.model.get('editable')) { /** * @typedef { Object } MessageActionAttributes * An object which represents a message action (as shown in the message dropdown); * @property { String } i18n_text * @property { Function } handler * @property { String } button_class * @property { String } icon_class * @property { String } name */ buttons.push({ 'i18n_text': this.model.get('correcting') ? __('Cancel Editing') : __('Edit'), 'handler': ev => this.onMessageEditButtonClicked(ev), 'button_class': 'chat-msg__action-edit', 'icon_class': 'fa fa-pencil-alt', 'name': 'edit' }); } const may_be_moderated = ['groupchat', 'mep'].includes(this.model.get('type')) && (await this.model.mayBeModerated()); const retractable = !this.is_retracted && (this.model.mayBeRetracted() || may_be_moderated); if (retractable) { buttons.push({ 'i18n_text': __('Retract'), 'handler': ev => this.onMessageRetractButtonClicked(ev), 'button_class': 'chat-msg__action-retract', 'icon_class': 'fas fa-trash-alt', 'name': 'retract' }); } if (!this.model.collection) { // While we were awaiting, this model got removed from the // collection (happens during tests) return []; } this.addMediaRenderingToggle(buttons); /** * *Hook* which allows plugins to add more message action buttons * @event _converse#getMessageActionButtons * @example * api.listen.on('getMessageActionButtons', (el, buttons) => { * buttons.push({ * 'i18n_text': 'Foo', * 'handler': ev => alert('Foo!'), * 'button_class': 'chat-msg__action-foo', * 'icon_class': 'fa fa-check', * 'name': 'foo' * }); * return buttons; * }); */ return core_api.hook('getMessageActionButtons', this, buttons); } } core_api.elements.define('converse-message-actions', MessageActions); ;// CONCATENATED MODULE: ./src/modals/templates/image.js /* harmony default export */ const templates_image = (o => { return $` `; }); ;// CONCATENATED MODULE: ./src/modals/image.js /* harmony default export */ const modals_image = (base.extend({ id: 'image-modal', toHTML() { return templates_image({ 'src': this.src, 'onload': ev => ev.target.parentElement.style.height = `${ev.target.height}px` }); } })); ;// CONCATENATED MODULE: ./node_modules/lit/directive.js //# sourceMappingURL=directive.js.map ;// CONCATENATED MODULE: ./src/templates/audio.js /* harmony default export */ const audio = ((url, hide_url) => $`${hide_url ? '' : $`${url}`}`); ;// CONCATENATED MODULE: ./src/shared/gif/stream.js class Stream { constructor(data) { if (data.toString().indexOf('ArrayBuffer') > 0) { data = new Uint8Array(data); } this.data = data; this.len = this.data.length; this.pos = 0; } readByte() { if (this.pos >= this.data.length) { throw new Error('Attempted to read past end of stream.'); } if (this.data instanceof Uint8Array) return this.data[this.pos++];else return this.data.charCodeAt(this.pos++) & 0xFF; } readBytes(n) { const bytes = []; for (let i = 0; i < n; i++) { bytes.push(this.readByte()); } return bytes; } read(n) { let s = ''; for (let i = 0; i < n; i++) { s += String.fromCharCode(this.readByte()); } return s; } readUnsigned() { // Little-endian. const a = this.readBytes(2); return (a[1] << 8) + a[0]; } } ;// CONCATENATED MODULE: ./src/shared/gif/utils.js /** * @copyright Shachaf Ben-Kiki and the Converse.js contributors * @description * Started as a fork of Shachaf Ben-Kiki's jsgif library * https://github.com/shachaf/jsgif * @license MIT License */ function bitsToNum(ba) { return ba.reduce(function (s, n) { return s * 2 + n; }, 0); } function byteToBitArr(bite) { const a = []; for (let i = 7; i >= 0; i--) { a.push(!!(bite & 1 << i)); } return a; } function lzwDecode(minCodeSize, data) { // TODO: Now that the GIF parser is a bit different, maybe this should get an array of bytes instead of a String? let pos = 0; // Maybe this streaming thing should be merged with the Stream? function readCode(size) { let code = 0; for (let i = 0; i < size; i++) { if (data.charCodeAt(pos >> 3) & 1 << (pos & 7)) { code |= 1 << i; } pos++; } return code; } const output = []; const clearCode = 1 << minCodeSize; const eoiCode = clearCode + 1; let codeSize = minCodeSize + 1; let dict = []; const clear = function () { dict = []; codeSize = minCodeSize + 1; for (let i = 0; i < clearCode; i++) { dict[i] = [i]; } dict[clearCode] = []; dict[eoiCode] = null; }; let code; let last; while (true) { // eslint-disable-line no-constant-condition last = code; code = readCode(codeSize); if (code === clearCode) { clear(); continue; } if (code === eoiCode) break; if (code < dict.length) { if (last !== clearCode) { dict.push(dict[last].concat(dict[code][0])); } } else { if (code !== dict.length) throw new Error('Invalid LZW code.'); dict.push(dict[last].concat(dict[last][0])); } output.push.apply(output, dict[code]); if (dict.length === 1 << codeSize && codeSize < 12) { // If we're at the last code and codeSize is 12, the next code will be a clearCode, and it'll be 12 bits long. codeSize++; } } // I don't know if this is technically an error, but some GIFs do it. //if (Math.ceil(pos / 8) !== data.length) throw new Error('Extraneous LZW bytes.'); return output; } function readSubBlocks(st) { let size, data; data = ''; do { size = st.readByte(); data += st.read(size); } while (size !== 0); return data; } /** * Parses GIF image color table information * @param { Stream } st * @param { Number } entries */ function parseCT(st, entries) { // Each entry is 3 bytes, for RGB. const ct = []; for (let i = 0; i < entries; i++) { ct.push(st.readBytes(3)); } return ct; } /** * Parses GIF image information * @param { Stream } st * @param { ByteStream } img * @param { Function } [callback] */ function parseImg(st, img, callback) { function deinterlace(pixels, width) { // Of course this defeats the purpose of interlacing. And it's *probably* // the least efficient way it's ever been implemented. But nevertheless... const newPixels = new Array(pixels.length); const rows = pixels.length / width; function cpRow(toRow, fromRow) { const fromPixels = pixels.slice(fromRow * width, (fromRow + 1) * width); newPixels.splice.apply(newPixels, [toRow * width, width].concat(fromPixels)); } // See appendix E. const offsets = [0, 4, 2, 1]; const steps = [8, 8, 4, 2]; let fromRow = 0; for (let pass = 0; pass < 4; pass++) { for (let toRow = offsets[pass]; toRow < rows; toRow += steps[pass]) { cpRow(toRow, fromRow); fromRow++; } } return newPixels; } img.leftPos = st.readUnsigned(); img.topPos = st.readUnsigned(); img.width = st.readUnsigned(); img.height = st.readUnsigned(); const bits = byteToBitArr(st.readByte()); img.lctFlag = bits.shift(); img.interlaced = bits.shift(); img.sorted = bits.shift(); img.reserved = bits.splice(0, 2); img.lctSize = bitsToNum(bits.splice(0, 3)); if (img.lctFlag) { img.lct = parseCT(st, 1 << img.lctSize + 1); } img.lzwMinCodeSize = st.readByte(); const lzwData = readSubBlocks(st); img.pixels = lzwDecode(img.lzwMinCodeSize, lzwData); if (img.interlaced) { // Move img.pixels = deinterlace(img.pixels, img.width); } callback === null || callback === void 0 ? void 0 : callback(img); } /** * Parses GIF header information * @param { Stream } st * @param { Function } [callback] */ function parseHeader(st, callback) { const hdr = {}; hdr.sig = st.read(3); hdr.ver = st.read(3); if (hdr.sig !== 'GIF') { throw new Error('Not a GIF file.'); } hdr.width = st.readUnsigned(); hdr.height = st.readUnsigned(); const bits = byteToBitArr(st.readByte()); hdr.gctFlag = bits.shift(); hdr.colorRes = bitsToNum(bits.splice(0, 3)); hdr.sorted = bits.shift(); hdr.gctSize = bitsToNum(bits.splice(0, 3)); hdr.bgColor = st.readByte(); hdr.pixelAspectRatio = st.readByte(); // if not 0, aspectRatio = (pixelAspectRatio + 15) / 64 if (hdr.gctFlag) { hdr.gct = parseCT(st, 1 << hdr.gctSize + 1); } callback === null || callback === void 0 ? void 0 : callback(hdr); } function parseExt(st, block, handler) { function parseGCExt(block) { st.readByte(); // blocksize, always 4 const bits = byteToBitArr(st.readByte()); block.reserved = bits.splice(0, 3); // Reserved; should be 000. block.disposalMethod = bitsToNum(bits.splice(0, 3)); block.userInput = bits.shift(); block.transparencyGiven = bits.shift(); block.delayTime = st.readUnsigned(); block.transparencyIndex = st.readByte(); block.terminator = st.readByte(); handler === null || handler === void 0 ? void 0 : handler.gce(block); } function parseComExt(block) { block.comment = readSubBlocks(st); handler.com && handler.com(block); } function parsePTExt(block) { // No one *ever* uses this. If you use it, deal with parsing it yourself. st.readByte(); // blocksize, always 12 block.ptHeader = st.readBytes(12); block.ptData = readSubBlocks(st); handler.pte && handler.pte(block); } function parseAppExt(block) { function parseNetscapeExt(block) { st.readByte(); // blocksize, always 3 block.unknown = st.readByte(); // ??? Always 1? What is this? block.iterations = st.readUnsigned(); block.terminator = st.readByte(); handler.app && handler.app.NETSCAPE && handler.app.NETSCAPE(block); } function parseUnknownAppExt(block) { block.appData = readSubBlocks(st); // FIXME: This won't work if a handler wants to match on any identifier. handler.app && handler.app[block.identifier] && handler.app[block.identifier](block); } st.readByte(); // blocksize, always 11 block.identifier = st.read(8); block.authCode = st.read(3); switch (block.identifier) { case 'NETSCAPE': parseNetscapeExt(block); break; default: parseUnknownAppExt(block); break; } } function parseUnknownExt(block) { block.data = readSubBlocks(st); handler.unknown && handler.unknown(block); } block.label = st.readByte(); switch (block.label) { case 0xF9: block.extType = 'gce'; parseGCExt(block); break; case 0xFE: block.extType = 'com'; parseComExt(block); break; case 0x01: block.extType = 'pte'; parsePTExt(block); break; case 0xFF: block.extType = 'app'; parseAppExt(block); break; default: block.extType = 'unknown'; parseUnknownExt(block); break; } } /** * @param { Stream } st * @param { GIFParserHandlers } handler */ function parseBlock(st, handler) { const block = {}; block.sentinel = st.readByte(); switch (String.fromCharCode(block.sentinel)) { // For ease of matching case '!': block.type = 'ext'; parseExt(st, block, handler); break; case ',': block.type = 'img'; parseImg(st, block, handler === null || handler === void 0 ? void 0 : handler.img); break; case ';': block.type = 'eof'; handler === null || handler === void 0 ? void 0 : handler.eof(block); break; default: throw new Error('Unknown block: 0x' + block.sentinel.toString(16)); // TODO: Pad this with a 0. } if (block.type !== 'eof') setTimeout(() => parseBlock(st, handler), 0); } /** * Takes a Stream and parses it for GIF data, calling the relevant handler * methods on the passed in `handler` object. * @param { Stream } st * @param { GIFParserHandlers } handler */ function parseGIF(st) { let handler = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; parseHeader(st, handler === null || handler === void 0 ? void 0 : handler.hdr); setTimeout(() => parseBlock(st, handler), 0); } ;// CONCATENATED MODULE: ./src/shared/gif/index.js /** * @copyright Shachaf Ben-Kiki, JC Brand * @description * Started as a fork of Shachaf Ben-Kiki's jsgif library * https://github.com/shachaf/jsgif * @license MIT License */ const DELAY_FACTOR = 10; class ConverseGif { /** * Creates a new ConverseGif instance * @param { HTMLElement } el * @param { Object } [options] * @param { Number } [options.width] - The width, in pixels, of the canvas * @param { Number } [options.height] - The height, in pixels, of the canvas * @param { Boolean } [options.loop=true] - Setting this to `true` will enable looping of the gif * @param { Boolean } [options.autoplay=true] - Same as the rel:autoplay attribute above, this arg overrides the img tag info. * @param { Number } [options.max_width] - Scale images over max_width down to max_width. Helpful with mobile. * @param { Function } [options.onIterationEnd] - Add a callback for when the gif reaches the end of a single loop (one iteration). The first argument passed will be the gif HTMLElement. * @param { Boolean } [options.show_progress_bar=true] * @param { String } [options.progress_bg_color='rgba(0,0,0,0.4)'] * @param { String } [options.progress_color='rgba(255,0,22,.8)'] * @param { Number } [options.progress_bar_height=5] */ constructor(el, opts) { this.options = Object.assign({ width: null, height: null, autoplay: true, loop: true, show_progress_bar: true, progress_bg_color: 'rgba(0,0,0,0.4)', progress_color: 'rgba(255,0,22,.8)', progress_bar_height: 5 }, opts); this.el = el; this.gif_el = el.querySelector('img'); this.canvas = el.querySelector('canvas'); this.ctx = this.canvas.getContext('2d'); // It's good practice to pre-render to an offscreen canvas this.offscreenCanvas = document.createElement('canvas'); this.ctx_scaled = false; this.disposal_method = null; this.disposal_restore_from_idx = null; this.frame = null; this.frame_offsets = []; // elements have .x and .y properties this.frames = []; this.last_disposal_method = null; this.last_img = null; this.load_error = null; this.playing = this.options.autoplay; this.transparency = null; this.frame_idx = 0; this.iteration_count = 0; this.start = null; this.initialize(); } async initialize() { if (this.options.width && this.options.height) { this.setSizes(this.options.width, this.options.height); } const data = await this.fetchGIF(this.gif_el.src); requestAnimationFrame(() => this.startParsing(data)); } initPlayer() { if (this.load_error) return; if (!(this.options.width && this.options.height)) { this.ctx.scale(this.getCanvasScale(), this.getCanvasScale()); } // Show the first frame this.frame_idx = 0; this.putFrame(this.frame_idx); if (this.options.autoplay) { var _this$frames$this$fra; const delay = (((_this$frames$this$fra = this.frames[this.frame_idx]) === null || _this$frames$this$fra === void 0 ? void 0 : _this$frames$this$fra.delay) ?? 0) * DELAY_FACTOR; setTimeout(() => this.play(), delay); } } /** * Gets the index of the frame "up next" * @returns {number} */ getNextFrameNo() { return (this.frame_idx + 1 + this.frames.length) % this.frames.length; } /** * Called once we've looped through all frames in the GIF * @returns { Boolean } - Returns `true` if the GIF is now paused (i.e. further iterations are not desired) */ onIterationEnd() { var _this$options$onItera, _this$options; this.iteration_count++; (_this$options$onItera = (_this$options = this.options).onIterationEnd) === null || _this$options$onItera === void 0 ? void 0 : _this$options$onItera.call(_this$options, this); if (!this.options.loop) { this.pause(); return true; } return false; } /** * Inner callback for the `requestAnimationFrame` function. * * This method gets wrapped by an arrow function so that the `previous_timestamp` and * `frame_delay` parameters can also be passed in. The `timestamp` * parameter comes from `requestAnimationFrame`. * * The purpose of this method is to call `putFrame` with the right delay * in order to render the GIF animation. * * Note, this method will cause the *next* upcoming frame to be rendered, * not the current one. * * This means `this.frame_idx` will be incremented before calling `this.putFrame`, so * `putFrame(0)` needs to be called *before* this method, otherwise the * animation will incorrectly start from frame #1 (this is done in `initPlayer`). * * @param { DOMHighRestTimestamp } timestamp - The timestamp as returned by `requestAnimationFrame` * @param { DOMHighRestTimestamp } previous_timestamp - The timestamp from the previous iteration of this method. * We need this in order to calculate whether we have waited long enough to * show the next frame. * @param { Number } frame_delay - The delay (in 1/100th of a second) * before the currently being shown frame should be replaced by a new one. */ onAnimationFrame(timestamp, previous_timestamp, frame_delay) { var _this$frames$this$fra2; if (!this.playing) { return; } if (timestamp - previous_timestamp < frame_delay) { this.hovering ? this.drawPauseIcon() : this.putFrame(this.frame_idx); // We need to wait longer requestAnimationFrame(ts => this.onAnimationFrame(ts, previous_timestamp, frame_delay)); return; } const next_frame = this.getNextFrameNo(); if (next_frame === 0 && this.onIterationEnd()) { return; } this.frame_idx = next_frame; this.putFrame(this.frame_idx); const delay = (((_this$frames$this$fra2 = this.frames[this.frame_idx]) === null || _this$frames$this$fra2 === void 0 ? void 0 : _this$frames$this$fra2.delay) || 8) * DELAY_FACTOR; requestAnimationFrame(ts => this.onAnimationFrame(ts, timestamp, delay)); } setSizes(w, h) { this.canvas.width = w * this.getCanvasScale(); this.canvas.height = h * this.getCanvasScale(); this.offscreenCanvas.width = w; this.offscreenCanvas.height = h; this.offscreenCanvas.style.width = w + 'px'; this.offscreenCanvas.style.height = h + 'px'; this.offscreenCanvas.getContext('2d').setTransform(1, 0, 0, 1, 0, 0); } setFrameOffset(frame, offset) { if (!this.frame_offsets[frame]) { this.frame_offsets[frame] = offset; return; } if (typeof offset.x !== 'undefined') { this.frame_offsets[frame].x = offset.x; } if (typeof offset.y !== 'undefined') { this.frame_offsets[frame].y = offset.y; } } doShowProgress(pos, length, draw) { if (draw && this.options.show_progress_bar) { let height = this.options.progress_bar_height; const top = (this.canvas.height - height) / (this.ctx_scaled ? this.getCanvasScale() : 1); const mid = pos / length * this.canvas.width / (this.ctx_scaled ? this.getCanvasScale() : 1); const width = this.canvas.width / (this.ctx_scaled ? this.getCanvasScale() : 1); height /= this.ctx_scaled ? this.getCanvasScale() : 1; this.ctx.fillStyle = this.options.progress_bg_color; this.ctx.fillRect(mid, top, width - mid, height); this.ctx.fillStyle = this.options.progress_color; this.ctx.fillRect(0, top, mid, height); } } /** * Starts parsing the GIF stream data by calling `parseGIF` and passing in * a map of handler functions. * @param { String } data - The GIF file data, as returned by the server */ startParsing(data) { const stream = new Stream(data); /** * @typedef { Object } GIFParserHandlers * A map of callback functions passed `parseGIF`. These functions are * called as various parts of the GIF file format are parsed. * @property { Function } hdr - Callback to handle the GIF header data * @property { Function } gce - Callback to handle the GIF Graphic Control Extension data * @property { Function } com - Callback to handle the comment extension block * @property { Function } img - Callback to handle image data * @property { Function } eof - Callback once the end of file has been reached */ const handler = { 'hdr': this.withProgress(stream, header => this.handleHeader(header)), 'gce': this.withProgress(stream, gce => this.handleGCE(gce)), 'com': this.withProgress(stream), 'img': this.withProgress(stream, img => this.doImg(img), true), 'eof': () => this.handleEOF(stream) }; try { parseGIF(stream, handler); } catch (err) { this.showError('parse'); } } drawError() { this.ctx.fillStyle = 'black'; this.ctx.fillRect(0, 0, this.options.width ? this.options.width : this.hdr.width, this.options.height ? this.options.height : this.hdr.height); this.ctx.strokeStyle = 'red'; this.ctx.lineWidth = 3; this.ctx.moveTo(0, 0); this.ctx.lineTo(this.options.width ? this.options.width : this.hdr.width, this.options.height ? this.options.height : this.hdr.height); this.ctx.moveTo(0, this.options.height ? this.options.height : this.hdr.height); this.ctx.lineTo(this.options.width ? this.options.width : this.hdr.width, 0); this.ctx.stroke(); } showError(errtype) { this.load_error = errtype; this.hdr = { width: this.gif_el.width, height: this.gif_el.height }; // Fake header. this.frames = []; this.drawError(); this.el.requestUpdate(); } handleHeader(header) { this.hdr = header; this.setSizes(this.options.width ?? this.hdr.width, this.options.height ?? this.hdr.height); } /** * Handler for GIF Graphic Control Extension (GCE) data */ handleGCE(gce) { this.pushFrame(gce.delayTime); this.clear(); this.transparency = gce.transparencyGiven ? gce.transparencyIndex : null; this.disposal_method = gce.disposalMethod; } /** * Handler for when the end of the GIF's file has been reached */ handleEOF(stream) { this.doDecodeProgress(stream, false); if (!(this.options.width && this.options.height)) { this.canvas.width = this.hdr.width * this.getCanvasScale(); this.canvas.height = this.hdr.height * this.getCanvasScale(); } this.initPlayer(); !this.options.autoplay && this.drawPlayIcon(); } pushFrame(delay) { if (!this.frame) return; this.frames.push({ data: this.frame.getImageData(0, 0, this.hdr.width, this.hdr.height), delay }); this.frame_offsets.push({ x: 0, y: 0 }); } doImg(img) { this.frame = this.frame || this.offscreenCanvas.getContext('2d'); const currIdx = this.frames.length; //ct = color table, gct = global color table const ct = img.lctFlag ? img.lct : this.hdr.gct; // TODO: What if neither exists? /* * Disposal method indicates the way in which the graphic is to * be treated after being displayed. * * Values : 0 - No disposal specified. The decoder is * not required to take any action. * 1 - Do not dispose. The graphic is to be left * in place. * 2 - Restore to background color. The area used by the * graphic must be restored to the background color. * 3 - Restore to previous. The decoder is required to * restore the area overwritten by the graphic with * what was there prior to rendering the graphic. * * Importantly, "previous" means the frame state * after the last disposal of method 0, 1, or 2. */ if (currIdx > 0) { if (this.last_disposal_method === 3) { // Restore to previous // If we disposed every frame including first frame up to this point, then we have // no composited frame to restore to. In this case, restore to background instead. if (this.disposal_restore_from_idx !== null) { this.frame.putImageData(this.frames[this.disposal_restore_from_idx].data, 0, 0); } else { this.frame.clearRect(this.last_img.leftPos, this.last_img.topPos, this.last_img.width, this.last_img.height); } } else { this.disposal_restore_from_idx = currIdx - 1; } if (this.last_disposal_method === 2) { // Restore to background color // Browser implementations historically restore to transparent; we do the same. // http://www.wizards-toolkit.org/discourse-server/viewtopic.php?f=1&t=21172#p86079 this.frame.clearRect(this.last_img.leftPos, this.last_img.topPos, this.last_img.width, this.last_img.height); } } // else, Undefined/Do not dispose. // frame contains final pixel data from the last frame; do nothing //Get existing pixels for img region after applying disposal method const imgData = this.frame.getImageData(img.leftPos, img.topPos, img.width, img.height); //apply color table colors img.pixels.forEach((pixel, i) => { // imgData.data === [R,G,B,A,R,G,B,A,...] if (pixel !== this.transparency) { imgData.data[i * 4 + 0] = ct[pixel][0]; imgData.data[i * 4 + 1] = ct[pixel][1]; imgData.data[i * 4 + 2] = ct[pixel][2]; imgData.data[i * 4 + 3] = 255; // Opaque. } }); this.frame.putImageData(imgData, img.leftPos, img.topPos); if (!this.ctx_scaled) { this.ctx.scale(this.getCanvasScale(), this.getCanvasScale()); this.ctx_scaled = true; } if (!this.last_img) { // This is the first receivd image, so we draw it this.ctx.drawImage(this.offscreenCanvas, 0, 0); } this.last_img = img; } /** * Draws a gif frame at a specific index inside the canvas. * @param { Number } i - The frame index */ putFrame(i) { let show_pause_on_hover = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : true; i = parseInt(i, 10); if (i > this.frames.length - 1) { i = 0; } if (i < 0) { i = 0; } const offset = this.frame_offsets[i]; this.offscreenCanvas.getContext('2d').putImageData(this.frames[i].data, offset.x, offset.y); this.ctx.globalCompositeOperation = 'copy'; this.ctx.drawImage(this.offscreenCanvas, 0, 0); if (show_pause_on_hover && this.hovering) { this.drawPauseIcon(); } } clear() { this.transparency = null; this.last_disposal_method = this.disposal_method; this.disposal_method = null; this.frame = null; } /** * Start playing the gif */ play() { this.playing = true; requestAnimationFrame(ts => this.onAnimationFrame(ts, 0, 0)); } /** * Pause the gif */ pause() { this.playing = false; requestAnimationFrame(() => this.drawPlayIcon()); } drawPauseIcon() { if (!this.playing) { return; } // Clear the potential play button by re-rendering the current frame this.putFrame(this.frame_idx, false); this.ctx.globalCompositeOperation = 'source-over'; // Draw dark overlay this.ctx.fillStyle = 'rgb(0, 0, 0, 0.25)'; this.ctx.fillRect(0, 0, this.canvas.width, this.canvas.height); const icon_size = this.canvas.height * 0.1; // Draw bars this.ctx.lineWidth = this.canvas.height * 0.04; this.ctx.beginPath(); this.ctx.moveTo(this.canvas.width / 2 - icon_size / 2, this.canvas.height / 2 - icon_size); this.ctx.lineTo(this.canvas.width / 2 - icon_size / 2, this.canvas.height / 2 + icon_size); this.ctx.fillStyle = 'rgb(200, 200, 200, 0.75)'; this.ctx.stroke(); this.ctx.beginPath(); this.ctx.moveTo(this.canvas.width / 2 + icon_size / 2, this.canvas.height / 2 - icon_size); this.ctx.lineTo(this.canvas.width / 2 + icon_size / 2, this.canvas.height / 2 + icon_size); this.ctx.fillStyle = 'rgb(200, 200, 200, 0.75)'; this.ctx.stroke(); // Draw circle this.ctx.lineWidth = this.canvas.height * 0.02; this.ctx.strokeStyle = 'rgb(200, 200, 200, 0.75)'; this.ctx.beginPath(); this.ctx.arc(this.canvas.width / 2, this.canvas.height / 2, icon_size * 1.5, 0, 2 * Math.PI); this.ctx.stroke(); } drawPlayIcon() { if (this.playing) { return; } // Clear the potential pause button by re-rendering the current frame this.putFrame(this.frame_idx, false); this.ctx.globalCompositeOperation = 'source-over'; // Draw dark overlay this.ctx.fillStyle = 'rgb(0, 0, 0, 0.25)'; this.ctx.fillRect(0, 0, this.canvas.width, this.canvas.height); // Draw triangle const triangle_size = this.canvas.height * 0.1; const region = new Path2D(); region.moveTo(this.canvas.width / 2 + triangle_size, this.canvas.height / 2); // start at the pointy end region.lineTo(this.canvas.width / 2 - triangle_size / 2, this.canvas.height / 2 + triangle_size); region.lineTo(this.canvas.width / 2 - triangle_size / 2, this.canvas.height / 2 - triangle_size); region.closePath(); this.ctx.fillStyle = 'rgb(200, 200, 200, 0.75)'; this.ctx.fill(region); // Draw circle const circle_size = triangle_size * 1.5; this.ctx.lineWidth = this.canvas.height * 0.02; this.ctx.strokeStyle = 'rgb(200, 200, 200, 0.75)'; this.ctx.beginPath(); this.ctx.arc(this.canvas.width / 2, this.canvas.height / 2, circle_size, 0, 2 * Math.PI); this.ctx.stroke(); } doDecodeProgress(stream, draw) { this.doShowProgress(stream.pos, stream.data.length, draw); } /** * @param{boolean=} draw Whether to draw progress bar or not; * this is not idempotent because of translucency. * Note that this means that the text will be unsynchronized * with the progress bar on non-frames; * but those are typically so small (GCE etc.) that it doesn't really matter */ withProgress(stream, fn, draw) { return block => { fn === null || fn === void 0 ? void 0 : fn(block); this.doDecodeProgress(stream, draw); }; } getCanvasScale() { let scale; if (this.options.max_width && this.hdr && this.hdr.width > this.options.max_width) { scale = this.options.max_width / this.hdr.width; } else { scale = 1; } return scale; } /** * Makes an HTTP request to fetch a GIF * @param { String } url * @returns { Promise } Returns a promise which resolves with the response data. */ fetchGIF(url) { const promise = getOpenPromise(); const h = new XMLHttpRequest(); h.open('GET', url, true); h === null || h === void 0 ? void 0 : h.overrideMimeType('text/plain; charset=x-user-defined'); h.onload = () => { if (h.status != 200) { this.showError('xhr - response'); return promise.reject(); } promise.resolve(h.response); }; h.onprogress = e => e.lengthComputable && this.doShowProgress(e.loaded, e.total, true); h.onerror = () => this.showError('xhr'); h.send(); return promise; } } ;// CONCATENATED MODULE: ./src/templates/file.js /* harmony default export */ const file = ((url, name) => { const i18n_download = __('Download file "%1$s"', name); return $`${i18n_download}`; }); ;// CONCATENATED MODULE: ./src/templates/form_captcha.js /* harmony default export */ const form_captcha = (o => $`
${o.label ? $`` : ''}
`); ;// CONCATENATED MODULE: ./src/templates/form_checkbox.js /* harmony default export */ const form_checkbox = (o => $`
`); ;// CONCATENATED MODULE: ./src/templates/form_help.js /* harmony default export */ const form_help = (o => $`

${o.text}

`); ;// CONCATENATED MODULE: ./src/templates/form_input.js /* harmony default export */ const form_input = (o => $`
${o.type !== 'hidden' ? $`` : ''} ${o.type === 'password' && o.fixed_username ? $` ` : ''}
`); ;// CONCATENATED MODULE: ./src/templates/form_select.js const tpl_option = o => $``; /* harmony default export */ const form_select = (o => { var _o$options; return $`
`; }); ;// CONCATENATED MODULE: ./src/templates/form_textarea.js /* harmony default export */ const form_textarea = (o => $` `); ;// CONCATENATED MODULE: ./src/templates/form_url.js /* harmony default export */ const form_url = (o => $` `); ;// CONCATENATED MODULE: ./src/templates/form_username.js /* harmony default export */ const form_username = (o => $`
${o.label ? $`` : ''}
${o.domain}
`); ;// CONCATENATED MODULE: ./src/templates/hyperlink.js function onClickXMPPURI(ev) { ev.preventDefault(); core_api.rooms.open(ev.target.href); } /* harmony default export */ const hyperlink = ((uri, url_text) => { let href_text = uri.normalizePath().toString(); if (!uri._parts.protocol && !url_text.startsWith('http://') && !url_text.startsWith('https://')) { href_text = 'http://' + href_text; } if (uri._parts.protocol === 'xmpp' && uri._parts.query === 'join') { return $` ${url_text}`; } return $`${url_text}`; }); ;// CONCATENATED MODULE: ./src/templates/video.js /* harmony default export */ const video = ((url, hide_url) => $`${hide_url ? '' : $`${url}`}`); ;// CONCATENATED MODULE: ./src/utils/html.js /** * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) * @description This is the DOM/HTML utilities module. */ const { sizzle: html_sizzle } = core_converse.env; const APPROVED_URL_PROTOCOLS = ['http', 'https', 'xmpp', 'mailto']; function getAutoCompleteProperty(name, options) { return { 'muc#roomconfig_lang': 'language', 'muc#roomconfig_roomsecret': options !== null && options !== void 0 && options.new_password ? 'new-password' : 'current-password' }[name]; } const XFORM_TYPE_MAP = { 'text-private': 'password', 'text-single': 'text', 'fixed': 'label', 'boolean': 'checkbox', 'hidden': 'hidden', 'jid-multi': 'textarea', 'list-single': 'dropdown', 'list-multi': 'dropdown' }; const XFORM_VALIDATE_TYPE_MAP = { 'xs:anyURI': 'url', 'xs:byte': 'number', 'xs:date': 'date', 'xs:dateTime': 'datetime', 'xs:int': 'number', 'xs:integer': 'number', 'xs:time': 'time' }; function getInputType(field) { const type = XFORM_TYPE_MAP[field.getAttribute('type')]; if (type == 'text') { const datatypes = field.getElementsByTagNameNS("http://jabber.org/protocol/xdata-validate", "validate"); if (datatypes.length === 1) { const datatype = datatypes[0].getAttribute("datatype"); return XFORM_VALIDATE_TYPE_MAP[datatype] || type; } } return type; } function slideOutWrapup(el) { /* Wrapup function for slideOut. */ el.removeAttribute('data-slider-marker'); el.classList.remove('collapsed'); el.style.overflow = ''; el.style.height = ''; } function getFileName(uri) { try { return decodeURI(uri.filename()); } catch (error) { headless_log.debug(error); return uri.filename(); } } /** * Returns the markup for a URL that points to a downloadable asset * (such as a video, image or audio file). * @method u#getOOBURLMarkup * @param { String } url * @returns { String } */ function getOOBURLMarkup(url) { const uri = getURI(url); if (uri === null) { return url; } if (isVideoURL(uri)) { return video(url); } else if (isAudioURL(uri)) { return audio(url); } else if (isImageURL(uri)) { return file(uri.toString(), getFileName(uri)); } else { return file(uri.toString(), getFileName(uri)); } } /** * Return the height of the passed in DOM element, * based on the heights of its children. * @method u#calculateElementHeight * @param {HTMLElement} el * @returns {integer} */ utils_core.calculateElementHeight = function (el) { return Array.from(el.children).reduce((result, child) => result + child.offsetHeight, 0); }; utils_core.getNextElement = function (el) { let selector = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : '*'; let next_el = el.nextElementSibling; while (next_el !== null && !html_sizzle.matchesSelector(next_el, selector)) { next_el = next_el.nextElementSibling; } return next_el; }; utils_core.getPreviousElement = function (el) { let selector = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : '*'; let prev_el = el.previousElementSibling; while (prev_el !== null && !html_sizzle.matchesSelector(prev_el, selector)) { prev_el = prev_el.previousElementSibling; } return prev_el; }; utils_core.getFirstChildElement = function (el) { let selector = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : '*'; let first_el = el.firstElementChild; while (first_el !== null && !html_sizzle.matchesSelector(first_el, selector)) { first_el = first_el.nextElementSibling; } return first_el; }; utils_core.getLastChildElement = function (el) { let selector = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : '*'; let last_el = el.lastElementChild; while (last_el !== null && !html_sizzle.matchesSelector(last_el, selector)) { last_el = last_el.previousElementSibling; } return last_el; }; utils_core.hasClass = function (className, el) { return el instanceof Element && el.classList.contains(className); }; utils_core.toggleClass = function (className, el) { utils_core.hasClass(className, el) ? utils_core.removeClass(className, el) : utils_core.addClass(className, el); }; /** * Add a class to an element. * @method u#addClass * @param {string} className * @param {Element} el */ utils_core.addClass = function (className, el) { el instanceof Element && el.classList.add(className); return el; }; /** * Remove a class from an element. * @method u#removeClass * @param {string} className * @param {Element} el */ utils_core.removeClass = function (className, el) { el instanceof Element && el.classList.remove(className); return el; }; utils_core.removeElement = function (el) { el instanceof Element && el.parentNode && el.parentNode.removeChild(el); return el; }; utils_core.getElementFromTemplateResult = function (tr) { const div = document.createElement('div'); x(tr, div); return div.firstElementChild; }; utils_core.showElement = el => { utils_core.removeClass('collapsed', el); utils_core.removeClass('hidden', el); }; utils_core.hideElement = function (el) { el instanceof Element && el.classList.add('hidden'); return el; }; function ancestor(el, selector) { let parent = el; while (parent !== null && !html_sizzle.matchesSelector(parent, selector)) { parent = parent.parentElement; } return parent; } /** * Return the element's siblings until one matches the selector. * @private * @method u#nextUntil * @param { HTMLElement } el * @param { String } selector */ utils_core.nextUntil = function (el, selector) { const matches = []; let sibling_el = el.nextElementSibling; while (sibling_el !== null && !sibling_el.matches(selector)) { matches.push(sibling_el); sibling_el = sibling_el.nextElementSibling; } return matches; }; /** * Helper method that replace HTML-escaped symbols with equivalent characters * (e.g. transform occurrences of '&' to '&') * @private * @method u#unescapeHTML * @param { String } string - a String containing the HTML-escaped symbols. */ utils_core.unescapeHTML = function (string) { var div = document.createElement('div'); div.innerHTML = string; return div.innerText; }; utils_core.escapeHTML = function (string) { return string.replace(/&/g, '&').replace(//g, '>').replace(/"/g, '"'); }; function isProtocolApproved(protocol) { let safeProtocolsList = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : APPROVED_URL_PROTOCOLS; return !!safeProtocolsList.includes(protocol); } // Will return false if URL is malformed or contains disallowed characters function isUrlValid(urlString) { try { const url = new URL(urlString); return !!url; } catch (error) { return false; } } function getHyperlinkTemplate(url) { const http_url = RegExp('^w{3}.', 'ig').test(url) ? `http://${url}` : url; const uri = getURI(url); if (uri !== null && isUrlValid(http_url) && (isProtocolApproved(uri._parts.protocol) || !uri._parts.protocol)) { return hyperlink(uri, url); } return url; } utils_core.slideInAllElements = function (elements) { let duration = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 300; return Promise.all(Array.from(elements).map(e => utils_core.slideIn(e, duration))); }; utils_core.slideToggleElement = function (el, duration) { if (utils_core.hasClass('collapsed', el) || utils_core.hasClass('hidden', el)) { return utils_core.slideOut(el, duration); } else { return utils_core.slideIn(el, duration); } }; /** * Shows/expands an element by sliding it out of itself * @private * @method u#slideOut * @param { HTMLElement } el - The HTML string * @param { Number } duration - The duration amount in milliseconds */ utils_core.slideOut = function (el) { let duration = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 200; return new Promise((resolve, reject) => { if (!el) { const err = 'An element needs to be passed in to slideOut'; headless_log.warn(err); reject(new Error(err)); return; } const marker = el.getAttribute('data-slider-marker'); if (marker) { el.removeAttribute('data-slider-marker'); window.cancelAnimationFrame(marker); } const end_height = utils_core.calculateElementHeight(el); if (window.converse_disable_effects) { // Effects are disabled (for tests) el.style.height = end_height + 'px'; slideOutWrapup(el); resolve(); return; } if (!utils_core.hasClass('collapsed', el) && !utils_core.hasClass('hidden', el)) { resolve(); return; } const steps = duration / 17; // We assume 17ms per animation which is ~60FPS let height = 0; function draw() { height += end_height / steps; if (height < end_height) { el.style.height = height + 'px'; el.setAttribute('data-slider-marker', window.requestAnimationFrame(draw)); } else { // We recalculate the height to work around an apparent // browser bug where browsers don't know the correct // offsetHeight beforehand. el.removeAttribute('data-slider-marker'); el.style.height = utils_core.calculateElementHeight(el) + 'px'; el.style.overflow = ''; el.style.height = ''; resolve(); } } el.style.height = '0'; el.style.overflow = 'hidden'; el.classList.remove('hidden'); el.classList.remove('collapsed'); el.setAttribute('data-slider-marker', window.requestAnimationFrame(draw)); }); }; utils_core.slideIn = function (el) { let duration = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 200; /* Hides/collapses an element by sliding it into itself. */ return new Promise((resolve, reject) => { if (!el) { const err = 'An element needs to be passed in to slideIn'; headless_log.warn(err); return reject(new Error(err)); } else if (utils_core.hasClass('collapsed', el)) { return resolve(el); } else if (window.converse_disable_effects) { // Effects are disabled (for tests) el.classList.add('collapsed'); el.style.height = ''; return resolve(el); } const marker = el.getAttribute('data-slider-marker'); if (marker) { el.removeAttribute('data-slider-marker'); window.cancelAnimationFrame(marker); } const original_height = el.offsetHeight, steps = duration / 17; // We assume 17ms per animation which is ~60FPS let height = original_height; el.style.overflow = 'hidden'; function draw() { height -= original_height / steps; if (height > 0) { el.style.height = height + 'px'; el.setAttribute('data-slider-marker', window.requestAnimationFrame(draw)); } else { el.removeAttribute('data-slider-marker'); el.classList.add('collapsed'); el.style.height = ''; resolve(el); } } el.setAttribute('data-slider-marker', window.requestAnimationFrame(draw)); }); }; function afterAnimationEnds(el, callback) { el.classList.remove('visible'); if (lodash_es_isFunction(callback)) { callback(); } } utils_core.isInDOM = function (el) { return document.querySelector('body').contains(el); }; utils_core.isVisible = function (el) { if (el === null) { return false; } if (utils_core.hasClass('hidden', el)) { return false; } // XXX: Taken from jQuery's "visible" implementation return el.offsetWidth > 0 || el.offsetHeight > 0 || el.getClientRects().length > 0; }; utils_core.fadeIn = function (el, callback) { if (!el) { headless_log.warn('An element needs to be passed in to fadeIn'); } if (window.converse_disable_effects) { el.classList.remove('hidden'); return afterAnimationEnds(el, callback); } if (utils_core.hasClass('hidden', el)) { el.classList.add('visible'); el.classList.remove('hidden'); el.addEventListener('webkitAnimationEnd', () => afterAnimationEnds(el, callback)); el.addEventListener('animationend', () => afterAnimationEnds(el, callback)); el.addEventListener('oanimationend', () => afterAnimationEnds(el, callback)); } else { afterAnimationEnds(el, callback); } }; /** * Takes an XML field in XMPP XForm (XEP-004: Data Forms) format returns a * [TemplateResult](https://lit.polymer-project.org/api/classes/_lit_html_.templateresult.html). * @method u#xForm2TemplateResult * @param { XMLElement } field - the field to convert * @param { XMLElement } stanza - the containing stanza * @param { Object } options * @returns { TemplateResult } */ utils_core.xForm2TemplateResult = function (field, stanza, options) { if (field.getAttribute('type') === 'list-single' || field.getAttribute('type') === 'list-multi') { const values = utils_core.queryChildren(field, 'value').map(el => el === null || el === void 0 ? void 0 : el.textContent); const options = utils_core.queryChildren(field, 'option').map(option => { var _option$querySelector; const value = (_option$querySelector = option.querySelector('value')) === null || _option$querySelector === void 0 ? void 0 : _option$querySelector.textContent; return { 'value': value, 'label': option.getAttribute('label'), 'selected': values.includes(value), 'required': !!field.querySelector('required') }; }); return form_select({ options, 'id': utils_core.getUniqueId(), 'label': field.getAttribute('label'), 'multiple': field.getAttribute('type') === 'list-multi', 'name': field.getAttribute('var'), 'required': !!field.querySelector('required') }); } else if (field.getAttribute('type') === 'fixed') { var _field$querySelector; const text = (_field$querySelector = field.querySelector('value')) === null || _field$querySelector === void 0 ? void 0 : _field$querySelector.textContent; return form_help({ text }); } else if (field.getAttribute('type') === 'jid-multi') { var _field$querySelector2; return form_textarea({ 'name': field.getAttribute('var'), 'label': field.getAttribute('label') || '', 'value': (_field$querySelector2 = field.querySelector('value')) === null || _field$querySelector2 === void 0 ? void 0 : _field$querySelector2.textContent, 'required': !!field.querySelector('required') }); } else if (field.getAttribute('type') === 'boolean') { var _field$querySelector3; const value = (_field$querySelector3 = field.querySelector('value')) === null || _field$querySelector3 === void 0 ? void 0 : _field$querySelector3.textContent; return form_checkbox({ 'id': utils_core.getUniqueId(), 'name': field.getAttribute('var'), 'label': field.getAttribute('label') || '', 'checked': (value === '1' || value === 'true') && 'checked="1"' || '', 'required': !!field.querySelector('required') }); } else if (field.getAttribute('var') === 'url') { var _field$querySelector4; return form_url({ 'label': field.getAttribute('label') || '', 'value': (_field$querySelector4 = field.querySelector('value')) === null || _field$querySelector4 === void 0 ? void 0 : _field$querySelector4.textContent }); } else if (field.getAttribute('var') === 'username') { var _field$querySelector5; return form_username({ 'domain': ' @' + options.domain, 'name': field.getAttribute('var'), 'type': getInputType(field), 'label': field.getAttribute('label') || '', 'value': (_field$querySelector5 = field.querySelector('value')) === null || _field$querySelector5 === void 0 ? void 0 : _field$querySelector5.textContent, 'required': !!field.querySelector('required') }); } else if (field.getAttribute('var') === 'ocr') { // Captcha const uri = field.querySelector('uri'); const el = html_sizzle('data[cid="' + uri.textContent.replace(/^cid:/, '') + '"]', stanza)[0]; return form_captcha({ 'label': field.getAttribute('label'), 'name': field.getAttribute('var'), 'data': el === null || el === void 0 ? void 0 : el.textContent, 'type': uri.getAttribute('type'), 'required': !!field.querySelector('required') }); } else { var _field$querySelector6; const name = field.getAttribute('var'); return form_input({ 'id': utils_core.getUniqueId(), 'label': field.getAttribute('label') || '', 'name': name, 'fixed_username': options === null || options === void 0 ? void 0 : options.fixed_username, 'autocomplete': getAutoCompleteProperty(name, options), 'placeholder': null, 'required': !!field.querySelector('required'), 'type': getInputType(field), 'value': (_field$querySelector6 = field.querySelector('value')) === null || _field$querySelector6 === void 0 ? void 0 : _field$querySelector6.textContent }); } }; Object.assign(utils_core, { getOOBURLMarkup, ancestor }); /* harmony default export */ const html = (utils_core); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/shared/components/styles/gif.scss var gif = __webpack_require__(4903); ;// CONCATENATED MODULE: ./src/shared/components/styles/gif.scss var gif_options = {}; gif_options.styleTagTransform = (styleTagTransform_default()); gif_options.setAttributes = (setAttributesWithoutAttributes_default()); gif_options.insert = insertBySelector_default().bind(null, "head"); gif_options.domAPI = (styleDomAPI_default()); gif_options.insertStyleElement = (insertStyleElement_default()); var gif_update = injectStylesIntoStyleTag_default()(gif/* default */.Z, gif_options); /* harmony default export */ const styles_gif = (gif/* default */.Z && gif/* default.locals */.Z.locals ? gif/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/shared/components/gif.js class ConverseGIFElement extends CustomElement { static get properties() { /** * @typedef { Object } ConverseGIFComponentProperties * @property { Boolean } autoplay * @property { Boolean } noloop * @property { String } progress_color * @property { String } nick * @property { ('url'|'empty'|'error') } fallback * @property { String } src */ return { 'autoplay': { type: Boolean }, 'noloop': { type: Boolean }, 'progress_color': { type: String }, 'fallback': { type: String }, 'src': { type: String } }; } constructor() { super(); this.autoplay = false; this.noloop = false; this.fallback = 'url'; } initGIF() { const options = { 'autoplay': this.autoplay, 'loop': !this.noloop }; if (this.progress_color) { options['progress_color'] = this.progress_color; } this.supergif = new ConverseGif(this, options); } updated(changed) { if (!this.supergif || changed.has('src')) { this.initGIF(); return; } if (changed.has('autoplay')) { this.supergif.options.autoplay = this.autoplay; } if (changed.has('noloop')) { this.supergif.options.loop = !this.noloop; } if (changed.has('progress_color')) { this.supergif.options.progress_color = this.progress_color; } } render() { var _this$supergif; return (_this$supergif = this.supergif) !== null && _this$supergif !== void 0 && _this$supergif.load_error && ['url', 'empty'].includes(this.fallback) ? this.renderErrorFallback() : $` this.setHover()} @mouseleave=${() => this.unsetHover()} @click=${ev => this.onControlsClicked(ev)}>`; } renderErrorFallback() { if (this.fallback === 'url') { return getHyperlinkTemplate(this.src); } else if (this.fallback === 'empty') { return ''; } } setHover() { if (this.supergif) { this.supergif.hovering = true; this.hover_timeout && clearTimeout(this.hover_timeout); this.hover_timeout = setTimeout(() => this.unsetHover(), 2000); } } unsetHover() { if (this.supergif) this.supergif.hovering = false; } onControlsClicked(ev) { ev.preventDefault(); if (this.supergif.playing) { this.supergif.pause(); } else { // When the user manually clicks play, we turn on looping this.supergif.options.loop = true; this.supergif.play(); } } } core_api.elements.define('converse-gif', ConverseGIFElement); ;// CONCATENATED MODULE: ./src/templates/gif.js /* harmony default export */ const templates_gif = ((url, hide_url) => $`${hide_url ? '' : $`${url}`}`); ;// CONCATENATED MODULE: ./node_modules/lit/async-directive.js //# sourceMappingURL=async-directive.js.map ;// CONCATENATED MODULE: ./src/shared/directives/image.js const { URI: image_URI } = core_converse.env; class ImageDirective extends async_directive_d { render(src, href, onLoad, onClick) { return href ? $`${this.renderImage(src, href, onLoad, onClick)}` : this.renderImage(src, href, onLoad, onClick); } renderImage(src, href, onLoad, onClick) { return $` this.onError(src, href, onLoad, onClick)} @load=${onLoad}/>`; } onError(src, href, onLoad, onClick) { if (isURLWithImageExtension(src)) { href && this.setValue(getHyperlinkTemplate(href)); } else { // Before giving up and falling back to just rendering a hyperlink, // we attach `.png` and try one more time. // This works with some Imgur URLs const uri = new image_URI(src); const filename = uri.filename(); uri.filename(`${filename}.png`); this.setValue(renderImage(uri.toString(), href, onLoad, onClick)); } } } /** * lit directive which attempts to render an element from a URL. * It will fall back to rendering an element if it can't. * * @param { String } src - The value that will be assigned to the `src` attribute of the `` element. * @param { String } href - The value that will be assigned to the `href` attribute of the `` element. * @param { Function } onLoad - A callback function to be called once the image has loaded. * @param { Function } onClick - A callback function to be called once the image has been clicked. */ const renderImage = directive_e(ImageDirective); ;// CONCATENATED MODULE: ./src/templates/image.js /* harmony default export */ const src_templates_image = (o => $`${renderImage(o.src || o.url, o.href, o.onLoad, o.onClick)}`); ;// CONCATENATED MODULE: ./src/shared/directives/styling.js async function transform(t) { try { await t.addTemplates(); } catch (e) { headless_log.error(e); } return t.payload; } class StylingDirective extends directive_i { render(txt, offset, options) { // eslint-disable-line class-methods-use-this const t = new RichText(txt, offset, Object.assign(options, { 'show_images': false, 'embed_videos': false, 'embed_audio': false })); return $`${until_c(transform(t), $`${t}`)}`; } } const renderStylingDirectiveBody = directive_e(StylingDirective); ;// CONCATENATED MODULE: ./src/shared/styling.js /** * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) * @description Utility functions to help with parsing XEP-393 message styling hints * @todo Other parsing helpers can be made more abstract and placed here. */ const bracketing_directives = ['*', '_', '~', '`']; const styling_directives = [...bracketing_directives, '```', '>']; const styling_map = { '*': { 'name': 'strong', 'type': 'span' }, '_': { 'name': 'emphasis', 'type': 'span' }, '~': { 'name': 'strike', 'type': 'span' }, '`': { 'name': 'preformatted', 'type': 'span' }, '```': { 'name': 'preformatted_block', 'type': 'block' }, '>': { 'name': 'quote', 'type': 'block' } }; const dont_escape = ['_', '>', '`', '~']; const styling_templates = { // m is the chatbox model // i is the offset of this directive relative to the start of the original message 'emphasis': (txt, i, options) => $`_${renderStylingDirectiveBody(txt, i, options)}_`, 'preformatted': txt => $`\`${txt}\``, 'preformatted_block': txt => $`
\`\`\`
${txt}
\`\`\`
`, 'quote': (txt, i, options) => $`
${renderStylingDirectiveBody(txt, i, options)}
`, 'strike': (txt, i, options) => $`~${renderStylingDirectiveBody(txt, i, options)}~`, 'strong': (txt, i, options) => $`*${renderStylingDirectiveBody(txt, i, options)}*` }; /** * Checks whether a given character "d" at index "i" of "text" is a valid opening or closing directive. * @param { String } d - The potential directive * @param { String } text - The text in which the directive appears * @param { Number } i - The directive index * @param { Boolean } opening - Check for a valid opening or closing directive */ function isValidDirective(d, text, i, opening) { // Ignore directives that are parts of words // More info on the Regexes used here: https://javascript.info/regexp-unicode#unicode-properties-p if (opening) { const regex = RegExp(dont_escape.includes(d) ? `^(\\p{L}|\\p{N})${d}` : `^(\\p{L}|\\p{N})\\${d}`, 'u'); if (i > 1 && regex.test(text.slice(i - 1))) { return false; } const is_quote = isQuoteDirective(d); if (is_quote && i > 0 && text[i - 1] !== '\n') { // Quote directives must be on newlines return false; } else if (bracketing_directives.includes(d) && text[i + 1] === d) { // Don't consider empty bracketing directives as valid (e.g. **, `` etc.) return false; } } else { const regex = RegExp(dont_escape.includes(d) ? `^${d}(\\p{L}|\\p{N})` : `^\\${d}(\\p{L}|\\p{N})`, 'u'); if (i < text.length - 1 && regex.test(text.slice(i))) { return false; } if (bracketing_directives.includes(d) && text[i - 1] === d) { // Don't consider empty directives as valid (e.g. **, `` etc.) return false; } } return true; } /** * Given a specific index "i" of "text", return the directive it matches or * null otherwise. * @param { String } text - The text in which the directive appears * @param { Number } i - The directive index * @param { Boolean } opening - Whether we're looking for an opening or closing directive */ function getDirective(text, i) { let opening = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : true; let d; if (/(^```\s*\n|^```\s*$)/.test(text.slice(i)) && (i === 0 || text[i - 1] === '\n' || text[i - 1] === '>')) { d = text.slice(i, i + 3); } else if (styling_directives.includes(text.slice(i, i + 1))) { d = text.slice(i, i + 1); if (!isValidDirective(d, text, i, opening)) return null; } else { return null; } return d; } /** * Given a directive "d", which occurs in "text" at index "i", check that it * has a valid closing directive and return the length from start to end of the * directive. * @param { String } d -The directive * @param { Number } i - The directive index * @param { String } text -The text in which the directive appears */ function getDirectiveLength(d, text, i) { if (!d) { return 0; } const begin = i; i += d.length; if (isQuoteDirective(d)) { i += text.slice(i).split(/\n[^>]/).shift().length; return i - begin; } else if (styling_map[d].type === 'span') { const line = text.slice(i).split('\n').shift(); let j = 0; let idx = line.indexOf(d); while (idx !== -1) { if (getDirective(text, i + idx, false) === d) { return idx + 2 * d.length; } idx = line.indexOf(d, j++); } return 0; } else { // block directives const substring = text.slice(i + 1); let j = 0; let idx = substring.indexOf(d); while (idx !== -1) { if (getDirective(text, i + 1 + idx, false) === d) { return idx + 1 + 2 * d.length; } idx = substring.indexOf(d, j++); } return 0; } } function getDirectiveAndLength(text, i) { const d = getDirective(text, i); const length = d ? getDirectiveLength(d, text, i) : 0; return length > 0 ? { d, length } : {}; } const isQuoteDirective = d => ['>', '>'].includes(d); function getDirectiveTemplate(d, text, offset, options) { const template = styling_templates[styling_map[d].name]; if (isQuoteDirective(d)) { const newtext = text.replace(/\n>/g, '\n') // Don't show the directive itself .replace(/\n$/, ''); // Trim line-break at the end return template(newtext, offset, options); } else { return template(text, offset, options); } } function containsDirectives(text) { for (let i = 0; i < styling_directives.length; i++) { if (text.includes(styling_directives[i])) { return true; } } } // EXTERNAL MODULE: ./node_modules/lodash/debounce.js var lodash_debounce = __webpack_require__(3279); var debounce_default = /*#__PURE__*/__webpack_require__.n(lodash_debounce); ;// CONCATENATED MODULE: ./src/shared/chat/templates/new-day.js /* harmony default export */ const new_day = (o => $`

`); ;// CONCATENATED MODULE: ./src/headless/plugins/emoji/regexes.js const ASCII_REGEX = '(\\*\\\\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|\\=#|\\:\\)|\\=\\]|\\=\\)|\\:\\])'; const ASCII_REPLACE_REGEX = new RegExp("]*>.*?<\/object>|]*>.*?<\/span>|<(?:object|embed|svg|img|div|span|p|a)[^>]*>|((\\s|^)" + ASCII_REGEX + "(?=\\s|$|[!,.?]))", "gi"); const CODEPOINTS_REGEX = /(?:\ud83d\udc68\ud83c\udffb\u200d\ud83e\udd1d\u200d\ud83d\udc68\ud83c[\udffc-\udfff]|\ud83d\udc68\ud83c\udffc\u200d\ud83e\udd1d\u200d\ud83d\udc68\ud83c[\udffb\udffd-\udfff]|\ud83d\udc68\ud83c\udffd\u200d\ud83e\udd1d\u200d\ud83d\udc68\ud83c[\udffb\udffc\udffe\udfff]|\ud83d\udc68\ud83c\udffe\u200d\ud83e\udd1d\u200d\ud83d\udc68\ud83c[\udffb-\udffd\udfff]|\ud83d\udc68\ud83c\udfff\u200d\ud83e\udd1d\u200d\ud83d\udc68\ud83c[\udffb-\udffe]|\ud83d\udc69\ud83c\udffb\u200d\ud83e\udd1d\u200d\ud83d\udc68\ud83c[\udffc-\udfff]|\ud83d\udc69\ud83c\udffb\u200d\ud83e\udd1d\u200d\ud83d\udc69\ud83c[\udffc-\udfff]|\ud83d\udc69\ud83c\udffc\u200d\ud83e\udd1d\u200d\ud83d\udc68\ud83c[\udffb\udffd-\udfff]|\ud83d\udc69\ud83c\udffc\u200d\ud83e\udd1d\u200d\ud83d\udc69\ud83c[\udffb\udffd-\udfff]|\ud83d\udc69\ud83c\udffd\u200d\ud83e\udd1d\u200d\ud83d\udc68\ud83c[\udffb\udffc\udffe\udfff]|\ud83d\udc69\ud83c\udffd\u200d\ud83e\udd1d\u200d\ud83d\udc69\ud83c[\udffb\udffc\udffe\udfff]|\ud83d\udc69\ud83c\udffe\u200d\ud83e\udd1d\u200d\ud83d\udc68\ud83c[\udffb-\udffd\udfff]|\ud83d\udc69\ud83c\udffe\u200d\ud83e\udd1d\u200d\ud83d\udc69\ud83c[\udffb-\udffd\udfff]|\ud83d\udc69\ud83c\udfff\u200d\ud83e\udd1d\u200d\ud83d\udc68\ud83c[\udffb-\udffe]|\ud83d\udc69\ud83c\udfff\u200d\ud83e\udd1d\u200d\ud83d\udc69\ud83c[\udffb-\udffe]|\ud83e\uddd1\ud83c\udffb\u200d\ud83e\udd1d\u200d\ud83e\uddd1\ud83c[\udffb-\udfff]|\ud83e\uddd1\ud83c\udffc\u200d\ud83e\udd1d\u200d\ud83e\uddd1\ud83c[\udffb-\udfff]|\ud83e\uddd1\ud83c\udffd\u200d\ud83e\udd1d\u200d\ud83e\uddd1\ud83c[\udffb-\udfff]|\ud83e\uddd1\ud83c\udffe\u200d\ud83e\udd1d\u200d\ud83e\uddd1\ud83c[\udffb-\udfff]|\ud83e\uddd1\ud83c\udfff\u200d\ud83e\udd1d\u200d\ud83e\uddd1\ud83c[\udffb-\udfff]|\ud83e\uddd1\u200d\ud83e\udd1d\u200d\ud83e\uddd1|\ud83d\udc6b\ud83c[\udffb-\udfff]|\ud83d\udc6c\ud83c[\udffb-\udfff]|\ud83d\udc6d\ud83c[\udffb-\udfff]|\ud83d[\udc6b-\udc6d])|(?:\ud83d[\udc68\udc69]|\ud83e\uddd1)(?:\ud83c[\udffb-\udfff])?\u200d(?:\u2695\ufe0f|\u2696\ufe0f|\u2708\ufe0f|\ud83c[\udf3e\udf73\udf93\udfa4\udfa8\udfeb\udfed]|\ud83d[\udcbb\udcbc\udd27\udd2c\ude80\ude92]|\ud83e[\uddaf-\uddb3\uddbc\uddbd])|(?:\ud83c[\udfcb\udfcc]|\ud83d[\udd74\udd75]|\u26f9)((?:\ud83c[\udffb-\udfff]|\ufe0f)\u200d[\u2640\u2642]\ufe0f)|(?:\ud83c[\udfc3\udfc4\udfca]|\ud83d[\udc6e\udc71\udc73\udc77\udc81\udc82\udc86\udc87\ude45-\ude47\ude4b\ude4d\ude4e\udea3\udeb4-\udeb6]|\ud83e[\udd26\udd35\udd37-\udd39\udd3d\udd3e\uddb8\uddb9\uddcd-\uddcf\uddd6-\udddd])(?:\ud83c[\udffb-\udfff])?\u200d[\u2640\u2642]\ufe0f|(?:\ud83d\udc68\u200d\u2764\ufe0f\u200d\ud83d\udc8b\u200d\ud83d\udc68|\ud83d\udc68\u200d\ud83d\udc68\u200d\ud83d\udc66\u200d\ud83d\udc66|\ud83d\udc68\u200d\ud83d\udc68\u200d\ud83d\udc67\u200d\ud83d[\udc66\udc67]|\ud83d\udc68\u200d\ud83d\udc69\u200d\ud83d\udc66\u200d\ud83d\udc66|\ud83d\udc68\u200d\ud83d\udc69\u200d\ud83d\udc67\u200d\ud83d[\udc66\udc67]|\ud83d\udc69\u200d\u2764\ufe0f\u200d\ud83d\udc8b\u200d\ud83d[\udc68\udc69]|\ud83d\udc69\u200d\ud83d\udc69\u200d\ud83d\udc66\u200d\ud83d\udc66|\ud83d\udc69\u200d\ud83d\udc69\u200d\ud83d\udc67\u200d\ud83d[\udc66\udc67]|\ud83d\udc68\u200d\u2764\ufe0f\u200d\ud83d\udc68|\ud83d\udc68\u200d\ud83d\udc66\u200d\ud83d\udc66|\ud83d\udc68\u200d\ud83d\udc67\u200d\ud83d[\udc66\udc67]|\ud83d\udc68\u200d\ud83d\udc68\u200d\ud83d[\udc66\udc67]|\ud83d\udc68\u200d\ud83d\udc69\u200d\ud83d[\udc66\udc67]|\ud83d\udc69\u200d\u2764\ufe0f\u200d\ud83d[\udc68\udc69]|\ud83d\udc69\u200d\ud83d\udc66\u200d\ud83d\udc66|\ud83d\udc69\u200d\ud83d\udc67\u200d\ud83d[\udc66\udc67]|\ud83d\udc69\u200d\ud83d\udc69\u200d\ud83d[\udc66\udc67]|\ud83c\udff3\ufe0f\u200d\u26a7\ufe0f|\ud83c\udff3\ufe0f\u200d\ud83c\udf08|\ud83c\udff4\u200d\u2620\ufe0f|\ud83d\udc15\u200d\ud83e\uddba|\ud83d\udc41\u200d\ud83d\udde8|\ud83d\udc68\u200d\ud83d[\udc66\udc67]|\ud83d\udc69\u200d\ud83d[\udc66\udc67]|\ud83d\udc6f\u200d\u2640\ufe0f|\ud83d\udc6f\u200d\u2642\ufe0f|\ud83e\udd3c\u200d\u2640\ufe0f|\ud83e\udd3c\u200d\u2642\ufe0f|\ud83e\uddde\u200d\u2640\ufe0f|\ud83e\uddde\u200d\u2642\ufe0f|\ud83e\udddf\u200d\u2640\ufe0f|\ud83e\udddf\u200d\u2642\ufe0f)|[#*0-9]\ufe0f?\u20e3|(?:[©®\u2122\u265f]\ufe0f)|(?:\ud83c[\udc04\udd70\udd71\udd7e\udd7f\ude02\ude1a\ude2f\ude37\udf21\udf24-\udf2c\udf36\udf7d\udf96\udf97\udf99-\udf9b\udf9e\udf9f\udfcd\udfce\udfd4-\udfdf\udff3\udff5\udff7]|\ud83d[\udc3f\udc41\udcfd\udd49\udd4a\udd6f\udd70\udd73\udd76-\udd79\udd87\udd8a-\udd8d\udda5\udda8\uddb1\uddb2\uddbc\uddc2-\uddc4\uddd1-\uddd3\udddc-\uddde\udde1\udde3\udde8\uddef\uddf3\uddfa\udecb\udecd-\udecf\udee0-\udee5\udee9\udef0\udef3]|[\u203c\u2049\u2139\u2194-\u2199\u21a9\u21aa\u231a\u231b\u2328\u23cf\u23ed-\u23ef\u23f1\u23f2\u23f8-\u23fa\u24c2\u25aa\u25ab\u25b6\u25c0\u25fb-\u25fe\u2600-\u2604\u260e\u2611\u2614\u2615\u2618\u2620\u2622\u2623\u2626\u262a\u262e\u262f\u2638-\u263a\u2640\u2642\u2648-\u2653\u2660\u2663\u2665\u2666\u2668\u267b\u267f\u2692-\u2697\u2699\u269b\u269c\u26a0\u26a1\u26a7\u26aa\u26ab\u26b0\u26b1\u26bd\u26be\u26c4\u26c5\u26c8\u26cf\u26d1\u26d3\u26d4\u26e9\u26ea\u26f0-\u26f5\u26f8\u26fa\u26fd\u2702\u2708\u2709\u270f\u2712\u2714\u2716\u271d\u2721\u2733\u2734\u2744\u2747\u2757\u2763\u2764\u27a1\u2934\u2935\u2b05-\u2b07\u2b1b\u2b1c\u2b50\u2b55\u3030\u303d\u3297\u3299])(?:\ufe0f|(?!\ufe0e))|(?:(?:\ud83c[\udfcb\udfcc]|\ud83d[\udd74\udd75\udd90]|[\u261d\u26f7\u26f9\u270c\u270d])(?:\ufe0f|(?!\ufe0e))|(?:\ud83c[\udf85\udfc2-\udfc4\udfc7\udfca]|\ud83d[\udc42\udc43\udc46-\udc50\udc66-\udc69\udc6e\udc70-\udc78\udc7c\udc81-\udc83\udc85-\udc87\udcaa\udd7a\udd95\udd96\ude45-\ude47\ude4b-\ude4f\udea3\udeb4-\udeb6\udec0\udecc]|\ud83e[\udd0f\udd18-\udd1c\udd1e\udd1f\udd26\udd30-\udd39\udd3d\udd3e\uddb5\uddb6\uddb8\uddb9\uddbb\uddcd-\uddcf\uddd1-\udddd]|[\u270a\u270b]))(?:\ud83c[\udffb-\udfff])?|(?:\ud83c\udff4\udb40\udc67\udb40\udc62\udb40\udc65\udb40\udc6e\udb40\udc67\udb40\udc7f|\ud83c\udff4\udb40\udc67\udb40\udc62\udb40\udc73\udb40\udc63\udb40\udc74\udb40\udc7f|\ud83c\udff4\udb40\udc67\udb40\udc62\udb40\udc77\udb40\udc6c\udb40\udc73\udb40\udc7f|\ud83c\udde6\ud83c[\udde8-\uddec\uddee\uddf1\uddf2\uddf4\uddf6-\uddfa\uddfc\uddfd\uddff]|\ud83c\udde7\ud83c[\udde6\udde7\udde9-\uddef\uddf1-\uddf4\uddf6-\uddf9\uddfb\uddfc\uddfe\uddff]|\ud83c\udde8\ud83c[\udde6\udde8\udde9\uddeb-\uddee\uddf0-\uddf5\uddf7\uddfa-\uddff]|\ud83c\udde9\ud83c[\uddea\uddec\uddef\uddf0\uddf2\uddf4\uddff]|\ud83c\uddea\ud83c[\udde6\udde8\uddea\uddec\udded\uddf7-\uddfa]|\ud83c\uddeb\ud83c[\uddee-\uddf0\uddf2\uddf4\uddf7]|\ud83c\uddec\ud83c[\udde6\udde7\udde9-\uddee\uddf1-\uddf3\uddf5-\uddfa\uddfc\uddfe]|\ud83c\udded\ud83c[\uddf0\uddf2\uddf3\uddf7\uddf9\uddfa]|\ud83c\uddee\ud83c[\udde8-\uddea\uddf1-\uddf4\uddf6-\uddf9]|\ud83c\uddef\ud83c[\uddea\uddf2\uddf4\uddf5]|\ud83c\uddf0\ud83c[\uddea\uddec-\uddee\uddf2\uddf3\uddf5\uddf7\uddfc\uddfe\uddff]|\ud83c\uddf1\ud83c[\udde6-\udde8\uddee\uddf0\uddf7-\uddfb\uddfe]|\ud83c\uddf2\ud83c[\udde6\udde8-\udded\uddf0-\uddff]|\ud83c\uddf3\ud83c[\udde6\udde8\uddea-\uddec\uddee\uddf1\uddf4\uddf5\uddf7\uddfa\uddff]|\ud83c\uddf4\ud83c\uddf2|\ud83c\uddf5\ud83c[\udde6\uddea-\udded\uddf0-\uddf3\uddf7-\uddf9\uddfc\uddfe]|\ud83c\uddf6\ud83c\udde6|\ud83c\uddf7\ud83c[\uddea\uddf4\uddf8\uddfa\uddfc]|\ud83c\uddf8\ud83c[\udde6-\uddea\uddec-\uddf4\uddf7-\uddf9\uddfb\uddfd-\uddff]|\ud83c\uddf9\ud83c[\udde6\udde8\udde9\uddeb-\udded\uddef-\uddf4\uddf7\uddf9\uddfb\uddfc\uddff]|\ud83c\uddfa\ud83c[\udde6\uddec\uddf2\uddf3\uddf8\uddfe\uddff]|\ud83c\uddfb\ud83c[\udde6\udde8\uddea\uddec\uddee\uddf3\uddfa]|\ud83c\uddfc\ud83c[\uddeb\uddf8]|\ud83c\uddfd\ud83c\uddf0|\ud83c\uddfe\ud83c[\uddea\uddf9]|\ud83c\uddff\ud83c[\udde6\uddf2\uddfc]|\ud83c[\udccf\udd8e\udd91-\udd9a\udde6-\uddff\ude01\ude32-\ude36\ude38-\ude3a\ude50\ude51\udf00-\udf20\udf2d-\udf35\udf37-\udf7c\udf7e-\udf84\udf86-\udf93\udfa0-\udfc1\udfc5\udfc6\udfc8\udfc9\udfcf-\udfd3\udfe0-\udff0\udff4\udff8-\udfff]|\ud83d[\udc00-\udc3e\udc40\udc44\udc45\udc51-\udc65\udc6a\udc6f\udc79-\udc7b\udc7d-\udc80\udc84\udc88-\udca9\udcab-\udcfc\udcff-\udd3d\udd4b-\udd4e\udd50-\udd67\udda4\uddfb-\ude44\ude48-\ude4a\ude80-\udea2\udea4-\udeb3\udeb7-\udebf\udec1-\udec5\uded0-\uded2\uded5\udeeb\udeec\udef4-\udefa\udfe0-\udfeb]|\ud83e[\udd0d\udd0e\udd10-\udd17\udd1d\udd20-\udd25\udd27-\udd2f\udd3a\udd3c\udd3f-\udd45\udd47-\udd71\udd73-\udd76\udd7a-\udda2\udda5-\uddaa\uddae-\uddb4\uddb7\uddba\uddbc-\uddca\uddd0\uddde-\uddff\ude70-\ude73\ude78-\ude7a\ude80-\ude82\ude90-\ude95]|[\u23e9-\u23ec\u23f0\u23f3\u267e\u26ce\u2705\u2728\u274c\u274e\u2753-\u2755\u2795-\u2797\u27b0\u27bf\ue50a])|\ufe0f/g; ;// CONCATENATED MODULE: ./src/headless/plugins/emoji/utils.js const { u: emoji_utils_u } = core_converse.env; // Closured cache const emojis_by_attribute = {}; const ASCII_LIST = { '*\\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', ':)': '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' }; function toCodePoint(unicode_surrogates) { const r = []; let p = 0; let i = 0; while (i < unicode_surrogates.length) { const c = unicode_surrogates.charCodeAt(i++); if (p) { r.push((0x10000 + (p - 0xD800 << 10) + (c - 0xDC00)).toString(16)); p = 0; } else if (0xD800 <= c && c <= 0xDBFF) { p = c; } else { r.push(c.toString(16)); } } return r.join('-'); } function fromCodePoint(codepoint) { let code = typeof codepoint === 'string' ? parseInt(codepoint, 16) : codepoint; if (code < 0x10000) { return String.fromCharCode(code); } code -= 0x10000; return String.fromCharCode(0xD800 + (code >> 10), 0xDC00 + (code & 0x3FF)); } function convert(unicode) { // Converts unicode code points and code pairs to their respective characters if (unicode.indexOf("-") > -1) { const parts = [], s = unicode.split('-'); for (let i = 0; i < s.length; i++) { let part = parseInt(s[i], 16); if (part >= 0x10000 && part <= 0x10FFFF) { const hi = Math.floor((part - 0x10000) / 0x400) + 0xD800; const lo = (part - 0x10000) % 0x400 + 0xDC00; part = String.fromCharCode(hi) + String.fromCharCode(lo); } else { part = String.fromCharCode(part); } parts.push(part); } return parts.join(''); } return fromCodePoint(unicode); } function convertASCII2Emoji(str) { // Replace ASCII smileys return str.replace(ASCII_REPLACE_REGEX, (entire, _, m2, m3) => { if (typeof m3 === 'undefined' || m3 === '' || !(emoji_utils_u.unescapeHTML(m3) in ASCII_LIST)) { // if the ascii doesnt exist just return the entire match return entire; } m3 = emoji_utils_u.unescapeHTML(m3); const unicode = ASCII_LIST[m3].toUpperCase(); return m2 + convert(unicode); }); } function getShortnameReferences(text) { if (!core_converse.emojis.initialized) { throw new Error('getShortnameReferences called before emojis are initialized. ' + 'To avoid this problem, first await the converse.emojis.initilaized_promise.'); } const references = [...text.matchAll(core_converse.emojis.shortnames_regex)].filter(ref => ref[0].length > 0); return references.map(ref => { const cp = core_converse.emojis.by_sn[ref[0]].cp; return { cp, 'begin': ref.index, 'end': ref.index + ref[0].length, 'shortname': ref[0], 'emoji': cp ? convert(cp) : null }; }); } function parseStringForEmojis(str, callback) { const UFE0Fg = /\uFE0F/g; const U200D = String.fromCharCode(0x200D); return String(str).replace(CODEPOINTS_REGEX, (emoji, _, offset) => { const icon_id = toCodePoint(emoji.indexOf(U200D) < 0 ? emoji.replace(UFE0Fg, '') : emoji); if (icon_id) callback(icon_id, emoji, offset); }); } function getCodePointReferences(text) { const references = []; parseStringForEmojis(text, (icon_id, emoji, offset) => { var _getEmojisByAtrribute; references.push({ 'begin': offset, 'cp': icon_id, 'emoji': emoji, 'end': offset + emoji.length, 'shortname': ((_getEmojisByAtrribute = getEmojisByAtrribute('cp')[icon_id]) === null || _getEmojisByAtrribute === void 0 ? void 0 : _getEmojisByAtrribute.sn) || '' }); }); return references; } function addEmojisMarkup(text) { let list = [text]; [...getShortnameReferences(text), ...getCodePointReferences(text)].sort((a, b) => b.begin - a.begin).forEach(ref => { const text = list.shift(); const emoji = ref.emoji || ref.shortname; list = [text.slice(0, ref.begin) + emoji + text.slice(ref.end), ...list]; }); return list; } /** * Replaces all shortnames in the passed in string with their * unicode (emoji) representation. * @namespace u * @method u.shortnamesToUnicode * @param { String } str - String containing the shortname(s) * @returns { String } */ function shortnamesToUnicode(str) { return addEmojisMarkup(convertASCII2Emoji(str)).pop(); } /** * Determines whether the passed in string is just a single emoji shortname; * @namespace u * @method u.isOnlyEmojis * @param { String } shortname - A string which migh be just an emoji shortname * @returns { Boolean } */ function isOnlyEmojis(text) { const words = text.trim().split(/\s+/); if (words.length === 0 || words.length > 3) { return false; } const emojis = words.filter(text => { const refs = getCodePointReferences(emoji_utils_u.shortnamesToUnicode(text)); return refs.length === 1 && (text === refs[0]['shortname'] || text === refs[0]['emoji']); }); return emojis.length === words.length; } /** * @namespace u * @method u.getEmojisByAtrribute * @param { 'category'|'cp'|'sn' } attr * The attribute according to which the returned map should be keyed. * @returns { Object } * Map of emojis with the passed in `attr` used as key and a list of emojis as values. */ function getEmojisByAtrribute(attr) { if (emojis_by_attribute[attr]) { return emojis_by_attribute[attr]; } if (attr === 'category') { return core_converse.emojis.json; } const all_variants = core_converse.emojis.list.map(e => e[attr]).filter((c, i, arr) => arr.indexOf(c) == i); emojis_by_attribute[attr] = {}; all_variants.forEach(v => emojis_by_attribute[attr][v] = core_converse.emojis.list.find(i => i[attr] === v)); return emojis_by_attribute[attr]; } Object.assign(emoji_utils_u, { getEmojisByAtrribute, isOnlyEmojis, shortnamesToUnicode }); ;// CONCATENATED MODULE: ./src/shared/chat/utils.js const { dayjs: utils_dayjs, u: shared_chat_utils_u } = core_converse.env; function onScrolledDown(model) { if (!model.isHidden()) { if (core_api.settings.get('allow_url_history_change')) { // Clear location hash if set to one of the messages in our history const hash = window.location.hash; hash && model.messages.get(hash.slice(1)) && shared_converse.router.history.navigate(); } } } /** * 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. * * Don't call this method directly, instead, call `markScrolled`, * which debounces this method. */ function _markScrolled(ev) { const el = ev.target; if (el.nodeName.toLowerCase() !== 'converse-chat-content') { return; } let scrolled = true; const is_at_bottom = Math.floor(el.scrollTop) === 0; const is_at_top = Math.ceil(el.clientHeight - el.scrollTop) >= el.scrollHeight - Math.ceil(el.scrollHeight / 20); if (is_at_bottom) { scrolled = false; onScrolledDown(el.model); } else if (is_at_top) { /** * Triggered once the chat's message area has been scrolled to the top * @event _converse#chatBoxScrolledUp * @property { _converse.ChatBoxView | _converse.ChatRoomView } view * @example _converse.api.listen.on('chatBoxScrolledUp', obj => { ... }); */ core_api.trigger('chatBoxScrolledUp', el); } if (el.model.get('scolled') !== scrolled) { el.model.ui.set({ scrolled }); } } const markScrolled = debounce_default()(ev => _markScrolled(ev), 50); /** * Given a message object, returns a TemplateResult indicating a new day if * the passed in message is more than a day later than its predecessor. * @param { _converse.Message } */ function getDayIndicator(message) { var _message$collection; const messages = (_message$collection = message.collection) === null || _message$collection === void 0 ? void 0 : _message$collection.models; if (!messages) { return; } const idx = messages.indexOf(message); const prev_message = messages[idx - 1]; if (!prev_message || utils_dayjs(message.get('time')).isAfter(utils_dayjs(prev_message.get('time')), 'day')) { const day_date = utils_dayjs(message.get('time')).startOf('day'); return new_day({ 'type': 'date', 'time': day_date.toISOString(), 'datestring': day_date.format("dddd MMM Do YYYY") }); } } function getHats(message) { if (message.get('type') === 'groupchat') { var _message$occupant; const allowed_hats = core_api.settings.get('muc_hats').filter(hat => hat).map(hat => hat.toLowerCase()); let vcard_roles = []; if (allowed_hats.includes('vcard_roles')) { vcard_roles = message.vcard ? message.vcard.get('role') : null; vcard_roles = vcard_roles ? vcard_roles.split(',').filter(hat => hat).map(hat => ({ title: hat })) : []; } const muc_role = message.occupant ? [message.occupant.get('role')] : []; const muc_affiliation = message.occupant ? [message.occupant.get('affiliation')] : []; const affiliation_role_hats = [...muc_role, ...muc_affiliation].filter(hat => hat).filter(hat => allowed_hats.includes(hat.toLowerCase())).map(hat => ({ title: hat })); const hats = allowed_hats.includes('xep317') ? ((_message$occupant = message.occupant) === null || _message$occupant === void 0 ? void 0 : _message$occupant.get('hats')) || [] : []; return [...hats, ...vcard_roles, ...affiliation_role_hats]; } return []; } function unique(arr) { return [...new Set(arr)]; } function getTonedEmojis() { if (!core_converse.emojis.toned) { core_converse.emojis.toned = unique(Object.values(core_converse.emojis.json.people).filter(person => person.sn.includes('_tone')).map(person => person.sn.replace(/_tone[1-5]/, ''))); } return core_converse.emojis.toned; } function getEmojiMarkup(data) { let options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : { unicode_only: false, add_title_wrapper: false }; const emoji = data.emoji; const shortname = data.shortname; if (emoji) { if (options.unicode_only) { return emoji; } else if (core_api.settings.get('use_system_emojis')) { if (options.add_title_wrapper) { return shortname ? $`${emoji}` : emoji; } else { return emoji; } } else { const path = core_api.settings.get('emoji_image_path'); return $`${emoji}`; } } else if (options.unicode_only) { return shortname; } else { return $`${shortname}`; } } function utils_addEmojisMarkup(text, options) { let list = [text]; [...getShortnameReferences(text), ...getCodePointReferences(text)].sort((a, b) => b.begin - a.begin).forEach(ref => { const text = list.shift(); const emoji = getEmojiMarkup(ref, options); if (typeof emoji === 'string') { list = [text.slice(0, ref.begin) + emoji + text.slice(ref.end), ...list]; } else { list = [text.slice(0, ref.begin), emoji, text.slice(ref.end), ...list]; } }); return list; } /** * Returns an emoji represented by the passed in shortname. * Scans the passed in text for shortnames and replaces them with * emoji unicode glyphs or alternatively if it's a custom emoji * without unicode representation then a lit TemplateResult * which represents image tag markup is returned. * * The shortname needs to be defined in `emojis.json` * and needs to have either a `cp` attribute for the codepoint, or * an `url` attribute which points to the source for the image. * * @namespace u * @method u.shortnamesToEmojis * @param { String } str - String containg the shortname(s) * @param { Object } options * @param { Boolean } options.unicode_only - Whether emojis are rendered as * unicode codepoints. If so, the returned result will be an array * with containing one string, because the emojis themselves will * also be strings. If set to false, emojis will be represented by * lit TemplateResult objects. * @param { Boolean } options.add_title_wrapper - Whether unicode * codepoints should be wrapped with a `` element with a * title, so that the shortname is shown upon hovering with the * mouse. * @returns {Array} An array of at least one string, or otherwise * strings and lit TemplateResult objects. */ function shortnamesToEmojis(str) { let options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : { unicode_only: false, add_title_wrapper: false }; str = convertASCII2Emoji(str); return utils_addEmojisMarkup(str, options); } Object.assign(shared_chat_utils_u, { shortnamesToEmojis }); ;// CONCATENATED MODULE: ./src/shared/rich-text.js const rich_text_isString = s => typeof s === 'string'; // We don't render more than two line-breaks, replace extra line-breaks with // the zero-width whitespace character const collapseLineBreaks = text => text.replace(/\n\n+/g, m => `\n${'\u200B'.repeat(m.length - 2)}\n`); const tpl_mention_with_nick = o => $`${o.mention}`; const tpl_mention = o => $`${o.mention}`; /** * @class RichText * A String subclass that is used to render rich text (i.e. text that contains * hyperlinks, images, mentions, styling etc.). * * The "rich" parts of the text is represented by lit TemplateResult * objects which are added via the {@link RichText.addTemplateResult} * method and saved as metadata. * * By default Converse adds TemplateResults to support emojis, hyperlinks, * images, map URIs and mentions. * * 3rd party plugins can listen for the `beforeMessageBodyTransformed` * and/or `afterMessageBodyTransformed` events and then call * `addTemplateResult` on the RichText instance in order to add their own * rich features. */ class RichText extends String { /** * Create a new {@link RichText} instance. * @param { String } text - The text to be annotated * @param { Integer } offset - The offset of this particular piece of text * from the start of the original message text. This is necessary because * RichText instances can be nested when templates call directives * which create new RichText instances (as happens with XEP-393 styling directives). * @param { Object } options * @param { String } options.nick - The current user's nickname (only relevant if the message is in a XEP-0045 MUC) * @param { Boolean } options.render_styling - Whether XEP-0393 message styling should be applied to the message * @param { Boolean } [options.embed_audio] - Whether audio URLs should be rendered as
${el.shouldShowAvatar() ? $` ` : ''}
${i18n_uploading} ${filename}, ${size}
`; }); ;// CONCATENATED MODULE: ./src/shared/components/message-versions.js const { dayjs: message_versions_dayjs } = core_converse.env; class MessageVersions extends CustomElement { static get properties() { return { 'model': { type: Object } }; } render() { const older_versions = this.model.get('older_versions'); return $`

Older versions

${Object.keys(older_versions).map(k => $`

: ${older_versions[k]}

`)}

Current version

${this.model.getMessageText()}

`; } } core_api.elements.define('converse-message-versions', MessageVersions); ;// CONCATENATED MODULE: ./src/modals/templates/message-versions.js /* harmony default export */ const message_versions = (model => $` `); ;// CONCATENATED MODULE: ./src/modals/message-versions.js /* harmony default export */ const modals_message_versions = (base.extend({ id: "message-versions-modal", toHTML() { return message_versions(this.model); } })); ;// CONCATENATED MODULE: ./src/plugins/muc-views/modals/templates/occupant.js /* harmony default export */ const templates_occupant = (o => { var _o$vcard, _o$vcard2; return $` `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/modals/occupant.js const OccupantModal = base.extend({ id: "muc-occupant", initialize() { base.prototype.initialize.apply(this, arguments); if (this.model) { this.listenTo(this.model, 'change', this.render); } /** * Triggered once the OccupantModal has been initialized * @event _converse#occupantModalInitialized * @type { Object } * @example _converse.api.listen.on('occupantModalInitialized', data); */ core_api.trigger('occupantModalInitialized', { 'model': this.model, 'message': this.message }); }, getVcard() { const model = this.model ?? this.message; if (model.vcard) { return model.vcard; } const jid = (model === null || model === void 0 ? void 0 : model.get('jid')) || (model === null || model === void 0 ? void 0 : model.get('from')); return jid ? shared_converse.vcards.get(jid) : null; }, toHTML() { var _this$model, _this$model2, _this$model3, _this$model3$get; const model = this.model ?? this.message; const jid = model === null || model === void 0 ? void 0 : model.get('jid'); const vcard = this.getVcard(); const display_name = model === null || model === void 0 ? void 0 : model.getDisplayName(); const nick = model.get('nick'); const occupant_id = model.get('occupant_id'); const role = (_this$model = this.model) === null || _this$model === void 0 ? void 0 : _this$model.get('role'); const affiliation = (_this$model2 = this.model) === null || _this$model2 === void 0 ? void 0 : _this$model2.get('affiliation'); const hats = (_this$model3 = this.model) !== null && _this$model3 !== void 0 && (_this$model3$get = _this$model3.get('hats')) !== null && _this$model3$get !== void 0 && _this$model3$get.length ? this.model.get('hats') : null; return templates_occupant({ jid, vcard, display_name, nick, occupant_id, role, affiliation, hats }); } }); shared_converse.OccupantModal = OccupantModal; /* harmony default export */ const modals_occupant = (OccupantModal); ;// CONCATENATED MODULE: ./src/modals/templates/user-details.js const remove_button = o => { const i18n_remove_contact = __('Remove as contact'); return $` `; }; /* harmony default export */ const user_details = (o => { const i18n_address = __('XMPP Address'); const i18n_email = __('Email'); const i18n_full_name = __('Full Name'); const i18n_nickname = __('Nickname'); const i18n_profile = __('The User\'s Profile Image'); const i18n_refresh = __('Refresh'); const i18n_role = __('Role'); const i18n_url = __('URL'); const avatar_data = { 'alt_text': i18n_profile, 'extra_classes': 'mb-3', 'height': '120', 'width': '120' }; return $` `; }); ;// CONCATENATED MODULE: ./src/modals/user-details.js const user_details_u = core_converse.env.utils; function removeContact(contact) { contact.removeFromRoster(() => contact.destroy(), e => { e && headless_log.error(e); core_api.alert('error', __('Error'), [__('Sorry, there was an error while trying to remove %1$s as a contact.', contact.getDisplayName())]); }); } const UserDetailsModal = base.extend({ id: 'user-details-modal', persistent: true, events: { 'click button.refresh-contact': 'refreshContact' }, initialize() { base.prototype.initialize.apply(this, arguments); this.model.rosterContactAdded.then(() => this.registerContactEventHandlers()); this.listenTo(this.model, 'change', this.render); this.registerContactEventHandlers(); /** * Triggered once the UserDetailsModal has been initialized * @event _converse#userDetailsModalInitialized * @type { _converse.ChatBox } * @example _converse.api.listen.on('userDetailsModalInitialized', (chatbox) => { ... }); */ core_api.trigger('userDetailsModalInitialized', this.model); }, toHTML() { var _this$model; const vcard = (_this$model = this.model) === null || _this$model === void 0 ? void 0 : _this$model.vcard; const vcard_json = vcard ? vcard.toJSON() : {}; return user_details(Object.assign(this.model.toJSON(), vcard_json, { '_converse': shared_converse, 'allow_contact_removal': core_api.settings.get('allow_contact_removal'), 'display_name': this.model.getDisplayName(), 'is_roster_contact': this.model.contact !== undefined, 'removeContact': ev => this.removeContact(ev), 'view': this, 'utils': user_details_u })); }, registerContactEventHandlers() { if (this.model.contact !== undefined) { this.listenTo(this.model.contact, 'change', this.render); this.listenTo(this.model.contact.vcard, 'change', this.render); this.model.contact.on('destroy', () => { delete this.model.contact; this.render(); }); } }, async refreshContact(ev) { if (ev && ev.preventDefault) { ev.preventDefault(); } const refresh_icon = this.el.querySelector('.fa-refresh'); user_details_u.addClass('fa-spin', refresh_icon); try { await core_api.vcard.update(this.model.contact.vcard, true); } catch (e) { headless_log.fatal(e); this.alert(__('Sorry, something went wrong while trying to refresh'), 'danger'); } user_details_u.removeClass('fa-spin', refresh_icon); }, removeContact(ev) { var _ev$preventDefault; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); if (!core_api.settings.get('allow_contact_removal')) { return; } const result = confirm(__("Are you sure you want to remove this contact?")); if (result === true) { // XXX: The `dismissHandler` in bootstrap.native tries to // reference the remove button after it's been cleared from // the DOM, so we delay removing the contact to give it time. setTimeout(() => removeContact(this.model.contact), 1); this.modal.hide(); } } }); shared_converse.UserDetailsModal = UserDetailsModal; /* harmony default export */ const modals_user_details = (UserDetailsModal); ;// CONCATENATED MODULE: ./src/shared/chat/templates/info-message.js const { dayjs: info_message_dayjs } = core_converse.env; /* harmony default export */ const info_message = (el => { const isodate = info_message_dayjs(el.model.get('time')).toISOString(); const i18n_retry = __('Retry'); return $`
${el.model.get('reason') ? $`${el.model.get('reason')}` : ``} ${el.model.get('error_text') ? $`${el.model.get('error_text')}` : ``} ${el.model.get('retry_event_id') ? $`${i18n_retry}` : ''}
`; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/mep-message.js const { dayjs: mep_message_dayjs } = core_converse.env; /* harmony default export */ const mep_message = (el => { const isodate = mep_message_dayjs(el.model.get('time')).toISOString(); return $`
${el.isRetracted() ? el.renderRetraction() : $` ${el.model.get('reason') ? $`` : ``} `}
`; }); ;// CONCATENATED MODULE: ./src/shared/components/image.js class Image extends CustomElement { static get properties() { return { 'src': { type: String }, 'onImgLoad': { type: Function }, // If specified, image is wrapped in a hyperlink that points to this URL. 'href': { type: String } }; } render() { if (isGIFURL(this.src) && shouldRenderMediaFromURL(this.src, 'image')) { return templates_gif(filterQueryParamsFromURL(this.src), true); } else { return src_templates_image({ 'src': filterQueryParamsFromURL(this.src), 'href': this.href, 'onClick': this.onImgClick, 'onLoad': this.onImgLoad }); } } } core_api.elements.define('converse-image', Image); ;// CONCATENATED MODULE: ./src/shared/chat/templates/unfurl.js function isValidURL(url) { // We don't consider relative URLs as valid return !!getURI(url).host(); } function isValidImage(image) { return image && isDomainAllowed(image, 'allowed_image_domains') && isValidURL(image); } const tpl_url_wrapper = (o, wrapped_template) => o.url && isValidURL(o.url) && !isGIFURL(o.image) ? $`${wrapped_template(o)}` : wrapped_template(o); const tpl_image = o => $``; /* harmony default export */ const unfurl = (o => { const show_image = isValidImage(o.image); const has_body_info = o.title || o.description || o.url; if (show_image || has_body_info) { return $`
${show_image ? tpl_url_wrapper(o, tpl_image) : ''} ${has_body_info ? $`
${o.title ? tpl_url_wrapper(o, o => $`
${o.title}
`) : ''} ${o.description ? $`

` : ''} ${o.url ? $`

${getURI(o.url).domain()}

` : ''}
` : ''}
`; } else { return ''; } }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/shared/chat/styles/unfurl.scss var styles_unfurl = __webpack_require__(5046); ;// CONCATENATED MODULE: ./src/shared/chat/styles/unfurl.scss var unfurl_options = {}; unfurl_options.styleTagTransform = (styleTagTransform_default()); unfurl_options.setAttributes = (setAttributesWithoutAttributes_default()); unfurl_options.insert = insertBySelector_default().bind(null, "head"); unfurl_options.domAPI = (styleDomAPI_default()); unfurl_options.insertStyleElement = (insertStyleElement_default()); var unfurl_update = injectStylesIntoStyleTag_default()(styles_unfurl/* default */.Z, unfurl_options); /* harmony default export */ const chat_styles_unfurl = (styles_unfurl/* default */.Z && styles_unfurl/* default.locals */.Z.locals ? styles_unfurl/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/shared/chat/unfurl.js class MessageUnfurl extends CustomElement { static get properties() { return { description: { type: String }, image: { type: String }, jid: { type: String }, title: { type: String }, url: { type: String } }; } initialize() { const settings = getAppSettings(); this.listenTo(settings, 'change:allowed_image_domains', () => this.requestUpdate()); this.listenTo(settings, 'change:render_media', () => this.requestUpdate()); } render() { return unfurl(Object.assign({ 'onload': () => this.onImageLoad() }, { description: this.description || '', image: this.image || '', title: this.title || '', url: this.url || '' })); } onImageLoad() { this.dispatchEvent(new CustomEvent('imageLoaded', { detail: this, 'bubbles': true })); } } core_api.elements.define('converse-message-unfurl', MessageUnfurl); ;// CONCATENATED MODULE: ./src/shared/chat/templates/message.js /* harmony default export */ const templates_message = ((el, o) => { var _el$model$vcard, _el$model$vcard2, _el$model$get; const i18n_new_messages = __('New messages'); const is_followup = el.model.isFollowup(); return $` ${o.is_first_unread ? $`

${i18n_new_messages}
` : ''}
${o.should_show_avatar && !is_followup ? $` ` : ''}
${!o.is_me_message && !is_followup ? $` ${o.username} ${o.hats.map(h => $`${h.title}`)} ${o.is_encrypted ? $`` : ''} ` : ''}
${o.is_me_message ? $`   ${o.is_me_message ? '**' : ''}${o.username} ` : ''} ${o.is_retracted ? el.renderRetraction() : el.renderMessageText()}
${(_el$model$get = el.model.get('ogp_metadata')) === null || _el$model$get === void 0 ? void 0 : _el$model$get.map(m => { var _el$chatbox; if (el.model.get('hide_url_previews') === true) { return ''; } if (!shouldRenderMediaFromURL(m['og:image'], 'image')) { return ''; } return $``; })}
`; }); ;// CONCATENATED MODULE: ./src/shared/chat/templates/message-text.js const tpl_edited_icon = el => { const i18n_edited = __('This message has been edited'); return $``; }; /* harmony default export */ const message_text = (el => { const i18n_show = __('Show more'); const is_groupchat_message = el.model.get('type') === 'groupchat'; const i18n_show_less = __('Show less'); const tpl_spoiler_hint = $` `; const spoiler_classes = el.model.get('is_spoiler') ? `spoiler ${el.model.get('is_spoiler_visible') ? '' : 'hidden'}` : ''; const text = el.model.getMessageText(); const show_oob = el.model.get('oob_url') && text !== el.model.get('oob_url'); return $` ${el.model.get('is_spoiler') ? tpl_spoiler_hint : ''} ${el.model.get('subject') ? $`
${el.model.get('subject')}
` : ''} ${el.model.get('received') && !el.model.isMeCommand() && !is_groupchat_message ? $`` : ''} ${el.model.get('edited') ? tpl_edited_icon(el) : ''} ${show_oob ? $`
${getOOBURLMarkup(el.model.get('oob_url'))}
` : ''}
${el.model.get('error_text') || el.model.get('error')}
`; }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/shared/chat/styles/retraction.scss var retraction = __webpack_require__(9523); ;// CONCATENATED MODULE: ./src/shared/chat/styles/retraction.scss var retraction_options = {}; retraction_options.styleTagTransform = (styleTagTransform_default()); retraction_options.setAttributes = (setAttributesWithoutAttributes_default()); retraction_options.insert = insertBySelector_default().bind(null, "head"); retraction_options.domAPI = (styleDomAPI_default()); retraction_options.insertStyleElement = (insertStyleElement_default()); var retraction_update = injectStylesIntoStyleTag_default()(retraction/* default */.Z, retraction_options); /* harmony default export */ const styles_retraction = (retraction/* default */.Z && retraction/* default.locals */.Z.locals ? retraction/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/shared/chat/templates/retraction.js /* harmony default export */ const templates_retraction = (el => { const retraction_text = el.isRetracted() ? el.getRetractionText() : null; return $`
${retraction_text}
${el.model.get('moderation_reason') ? $`${el.model.get('moderation_reason')}` : ''}`; }); ;// CONCATENATED MODULE: ./src/templates/spinner.js /* harmony default export */ const spinner = (function () { var _o$classes; let o = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; if ((_o$classes = o.classes) !== null && _o$classes !== void 0 && _o$classes.includes('hor_centered')) { return $`
`; } else { return $``; } }); ;// CONCATENATED MODULE: ./src/shared/chat/message.js const { Strophe: chat_message_Strophe, dayjs: message_dayjs } = core_converse.env; class Message extends CustomElement { static get properties() { return { jid: { type: String }, mid: { type: String } }; } async initialize() { await this.setModels(); if (!this.model) { // Happen during tests due to a race condition headless_log.error('Could not find module for converse-chat-message'); return; } const settings = getAppSettings(); this.listenTo(settings, 'change:render_media', () => { // Reset individual show/hide state of media this.model.save('hide_url_previews', undefined); this.requestUpdate(); }); this.listenTo(this.chatbox, 'change:first_unread_id', () => this.requestUpdate()); this.listenTo(this.model, 'change', () => this.requestUpdate()); this.model.vcard && this.listenTo(this.model.vcard, 'change', () => this.requestUpdate()); if (this.model.get('type') === 'groupchat') { if (this.model.occupant) { this.listenTo(this.model.occupant, 'change', () => this.requestUpdate()); } else { this.listenTo(this.model, 'occupantAdded', () => { this.requestUpdate(); this.listenTo(this.model.occupant, 'change', () => this.requestUpdate()); }); } } } async setModels() { this.chatbox = await core_api.chatboxes.get(this.jid); await this.chatbox.initialized; await this.chatbox.messages.fetched; this.model = this.chatbox.messages.get(this.mid); this.model && this.requestUpdate(); } render() { if (!this.model) { return ''; } else if (this.show_spinner) { return spinner(); } else if (this.model.get('file') && this.model.get('upload') !== shared_converse.SUCCESS) { return this.renderFileProgress(); } else if (['mep'].includes(this.model.get('type'))) { return this.renderMEPMessage(); } else if (['error', 'info'].includes(this.model.get('type'))) { return this.renderInfoMessage(); } else { return this.renderChatMessage(); } } getProps() { return Object.assign(this.model.toJSON(), this.getDerivedMessageProps()); } renderRetraction() { return templates_retraction(this); } renderMessageText() { return message_text(this); } renderMEPMessage() { return mep_message(this); } renderInfoMessage() { return info_message(this); } renderFileProgress() { if (!this.model.file) { // Can happen when file upload failed and page was reloaded return ''; } return file_progress(this); } renderChatMessage() { return templates_message(this, this.getProps()); } shouldShowAvatar() { return core_api.settings.get('show_message_avatar') && !this.model.isMeCommand() && ['chat', 'groupchat'].includes(this.model.get('type')); } onUnfurlAnimationEnd() { if (this.model.get('url_preview_transition') === 'fade-out') { this.model.save({ 'hide_url_previews': true, 'url_preview_transition': 'fade-in' }); } } async onRetryClicked() { this.show_spinner = true; this.requestUpdate(); await core_api.trigger(this.model.get('retry_event_id'), { 'synchronous': true }); this.model.destroy(); this.parentElement.removeChild(this); } isRetracted() { return this.model.get('retracted') || this.model.get('moderated') === 'retracted'; } hasMentions() { const is_groupchat = this.model.get('type') === 'groupchat'; return is_groupchat && this.model.get('sender') === 'them' && this.chatbox.isUserMentioned(this.model); } getOccupantAffiliation() { var _this$model$occupant; return (_this$model$occupant = this.model.occupant) === null || _this$model$occupant === void 0 ? void 0 : _this$model$occupant.get('affiliation'); } getOccupantRole() { var _this$model$occupant2; return (_this$model$occupant2 = this.model.occupant) === null || _this$model$occupant2 === void 0 ? void 0 : _this$model$occupant2.get('role'); } getExtraMessageClasses() { const extra_classes = [this.model.isFollowup() ? 'chat-msg--followup' : null, this.model.get('is_delayed') ? 'delayed' : null, this.model.isMeCommand() ? 'chat-msg--action' : null, this.isRetracted() ? 'chat-msg--retracted' : null, this.model.get('type'), this.shouldShowAvatar() ? 'chat-msg--with-avatar' : null].map(c => c); if (this.model.get('type') === 'groupchat') { extra_classes.push(this.getOccupantRole() ?? ''); extra_classes.push(this.getOccupantAffiliation() ?? ''); if (this.model.get('sender') === 'them' && this.hasMentions()) { extra_classes.push('mentioned'); } } this.model.get('correcting') && extra_classes.push('correcting'); return extra_classes.filter(c => c).join(" "); } getDerivedMessageProps() { const format = core_api.settings.get('time_format'); return { 'pretty_time': message_dayjs(this.model.get('edited') || this.model.get('time')).format(format), 'has_mentions': this.hasMentions(), 'hats': getHats(this.model), 'is_first_unread': this.chatbox.get('first_unread_id') === this.model.get('id'), 'is_me_message': this.model.isMeCommand(), 'is_retracted': this.isRetracted(), 'username': this.model.getDisplayName(), 'should_show_avatar': this.shouldShowAvatar() }; } getRetractionText() { if (['groupchat', 'mep'].includes(this.model.get('type')) && this.model.get('moderated_by')) { const retracted_by_mod = this.model.get('moderated_by'); const chatbox = this.model.collection.chatbox; if (!this.model.mod) { this.model.mod = chatbox.occupants.findOccupant({ 'jid': retracted_by_mod }) || chatbox.occupants.findOccupant({ 'nick': chat_message_Strophe.getResourceFromJid(retracted_by_mod) }); } const modname = this.model.mod ? this.model.mod.getDisplayName() : 'A moderator'; return __('%1$s has removed this message', modname); } else { return __('%1$s has removed this message', this.model.getDisplayName()); } } showUserModal(ev) { if (this.model.get('sender') === 'me') { core_api.modal.show(shared_converse.ProfileModal, { model: this.model }, ev); } else if (this.model.get('type') === 'groupchat') { ev.preventDefault(); core_api.modal.show(modals_occupant, { 'model': this.model.occupant, 'message': this.model }, ev); } else { ev.preventDefault(); const chatbox = this.model.collection.chatbox; core_api.modal.show(modals_user_details, { model: chatbox }, ev); } } showMessageVersionsModal(ev) { ev.preventDefault(); core_api.modal.show(modals_message_versions, { 'model': this.model }, ev); } toggleSpoilerMessage(ev) { ev === null || ev === void 0 ? void 0 : ev.preventDefault(); this.model.save({ 'is_spoiler_visible': !this.model.get('is_spoiler_visible') }); } } core_api.elements.define('converse-chat-message', Message); ;// CONCATENATED MODULE: ./src/shared/chat/message-history.js class MessageHistory extends CustomElement { static get properties() { return { model: { type: Object }, messages: { type: Array } }; } render() { const msgs = this.messages; if (msgs.length) { return repeat_c(msgs, m => m.get('id'), m => $`${this.renderMessage(m)}`); } else { return ''; } } renderMessage(model) { if (model.get('dangling_retraction') || model.get('is_only_key')) { return ''; } const template_hook = model.get('template_hook'); if (typeof template_hook === 'string') { const template_promise = core_api.hook(template_hook, model, ''); return until_c(template_promise, ''); } else { const template = $``; const day = getDayIndicator(model); return day ? [day, template] : template; } } } core_api.elements.define('converse-message-history', MessageHistory); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/shared/chat/styles/chat-content.scss var chat_content = __webpack_require__(8054); ;// CONCATENATED MODULE: ./src/shared/chat/styles/chat-content.scss var chat_content_options = {}; chat_content_options.styleTagTransform = (styleTagTransform_default()); chat_content_options.setAttributes = (setAttributesWithoutAttributes_default()); chat_content_options.insert = insertBySelector_default().bind(null, "head"); chat_content_options.domAPI = (styleDomAPI_default()); chat_content_options.insertStyleElement = (insertStyleElement_default()); var chat_content_update = injectStylesIntoStyleTag_default()(chat_content/* default */.Z, chat_content_options); /* harmony default export */ const styles_chat_content = (chat_content/* default */.Z && chat_content/* default.locals */.Z.locals ? chat_content/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/shared/chat/chat-content.js class ChatContent extends CustomElement { static get properties() { return { jid: { type: String } }; } disconnectedCallback() { super.disconnectedCallback(); this.removeEventListener('scroll', markScrolled); } async initialize() { await this.setModels(); this.listenTo(this.model, 'change:hidden_occupants', this.requestUpdate); this.listenTo(this.model.messages, 'add', this.requestUpdate); this.listenTo(this.model.messages, 'change', this.requestUpdate); this.listenTo(this.model.messages, 'remove', this.requestUpdate); this.listenTo(this.model.messages, 'rendered', this.requestUpdate); this.listenTo(this.model.messages, 'reset', this.requestUpdate); this.listenTo(this.model.notifications, 'change', this.requestUpdate); this.listenTo(this.model.ui, 'change', this.requestUpdate); this.listenTo(this.model.ui, 'change:scrolled', this.scrollDown); if (this.model.occupants) { this.listenTo(this.model.occupants, 'change', this.requestUpdate); } this.addEventListener('scroll', markScrolled); } async setModels() { this.model = await core_api.chatboxes.get(this.jid); await this.model.initialized; this.requestUpdate(); } render() { var _this$model$ui; if (!this.model) { return ''; } // This element has "flex-direction: reverse", so elements here are // shown in reverse order. return $`
${this.model.getNotificationsText()}
${(_this$model$ui = this.model.ui) !== null && _this$model$ui !== void 0 && _this$model$ui.get('chat-content-spinner-top') ? spinner() : ''} `; } scrollDown() { if (this.model.ui.get('scrolled')) { return; } if (this.scrollTo) { const behavior = this.scrollTop ? 'smooth' : 'auto'; this.scrollTo({ 'top': 0, behavior }); } else { this.scrollTop = 0; } /** * Triggered once the converse-chat-content element has been scrolled down to the bottom. * @event _converse#chatBoxScrolledDown * @type {object} * @property { _converse.ChatBox | _converse.ChatRoom } chatbox - The chat model * @example _converse.api.listen.on('chatBoxScrolledDown', obj => { ... }); */ core_api.trigger('chatBoxScrolledDown', { 'chatbox': this.model }); } } core_api.elements.define('converse-chat-content', ChatContent); ;// CONCATENATED MODULE: ./node_modules/lit-html/directives/unsafe-html.js /** * @license * Copyright 2017 Google LLC * SPDX-License-Identifier: BSD-3-Clause */ class unsafe_html_e extends directive_i { constructor(i) { if (super(i), this.it = w, i.type !== directive_t.CHILD) throw Error(this.constructor.directiveName + "() can only be used in child bindings"); } render(r) { if (r === w || null == r) return this.ft = void 0, this.it = r; if (r === b) return r; if ("string" != typeof r) throw Error(this.constructor.directiveName + "() called with a non-string value"); if (r === this.it) return this.ft; this.it = r; const s = [r]; return s.raw = s, this.ft = { _$litType$: this.constructor.resultType, strings: s, values: [] }; } } unsafe_html_e.directiveName = "unsafeHTML", unsafe_html_e.resultType = 1; const unsafe_html_o = directive_e(unsafe_html_e); ;// CONCATENATED MODULE: ./node_modules/lit/directives/unsafe-html.js //# sourceMappingURL=unsafe-html.js.map ;// CONCATENATED MODULE: ./src/shared/chat/help-messages.js class ChatHelp extends CustomElement { static get properties() { return { chat_type: { type: String }, messages: { type: Array }, model: { type: Object }, type: { type: String } }; } render() { const isodate = new Date().toISOString(); return [$``, ...this.messages.map(m => this.renderHelpMessage({ isodate, 'markup': purify_default().sanitize(m, { 'ALLOWED_TAGS': ['strong'] }) }))]; } close() { this.model.set({ 'show_help_messages': false }); } renderHelpMessage(o) { return $`
${unsafe_html_o(o.markup)}
`; } } core_api.elements.define('converse-chat-help', ChatHelp); ;// CONCATENATED MODULE: ./src/shared/chat/templates/emoji-picker.js const emoji_picker_u = core_converse.env.utils; const emoji_category = o => { return $`
  • ${o.emoji}
  • `; }; const emoji_picker_header = o => { const cats = core_api.settings.get('emoji_categories'); const transform = c => cats[c] ? emoji_category(Object.assign({ 'category': c, 'emoji': o.sn2Emoji(cats[c]) }, o)) : ''; return $`
      ${Object.keys(cats).map(transform)}
    `; }; const emoji_item = o => { return $`
  • ${emoji_picker_u.shortnamesToEmojis(o.emoji.sn)}
  • `; }; const tpl_search_results = o => { const i18n_search_results = __('Search results'); return $` ${i18n_search_results}
      ${o.search_results.map(emoji => emoji_item(Object.assign({ emoji }, o)))}
    `; }; const emojis_for_category = o => { return $` ${__(core_api.settings.get('emoji_category_labels')[o.category])}
      ${Object.values(core_converse.emojis.json[o.category]).map(emoji => emoji_item(Object.assign({ emoji }, o)))}
    `; }; const tpl_all_emojis = o => { const cats = core_api.settings.get('emoji_categories'); return $` ${Object.keys(cats).map(c => cats[c] ? emojis_for_category(Object.assign({ 'category': c }, o)) : '')} `; }; const skintone_emoji = o => { return $`
  • ${emoji_picker_u.shortnamesToEmojis(':' + o.skintone + ':')}
  • `; }; const tpl_emoji_picker = o => { const i18n_search = __('Search'); const skintones = ['tone1', 'tone2', 'tone3', 'tone4', 'tone5']; return $`
    ${o.query ? '' : emoji_picker_header(o)}
    ${o.render_emojis ? $`` : ''}
      ${skintones.map(skintone => skintone_emoji(Object.assign({ skintone }, o)))}
    `; }; ;// CONCATENATED MODULE: ./src/shared/chat/emoji-picker-content.js const { sizzle: emoji_picker_content_sizzle } = core_converse.env; class EmojiPickerContent extends CustomElement { static get properties() { return { 'chatview': { type: Object }, 'search_results': { type: Array }, 'current_skintone': { type: String }, 'model': { type: Object }, 'query': { type: String } }; } render() { const props = { 'current_skintone': this.current_skintone, 'insertEmoji': ev => this.insertEmoji(ev), 'query': this.query, 'search_results': this.search_results, 'shouldBeHidden': shortname => this.shouldBeHidden(shortname) }; return $`
    ${tpl_search_results(props)} ${tpl_all_emojis(props)}
    `; } firstUpdated() { this.initIntersectionObserver(); } initIntersectionObserver() { if (!window.IntersectionObserver) { return; } if (this.observer) { this.observer.disconnect(); } else { const options = { root: this.querySelector('.emoji-picker__lists'), threshold: [0.1] }; const handler = ev => this.setCategoryOnVisibilityChange(ev); this.observer = new IntersectionObserver(handler, options); } emoji_picker_content_sizzle('.emoji-picker', this).forEach(a => this.observer.observe(a)); } setCategoryOnVisibilityChange(entries) { const selected = this.parentElement.navigator.selected; const intersection_with_selected = entries.filter(i => i.target.contains(selected)).pop(); let current; // Choose the intersection that contains the currently selected // element, or otherwise the one with the largest ratio. if (intersection_with_selected) { current = intersection_with_selected; } else { current = entries.reduce((p, c) => c.intersectionRatio >= ((p === null || p === void 0 ? void 0 : p.intersectionRatio) || 0) ? c : p, null); } if (current && current.isIntersecting) { const category = current.target.getAttribute('data-category'); if (category !== this.model.get('current_category')) { this.parentElement.preserve_scroll = true; this.model.save({ 'current_category': category }); } } } insertEmoji(ev) { ev.preventDefault(); ev.stopPropagation(); const target = ev.target.nodeName === 'IMG' ? ev.target.parentElement : ev.target; this.parentElement.insertIntoTextArea(target.getAttribute('data-emoji')); } shouldBeHidden(shortname) { // Helper method for the template which decides whether an // emoji should be hidden, based on which skin tone is // currently being applied. if (shortname.includes('_tone')) { if (!this.current_skintone || !shortname.includes(this.current_skintone)) { return true; } } else { if (this.current_skintone && getTonedEmojis().includes(shortname)) { return true; } } if (this.query && !shared_converse.FILTER_CONTAINS(shortname, this.query)) { return true; } return false; } } core_api.elements.define('converse-emoji-picker-content', EmojiPickerContent); ;// CONCATENATED MODULE: ./src/shared/chat/emoji-dropdown.js const emoji_dropdown_u = core_converse.env.utils; class EmojiDropdown extends Dropdown { static get properties() { return { chatview: { type: Object } }; } constructor() { super(); // This is an optimization, we lazily render the emoji picker, otherwise tests slow to a crawl. this.render_emojis = false; } initModel() { if (!this.init_promise) { this.init_promise = (async () => { await core_api.emojis.initialize(); const id = `converse.emoji-${shared_converse.bare_jid}-${this.chatview.model.get('jid')}`; this.model = new shared_converse.EmojiPicker({ 'id': id }); initStorage(this.model, id); await new Promise(resolve => this.model.fetch({ 'success': resolve, 'error': resolve })); // We never want still be in the autocompleting state upon page load this.model.set({ 'autocompleting': null, 'ac_position': null }); })(); } return this.init_promise; } render() { const is_groupchat = this.chatview.model.get('type') === shared_converse.CHATROOMS_TYPE; const color = is_groupchat ? '--muc-toolbar-btn-color' : '--chat-toolbar-btn-color'; return $`
    `; } connectedCallback() { super.connectedCallback(); this.render_emojis = false; } toggleMenu(ev) { ev.stopPropagation(); ev.preventDefault(); if (emoji_dropdown_u.hasClass('show', this.menu)) { if (emoji_dropdown_u.ancestor(ev.target, '.toggle-emojis')) { this.hideMenu(); } } else { this.showMenu(); } } async showMenu() { await this.initModel(); if (!this.render_emojis) { // Trigger an update so that emojis are rendered this.render_emojis = true; this.requestUpdate(); await this.updateComplete; } super.showMenu(); setTimeout(() => { var _this$querySelector; return (_this$querySelector = this.querySelector('.emoji-search')) === null || _this$querySelector === void 0 ? void 0 : _this$querySelector.focus(); }); } } core_api.elements.define('converse-emoji-dropdown', EmojiDropdown); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/shared/chat/styles/emoji.scss var emoji = __webpack_require__(8125); ;// CONCATENATED MODULE: ./src/shared/chat/styles/emoji.scss var emoji_options = {}; emoji_options.styleTagTransform = (styleTagTransform_default()); emoji_options.setAttributes = (setAttributesWithoutAttributes_default()); emoji_options.insert = insertBySelector_default().bind(null, "head"); emoji_options.domAPI = (styleDomAPI_default()); emoji_options.insertStyleElement = (insertStyleElement_default()); var emoji_update = injectStylesIntoStyleTag_default()(emoji/* default */.Z, emoji_options); /* harmony default export */ const styles_emoji = (emoji/* default */.Z && emoji/* default.locals */.Z.locals ? emoji/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/shared/chat/emoji-picker.js const chat_emoji_picker_u = core_converse.env.utils; class EmojiPicker extends CustomElement { static get properties() { return { 'chatview': { type: Object }, 'current_category': { type: String, 'reflect': true }, 'current_skintone': { type: String, 'reflect': true }, 'model': { type: Object }, 'query': { type: String, 'reflect': true }, // This is an optimization, we lazily render the emoji picker, otherwise tests slow to a crawl. 'render_emojis': { type: Boolean } }; } firstUpdated() { super.firstUpdated(); this.listenTo(this.model, 'change', o => this.onModelChanged(o.changed)); this.initArrowNavigation(); } constructor() { super(); this.query = ''; this._search_results = []; this.debouncedFilter = lodash_es_debounce(input => this.model.set({ 'query': input.value }), 250); } get search_results() { return this._search_results; } set search_results(value) { this._search_results = value; this.requestUpdate(); } render() { return tpl_emoji_picker({ 'chatview': this.chatview, 'current_category': this.current_category, 'current_skintone': this.current_skintone, 'model': this.model, 'onCategoryPicked': ev => this.chooseCategory(ev), 'onSearchInputBlurred': ev => this.chatview.emitFocused(ev), 'onSearchInputFocus': ev => this.onSearchInputFocus(ev), 'onSearchInputKeyDown': ev => this.onSearchInputKeyDown(ev), 'onSkintonePicked': ev => this.chooseSkinTone(ev), 'query': this.query, 'search_results': this.search_results, 'render_emojis': this.render_emojis, 'sn2Emoji': shortname => chat_emoji_picker_u.shortnamesToEmojis(this.getTonedShortname(shortname)) }); } updated(changed) { changed.has('query') && this.updateSearchResults(changed); changed.has('current_category') && this.setScrollPosition(); } onModelChanged(changed) { if ('current_category' in changed) this.current_category = changed.current_category; if ('current_skintone' in changed) this.current_skintone = changed.current_skintone; if ('query' in changed) this.query = changed.query; } setScrollPosition() { if (this.preserve_scroll) { this.preserve_scroll = false; return; } const el = this.querySelector('.emoji-lists__container--browse'); const heading = this.querySelector(`#emoji-picker-${this.current_category}`); if (heading) { // +4 due to 2px padding on list elements el.scrollTop = heading.offsetTop - heading.offsetHeight * 3 + 4; } } updateSearchResults(changed) { const old_query = changed.get('query'); const contains = shared_converse.FILTER_CONTAINS; if (this.query) { if (this.query === old_query) { return this.search_results; } else if (old_query && this.query.includes(old_query)) { this.search_results = this.search_results.filter(e => contains(e.sn, this.query)); } else { this.search_results = core_converse.emojis.list.filter(e => contains(e.sn, this.query)); } } else if (this.search_results.length) { // Avoid re-rendering by only setting to new empty array if it wasn't empty before this.search_results = []; } } registerEvents() { this.onGlobalKeyDown = ev => this._onGlobalKeyDown(ev); const body = document.querySelector('body'); body.addEventListener('keydown', this.onGlobalKeyDown); } connectedCallback() { super.connectedCallback(); this.registerEvents(); } disconnectedCallback() { const body = document.querySelector('body'); body.removeEventListener('keydown', this.onGlobalKeyDown); this.disableArrowNavigation(); super.disconnectedCallback(); } _onGlobalKeyDown(ev) { if (!this.navigator) { return; } if (ev.keyCode === KEYCODES.ENTER && chat_emoji_picker_u.isVisible(this)) { this.onEnterPressed(ev); } else if (ev.keyCode === KEYCODES.DOWN_ARROW && !this.navigator.enabled && chat_emoji_picker_u.isVisible(this)) { this.enableArrowNavigation(ev); } else if (ev.keyCode === KEYCODES.ESCAPE) { this.disableArrowNavigation(); setTimeout(() => this.chatview.querySelector('.chat-textarea').focus(), 50); } } setCategoryForElement(el) { const old_category = this.current_category; const category = (el === null || el === void 0 ? void 0 : el.getAttribute('data-category')) || old_category; if (old_category !== category) { this.model.save({ 'current_category': category }); } } insertIntoTextArea(value) { const autocompleting = this.model.get('autocompleting'); const ac_position = this.model.get('ac_position'); this.model.set({ 'autocompleting': null, 'query': '', 'ac_position': null }); this.disableArrowNavigation(); const jid = this.chatview.model.get('jid'); const options = { 'bubbles': true, 'detail': { value, autocompleting, ac_position, jid } }; this.dispatchEvent(new CustomEvent("emojiSelected", options)); } chooseSkinTone(ev) { ev.preventDefault(); ev.stopPropagation(); const target = ev.target.nodeName === 'IMG' ? ev.target.parentElement : ev.target; const skintone = target.getAttribute("data-skintone").trim(); if (this.current_skintone === skintone) { this.model.save({ 'current_skintone': '' }); } else { this.model.save({ 'current_skintone': skintone }); } } chooseCategory(ev) { ev.preventDefault && ev.preventDefault(); ev.stopPropagation && ev.stopPropagation(); const el = ev.target.matches('li') ? ev.target : chat_emoji_picker_u.ancestor(ev.target, 'li'); this.setCategoryForElement(el); this.navigator.select(el); !this.navigator.enabled && this.navigator.enable(); } onSearchInputKeyDown(ev) { if (ev.keyCode === KEYCODES.TAB) { if (ev.target.value) { ev.preventDefault(); const match = core_converse.emojis.shortnames.find(sn => shared_converse.FILTER_CONTAINS(sn, ev.target.value)); match && this.model.set({ 'query': match }); } else if (!this.navigator.enabled) { this.enableArrowNavigation(ev); } } else if (ev.keyCode === KEYCODES.DOWN_ARROW && !this.navigator.enabled) { this.enableArrowNavigation(ev); } else if (ev.keyCode !== KEYCODES.ENTER && ev.keyCode !== KEYCODES.DOWN_ARROW) { this.debouncedFilter(ev.target); } } onEnterPressed(ev) { ev.preventDefault(); ev.stopPropagation(); if (core_converse.emojis.shortnames.includes(ev.target.value)) { this.insertIntoTextArea(ev.target.value); } else if (this.search_results.length === 1) { this.insertIntoTextArea(this.search_results[0].sn); } else if (this.navigator.selected && this.navigator.selected.matches('.insert-emoji')) { this.insertIntoTextArea(this.navigator.selected.getAttribute('data-emoji')); } else if (this.navigator.selected && this.navigator.selected.matches('.emoji-category')) { this.chooseCategory({ 'target': this.navigator.selected }); } } onSearchInputFocus(ev) { this.chatview.emitBlurred(ev); this.disableArrowNavigation(); } getTonedShortname(shortname) { if (getTonedEmojis().includes(shortname) && this.current_skintone) { return `${shortname.slice(0, shortname.length - 1)}_${this.current_skintone}:`; } return shortname; } initArrowNavigation() { if (!this.navigator) { const default_selector = 'li:not(.hidden):not(.emoji-skintone), .emoji-search'; const options = { 'jump_to_picked': '.emoji-category', 'jump_to_picked_selector': '.emoji-category.picked', 'jump_to_picked_direction': dom_navigator.DIRECTION.down, 'picked_selector': '.picked', 'scroll_container': this.querySelector('.emoji-picker__lists'), 'getSelector': direction => { if (direction === dom_navigator.DIRECTION.down) { const c = this.navigator.selected && this.navigator.selected.getAttribute('data-category'); return c ? `ul[data-category="${c}"] li:not(.hidden):not(.emoji-skintone), .emoji-search` : default_selector; } else { return default_selector; } }, 'onSelected': el => { el.matches('.insert-emoji') && this.setCategoryForElement(el.parentElement); el.matches('.insert-emoji, .emoji-category') && el.firstElementChild.focus(); el.matches('.emoji-search') && el.focus(); } }; this.navigator = new dom_navigator(this, options); } } disableArrowNavigation() { var _this$navigator; (_this$navigator = this.navigator) === null || _this$navigator === void 0 ? void 0 : _this$navigator.disable(); } enableArrowNavigation(ev) { var _ev$preventDefault, _ev$stopPropagation; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); ev === null || ev === void 0 ? void 0 : (_ev$stopPropagation = ev.stopPropagation) === null || _ev$stopPropagation === void 0 ? void 0 : _ev$stopPropagation.call(ev); this.disableArrowNavigation(); this.navigator.enable(); this.navigator.handleKeydown(ev); } } core_api.elements.define('converse-emoji-picker', EmojiPicker); ;// CONCATENATED MODULE: ./src/shared/chat/templates/message-limit.js /* harmony default export */ const message_limit = (counter => { const i18n_chars_remaining = __('Message characters remaining'); return $`${counter}`; }); ;// CONCATENATED MODULE: ./src/shared/chat/message-limit.js class MessageLimitIndicator extends CustomElement { static get properties() { return { model: { type: Object } }; } connectedCallback() { super.connectedCallback(); this.listenTo(this.model, 'change:draft', this.requestUpdate); } render() { const limit = core_api.settings.get('message_limit'); if (!limit) return ''; const chars = this.model.get('draft') || ''; return message_limit(limit - chars.length); } } core_api.elements.define('converse-message-limit-indicator', MessageLimitIndicator); ;// CONCATENATED MODULE: ./src/shared/chat/templates/toolbar.js function tpl_send_button() { const i18n_send_message = __('Send the message'); return $``; } /* harmony default export */ const toolbar = (el => { return $` ${until_c(el.getButtons(), '')} ${el.show_send_button ? tpl_send_button() : ''} `; }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/shared/chat/styles/toolbar.scss var styles_toolbar = __webpack_require__(6176); ;// CONCATENATED MODULE: ./src/shared/chat/styles/toolbar.scss var toolbar_options = {}; toolbar_options.styleTagTransform = (styleTagTransform_default()); toolbar_options.setAttributes = (setAttributesWithoutAttributes_default()); toolbar_options.insert = insertBySelector_default().bind(null, "head"); toolbar_options.domAPI = (styleDomAPI_default()); toolbar_options.insertStyleElement = (insertStyleElement_default()); var toolbar_update = injectStylesIntoStyleTag_default()(styles_toolbar/* default */.Z, toolbar_options); /* harmony default export */ const chat_styles_toolbar = (styles_toolbar/* default */.Z && styles_toolbar/* default.locals */.Z.locals ? styles_toolbar/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/shared/chat/toolbar.js const toolbar_Strophe = core_converse.env.Strophe; class ChatToolbar extends CustomElement { static get properties() { return { hidden_occupants: { type: Boolean }, is_groupchat: { type: Boolean }, message_limit: { type: Number }, model: { type: Object }, show_call_button: { type: Boolean }, show_emoji_button: { type: Boolean }, show_send_button: { type: Boolean }, show_spoiler_button: { type: Boolean } }; } connectedCallback() { super.connectedCallback(); this.listenTo(this.model, 'change:composing_spoiler', () => this.requestUpdate()); } render() { return toolbar(this); } firstUpdated() { /** * Triggered once the _converse.ChatBoxView's toolbar has been rendered * @event _converse#renderToolbar * @type { _converse.ChatBoxView } * @example _converse.api.listen.on('renderToolbar', this => { ... }); */ core_api.trigger('renderToolbar', this); } getButtons() { var _api$settings$get; const buttons = []; if (this.show_emoji_button) { const chatview = shared_converse.chatboxviews.get(this.model.get('jid')); buttons.push($``); } if (this.show_call_button) { const color = this.is_groupchat ? '--muc-toolbar-btn-color' : '--chat-toolbar-btn-color'; const i18n_start_call = __('Start a call'); buttons.push($` `); } const message_limit = core_api.settings.get('message_limit'); if (message_limit) { buttons.push($` `); } if (this.show_spoiler_button) { buttons.push(this.getSpoilerButton()); } const http_upload_promise = core_api.disco.supports(toolbar_Strophe.NS.HTTPUPLOAD, shared_converse.domain); buttons.push($`${until_c(http_upload_promise.then(is_supported => this.getHTTPUploadButton(is_supported)), '')}`); if (this.is_groupchat && (_api$settings$get = core_api.settings.get('visible_toolbar_buttons')) !== null && _api$settings$get !== void 0 && _api$settings$get.toggle_occupants) { const i18n_hide_occupants = __('Hide participants'); const i18n_show_occupants = __('Show participants'); buttons.push($` `); } /** * *Hook* which allows plugins to add more buttons to a chat's toolbar * @event _converse#getToolbarButtons * @example * api.listen.on('getToolbarButtons', (toolbar_el, buttons) { * buttons.push(html` * ` * ); * return buttons; * } */ return shared_converse.api.hook('getToolbarButtons', this, buttons); } getHTTPUploadButton(is_supported) { if (is_supported) { const i18n_choose_file = __('Choose a file to send'); const color = this.is_groupchat ? '--muc-toolbar-btn-color' : '--chat-toolbar-btn-color'; return $` `; } else { return ''; } } getSpoilerButton() { var _model$presence; const model = this.model; if (!this.is_groupchat && !((_model$presence = model.presence) !== null && _model$presence !== void 0 && _model$presence.resources.length)) { return; } let i18n_toggle_spoiler; if (model.get('composing_spoiler')) { i18n_toggle_spoiler = __("Click to write as a normal (non-spoiler) message"); } else { i18n_toggle_spoiler = __("Click to write your message as a spoiler"); } const color = this.is_groupchat ? '--muc-toolbar-btn-color' : '--chat-toolbar-btn-color'; const markup = $` `; if (this.is_groupchat) { return markup; } else { const contact_jid = model.get('jid'); const spoilers_promise = Promise.all(model.presence.resources.map(r => core_api.disco.supports(toolbar_Strophe.NS.SPOILER, `${contact_jid}/${r.get('name')}`))).then(results => results.reduce((acc, val) => acc && val, true)); return $`${until_c(spoilers_promise.then(() => markup), '')}`; } } toggleFileUpload(ev) { var _ev$preventDefault, _ev$stopPropagation; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); ev === null || ev === void 0 ? void 0 : (_ev$stopPropagation = ev.stopPropagation) === null || _ev$stopPropagation === void 0 ? void 0 : _ev$stopPropagation.call(ev); this.querySelector('.fileupload').click(); } onFileSelection(evt) { this.model.sendFiles(evt.target.files); } toggleComposeSpoilerMessage(ev) { var _ev$preventDefault2, _ev$stopPropagation2; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault2 = ev.preventDefault) === null || _ev$preventDefault2 === void 0 ? void 0 : _ev$preventDefault2.call(ev); ev === null || ev === void 0 ? void 0 : (_ev$stopPropagation2 = ev.stopPropagation) === null || _ev$stopPropagation2 === void 0 ? void 0 : _ev$stopPropagation2.call(ev); this.model.set('composing_spoiler', !this.model.get('composing_spoiler')); } toggleOccupants(ev) { var _ev$preventDefault3, _ev$stopPropagation3; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault3 = ev.preventDefault) === null || _ev$preventDefault3 === void 0 ? void 0 : _ev$preventDefault3.call(ev); ev === null || ev === void 0 ? void 0 : (_ev$stopPropagation3 = ev.stopPropagation) === null || _ev$stopPropagation3 === void 0 ? void 0 : _ev$stopPropagation3.call(ev); this.model.save({ 'hidden_occupants': !this.model.get('hidden_occupants') }); } toggleCall(ev) { var _ev$preventDefault4, _ev$stopPropagation4; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault4 = ev.preventDefault) === null || _ev$preventDefault4 === void 0 ? void 0 : _ev$preventDefault4.call(ev); ev === null || ev === void 0 ? void 0 : (_ev$stopPropagation4 = ev.stopPropagation) === null || _ev$stopPropagation4 === void 0 ? void 0 : _ev$stopPropagation4.call(ev); /** * When a call button (i.e. with class .toggle-call) on a chatbox has been clicked. * @event _converse#callButtonClicked * @type { object } * @property { Strophe.Connection } _converse.connection - The XMPP Connection object * @property { _converse.ChatBox | _converse.ChatRoom } _converse.connection - The XMPP Connection object * @example _converse.api.listen.on('callButtonClicked', (connection, model) => { ... }); */ core_api.trigger('callButtonClicked', { connection: shared_converse.connection, model: this.model }); } } window.customElements.define('converse-chat-toolbar', ChatToolbar); ;// CONCATENATED MODULE: ./src/plugins/chatview/utils.js function clearHistory(jid) { if (shared_converse.router.history.getFragment() === `converse/chat?jid=${jid}`) { shared_converse.router.navigate(''); } } async function getHeadingDropdownItem(promise_or_data) { const data = await promise_or_data; return data ? $` ${data.i18n_text} ` : ''; } async function getHeadingStandaloneButton(promise_or_data) { const data = await promise_or_data; return $` `; } async function clearMessages(chat) { const result = confirm(__('Are you sure you want to clear the messages from this conversation?')); if (result === true) { await chat.clearMessages(); } } async function parseMessageForCommands(chat, text) { const match = text.replace(/^\s*/, '').match(/^\/(.*)\s*$/); if (match) { let handled = false; /** * *Hook* which allows plugins to add more commands to a chat's textbox. * Data provided is the chatbox model and the text typed - {model, text}. * Check `handled` to see if the hook was already handled. * @event _converse#parseMessageForCommands * @example * api.listen.on('parseMessageForCommands', (data, handled) { * if (!handled) { * const command = (data.text.match(/^\/([a-zA-Z]*) ?/) || ['']).pop().toLowerCase(); * // custom code comes here * } * return handled; * } */ handled = await core_api.hook('parseMessageForCommands', { model: chat, text }, handled); if (handled) { return true; } if (match[1] === 'clear') { clearMessages(chat); return true; } else if (match[1] === 'close') { var _converse$chatboxview; (_converse$chatboxview = shared_converse.chatboxviews.get(chat.get('jid'))) === null || _converse$chatboxview === void 0 ? void 0 : _converse$chatboxview.close(); return true; } else if (match[1] === 'help') { chat.set({ 'show_help_messages': false }, { 'silent': true }); chat.set({ 'show_help_messages': true }); return true; } } return false; } function resetElementHeight(ev) { if (ev.target.value) { const height = ev.target.scrollHeight + 'px'; if (ev.target.style.height != height) { ev.target.style.height = 'auto'; ev.target.style.height = height; } } else { ev.target.style = ''; } } ;// CONCATENATED MODULE: ./src/plugins/chatview/templates/chat-head.js async function getStandaloneButtons(promise) { const heading_btns = await promise; const standalone_btns = heading_btns.filter(b => b.standalone); return standalone_btns.map(b => getHeadingStandaloneButton(b)); } async function getDropdownButtons(promise) { const heading_btns = await promise; const dropdown_btns = heading_btns.filter(b => !b.standalone); return dropdown_btns.map(b => getHeadingDropdownItem(b)); } /* harmony default export */ const chat_head = (o => { var _o$model$vcard, _o$model$vcard2; const i18n_profile = __("The User's Profile Image"); const avatar = $` `; const display_name = o.model.getDisplayName(); const tpl_dropdown_btns = () => getDropdownButtons(o.heading_buttons_promise).then(btns => btns.length ? $`` : ''); const tpl_standalone_btns = () => getStandaloneButtons(o.heading_buttons_promise).then(btns => btns.reverse().map(b => until_c(b, ''))); return $`
    ${!shared_converse.api.settings.get("singleton") ? $`` : ''} ${o.type !== shared_converse.HEADLINES_TYPE ? $`${avatar}` : ''}
    ${o.type !== shared_converse.HEADLINES_TYPE ? $`${display_name}` : display_name}
    ${until_c(tpl_dropdown_btns(), '')} ${until_c(tpl_standalone_btns(), '')}
    ${o.status ? $`

    ${o.status}

    ` : ''} `; }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/chatview/styles/chat-head.scss var styles_chat_head = __webpack_require__(7802); ;// CONCATENATED MODULE: ./src/plugins/chatview/styles/chat-head.scss var chat_head_options = {}; chat_head_options.styleTagTransform = (styleTagTransform_default()); chat_head_options.setAttributes = (setAttributesWithoutAttributes_default()); chat_head_options.insert = insertBySelector_default().bind(null, "head"); chat_head_options.domAPI = (styleDomAPI_default()); chat_head_options.insertStyleElement = (insertStyleElement_default()); var chat_head_update = injectStylesIntoStyleTag_default()(styles_chat_head/* default */.Z, chat_head_options); /* harmony default export */ const chatview_styles_chat_head = (styles_chat_head/* default */.Z && styles_chat_head/* default.locals */.Z.locals ? styles_chat_head/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/chatview/heading.js class ChatHeading extends CustomElement { initialize() { var _this$model$rosterCon; this.model = shared_converse.chatboxes.get(this.getAttribute('jid')); this.listenTo(this.model, 'change:status', this.requestUpdate); this.listenTo(this.model, 'vcard:add', this.requestUpdate); this.listenTo(this.model, 'vcard:change', this.requestUpdate); if (this.model.contact) { this.listenTo(this.model.contact, 'destroy', this.requestUpdate); } (_this$model$rosterCon = this.model.rosterContactAdded) === null || _this$model$rosterCon === void 0 ? void 0 : _this$model$rosterCon.then(() => { this.listenTo(this.model.contact, 'change:nickname', this.requestUpdate); this.requestUpdate(); }); } render() { return chat_head(Object.assign(this.model.toJSON(), { 'heading_buttons_promise': this.getHeadingButtons(), 'model': this.model, 'showUserDetailsModal': ev => this.showUserDetailsModal(ev) })); } showUserDetailsModal(ev) { ev.preventDefault(); core_api.modal.show(modals_user_details, { model: this.model }, ev); } close(ev) { ev.preventDefault(); this.model.close(); } /** * Returns a list of objects which represent buttons for the chat's header. * @async * @emits _converse#getHeadingButtons */ getHeadingButtons() { const buttons = [ /** * @typedef { Object } HeadingButtonAttributes * An object representing a chat heading button * @property { Boolean } standalone * True if shown on its own, false if it must be in the dropdown menu. * @property { Function } handler * A handler function to be called when the button is clicked. * @property { String } a_class - HTML classes to show on the button * @property { String } i18n_text - The user-visiible name of the button * @property { String } i18n_title - The tooltip text for this button * @property { String } icon_class - What kind of CSS class to use for the icon * @property { String } name - The internal name of the button */ { 'a_class': 'show-user-details-modal', 'handler': ev => this.showUserDetailsModal(ev), 'i18n_text': __('Details'), 'i18n_title': __('See more information about this person'), 'icon_class': 'fa-id-card', 'name': 'details', 'standalone': core_api.settings.get('view_mode') === 'overlayed' }]; if (!core_api.settings.get('singleton')) { buttons.push({ 'a_class': 'close-chatbox-button', 'handler': ev => this.close(ev), 'i18n_text': __('Close'), 'i18n_title': __('Close and end this conversation'), 'icon_class': 'fa-times', 'name': 'close', 'standalone': core_api.settings.get('view_mode') === 'overlayed' }); } const el = shared_converse.chatboxviews.get(this.getAttribute('jid')); if (el) { /** * *Hook* which allows plugins to add more buttons to a chat's heading. * * Note: This hook is fired for both 1 on 1 chats and groupchats. * If you only care about one, you need to add a check in your code. * * @event _converse#getHeadingButtons * @param { HTMLElement } el * The `converse-chat` (or `converse-muc`) DOM element that represents the chat * @param { Array. } * An array of the existing buttons. New buttons may be added, * and existing ones removed or modified. * @example * api.listen.on('getHeadingButtons', (el, buttons) => { * buttons.push({ * 'i18n_title': __('Foo'), * 'i18n_text': __('Foo Bar'), * 'handler': ev => alert('Foo!'), * 'a_class': 'toggle-foo', * 'icon_class': 'fa-foo', * 'name': 'foo' * }); * return buttons; * }); */ return shared_converse.api.hook('getHeadingButtons', el, buttons); } else { return buttons; // Happens during tests } } } core_api.elements.define('converse-chat-heading', ChatHeading); ;// CONCATENATED MODULE: ./src/plugins/chatview/templates/message-form.js /* harmony default export */ const message_form = (o => { const label_message = o.composing_spoiler ? __('Hidden message') : __('Message'); const label_spoiler_hint = __('Optional hint'); const show_send_button = core_api.settings.get('show_send_button'); return $`
    `; }); ;// CONCATENATED MODULE: ./node_modules/@converse/skeletor/src/element.js function element_defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } const element_paddedLt = /^\s* this.render()); this.handleEmojiSelection = _ref => { let { detail } = _ref; if (this.model.get('jid') === detail.jid) { this.insertIntoTextArea(detail.value, detail.autocompleting, false, detail.ac_position); } }; document.addEventListener("emojiSelected", this.handleEmojiSelection); this.render(); } disconnectedCallback() { super.disconnectedCallback(); document.removeEventListener("emojiSelected", this.handleEmojiSelection); } toHTML() { var _this$querySelector, _this$querySelector2; return message_form(Object.assign(this.model.toJSON(), { 'onDrop': ev => this.onDrop(ev), 'hint_value': (_this$querySelector = this.querySelector('.spoiler-hint')) === null || _this$querySelector === void 0 ? void 0 : _this$querySelector.value, 'message_value': (_this$querySelector2 = this.querySelector('.chat-textarea')) === null || _this$querySelector2 === void 0 ? void 0 : _this$querySelector2.value, 'onChange': ev => this.model.set({ 'draft': ev.target.value }), 'onKeyDown': ev => this.onKeyDown(ev), 'onKeyUp': ev => this.onKeyUp(ev), 'onPaste': ev => this.onPaste(ev), 'viewUnreadMessages': ev => this.viewUnreadMessages(ev) })); } /** * Insert a particular string value into the textarea of this chat box. * @param {string} value - The value to be inserted. * @param {(boolean|string)} [replace] - Whether an existing value * should be replaced. If set to `true`, the entire textarea will * be replaced with the new value. If set to a string, then only * that string will be replaced *if* a position is also specified. * @param {integer} [position] - The end index of the string to be * replaced with the new value. */ insertIntoTextArea(value) { let replace = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false; let correcting = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false; let position = arguments.length > 3 ? arguments[3] : undefined; const textarea = this.querySelector('.chat-textarea'); if (correcting) { message_form_u.addClass('correcting', textarea); } else { message_form_u.removeClass('correcting', textarea); } if (replace) { if (position && typeof replace == 'string') { textarea.value = textarea.value.replace(new RegExp(replace, 'g'), (match, offset) => offset == position - replace.length ? value + ' ' : match); } else { textarea.value = value; } } else { let existing = textarea.value; if (existing && existing[existing.length - 1] !== ' ') { existing = existing + ' '; } textarea.value = existing + value + ' '; } const ev = document.createEvent('HTMLEvents'); ev.initEvent('change', false, true); textarea.dispatchEvent(ev); message_form_u.placeCaretAtEnd(textarea); } onMessageCorrecting(message) { if (message.get('correcting')) { this.insertIntoTextArea(prefixMentions(message), true, true); } else { const currently_correcting = this.model.messages.findWhere('correcting'); if (currently_correcting && currently_correcting !== message) { this.insertIntoTextArea(prefixMentions(message), true, true); } else { this.insertIntoTextArea('', true, false); } } } onEscapePressed(ev) { const idx = this.model.messages.findLastIndex('correcting'); const message = idx >= 0 ? this.model.messages.at(idx) : null; if (message) { ev.preventDefault(); message.save('correcting', false); this.insertIntoTextArea('', true, false); } } onPaste(ev) { ev.stopPropagation(); if (ev.clipboardData.files.length !== 0) { ev.preventDefault(); // Workaround for quirk in at least Firefox 60.7 ESR: // It seems that pasted files disappear from the event payload after // the event has finished, which apparently happens during async // processing in sendFiles(). So we copy the array here. this.model.sendFiles(Array.from(ev.clipboardData.files)); return; } this.model.set({ 'draft': ev.clipboardData.getData('text/plain') }); } onKeyUp(ev) { this.model.set({ 'draft': ev.target.value }); } onKeyDown(ev) { if (ev.ctrlKey) { // When ctrl is pressed, no chars are entered into the textarea. return; } if (!ev.shiftKey && !ev.altKey && !ev.metaKey) { if (ev.keyCode === core_converse.keycodes.TAB) { const value = message_form_u.getCurrentWord(ev.target, null, /(:.*?:)/g); if (value.startsWith(':')) { ev.preventDefault(); ev.stopPropagation(); this.model.trigger('emoji-picker-autocomplete', ev.target, value); } } else if (ev.keyCode === core_converse.keycodes.FORWARD_SLASH) { // Forward slash is used to run commands. Nothing to do here. return; } else if (ev.keyCode === core_converse.keycodes.ESCAPE) { return this.onEscapePressed(ev, this); } else if (ev.keyCode === core_converse.keycodes.ENTER) { return this.onFormSubmitted(ev); } else if (ev.keyCode === core_converse.keycodes.UP_ARROW && !ev.target.selectionEnd) { const textarea = this.querySelector('.chat-textarea'); if (!textarea.value || message_form_u.hasClass('correcting', textarea)) { return this.model.editEarlierMessage(); } } else if (ev.keyCode === core_converse.keycodes.DOWN_ARROW && ev.target.selectionEnd === ev.target.value.length && message_form_u.hasClass('correcting', this.querySelector('.chat-textarea'))) { return this.model.editLaterMessage(); } } if ([core_converse.keycodes.SHIFT, core_converse.keycodes.META, core_converse.keycodes.META_RIGHT, core_converse.keycodes.ESCAPE, core_converse.keycodes.ALT].includes(ev.keyCode)) { return; } if (this.model.get('chat_state') !== shared_converse.COMPOSING) { // Set chat state to composing if keyCode is not a forward-slash // (which would imply an internal command and not a message). this.model.setChatState(shared_converse.COMPOSING); } } async onFormSubmitted(ev) { var _ev$preventDefault, _this$querySelector3; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); const textarea = this.querySelector('.chat-textarea'); const message_text = textarea.value.trim(); if (core_api.settings.get('message_limit') && message_text.length > core_api.settings.get('message_limit') || !message_text.replace(/\s/g, '').length) { return; } if (!shared_converse.connection.authenticated) { const err_msg = __('Sorry, the connection has been lost, and your message could not be sent'); core_api.alert('error', __('Error'), err_msg); core_api.connection.reconnect(); return; } let spoiler_hint, hint_el = {}; if (this.model.get('composing_spoiler')) { hint_el = this.querySelector('form.sendXMPPMessage input.spoiler-hint'); spoiler_hint = hint_el.value; } message_form_u.addClass('disabled', textarea); textarea.setAttribute('disabled', 'disabled'); (_this$querySelector3 = this.querySelector('converse-emoji-dropdown')) === null || _this$querySelector3 === void 0 ? void 0 : _this$querySelector3.hideMenu(); const is_command = await parseMessageForCommands(this.model, message_text); const message = is_command ? null : await this.model.sendMessage({ 'body': message_text, spoiler_hint }); if (is_command || message) { hint_el.value = ''; textarea.value = ''; message_form_u.removeClass('correcting', textarea); textarea.style.height = 'auto'; this.model.set({ 'draft': '' }); } if (core_api.settings.get('view_mode') === 'overlayed') { // XXX: Chrome flexbug workaround. The .chat-content area // doesn't resize when the textarea is resized to its original size. const chatview = shared_converse.chatboxviews.get(this.getAttribute('jid')); const msgs_container = chatview.querySelector('.chat-content__messages'); msgs_container.parentElement.style.display = 'none'; } textarea.removeAttribute('disabled'); message_form_u.removeClass('disabled', textarea); if (core_api.settings.get('view_mode') === 'overlayed') { // XXX: Chrome flexbug workaround. const chatview = shared_converse.chatboxviews.get(this.getAttribute('jid')); const msgs_container = chatview.querySelector('.chat-content__messages'); msgs_container.parentElement.style.display = ''; } // Suppress events, otherwise superfluous CSN gets set // immediately after the message, causing rate-limiting issues. this.model.setChatState(shared_converse.ACTIVE, { 'silent': true }); textarea.focus(); } } core_api.elements.define('converse-message-form', MessageForm); ;// CONCATENATED MODULE: ./src/plugins/chatview/templates/bottom-panel.js /* harmony default export */ const bottom_panel = (o => { const unread_msgs = __('You have unread messages'); const message_limit = core_api.settings.get('message_limit'); const show_call_button = core_api.settings.get('visible_toolbar_buttons').call; const show_emoji_button = core_api.settings.get('visible_toolbar_buttons').emoji; const show_send_button = core_api.settings.get('show_send_button'); const show_spoiler_button = core_api.settings.get('visible_toolbar_buttons').spoiler; const show_toolbar = core_api.settings.get('show_toolbar'); return $` ${o.model.ui.get('scrolled') && o.model.get('num_unread') ? $`
    o.viewUnreadMessages(ev)}>▼ ${unread_msgs} ▼
    ` : ''} ${core_api.settings.get('show_toolbar') ? $` ` : ''} `; }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/chatview/styles/chat-bottom-panel.scss var chat_bottom_panel = __webpack_require__(352); ;// CONCATENATED MODULE: ./src/plugins/chatview/styles/chat-bottom-panel.scss var chat_bottom_panel_options = {}; chat_bottom_panel_options.styleTagTransform = (styleTagTransform_default()); chat_bottom_panel_options.setAttributes = (setAttributesWithoutAttributes_default()); chat_bottom_panel_options.insert = insertBySelector_default().bind(null, "head"); chat_bottom_panel_options.domAPI = (styleDomAPI_default()); chat_bottom_panel_options.insertStyleElement = (insertStyleElement_default()); var chat_bottom_panel_update = injectStylesIntoStyleTag_default()(chat_bottom_panel/* default */.Z, chat_bottom_panel_options); /* harmony default export */ const styles_chat_bottom_panel = (chat_bottom_panel/* default */.Z && chat_bottom_panel/* default.locals */.Z.locals ? chat_bottom_panel/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/chatview/bottom-panel.js function bottom_panel_defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } class ChatBottomPanel extends ElementView { constructor() { super(); bottom_panel_defineProperty(this, "events", { 'click .send-button': 'sendButtonClicked', 'click .toggle-clear': 'clearMessages' }); this.debouncedRender = lodash_es_debounce(this.render, 100); } async connectedCallback() { super.connectedCallback(); await this.initialize(); this.render(); // don't call in initialize, since the MUCBottomPanel subclasses it // and we want to render after it has finished as wel. } async initialize() { this.model = await core_api.chatboxes.get(this.getAttribute('jid')); await this.model.initialized; this.listenTo(this.model, 'change:num_unread', this.debouncedRender); this.listenTo(this.model, 'emoji-picker-autocomplete', this.autocompleteInPicker); this.addEventListener('focusin', ev => this.emitFocused(ev)); this.addEventListener('focusout', ev => this.emitBlurred(ev)); } render() { x(bottom_panel({ 'model': this.model, 'viewUnreadMessages': ev => this.viewUnreadMessages(ev) }), this); } sendButtonClicked(ev) { var _this$querySelector; (_this$querySelector = this.querySelector('converse-message-form')) === null || _this$querySelector === void 0 ? void 0 : _this$querySelector.onFormSubmitted(ev); } viewUnreadMessages(ev) { var _ev$preventDefault; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); this.model.ui.set({ 'scrolled': false }); } emitFocused(ev) { var _converse$chatboxview; (_converse$chatboxview = shared_converse.chatboxviews.get(this.getAttribute('jid'))) === null || _converse$chatboxview === void 0 ? void 0 : _converse$chatboxview.emitFocused(ev); } emitBlurred(ev) { var _converse$chatboxview2; (_converse$chatboxview2 = shared_converse.chatboxviews.get(this.getAttribute('jid'))) === null || _converse$chatboxview2 === void 0 ? void 0 : _converse$chatboxview2.emitBlurred(ev); } onDrop(evt) { if (evt.dataTransfer.files.length == 0) { // There are no files to be dropped, so this isn’t a file // transfer operation. return; } evt.preventDefault(); this.model.sendFiles(evt.dataTransfer.files); } onDragOver(ev) { // eslint-disable-line class-methods-use-this ev.preventDefault(); } clearMessages(ev) { var _ev$preventDefault2; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault2 = ev.preventDefault) === null || _ev$preventDefault2 === void 0 ? void 0 : _ev$preventDefault2.call(ev); clearMessages(this.model); } async autocompleteInPicker(input, value) { await core_api.emojis.initialize(); const emoji_picker = this.querySelector('converse-emoji-picker'); if (emoji_picker) { emoji_picker.model.set({ 'ac_position': input.selectionStart, 'autocompleting': value, 'query': value }); const emoji_dropdown = this.querySelector('converse-emoji-dropdown'); emoji_dropdown === null || emoji_dropdown === void 0 ? void 0 : emoji_dropdown.showMenu(); } } } core_api.elements.define('converse-chat-bottom-panel', ChatBottomPanel); ;// CONCATENATED MODULE: ./src/shared/chat/baseview.js class BaseChatView extends CustomElement { static get properties() { return { jid: { type: String } }; } disconnectedCallback() { super.disconnectedCallback(); shared_converse.chatboxviews.remove(this.jid, this); } updated() { if (this.model && this.jid !== this.model.get('jid')) { this.stopListening(); shared_converse.chatboxviews.remove(this.model.get('jid'), this); delete this.model; this.requestUpdate(); this.initialize(); } } close(ev) { var _ev$preventDefault; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); return this.model.close(ev); } maybeFocus() { core_api.settings.get('auto_focus') && this.focus(); } focus() { const textarea_el = this.getElementsByClassName('chat-textarea')[0]; if (textarea_el && document.activeElement !== textarea_el) { textarea_el.focus(); } return this; } emitBlurred(ev) { if (this.contains(document.activeElement) || this.contains(ev.relatedTarget)) { // Something else in this chatbox is still focused return; } /** * Triggered when the focus has been removed from a particular chat. * @event _converse#chatBoxBlurred * @type { _converse.ChatBoxView | _converse.ChatRoomView } * @example _converse.api.listen.on('chatBoxBlurred', (view, event) => { ... }); */ core_api.trigger('chatBoxBlurred', this, ev); } emitFocused(ev) { if (this.contains(ev.relatedTarget)) { // Something else in this chatbox was already focused return; } /** * Triggered when the focus has been moved to a particular chat. * @event _converse#chatBoxFocused * @type { _converse.ChatBoxView | _converse.ChatRoomView } * @example _converse.api.listen.on('chatBoxFocused', (view, event) => { ... }); */ core_api.trigger('chatBoxFocused', this, ev); } getBottomPanel() { if (this.model.get('type') === shared_converse.CHATROOMS_TYPE) { return this.querySelector('converse-muc-bottom-panel'); } else { return this.querySelector('converse-chat-bottom-panel'); } } getMessageForm() { if (this.model.get('type') === shared_converse.CHATROOMS_TYPE) { return this.querySelector('converse-muc-message-form'); } else { return this.querySelector('converse-message-form'); } } /** * Scrolls the chat down. * * This method will always scroll the chat down, regardless of * whether the user scrolled up manually or not. * @param { Event } [ev] - An optional event that is the cause for needing to scroll down. */ scrollDown(ev) { var _ev$preventDefault2, _ev$stopPropagation; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault2 = ev.preventDefault) === null || _ev$preventDefault2 === void 0 ? void 0 : _ev$preventDefault2.call(ev); ev === null || ev === void 0 ? void 0 : (_ev$stopPropagation = ev.stopPropagation) === null || _ev$stopPropagation === void 0 ? void 0 : _ev$stopPropagation.call(ev); if (this.model.ui.get('scrolled')) { this.model.ui.set({ 'scrolled': false }); } onScrolledDown(this.model); } onWindowStateChanged(data) { if (data.state === 'visible') { if (!this.model.isHidden()) { this.model.clearUnreadMsgCounter(); } } else if (data.state === 'hidden') { this.model.setChatState(shared_converse.INACTIVE, { 'silent': true }); this.model.sendChatState(); } } } ;// CONCATENATED MODULE: ./src/plugins/chatview/templates/chat.js /* harmony default export */ const chat = (o => $`
    ${o.model ? $`
    ${o.show_help_messages ? $`
    ` : ''}
    ` : ''}
    `); ;// CONCATENATED MODULE: ./src/plugins/chatview/chat.js function chat_defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } /** * The view of an open/ongoing chat conversation. * @class * @namespace _converse.ChatBoxView * @memberOf _converse */ class ChatView extends BaseChatView { constructor() { super(...arguments); chat_defineProperty(this, "length", 200); } async initialize() { shared_converse.chatboxviews.add(this.jid, this); this.model = shared_converse.chatboxes.get(this.jid); this.listenTo(shared_converse, 'windowStateChanged', this.onWindowStateChanged); this.listenTo(this.model, 'change:hidden', () => !this.model.get('hidden') && this.afterShown()); this.listenTo(this.model, 'change:show_help_messages', this.requestUpdate); await this.model.messages.fetched; !this.model.get('hidden') && this.afterShown(); /** * Triggered once the {@link _converse.ChatBoxView} has been initialized * @event _converse#chatBoxViewInitialized * @type { _converse.HeadlinesBoxView } * @example _converse.api.listen.on('chatBoxViewInitialized', view => { ... }); */ core_api.trigger('chatBoxViewInitialized', this); } render() { return chat(Object.assign({ 'model': this.model, 'help_messages': this.getHelpMessages(), 'show_help_messages': this.model.get('show_help_messages') }, this.model.toJSON())); } getHelpMessages() { // eslint-disable-line class-methods-use-this return [`/clear: ${__('Remove messages')}`, `/close: ${__('Close this chat')}`, `/me: ${__('Write in the third person')}`, `/help: ${__('Show this menu')}`]; } showControlBox() { var _converse$chatboxview; // eslint-disable-line class-methods-use-this // Used in mobile view, to navigate back to the controlbox (_converse$chatboxview = shared_converse.chatboxviews.get('controlbox')) === null || _converse$chatboxview === void 0 ? void 0 : _converse$chatboxview.show(); } afterShown() { this.model.setChatState(shared_converse.ACTIVE); this.model.clearUnreadMsgCounter(); this.maybeFocus(); } } core_api.elements.define('converse-chat', ChatView); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/chatview/styles/index.scss var chatview_styles = __webpack_require__(5599); ;// CONCATENATED MODULE: ./src/plugins/chatview/styles/index.scss var styles_options = {}; styles_options.styleTagTransform = (styleTagTransform_default()); styles_options.setAttributes = (setAttributesWithoutAttributes_default()); styles_options.insert = insertBySelector_default().bind(null, "head"); styles_options.domAPI = (styleDomAPI_default()); styles_options.insertStyleElement = (insertStyleElement_default()); var styles_update = injectStylesIntoStyleTag_default()(chatview_styles/* default */.Z, styles_options); /* harmony default export */ const plugins_chatview_styles = (chatview_styles/* default */.Z && chatview_styles/* default.locals */.Z.locals ? chatview_styles/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/chatview/index.js /** * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ const { Strophe: chatview_Strophe } = core_converse.env; core_converse.plugins.add('converse-chatview', { /* Plugin dependencies are other plugins which might be * overridden or relied upon, and therefore need to be loaded before * this plugin. * * If the setting "strict_plugin_dependencies" is set to true, * an error will be raised if the plugin is not found. By default it's * false, which means these plugins are only loaded opportunistically. * * NB: These plugins need to have already been loaded via require.js. */ dependencies: ['converse-chatboxviews', 'converse-chat', 'converse-disco', 'converse-modal'], initialize() { /* The initialize function gets called as soon as the plugin is * loaded by converse.js's plugin machinery. */ core_api.settings.extend({ 'allowed_audio_domains': null, 'allowed_image_domains': null, 'allowed_video_domains': null, 'auto_focus': true, 'debounced_content_rendering': true, 'filter_url_query_params': null, 'image_urls_regex': null, 'message_limit': 0, 'muc_hats': ['xep317'], 'render_media': true, 'show_message_avatar': true, 'show_retraction_warning': true, 'show_send_button': true, 'show_toolbar': true, 'time_format': 'HH:mm', 'use_system_emojis': true, 'visible_toolbar_buttons': { 'call': false, 'clear': true, 'emoji': true, 'spoiler': true } }); shared_converse.ChatBoxView = ChatView; core_api.listen.on('connected', () => core_api.disco.own.features.add(chatview_Strophe.NS.SPOILER)); core_api.listen.on('chatBoxClosed', model => clearHistory(model.get('jid'))); } }); ;// CONCATENATED MODULE: ./src/shared/components/brand-byline.js class ConverseBrandByline extends CustomElement { render() { // eslint-disable-line class-methods-use-this const is_fullscreen = core_api.settings.get('view_mode') === 'fullscreen'; return $` ${is_fullscreen ? $`

    ${shared_converse.VERSION_NAME}

    Open Source XMPP chat client brought to you by Opkode

    Translate it into your own language

    ` : ''} `; } } core_api.elements.define('converse-brand-byline', ConverseBrandByline); ;// CONCATENATED MODULE: ./src/shared/components/brand-logo.js class ConverseBrandLogo extends CustomElement { render() { // eslint-disable-line class-methods-use-this const is_fullscreen = core_api.settings.get('view_mode') === 'fullscreen'; return $` converse.js ${is_fullscreen ? $` ` : ''} `; } } core_api.elements.define('converse-brand-logo', ConverseBrandLogo); ;// CONCATENATED MODULE: ./node_modules/lit/html.js //# sourceMappingURL=html.js.map ;// CONCATENATED MODULE: ./src/shared/components/brand-heading.js class ConverseBrandHeading extends CustomElement { render() { // eslint-disable-line class-methods-use-this return $` `; } } core_api.elements.define('converse-brand-heading', ConverseBrandHeading); ;// CONCATENATED MODULE: ./src/plugins/controlbox/constants.js const { Strophe: constants_Strophe } = core_converse.env; const REPORTABLE_STATUSES = [constants_Strophe.Status.ERROR, constants_Strophe.Status.CONNECTING, constants_Strophe.Status.CONNFAIL, constants_Strophe.Status.AUTHENTICATING, constants_Strophe.Status.AUTHFAIL, constants_Strophe.Status.DISCONNECTING, constants_Strophe.Status.RECONNECTING]; const PRETTY_CONNECTION_STATUS = Object.fromEntries([[constants_Strophe.Status.ERROR, 'Error'], [constants_Strophe.Status.CONNECTING, 'Connecting'], [constants_Strophe.Status.CONNFAIL, 'Connection failure'], [constants_Strophe.Status.AUTHENTICATING, 'Authenticating'], [constants_Strophe.Status.AUTHFAIL, 'Authentication failure'], [constants_Strophe.Status.CONNECTED, 'Connected'], [constants_Strophe.Status.DISCONNECTED, 'Disconnected'], [constants_Strophe.Status.DISCONNECTING, 'Disconnecting'], [constants_Strophe.Status.ATTACHED, 'Attached'], [constants_Strophe.Status.REDIRECT, 'Redirect'], [constants_Strophe.Status.CONNTIMEOUT, 'Connection timeout'], [constants_Strophe.Status.RECONNECTING, 'Reconnecting']]); const CONNECTION_STATUS_CSS_CLASS = Object.fromEntries([[constants_Strophe.Status.ERROR, 'error'], [constants_Strophe.Status.CONNECTING, 'info'], [constants_Strophe.Status.CONNFAIL, 'error'], [constants_Strophe.Status.AUTHENTICATING, 'info'], [constants_Strophe.Status.AUTHFAIL, 'error'], [constants_Strophe.Status.CONNECTED, 'info'], [constants_Strophe.Status.DISCONNECTED, 'error'], [constants_Strophe.Status.DISCONNECTING, 'warn'], [constants_Strophe.Status.ATTACHED, 'info'], [constants_Strophe.Status.REDIRECT, 'info'], [constants_Strophe.Status.RECONNECTING, 'warn']]); ;// CONCATENATED MODULE: ./src/plugins/controlbox/templates/loginform.js const trust_checkbox = checked => { const i18n_hint_trusted = __('To improve performance, we cache your data in this browser. ' + 'Uncheck this box if this is a public computer or if you want your data to be deleted when you log out. ' + 'It\'s important that you explicitly log out, otherwise not all cached data might be deleted. ' + 'Please note, when using an untrusted device, OMEMO encryption is NOT available.'); const i18n_trusted = __('This is a trusted device'); return $` `; }; const connection_url_input = () => { const i18n_connection_url = __('Connection URL'); const i18n_form_help = __('HTTP or websocket URL that is used to connect to your XMPP server'); const i18n_placeholder = __('e.g. wss://example.org/xmpp-websocket'); return $`

    ${i18n_form_help}

    `; }; const password_input = () => { const i18n_password = __('Password'); return $`
    `; }; const register_link = () => { const i18n_create_account = __("Create an account"); const i18n_hint_no_account = __("Don't have a chat account?"); return $`

    ${i18n_hint_no_account}

    `; }; const show_register_link = () => { return core_api.settings.get('allow_registration') && !core_api.settings.get("auto_login") && shared_converse.pluggable.plugins["converse-register"].enabled(shared_converse); }; const auth_fields = el => { const authentication = core_api.settings.get('authentication'); const i18n_login = __('Log in'); const i18n_xmpp_address = __("XMPP Address"); const locked_domain = core_api.settings.get('locked_domain'); const default_domain = core_api.settings.get('default_domain'); const placeholder_username = (locked_domain || default_domain) && __('Username') || __('user@domain'); const show_trust_checkbox = core_api.settings.get('allow_user_trust_override'); return $`
    ${authentication !== shared_converse.EXTERNAL ? password_input() : ''} ${core_api.settings.get('show_connection_url_input') ? connection_url_input() : ''} ${show_trust_checkbox ? trust_checkbox(show_trust_checkbox === 'off' ? false : true) : ''}
    ${show_register_link() ? register_link() : ''} `; }; const form_fields = el => { const authentication = core_api.settings.get('authentication'); const { ANONYMOUS, EXTERNAL, LOGIN, PREBIND } = shared_converse; const i18n_disconnected = __('Disconnected'); const i18n_anon_login = __('Click here to log in anonymously'); return $` ${authentication == LOGIN || authentication == EXTERNAL ? auth_fields(el) : ''} ${authentication == ANONYMOUS ? $`` : ''} ${authentication == PREBIND ? $`

    ${i18n_disconnected}

    ` : ''} `; }; /* harmony default export */ const loginform = (el => { const connection_status = shared_converse.connfeedback.get('connection_status'); let feedback_class, pretty_status; if (REPORTABLE_STATUSES.includes(connection_status)) { pretty_status = PRETTY_CONNECTION_STATUS[connection_status]; feedback_class = CONNECTION_STATUS_CSS_CLASS[connection_status]; } const conn_feedback_message = shared_converse.connfeedback.get('message'); return $`
    ${shared_converse.CONNECTION_STATUS[connection_status] === 'CONNECTING' ? spinner({ 'classes': 'hor_centered' }) : form_fields(el)}
    `; }); ;// CONCATENATED MODULE: ./src/plugins/controlbox/utils.js const { Strophe: controlbox_utils_Strophe, u: controlbox_utils_u } = core_converse.env; function addControlBox() { var _converse$chatboxview; const m = shared_converse.chatboxes.add(new shared_converse.ControlBox({ 'id': 'controlbox' })); (_converse$chatboxview = shared_converse.chatboxviews.get('controlbox')) === null || _converse$chatboxview === void 0 ? void 0 : _converse$chatboxview.setModel(); return m; } function showControlBox(ev) { var _ev$preventDefault; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); const controlbox = shared_converse.chatboxes.get('controlbox') || addControlBox(); controlbox_utils_u.safeSave(controlbox, { 'closed': false }); } function navigateToControlBox(jid) { showControlBox(); const model = shared_converse.chatboxes.get(jid); controlbox_utils_u.safeSave(model, { 'hidden': true }); } function disconnect() { /* 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. */ const view = shared_converse.chatboxviews.get('controlbox'); view.model.set({ 'connected': false }); return view; } function controlbox_utils_clearSession() { const chatboxviews = shared_converse === null || shared_converse === void 0 ? void 0 : shared_converse.chatboxviews; const view = chatboxviews && chatboxviews.get('controlbox'); if (view) { controlbox_utils_u.safeSave(view.model, { 'connected': false }); if (view !== null && view !== void 0 && view.controlbox_pane) { view.controlbox_pane.remove(); delete view.controlbox_pane; } } } function onChatBoxesFetched() { const controlbox = shared_converse.chatboxes.get('controlbox') || addControlBox(); controlbox.save({ 'connected': true }); } /** * Given the login `
    ` element, parse its data and update the * converse settings with the supplied JID, password and connection URL. * @param { HTMLElement } form * @param { Object } settings - Extra settings that may be passed in and will * also be set together with the form settings. */ function updateSettingsWithFormData(form) { let settings = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; const form_data = new FormData(form); const connection_url = form_data.get('connection-url'); if (connection_url !== null && connection_url !== void 0 && connection_url.startsWith('ws')) { settings['websocket_url'] = connection_url; } else if (connection_url !== null && connection_url !== void 0 && connection_url.startsWith('http')) { settings['bosh_service_url'] = connection_url; } let jid = form_data.get('jid'); if (core_api.settings.get('locked_domain')) { const last_part = '@' + core_api.settings.get('locked_domain'); if (jid.endsWith(last_part)) { jid = jid.substr(0, jid.length - last_part.length); } jid = controlbox_utils_Strophe.escapeNode(jid) + last_part; } else if (core_api.settings.get('default_domain') && !jid.includes('@')) { jid = jid + '@' + core_api.settings.get('default_domain'); } settings['jid'] = jid; settings['password'] = form_data.get('password'); core_api.settings.set(settings); shared_converse.config.save({ 'trusted': form_data.get('trusted') && true || false }); } function validateJID(form) { const jid_element = form.querySelector('input[name=jid]'); if (jid_element.value && !core_api.settings.get('locked_domain') && !core_api.settings.get('default_domain') && !controlbox_utils_u.isValidJID(jid_element.value)) { jid_element.setCustomValidity(__('Please enter a valid XMPP address')); return false; } jid_element.setCustomValidity(''); return true; } ;// CONCATENATED MODULE: ./src/plugins/controlbox/loginform.js const { Strophe: loginform_Strophe, u: loginform_u } = core_converse.env; class LoginForm extends CustomElement { initialize() { this.listenTo(shared_converse.connfeedback, 'change', () => this.requestUpdate()); this.handler = () => this.requestUpdate(); } connectedCallback() { super.connectedCallback(); core_api.settings.listen.on('change', this.handler); } disconnectedCallback() { super.disconnectedCallback(); core_api.settings.listen.not('change', this.handler); } render() { return loginform(this); } firstUpdated() { this.initPopovers(); } async onLoginFormSubmitted(ev) { ev === null || ev === void 0 ? void 0 : ev.preventDefault(); if (core_api.settings.get('authentication') === shared_converse.ANONYMOUS) { return this.connect(shared_converse.jid); } if (!validateJID(ev.target)) { return; } updateSettingsWithFormData(ev.target); if (!core_api.settings.get('bosh_service_url') && !core_api.settings.get('websocket_url')) { // We don't have a connection URL available, so we try here to discover // XEP-0156 connection methods now, and if not found we present the user // with the option to enter their own connection URL await this.discoverConnectionMethods(ev); } if (core_api.settings.get('bosh_service_url') || core_api.settings.get('websocket_url')) { // FIXME: The connection class will still try to discover XEP-0156 connection methods this.connect(); } else { core_api.settings.set('show_connection_url_input', true); } } // eslint-disable-next-line class-methods-use-this async discoverConnectionMethods(ev) { var _converse$connection; if (!core_api.settings.get("discover_connection_methods")) { return; } const form_data = new FormData(ev.target); const jid = form_data.get('jid'); const domain = loginform_Strophe.getDomainFromJid(jid); if (!((_converse$connection = shared_converse.connection) !== null && _converse$connection !== void 0 && _converse$connection.jid) || jid && !loginform_u.isSameDomain(shared_converse.connection.jid, jid)) { await shared_converse.initConnection(); } return shared_converse.connection.discoverConnectionMethods(domain); } initPopovers() { Array.from(this.querySelectorAll('[data-title]')).forEach(el => { new (bootstrap_native_default()).Popover(el, { 'trigger': core_api.settings.get('view_mode') === 'mobile' && 'click' || 'hover', 'dismissible': core_api.settings.get('view_mode') === 'mobile' && true || false, 'container': this.parentElement.parentElement.parentElement }); }); } // eslint-disable-next-line class-methods-use-this connect(jid) { var _converse$connection2; if (['converse/login', 'converse/register'].includes(shared_converse.router.history.getFragment())) { shared_converse.router.navigate('', { 'replace': true }); } (_converse$connection2 = shared_converse.connection) === null || _converse$connection2 === void 0 ? void 0 : _converse$connection2.reset(); core_api.user.login(jid); } } core_api.elements.define('converse-login-form', LoginForm); ;// CONCATENATED MODULE: ./src/plugins/controlbox/templates/navback.js /* harmony default export */ const navback = (jid => { return $` navigateToControlBox(jid)}>`; }); ;// CONCATENATED MODULE: ./src/plugins/controlbox/navback.js class ControlBoxNavback extends CustomElement { static get properties() { return { 'jid': { type: String } }; } render() { return navback(this.jid); } } core_api.elements.define('converse-controlbox-navback', ControlBoxNavback); /* harmony default export */ const controlbox_navback = ((/* unused pure expression or super */ null && (ControlBoxNavback))); ;// CONCATENATED MODULE: ./src/plugins/controlbox/model.js const { dayjs: model_dayjs } = core_converse.env; /** * The ControlBox is the section of the chat that contains the open groupchats, * bookmarks and roster. * * In `overlayed` `view_mode` it's a box like the chat boxes, in `fullscreen` * `view_mode` it's a left-aligned sidebar. * @mixin */ const ControlBox = Model.extend({ defaults() { return { 'bookmarked': false, 'box_id': 'controlbox', 'chat_state': undefined, 'closed': !core_api.settings.get('show_controlbox_by_default'), 'num_unread': 0, 'time_opened': model_dayjs(0).valueOf(), 'type': shared_converse.CONTROLBOX_TYPE, 'url': '' }; }, validate(attrs) { if (attrs.type === shared_converse.CONTROLBOX_TYPE) { if (core_api.settings.get('view_mode') === 'embedded' && core_api.settings.get('singleton')) { return 'Controlbox not relevant in embedded view mode'; } return; } return shared_converse.ChatBox.prototype.validate.call(this, attrs); }, maybeShow(force) { if (!force && this.get('id') === 'controlbox') { // Must return the chatbox return this; } return shared_converse.ChatBox.prototype.maybeShow.call(this, force); }, onReconnection() { this.save('connected', true); } }); /* harmony default export */ const controlbox_model = (ControlBox); ;// CONCATENATED MODULE: ./src/plugins/controlbox/templates/toggle.js /* harmony default export */ const toggle = (o => { const i18n_toggle = core_api.connection.connected() ? __('Chat Contacts') : __('Toggle chat'); return $`${i18n_toggle}`; }); ;// CONCATENATED MODULE: ./src/plugins/controlbox/toggle.js class ControlBoxToggle extends CustomElement { async connectedCallback() { super.connectedCallback(); await core_api.waitUntil('initialized'); this.model = shared_converse.chatboxes.get('controlbox'); this.listenTo(this.model, 'change:closed', () => this.requestUpdate()); this.requestUpdate(); } render() { var _this$model; return toggle({ 'onClick': showControlBox, 'hide': !((_this$model = this.model) !== null && _this$model !== void 0 && _this$model.get('closed')) }); } } core_api.elements.define('converse-controlbox-toggle', ControlBoxToggle); /* harmony default export */ const controlbox_toggle = (ControlBoxToggle); ;// CONCATENATED MODULE: ./src/plugins/controlbox/templates/controlbox.js const { Strophe: controlbox_Strophe } = core_converse.env; function whenNotConnected(o) { const connection_status = shared_converse.connfeedback.get('connection_status'); console.log("connection_status"); console.log(connection_status); if ([controlbox_Strophe.Status.RECONNECTING, controlbox_Strophe.Status.CONNECTING].includes(connection_status)) { return spinner(); } if (o['active-form'] === 'register') { return $``; } return $`}`; } /* harmony default export */ const controlbox = (el => { const o = el.model.toJSON(); const sticky_controlbox = core_api.settings.get('sticky_controlbox'); return $`
    ${sticky_controlbox ? '' : $` el.close(ev)}> `}
    ${o.connected ? $`
    }` : whenNotConnected(o)}
    `; }); ;// CONCATENATED MODULE: ./src/plugins/controlbox/controlbox.js const controlbox_u = core_converse.env.utils; /** * The ControlBox is the section of the chat that contains the open groupchats, * bookmarks and roster. * * In `overlayed` `view_mode` it's a box like the chat boxes, in `fullscreen` * `view_mode` it's a left-aligned sidebar. */ class controlbox_ControlBox extends CustomElement { initialize() { this.setModel(); shared_converse.chatboxviews.add('controlbox', this); if (this.model.get('connected') && this.model.get('closed') === undefined) { this.model.set('closed', !core_api.settings.get('show_controlbox_by_default')); } this.requestUpdate(); /** * Triggered when the _converse.ControlBoxView has been initialized and therefore * exists. The controlbox contains the login and register forms when the user is * logged out and a list of the user's contacts and group chats when logged in. * @event _converse#controlBoxInitialized * @type { _converse.ControlBoxView } * @example _converse.api.listen.on('controlBoxInitialized', view => { ... }); */ core_api.trigger('controlBoxInitialized', this); } setModel() { this.model = shared_converse.chatboxes.get('controlbox'); this.listenTo(shared_converse.connfeedback, 'change:connection_status', () => this.requestUpdate()); this.listenTo(this.model, 'change:active-form', () => this.requestUpdate()); this.listenTo(this.model, 'change:connected', () => this.requestUpdate()); this.listenTo(this.model, 'change:closed', () => !this.model.get('closed') && this.afterShown()); this.requestUpdate(); } render() { return this.model ? controlbox(this) : ''; } close(ev) { var _ev$preventDefault; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); if ((ev === null || ev === void 0 ? void 0 : ev.name) === 'closeAllChatBoxes' && (shared_converse.disconnection_cause !== shared_converse.LOGOUT || core_api.settings.get('show_controlbox_by_default'))) { return; } if (core_api.settings.get('sticky_controlbox')) { return; } controlbox_u.safeSave(this.model, { 'closed': true }); core_api.trigger('controlBoxClosed', this); return this; } afterShown() { /** * Triggered once the controlbox has been opened * @event _converse#controlBoxOpened * @type {_converse.ControlBox} */ core_api.trigger('controlBoxOpened', this); return this; } } core_api.elements.define('converse-controlbox', controlbox_ControlBox); /* harmony default export */ const controlbox_controlbox = (controlbox_ControlBox); ;// CONCATENATED MODULE: ./src/plugins/controlbox/api.js const { u: controlbox_api_u } = core_converse.env; /* harmony default export */ const controlbox_api = ({ /** * The "controlbox" namespace groups methods pertaining to the * controlbox view * * @namespace _converse.api.controlbox * @memberOf _converse.api */ controlbox: { /** * Opens the controlbox * @method _converse.api.controlbox.open * @returns { Promise<_converse.ControlBox> } */ async open() { await core_api.waitUntil('chatBoxesFetched'); const model = (await core_api.chatboxes.get('controlbox')) || core_api.chatboxes.create('controlbox', {}, shared_converse.Controlbox); controlbox_api_u.safeSave(model, { 'closed': false }); return model; }, /** * Returns the controlbox view. * @method _converse.api.controlbox.get * @returns { View } View representing the controlbox * @example const view = _converse.api.controlbox.get(); */ get() { return shared_converse.chatboxviews.get('controlbox'); } } }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/controlbox/styles/_controlbox.scss var _controlbox = __webpack_require__(1875); ;// CONCATENATED MODULE: ./src/plugins/controlbox/styles/_controlbox.scss var _controlbox_options = {}; _controlbox_options.styleTagTransform = (styleTagTransform_default()); _controlbox_options.setAttributes = (setAttributesWithoutAttributes_default()); _controlbox_options.insert = insertBySelector_default().bind(null, "head"); _controlbox_options.domAPI = (styleDomAPI_default()); _controlbox_options.insertStyleElement = (insertStyleElement_default()); var _controlbox_update = injectStylesIntoStyleTag_default()(_controlbox/* default */.Z, _controlbox_options); /* harmony default export */ const styles_controlbox = (_controlbox/* default */.Z && _controlbox/* default.locals */.Z.locals ? _controlbox/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/controlbox/index.js /** * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.plugins.add('converse-controlbox', { /* Plugin dependencies are other plugins which might be * overridden or relied upon, and therefore need to be loaded before * this plugin. * * If the setting "strict_plugin_dependencies" is set to true, * an error will be raised if the plugin is not found. By default it's * false, which means these plugins are only loaded opportunistically. * * NB: These plugins need to have already been loaded via require.js. */ dependencies: ['converse-modal', 'converse-chatboxes', 'converse-chat', 'converse-rosterview', 'converse-chatview'], enabled(_converse) { return !_converse.api.settings.get('singleton'); }, 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. ChatBoxes: { model(attrs, options) { if (attrs && attrs.id == 'controlbox') { return new controlbox_model(attrs, options); } else { return this.__super__.model.apply(this, arguments); } } } }, initialize() { core_api.settings.extend({ allow_logout: true, allow_user_trust_override: true, default_domain: undefined, locked_domain: undefined, show_connection_url_input: false, show_controlbox_by_default: false, sticky_controlbox: false }); core_api.promises.add('controlBoxInitialized'); Object.assign(core_api, controlbox_api); shared_converse.ControlBoxView = controlbox_controlbox; shared_converse.ControlBox = controlbox_model; shared_converse.ControlBoxToggle = controlbox_toggle; core_api.listen.on('chatBoxesFetched', onChatBoxesFetched); core_api.listen.on('clearSession', controlbox_utils_clearSession); core_api.listen.on('will-reconnect', disconnect); core_api.waitUntil('chatBoxViewsInitialized').then(addControlBox).catch(e => headless_log.fatal(e)); } }); ;// CONCATENATED MODULE: ./src/plugins/dragresize/utils.js const { u: dragresize_utils_u } = core_converse.env; function onStartVerticalResize(ev) { let trigger = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : true; if (!core_api.settings.get('allow_dragresize')) { return true; } ev.preventDefault(); // Record element attributes for mouseMove(). const flyout = dragresize_utils_u.ancestor(ev.target, '.box-flyout'); const style = window.getComputedStyle(flyout); const chatbox_el = flyout.parentElement; chatbox_el.height = parseInt(style.height.replace(/px$/, ''), 10); shared_converse.resizing = { 'chatbox': chatbox_el, 'direction': 'top' }; chatbox_el.prev_pageY = ev.pageY; if (trigger) { /** * Triggered once the user starts to vertically resize a {@link _converse.ChatBoxView} * @event _converse#startVerticalResize * @example _converse.api.listen.on('startVerticalResize', (view) => { ... }); */ core_api.trigger('startVerticalResize', chatbox_el); } } function onStartHorizontalResize(ev) { let trigger = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : true; if (!core_api.settings.get('allow_dragresize')) { return true; } ev.preventDefault(); const flyout = dragresize_utils_u.ancestor(ev.target, '.box-flyout'); const style = window.getComputedStyle(flyout); const chatbox_el = flyout.parentElement; chatbox_el.width = parseInt(style.width.replace(/px$/, ''), 10); shared_converse.resizing = { 'chatbox': chatbox_el, 'direction': 'left' }; chatbox_el.prev_pageX = ev.pageX; if (trigger) { /** * Triggered once the user starts to horizontally resize a {@link _converse.ChatBoxView} * @event _converse#startHorizontalResize * @example _converse.api.listen.on('startHorizontalResize', (view) => { ... }); */ core_api.trigger('startHorizontalResize', chatbox_el); } } function onStartDiagonalResize(ev) { onStartHorizontalResize(ev, false); onStartVerticalResize(ev, false); shared_converse.resizing.direction = 'topleft'; /** * Triggered once the user starts to diagonally resize a {@link _converse.ChatBoxView} * @event _converse#startDiagonalResize * @example _converse.api.listen.on('startDiagonalResize', (view) => { ... }); */ core_api.trigger('startDiagonalResize', this); } /** * Applies some resistance to `value` around the `default_value`. * If value is close enough to `default_value`, then it is returned, otherwise * `value` is returned. * @param { Integer } value * @param { Integer } default_value * @returns { Integer } */ function applyDragResistance(value, default_value) { if (value === undefined) { return undefined; } else if (default_value === undefined) { return value; } const resistance = 10; if (value !== default_value && Math.abs(value - default_value) < resistance) { return default_value; } return value; } function onMouseMove(ev) { if (!shared_converse.resizing || !core_api.settings.get('allow_dragresize')) { return true; } ev.preventDefault(); shared_converse.resizing.chatbox.resizeChatBox(ev); } function onMouseUp(ev) { if (!shared_converse.resizing || !core_api.settings.get('allow_dragresize')) { return true; } ev.preventDefault(); const height = applyDragResistance(shared_converse.resizing.chatbox.height, shared_converse.resizing.chatbox.model.get('default_height')); const width = applyDragResistance(shared_converse.resizing.chatbox.width, shared_converse.resizing.chatbox.model.get('default_width')); if (core_api.connection.connected()) { shared_converse.resizing.chatbox.model.save({ 'height': height }); shared_converse.resizing.chatbox.model.save({ 'width': width }); } else { shared_converse.resizing.chatbox.model.set({ 'height': height }); shared_converse.resizing.chatbox.model.set({ 'width': width }); } shared_converse.resizing = null; } ;// CONCATENATED MODULE: ./src/plugins/dragresize/templates/dragresize.js /* harmony default export */ const dragresize = (() => $`
    `); ;// CONCATENATED MODULE: ./src/plugins/dragresize/components/dragresize.js class ConverseDragResize extends CustomElement { render() { // eslint-disable-line class-methods-use-this return dragresize(); } } customElements.define('converse-dragresize', ConverseDragResize); ;// CONCATENATED MODULE: ./src/plugins/dragresize/mixin.js const DragResizableMixin = { initDragResize() { var _converse$connection; const view = this; const debouncedSetDimensions = lodash_es_debounce(() => view.setDimensions()); window.addEventListener('resize', view.debouncedSetDimensions); this.listenTo(this.model, 'destroy', () => window.removeEventListener('resize', debouncedSetDimensions)); // Determine and store the default box size. // We need this information for the drag-resizing feature. const flyout = this.querySelector('.box-flyout'); const style = window.getComputedStyle(flyout); if (this.model.get('height') === undefined) { const height = parseInt(style.height.replace(/px$/, ''), 10); const width = parseInt(style.width.replace(/px$/, ''), 10); this.model.set('height', height); this.model.set('default_height', height); this.model.set('width', width); this.model.set('default_width', width); } const min_width = style['min-width']; const min_height = style['min-height']; 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; if ((_converse$connection = shared_converse.connection) !== null && _converse$connection !== void 0 && _converse$connection.connected) { this.height = this.model.get('height'); this.width = this.model.get('width'); } return this; }, resizeChatBox(ev) { let diff; if (shared_converse.resizing.direction.indexOf('top') === 0) { diff = ev.pageY - this.prev_pageY; if (diff) { 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); } } if (shared_converse.resizing.direction.includes('left')) { diff = this.prev_pageX - ev.pageX; if (diff) { 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); } } }, 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')); }, setChatBoxHeight(height) { if (height) { height = applyDragResistance(height, this.model.get('default_height')) + 'px'; } else { height = ''; } const flyout_el = this.querySelector('.box-flyout'); if (flyout_el !== null) { flyout_el.style.height = height; } }, setChatBoxWidth(width) { if (width) { width = applyDragResistance(width, this.model.get('default_width')) + 'px'; } else { width = ''; } this.style.width = width; const flyout_el = this.querySelector('.box-flyout'); if (flyout_el !== null) { flyout_el.style.width = width; } }, adjustToViewport() { /* Event handler called when viewport gets resized. We remove * custom width/height from chat boxes. */ const viewport_width = Math.max(document.documentElement.clientWidth, window.innerWidth || 0); const 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); } } }; /* harmony default export */ const mixin = (DragResizableMixin); ;// CONCATENATED MODULE: ./src/plugins/dragresize/index.js /** * @module converse-dragresize * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.plugins.add('converse-dragresize', { /* Plugin dependencies are other plugins which might be * overridden or relied upon, and therefore need to be loaded before * this plugin. * * If the setting "strict_plugin_dependencies" is set to true, * an error will be raised if the plugin is not found. By default it's * false, which means these plugins are only loaded opportunistically. * * NB: These plugins need to have already been loaded via require.js. */ dependencies: ['converse-chatview', 'converse-headlines-view', 'converse-muc-views'], enabled(_converse) { return _converse.api.settings.get('view_mode') == 'overlayed'; }, 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. ChatBox: { initialize() { const result = this.__super__.initialize.apply(this, arguments); const height = this.get('height'); const width = this.get('width'); const save = this.get('id') === 'controlbox' ? a => this.set(a) : a => this.save(a); save({ 'height': applyDragResistance(height, this.get('default_height')), 'width': applyDragResistance(width, this.get('default_width')) }); return result; } } }, initialize() { /* The initialize function gets called as soon as the plugin is * loaded by converse.js's plugin machinery. */ core_api.settings.extend({ 'allow_dragresize': true }); Object.assign(shared_converse.ChatBoxView.prototype, mixin); Object.assign(shared_converse.ChatRoomView.prototype, mixin); if (shared_converse.ControlBoxView) { Object.assign(shared_converse.ControlBoxView.prototype, mixin); } /************************ BEGIN Event Handlers ************************/ function registerGlobalEventHandlers() { document.addEventListener('mousemove', onMouseMove); document.addEventListener('mouseup', onMouseUp); } function unregisterGlobalEventHandlers() { document.removeEventListener('mousemove', onMouseMove); document.removeEventListener('mouseup', onMouseUp); } core_api.listen.on('registeredGlobalEventHandlers', registerGlobalEventHandlers); core_api.listen.on('unregisteredGlobalEventHandlers', unregisterGlobalEventHandlers); core_api.listen.on('beforeShowingChatView', view => view.initDragResize().setDimensions()); } }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/singleton/singleton.scss var singleton = __webpack_require__(1833); ;// CONCATENATED MODULE: ./src/plugins/singleton/singleton.scss var singleton_options = {}; singleton_options.styleTagTransform = (styleTagTransform_default()); singleton_options.setAttributes = (setAttributesWithoutAttributes_default()); singleton_options.insert = insertBySelector_default().bind(null, "head"); singleton_options.domAPI = (styleDomAPI_default()); singleton_options.insertStyleElement = (insertStyleElement_default()); var singleton_update = injectStylesIntoStyleTag_default()(singleton/* default */.Z, singleton_options); /* harmony default export */ const singleton_singleton = (singleton/* default */.Z && singleton/* default.locals */.Z.locals ? singleton/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/singleton/index.js /** * @copyright JC Brand * @license Mozilla Public License (MPLv2) * @description A plugin which restricts Converse to only one chat. */ core_converse.plugins.add('converse-singleton', { enabled(_converse) { return _converse.api.settings.get("singleton"); }, initialize() { core_api.settings.extend({ 'allow_logout': false, // No point in logging out when we have auto_login as true. 'allow_muc_invitations': false, // Doesn't make sense to allow because only // roster contacts can be invited 'hide_muc_server': true }); const auto_join_rooms = core_api.settings.get('auto_join_rooms'); const auto_join_private_chats = core_api.settings.get('auto_join_private_chats'); if (!Array.isArray(auto_join_rooms) && !Array.isArray(auto_join_private_chats)) { throw new Error("converse-singleton: auto_join_rooms must be an Array"); } if (auto_join_rooms.length === 0 && auto_join_private_chats.length === 0) { throw new Error("If you set singleton set to true, you need " + "to specify auto_join_rooms or auto_join_private_chats"); } if (auto_join_rooms.length > 0 && auto_join_private_chats.length > 0) { throw new Error("It doesn't make sense to have singleton set to true and " + "auto_join_rooms or auto_join_private_chats set to more then one, " + "since only one chat room may be open at any time."); } } }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/fullscreen/styles/fullscreen.scss var fullscreen = __webpack_require__(2791); ;// CONCATENATED MODULE: ./src/plugins/fullscreen/styles/fullscreen.scss var fullscreen_options = {}; fullscreen_options.styleTagTransform = (styleTagTransform_default()); fullscreen_options.setAttributes = (setAttributesWithoutAttributes_default()); fullscreen_options.insert = insertBySelector_default().bind(null, "head"); fullscreen_options.domAPI = (styleDomAPI_default()); fullscreen_options.insertStyleElement = (insertStyleElement_default()); var fullscreen_update = injectStylesIntoStyleTag_default()(fullscreen/* default */.Z, fullscreen_options); /* harmony default export */ const styles_fullscreen = (fullscreen/* default */.Z && fullscreen/* default.locals */.Z.locals ? fullscreen/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/fullscreen/index.js /** * @module converse-fullscreen * @license Mozilla Public License (MPLv2) * @copyright 2022, the Converse.js contributors */ core_converse.plugins.add('converse-fullscreen', { enabled() { return isUniView(); }, initialize() { core_api.settings.extend({ chatview_avatar_height: 50, chatview_avatar_width: 50, hide_open_bookmarks: true, show_controlbox_by_default: true, sticky_controlbox: true }); } }); ;// CONCATENATED MODULE: ./src/plugins/headlines-view/templates/chat-head.js /* harmony default export */ const templates_chat_head = (o => { const tpl_standalone_btns = o => o.standalone_btns.reverse().map(b => until_c(b, '')); return $`
    ${!shared_converse.api.settings.get("singleton") ? $`` : ''}
    ${o.display_name}
    ${o.dropdown_btns.length ? $`` : ''} ${o.standalone_btns.length ? tpl_standalone_btns(o) : ''}
    ${o.status ? $`

    ${o.status}

    ` : ''} `; }); ;// CONCATENATED MODULE: ./src/plugins/headlines-view/heading.js class HeadlinesHeading extends ElementView { async connectedCallback() { super.connectedCallback(); this.model = shared_converse.chatboxes.get(this.getAttribute('jid')); await this.model.initialized; this.render(); } async render() { const tpl = await this.generateHeadingTemplate(); x(tpl, this); } async generateHeadingTemplate() { const heading_btns = await this.getHeadingButtons(); const standalone_btns = heading_btns.filter(b => b.standalone); const dropdown_btns = heading_btns.filter(b => !b.standalone); return templates_chat_head(Object.assign(this.model.toJSON(), { 'display_name': this.model.getDisplayName(), 'dropdown_btns': dropdown_btns.map(b => getHeadingDropdownItem(b)), 'standalone_btns': standalone_btns.map(b => getHeadingStandaloneButton(b)) })); } /** * Returns a list of objects which represent buttons for the headlines header. * @async * @emits _converse#getHeadingButtons * @method HeadlinesHeading#getHeadingButtons */ getHeadingButtons() { const buttons = []; if (!core_api.settings.get('singleton')) { buttons.push({ 'a_class': 'close-chatbox-button', 'handler': ev => this.close(ev), 'i18n_text': __('Close'), 'i18n_title': __('Close these announcements'), 'icon_class': 'fa-times', 'name': 'close', 'standalone': core_api.settings.get('view_mode') === 'overlayed' }); } return shared_converse.api.hook('getHeadingButtons', this, buttons); } close(ev) { ev.preventDefault(); this.model.close(); } } core_api.elements.define('converse-headlines-heading', HeadlinesHeading); ;// CONCATENATED MODULE: ./src/plugins/headlines-view/templates/headlines.js /* harmony default export */ const headlines = (model => $`
    ${model ? $`
    ` : ''}
    `); ;// CONCATENATED MODULE: ./src/plugins/headlines-view/view.js class HeadlinesView extends BaseChatView { async initialize() { shared_converse.chatboxviews.add(this.jid, this); this.model = shared_converse.chatboxes.get(this.jid); this.model.disable_mam = true; // Don't do MAM queries for this box this.listenTo(shared_converse, 'windowStateChanged', this.onWindowStateChanged); this.listenTo(this.model, 'change:hidden', () => this.afterShown()); this.listenTo(this.model, 'destroy', this.remove); this.listenTo(this.model.messages, 'add', this.requestUpdate); this.listenTo(this.model.messages, 'remove', this.requestUpdate); this.listenTo(this.model.messages, 'reset', this.requestUpdate); await this.model.messages.fetched; this.model.maybeShow(); /** * Triggered once the {@link _converse.HeadlinesBoxView} has been initialized * @event _converse#headlinesBoxViewInitialized * @type { _converse.HeadlinesBoxView } * @example _converse.api.listen.on('headlinesBoxViewInitialized', view => { ... }); */ core_api.trigger('headlinesBoxViewInitialized', this); } render() { return headlines(this.model); } async close(ev) { var _ev$preventDefault; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); if (shared_converse.router.history.getFragment() === 'converse/chat?jid=' + this.model.get('jid')) { shared_converse.router.navigate(''); } await this.model.close(ev); return this; } getNotifications() { // eslint-disable-line class-methods-use-this // Override method in ChatBox. We don't show notifications for // headlines boxes. return []; } afterShown() { this.model.clearUnreadMsgCounter(); } } core_api.elements.define('converse-headlines', HeadlinesView); ;// CONCATENATED MODULE: ./src/templates/headline_list.js const tpl_headline_box = o => $` `; /* harmony default export */ const headline_list = (o => $`
    ${o.headlineboxes.map(headlinebox => tpl_headline_box(Object.assign({ headlinebox }, o)))}
    `); ;// CONCATENATED MODULE: ./src/plugins/headlines-view/templates/panel.js /* harmony default export */ const panel = (o => $`
    ${o.heading_headline}
    ${headline_list(o)} `); ;// CONCATENATED MODULE: ./src/plugins/headlines-view/panel.js function panel_defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } /** * View which renders headlines section of the control box. * @class * @namespace _converse.HeadlinesPanel * @memberOf _converse */ class HeadlinesPanel extends ElementView { constructor() { super(...arguments); panel_defineProperty(this, "events", { 'click .open-headline': 'openHeadline' }); } initialize() { this.model = shared_converse.chatboxes; this.listenTo(this.model, 'add', this.renderIfHeadline); this.listenTo(this.model, 'remove', this.renderIfHeadline); this.listenTo(this.model, 'destroy', this.renderIfHeadline); this.render(); } toHTML() { return panel({ 'heading_headline': __('Announcements'), 'headlineboxes': this.model.filter(m => m.get('type') === shared_converse.HEADLINES_TYPE), 'open_title': __('Click to open this server message') }); } renderIfHeadline(model) { return model && model.get('type') === shared_converse.HEADLINES_TYPE && this.render(); } openHeadline(ev) { // eslint-disable-line class-methods-use-this ev.preventDefault(); const jid = ev.target.getAttribute('data-headline-jid'); const chat = shared_converse.chatboxes.get(jid); chat.maybeShow(true); } } core_api.elements.define('converse-headlines-panel', HeadlinesPanel); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/headlines-view/styles/headlines.scss var styles_headlines = __webpack_require__(5956); ;// CONCATENATED MODULE: ./src/plugins/headlines-view/styles/headlines.scss var headlines_options = {}; headlines_options.styleTagTransform = (styleTagTransform_default()); headlines_options.setAttributes = (setAttributesWithoutAttributes_default()); headlines_options.insert = insertBySelector_default().bind(null, "head"); headlines_options.domAPI = (styleDomAPI_default()); headlines_options.insertStyleElement = (insertStyleElement_default()); var headlines_update = injectStylesIntoStyleTag_default()(styles_headlines/* default */.Z, headlines_options); /* harmony default export */ const headlines_view_styles_headlines = (styles_headlines/* default */.Z && styles_headlines/* default.locals */.Z.locals ? styles_headlines/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/headlines-view/index.js /** * @module converse-headlines-view * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.plugins.add('converse-headlines-view', { /* Plugin dependencies are other plugins which might be * overridden or relied upon, and therefore need to be loaded before * this plugin. * * If the setting "strict_plugin_dependencies" is set to true, * an error will be raised if the plugin is not found. By default it's * false, which means these plugins are only loaded opportunistically. * * NB: These plugins need to have already been loaded by the bundler */ dependencies: ['converse-headlines', 'converse-chatview'], initialize() { shared_converse.HeadlinesPanel = HeadlinesPanel; } }); ;// CONCATENATED MODULE: ./src/plugins/mam-views/templates/placeholder.js /* harmony default export */ const placeholder = (el => { return el.model.get('fetching') ? spinner({ 'classes': 'hor_centered' }) : $`
    `; }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/mam-views/styles/placeholder.scss var styles_placeholder = __webpack_require__(9679); ;// CONCATENATED MODULE: ./src/plugins/mam-views/styles/placeholder.scss var placeholder_options = {}; placeholder_options.styleTagTransform = (styleTagTransform_default()); placeholder_options.setAttributes = (setAttributesWithoutAttributes_default()); placeholder_options.insert = insertBySelector_default().bind(null, "head"); placeholder_options.domAPI = (styleDomAPI_default()); placeholder_options.insertStyleElement = (insertStyleElement_default()); var placeholder_update = injectStylesIntoStyleTag_default()(styles_placeholder/* default */.Z, placeholder_options); /* harmony default export */ const mam_views_styles_placeholder = (styles_placeholder/* default */.Z && styles_placeholder/* default.locals */.Z.locals ? styles_placeholder/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/mam-views/placeholder.js class Placeholder extends CustomElement { static get properties() { return { 'model': { type: Object } }; } render() { return placeholder(this); } async fetchMissingMessages(ev) { var _ev$preventDefault; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); this.model.set('fetching', true); const options = { 'before': this.model.get('before'), 'start': this.model.get('start') }; await fetchArchivedMessages(this.model.collection.chatbox, options); this.model.destroy(); } } core_api.elements.define('converse-mam-placeholder', Placeholder); ;// CONCATENATED MODULE: ./src/plugins/mam-views/utils.js function getPlaceholderTemplate(message, tpl) { if (message instanceof MAMPlaceholderMessage) { return $``; } else { return tpl; } } async function fetchMessagesOnScrollUp(view) { if (view.model.ui.get('chat-content-spinner-top')) { return; } if (view.model.messages.length) { const is_groupchat = view.model.get('type') === shared_converse.CHATROOMS_TYPE; const oldest_message = view.model.getOldestMessage(); if (oldest_message) { const by_jid = is_groupchat ? view.model.get('jid') : shared_converse.bare_jid; const stanza_id = oldest_message && oldest_message.get(`stanza_id ${by_jid}`); view.model.ui.set('chat-content-spinner-top', true); try { if (stanza_id) { await fetchArchivedMessages(view.model, { 'before': stanza_id }); } else { await fetchArchivedMessages(view.model, { 'end': oldest_message.get('time') }); } } catch (e) { headless_log.error(e); view.model.ui.set('chat-content-spinner-top', false); return; } if (core_api.settings.get('allow_url_history_change')) { shared_converse.router.history.navigate(`#${oldest_message.get('msgid')}`); } setTimeout(() => view.model.ui.set('chat-content-spinner-top', false), 250); } } } ;// CONCATENATED MODULE: ./src/plugins/mam-views/index.js /** * @description UI code XEP-0313 Message Archive Management * @copyright 2021, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.plugins.add('converse-mam-views', { dependencies: ['converse-mam', 'converse-chatview', 'converse-muc-views'], initialize() { core_api.listen.on('chatBoxScrolledUp', fetchMessagesOnScrollUp); core_api.listen.on('getMessageTemplate', getPlaceholderTemplate); } }); ;// CONCATENATED MODULE: ./src/plugins/minimize/toggle.js const MinimizedChatsToggle = Model.extend({ defaults: { 'collapsed': false } }); /* harmony default export */ const minimize_toggle = (MinimizedChatsToggle); ;// CONCATENATED MODULE: ./src/plugins/minimize/templates/chats-panel.js /* harmony default export */ const chats_panel = (o => $``); ;// CONCATENATED MODULE: ./src/plugins/minimize/view.js class MinimizedChats extends CustomElement { async initialize() { this.model = shared_converse.chatboxes; await this.initToggle(); this.listenTo(this.minchats, 'change:collapsed', this.requestUpdate); this.listenTo(this.model, 'add', this.requestUpdate); this.listenTo(this.model, 'change:fullname', this.requestUpdate); this.listenTo(this.model, 'change:jid', this.requestUpdate); this.listenTo(this.model, 'change:minimized', this.requestUpdate); this.listenTo(this.model, 'change:name', this.requestUpdate); this.listenTo(this.model, 'change:num_unread', this.requestUpdate); this.listenTo(this.model, 'remove', this.requestUpdate); this.listenTo(shared_converse, 'connected', this.requestUpdate); this.listenTo(shared_converse, 'reconnected', this.requestUpdate); this.listenTo(shared_converse, 'disconnected', this.requestUpdate); } render() { const chats = this.model.where({ 'minimized': true }); const num_unread = chats.reduce((acc, chat) => acc + chat.get('num_unread'), 0); const num_minimized = chats.reduce((acc, chat) => acc + (chat.get('minimized') ? 1 : 0), 0); const collapsed = this.minchats.get('collapsed'); const data = { chats, num_unread, num_minimized, collapsed }; data.toggle = ev => this.toggle(ev); return chats_panel(data); } async initToggle() { const id = `converse.minchatstoggle-${shared_converse.bare_jid}`; this.minchats = new minimize_toggle({ id }); initStorage(this.minchats, id, 'session'); await new Promise(resolve => this.minchats.fetch({ 'success': resolve, 'error': resolve })); } toggle(ev) { ev === null || ev === void 0 ? void 0 : ev.preventDefault(); this.minchats.save({ 'collapsed': !this.minchats.get('collapsed') }); } } core_api.elements.define('converse-minimized-chats', MinimizedChats); ;// CONCATENATED MODULE: ./src/plugins/minimize/templates/trimmed_chat.js /* harmony default export */ const trimmed_chat = (o => { const i18n_tooltip = __('Click to restore this chat'); const close_color = o.type === 'chatroom' ? "var(--chatroom-head-color)" : "var(--chat-head-text-color)"; return $` `; }); ;// CONCATENATED MODULE: ./src/plugins/minimize/utils.js const { dayjs: minimize_utils_dayjs, u: minimize_utils_u } = core_converse.env; function initializeChat(chat) { chat.on('change:hidden', m => !m.get('hidden') && maximize(chat), chat); if (chat.get('id') === 'controlbox') { return; } chat.save({ 'minimized': chat.get('minimized') || false, 'time_minimized': chat.get('time_minimized') || minimize_utils_dayjs() }); } function getChatBoxWidth(view) { if (view.model.get('id') === 'controlbox') { // We return the width of the controlbox or its toggle, // depending on which is visible. if (minimize_utils_u.isVisible(view)) { return minimize_utils_u.getOuterWidth(view, true); } else { const toggle = document.querySelector('converse-controlbox-toggle'); return toggle ? minimize_utils_u.getOuterWidth(toggle, true) : 0; } } else if (!view.model.get('minimized') && minimize_utils_u.isVisible(view)) { return minimize_utils_u.getOuterWidth(view, true); } return 0; } function getShownChats() { return shared_converse.chatboxviews.filter(el => // The controlbox can take a while to close, // so we need to check its state. That's why we checked the 'closed' state. !el.model.get('minimized') && !el.model.get('closed') && minimize_utils_u.isVisible(el)); } function getMinimizedWidth() { const minimized_el = document.querySelector('converse-minimized-chats'); return shared_converse.chatboxes.pluck('minimized').includes(true) ? minimize_utils_u.getOuterWidth(minimized_el, true) : 0; } function getBoxesWidth(newchat) { const new_id = newchat ? newchat.model.get('id') : null; const newchat_width = newchat ? minimize_utils_u.getOuterWidth(newchat, true) : 0; return Object.values(shared_converse.chatboxviews.xget(new_id)).reduce((memo, view) => memo + getChatBoxWidth(view), newchat_width); } /** * 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. * @private * @method _converse.ChatBoxViews#trimChats * @param { _converse.ChatBoxView|_converse.ChatRoomView|_converse.ControlBoxView|_converse.HeadlinesBoxView } [newchat] */ function trimChats(newchat) { if (shared_converse.isTestEnv() || core_api.settings.get('no_trimming') || core_api.settings.get("view_mode") !== 'overlayed') { return; } const shown_chats = getShownChats(); if (shown_chats.length <= 1) { return; } const body_width = minimize_utils_u.getOuterWidth(document.querySelector('body'), true); if (getChatBoxWidth(shown_chats[0]) === body_width) { // 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; } const minimized_el = document.querySelector('converse-minimized-chats'); if (minimized_el) { while (getMinimizedWidth() + getBoxesWidth(newchat) > body_width) { const new_id = newchat ? newchat.model.get('id') : null; const oldest_chat = getOldestMaximizedChat([new_id]); if (oldest_chat) { const model = shared_converse.chatboxes.get(oldest_chat.get('id')); model === null || model === void 0 ? void 0 : model.save('hidden', true); minimize(oldest_chat); } else { break; } } } } function getOldestMaximizedChat(exclude_ids) { // Get oldest view (if its id is not excluded) exclude_ids.push('controlbox'); let i = 0; let model = shared_converse.chatboxes.sort().at(i); while (exclude_ids.includes(model.get('id')) || model.get('minimized') === true) { i++; model = shared_converse.chatboxes.at(i); if (!model) { return null; } } return model; } function addMinimizeButtonToChat(view, buttons) { const data = { 'a_class': 'toggle-chatbox-button', 'handler': ev => minimize(ev, view.model), 'i18n_text': __('Minimize'), 'i18n_title': __('Minimize this chat'), 'icon_class': "fa-minus", 'name': 'minimize', 'standalone': shared_converse.api.settings.get("view_mode") === 'overlayed' }; const names = buttons.map(t => t.name); const idx = names.indexOf('close'); return idx > -1 ? [...buttons.slice(0, idx), data, ...buttons.slice(idx)] : [data, ...buttons]; } function addMinimizeButtonToMUC(view, buttons) { const data = { 'a_class': 'toggle-chatbox-button', 'handler': ev => minimize(ev, view.model), 'i18n_text': __('Minimize'), 'i18n_title': __('Minimize this groupchat'), 'icon_class': "fa-minus", 'name': 'minimize', 'standalone': shared_converse.api.settings.get("view_mode") === 'overlayed' }; const names = buttons.map(t => t.name); const idx = names.indexOf('signout'); return idx > -1 ? [...buttons.slice(0, idx), data, ...buttons.slice(idx)] : [data, ...buttons]; } function maximize(ev, chatbox) { if (ev !== null && ev !== void 0 && ev.preventDefault) { ev.preventDefault(); } else { chatbox = ev; } minimize_utils_u.safeSave(chatbox, { 'hidden': false, 'minimized': false, 'time_opened': new Date().getTime() }); } function minimize(ev, model) { if (ev !== null && ev !== void 0 && ev.preventDefault) { ev.preventDefault(); } else { model = ev; } model.setChatState(shared_converse.INACTIVE); minimize_utils_u.safeSave(model, { 'hidden': true, 'minimized': true, 'time_minimized': new Date().toISOString() }); } /** * Handler which gets called when a {@link _converse#ChatBox} has it's * `minimized` property set to false. * * Will trigger {@link _converse#chatBoxMaximized} * @returns {_converse.ChatBoxView|_converse.ChatRoomView} */ function onMaximized(model) { if (!model.isScrolledUp()) { model.clearUnreadMsgCounter(); } model.setChatState(shared_converse.ACTIVE); /** * Triggered when a previously minimized chat gets maximized * @event _converse#chatBoxMaximized * @type { _converse.ChatBoxView } * @example _converse.api.listen.on('chatBoxMaximized', view => { ... }); */ core_api.trigger('chatBoxMaximized', model); } /** * Handler which gets called when a {@link _converse#ChatBox} has it's * `minimized` property set to true. * * Will trigger {@link _converse#chatBoxMinimized} * @returns {_converse.ChatBoxView|_converse.ChatRoomView} */ function onMinimized(model) { /** * Triggered when a previously maximized chat gets Minimized * @event _converse#chatBoxMinimized * @type { _converse.ChatBoxView } * @example _converse.api.listen.on('chatBoxMinimized', view => { ... }); */ core_api.trigger('chatBoxMinimized', model); } function onMinimizedChanged(model) { if (model.get('minimized')) { onMinimized(model); } else { onMaximized(model); } } ;// CONCATENATED MODULE: ./src/plugins/minimize/components/minimized-chat.js class MinimizedChat extends CustomElement { static get properties() { return { model: { type: Object }, title: { type: String }, type: { type: String }, num_unread: { type: Number } }; } render() { const data = { 'close': ev => this.close(ev), 'num_unread': this.num_unread, 'restore': ev => this.restore(ev), 'title': this.title, 'type': this.type }; return trimmed_chat(data); } close(ev) { ev === null || ev === void 0 ? void 0 : ev.preventDefault(); this.model.close(); } restore(ev) { ev === null || ev === void 0 ? void 0 : ev.preventDefault(); maximize(this.model); } } core_api.elements.define('converse-minimized-chat', MinimizedChat); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/minimize/styles/minimize.scss var styles_minimize = __webpack_require__(4915); ;// CONCATENATED MODULE: ./src/plugins/minimize/styles/minimize.scss var minimize_options = {}; minimize_options.styleTagTransform = (styleTagTransform_default()); minimize_options.setAttributes = (setAttributesWithoutAttributes_default()); minimize_options.insert = insertBySelector_default().bind(null, "head"); minimize_options.domAPI = (styleDomAPI_default()); minimize_options.insertStyleElement = (insertStyleElement_default()); var minimize_update = injectStylesIntoStyleTag_default()(styles_minimize/* default */.Z, minimize_options); /* harmony default export */ const minimize_styles_minimize = (styles_minimize/* default */.Z && styles_minimize/* default.locals */.Z.locals ? styles_minimize/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/minimize/index.js /** * @module converse-minimize * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.plugins.add('converse-minimize', { /* 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. */ dependencies: ["converse-chatview", "converse-controlbox", "converse-muc-views", "converse-headlines-view", "converse-dragresize"], enabled(_converse) { return _converse.api.settings.get("view_mode") === 'overlayed'; }, 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: { maybeShow(force) { if (!force && this.get('minimized')) { // Must return the chatbox return this; } return this.__super__.maybeShow.apply(this, arguments); }, isHidden() { return this.__super__.isHidden.call(this) || this.get('minimized'); } }, ChatBoxView: { isNewMessageHidden() { return this.model.get('minimized') || this.__super__.isNewMessageHidden.apply(this, arguments); }, setChatBoxHeight(height) { if (!this.model.get('minimized')) { return this.__super__.setChatBoxHeight.call(this, height); } }, setChatBoxWidth(width) { if (!this.model.get('minimized')) { return this.__super__.setChatBoxWidth.call(this, width); } } } }, initialize() { core_api.settings.extend({ 'no_trimming': false }); core_api.promises.add('minimizedChatsInitialized'); shared_converse.MinimizedChatsToggle = minimize_toggle; shared_converse.minimize = { trimChats: trimChats, minimize: minimize, maximize: maximize }; function onChatInitialized(model) { initializeChat(model); model.on('change:minimized', () => onMinimizedChanged(model)); } core_api.listen.on('chatBoxViewInitialized', view => shared_converse.minimize.trimChats(view)); core_api.listen.on('chatRoomViewInitialized', view => shared_converse.minimize.trimChats(view)); core_api.listen.on('controlBoxOpened', view => shared_converse.minimize.trimChats(view)); core_api.listen.on('chatBoxInitialized', onChatInitialized); core_api.listen.on('chatRoomInitialized', onChatInitialized); core_api.listen.on('getHeadingButtons', (view, buttons) => { if (view.model.get('type') === shared_converse.CHATROOMS_TYPE) { return addMinimizeButtonToMUC(view, buttons); } else { return addMinimizeButtonToChat(view, buttons); } }); const debouncedTrimChats = lodash_es_debounce(() => shared_converse.minimize.trimChats(), 250); core_api.listen.on('registeredGlobalEventHandlers', () => window.addEventListener("resize", debouncedTrimChats)); core_api.listen.on('unregisteredGlobalEventHandlers', () => window.removeEventListener("resize", debouncedTrimChats)); } }); ;// CONCATENATED MODULE: ./src/shared/autocomplete/utils.js const autocomplete_utils_u = core_converse.env.utils; const utils_helpers = { getElement(expr, el) { return typeof expr === 'string' ? (el || document).querySelector(expr) : expr || null; }, bind(element, o) { if (element) { for (var event in o) { if (!Object.prototype.hasOwnProperty.call(o, event)) { continue; } const callback = o[event]; event.split(/\s+/).forEach(event => element.addEventListener(event, callback)); } } }, unbind(element, o) { if (element) { for (var event in o) { if (!Object.prototype.hasOwnProperty.call(o, event)) { continue; } const callback = o[event]; event.split(/\s+/).forEach(event => element.removeEventListener(event, callback)); } } }, regExpEscape(s) { return s.replace(/[-\\^$*+?.()|[\]{}]/g, '\\$&'); }, isMention(word, ac_triggers) { return ac_triggers.includes(word[0]) || autocomplete_utils_u.isMentionBoundary(word[0]) && ac_triggers.includes(word[1]); } }; const FILTER_CONTAINS = function (text, input) { return RegExp(utils_helpers.regExpEscape(input.trim()), 'i').test(text); }; const FILTER_STARTSWITH = function (text, input) { return RegExp('^' + utils_helpers.regExpEscape(input.trim()), 'i').test(text); }; const SORT_BY_LENGTH = function (a, b) { if (a.length !== b.length) { return a.length - b.length; } return a < b ? -1 : 1; }; const SORT_BY_QUERY_POSITION = function (a, b) { const query = a.query.toLowerCase(); const x = a.label.toLowerCase().indexOf(query); const y = b.label.toLowerCase().indexOf(query); if (x === y) { return SORT_BY_LENGTH(a, b); } return (x === -1 ? Infinity : x) < (y === -1 ? Infinity : y) ? -1 : 1; }; const ITEM = (text, input) => { input = input.trim(); const element = document.createElement('li'); element.setAttribute('aria-selected', 'false'); const regex = new RegExp('(' + input + ')', 'ig'); const parts = input ? text.split(regex) : [text]; parts.forEach(txt => { if (input && txt.match(regex)) { const match = document.createElement('mark'); match.textContent = txt; element.appendChild(match); } else { element.appendChild(document.createTextNode(txt)); } }); return element; }; ;// CONCATENATED MODULE: ./src/shared/autocomplete/suggestion.js /** * An autocomplete suggestion */ class Suggestion extends String { /** * @param { Any } data - The auto-complete data. Ideally an object e.g. { label, value }, * which specifies the value and human-presentable label of the suggestion. * @param { string } query - The query string being auto-completed */ constructor(data, query) { super(); const o = Array.isArray(data) ? { label: data[0], value: data[1] } : typeof data === 'object' && 'label' in data && 'value' in data ? data : { label: data, value: data }; this.label = o.label || o.value; this.value = o.value; this.query = query; } get lenth() { return this.label.length; } toString() { return '' + this.label; } valueOf() { return this.toString(); } } /* harmony default export */ const suggestion = (Suggestion); ;// CONCATENATED MODULE: ./src/shared/autocomplete/autocomplete.js /** * @copyright Lea Verou and the Converse.js contributors * @description * Started as a fork of Lea Verou's "Awesomplete" * https://leaverou.github.io/awesomplete/ * @license Mozilla Public License (MPLv2) */ const autocomplete_u = core_converse.env.utils; class AutoComplete { constructor(el) { let config = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; this.suggestions = []; this.is_opened = false; if (autocomplete_u.hasClass('suggestion-box', el)) { this.container = el; } else { this.container = el.querySelector('.suggestion-box'); } this.input = this.container.querySelector('.suggestion-box__input'); this.input.setAttribute("aria-autocomplete", "list"); this.ul = this.container.querySelector('.suggestion-box__results'); this.status = this.container.querySelector('.suggestion-box__additions'); Object.assign(this, { 'match_current_word': false, // Match only the current word, otherwise all input is matched 'ac_triggers': [], // Array of keys (`ev.key`) values that will trigger auto-complete 'include_triggers': [], // Array of trigger keys which should be included in the returned value 'min_chars': 2, 'max_items': 10, 'auto_evaluate': true, // Should evaluation happen automatically without any particular key as trigger? 'auto_first': false, // Should the first element be automatically selected? 'data': a => a, 'filter': FILTER_CONTAINS, 'sort': config.sort === false ? false : SORT_BY_QUERY_POSITION, 'item': ITEM }, config); this.index = -1; this.bindEvents(); if (this.input.hasAttribute("list")) { this.list = "#" + this.input.getAttribute("list"); this.input.removeAttribute("list"); } else { this.list = this.input.getAttribute("data-list") || config.list || []; } } bindEvents() { // Bind events const input = { "blur": () => this.close({ 'reason': 'blur' }) }; if (this.auto_evaluate) { input["input"] = () => this.evaluate(); } this._events = { 'input': input, 'form': { "submit": () => this.close({ 'reason': 'submit' }) }, 'ul': { "mousedown": ev => this.onMouseDown(ev), "mouseover": ev => this.onMouseOver(ev) } }; utils_helpers.bind(this.input, this._events.input); utils_helpers.bind(this.input.form, this._events.form); utils_helpers.bind(this.ul, this._events.ul); } set list(list) { if (Array.isArray(list) || typeof list === "function") { this._list = list; } else if (typeof list === "string" && list.includes(",")) { this._list = list.split(/\s*,\s*/); } else { var _helpers$getElement; // Element or CSS selector const children = ((_helpers$getElement = utils_helpers.getElement(list)) === null || _helpers$getElement === void 0 ? void 0 : _helpers$getElement.children) || []; this._list = Array.from(children).filter(el => !el.disabled).map(el => { const text = el.textContent.trim(); const value = el.value || text; const label = el.label || text; return value !== "" ? { label, value } : null; }).filter(i => i); } if (document.activeElement === this.input) { this.evaluate(); } } get list() { return this._list; } get selected() { return this.index > -1; } get opened() { return this.is_opened; } close(o) { if (!this.opened) { return; } this.ul.setAttribute("hidden", ""); this.is_opened = false; this.index = -1; this.trigger("suggestion-box-close", o || {}); } insertValue(suggestion) { if (this.match_current_word) { autocomplete_u.replaceCurrentWord(this.input, suggestion.value); } else { this.input.value = suggestion.value; } } open() { this.ul.removeAttribute("hidden"); this.is_opened = true; if (this.auto_first && this.index === -1) { this.goto(0); } this.trigger("suggestion-box-open"); } destroy() { //remove events from the input and its form utils_helpers.unbind(this.input, this._events.input); utils_helpers.unbind(this.input.form, this._events.form); this.input.removeAttribute("aria-autocomplete"); } next() { const count = this.ul.children.length; this.goto(this.index < count - 1 ? this.index + 1 : count ? 0 : -1); } previous() { const count = this.ul.children.length, pos = this.index - 1; this.goto(this.selected && pos !== -1 ? pos : count - 1); } goto(i) { // Should not be used directly, highlights specific item without any checks! const list = this.ul.children; if (this.selected) { list[this.index].setAttribute("aria-selected", "false"); } this.index = i; if (i > -1 && list.length > 0) { list[i].setAttribute("aria-selected", "true"); list[i].focus(); this.status.textContent = list[i].textContent; // scroll to highlighted element in case parent's height is fixed this.ul.scrollTop = list[i].offsetTop - this.ul.clientHeight + list[i].clientHeight; this.trigger("suggestion-box-highlight", { 'text': this.suggestions[this.index] }); } } select(selected) { if (selected) { this.index = autocomplete_u.siblingIndex(selected); } else { selected = this.ul.children[this.index]; } if (selected) { const suggestion = this.suggestions[this.index]; this.insertValue(suggestion); this.close({ 'reason': 'select' }); this.auto_completing = false; this.trigger("suggestion-box-selectcomplete", { 'text': suggestion }); } } onMouseOver(ev) { const li = autocomplete_u.ancestor(ev.target, 'li'); if (li) { this.goto(Array.prototype.slice.call(this.ul.children).indexOf(li)); } } onMouseDown(ev) { if (ev.button !== 0) { return; // Only select on left click } const li = autocomplete_u.ancestor(ev.target, 'li'); if (li) { ev.preventDefault(); this.select(li, ev.target); } } onKeyDown(ev) { if (this.opened) { if ([core_converse.keycodes.ENTER, core_converse.keycodes.TAB].includes(ev.keyCode) && this.selected) { ev.preventDefault(); ev.stopPropagation(); this.select(); return true; } else if (ev.keyCode === core_converse.keycodes.ESCAPE) { this.close({ 'reason': 'esc' }); return true; } else if ([core_converse.keycodes.UP_ARROW, core_converse.keycodes.DOWN_ARROW].includes(ev.keyCode)) { ev.preventDefault(); ev.stopPropagation(); this[ev.keyCode === core_converse.keycodes.UP_ARROW ? "previous" : "next"](); return true; } } if ([core_converse.keycodes.SHIFT, core_converse.keycodes.META, core_converse.keycodes.META_RIGHT, core_converse.keycodes.ESCAPE, core_converse.keycodes.ALT].includes(ev.keyCode)) { return; } if (this.ac_triggers.includes(ev.key)) { if (ev.key === "Tab") { ev.preventDefault(); } this.auto_completing = true; } else if (ev.key === "Backspace") { const word = autocomplete_u.getCurrentWord(ev.target, ev.target.selectionEnd - 1); if (utils_helpers.isMention(word, this.ac_triggers)) { this.auto_completing = true; } } } async evaluate(ev) { const selecting = this.selected && ev && (ev.keyCode === core_converse.keycodes.UP_ARROW || ev.keyCode === core_converse.keycodes.DOWN_ARROW); if (!this.auto_evaluate && !this.auto_completing || selecting) { return; } const list = typeof this._list === "function" ? await this._list() : this._list; if (list.length === 0) { return; } let value = this.match_current_word ? autocomplete_u.getCurrentWord(this.input) : this.input.value; const contains_trigger = utils_helpers.isMention(value, this.ac_triggers); if (contains_trigger) { this.auto_completing = true; if (!this.include_triggers.includes(ev.key)) { value = autocomplete_u.isMentionBoundary(value[0]) ? value.slice('2') : value.slice('1'); } } if ((contains_trigger || value.length) && value.length >= this.min_chars) { this.index = -1; // Populate list with options that match this.ul.innerHTML = ""; this.suggestions = list.map(item => new suggestion(this.data(item, value), value)).filter(item => this.filter(item, value)); if (this.sort !== false) { this.suggestions = this.suggestions.sort(this.sort); } this.suggestions = this.suggestions.slice(0, this.max_items); this.suggestions.forEach(text => this.ul.appendChild(this.item(text, value))); if (this.ul.children.length === 0) { this.close({ 'reason': 'nomatches' }); } else { this.open(); } } else { this.close({ 'reason': 'nomatches' }); if (!contains_trigger) { this.auto_completing = false; } } } } // Make it an event emitter Object.assign(AutoComplete.prototype, Events); /* harmony default export */ const autocomplete = (AutoComplete); ;// CONCATENATED MODULE: ./src/shared/autocomplete/component.js class AutoCompleteComponent extends CustomElement { static get properties() { return { 'getAutoCompleteList': { type: Function }, 'auto_evaluate': { type: Boolean }, 'auto_first': { type: Boolean }, // Should the first element be automatically selected? 'filter': { type: String }, 'include_triggers': { type: String }, 'min_chars': { type: Number }, 'name': { type: String }, 'placeholder': { type: String }, 'triggers': { type: String } }; } constructor() { super(); this.auto_evaluate = true; // Should evaluation happen automatically without any particular key as trigger? this.auto_first = false; // Should the first element be automatically selected? this.filter = 'contains'; this.include_triggers = ''; // Space separated chars which should be included in the returned value this.match_current_word = false; // Match only the current word, otherwise all input is matched this.max_items = 10; this.min_chars = 1; this.triggers = ''; // String of space separated chars } render() { return $`
    `; } firstUpdated() { this.auto_complete = new autocomplete(this.firstElementChild, { 'ac_triggers': this.triggers.split(' '), 'auto_evaluate': this.auto_evaluate, 'auto_first': this.auto_first, 'filter': this.filter == 'contains' ? FILTER_CONTAINS : FILTER_STARTSWITH, 'include_triggers': [], 'list': () => this.getAutoCompleteList(), 'match_current_word': true, 'max_items': this.max_items, 'min_chars': this.min_chars }); this.auto_complete.on('suggestion-box-selectcomplete', () => this.auto_completing = false); } onKeyDown(ev) { this.auto_complete.onKeyDown(ev); } onKeyUp(ev) { this.auto_complete.evaluate(ev); } } core_api.elements.define('converse-autocomplete', AutoCompleteComponent); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/shared/autocomplete/styles/_autocomplete.scss var _autocomplete = __webpack_require__(4921); ;// CONCATENATED MODULE: ./src/shared/autocomplete/styles/_autocomplete.scss var _autocomplete_options = {}; _autocomplete_options.styleTagTransform = (styleTagTransform_default()); _autocomplete_options.setAttributes = (setAttributesWithoutAttributes_default()); _autocomplete_options.insert = insertBySelector_default().bind(null, "head"); _autocomplete_options.domAPI = (styleDomAPI_default()); _autocomplete_options.insertStyleElement = (insertStyleElement_default()); var _autocomplete_update = injectStylesIntoStyleTag_default()(_autocomplete/* default */.Z, _autocomplete_options); /* harmony default export */ const styles_autocomplete = (_autocomplete/* default */.Z && _autocomplete/* default.locals */.Z.locals ? _autocomplete/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/shared/autocomplete/index.js shared_converse.FILTER_CONTAINS = FILTER_CONTAINS; shared_converse.FILTER_STARTSWITH = FILTER_STARTSWITH; shared_converse.AutoComplete = autocomplete; ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/ad-hoc-command-form.js /* harmony default export */ const ad_hoc_command_form = ((o, command) => { const i18n_hide = __('Hide'); const i18n_run = __('Execute'); return $` ${command.alert ? $`` : ''}

    ${command.instructions}

    ${command.fields}
    `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/ad-hoc-command.js /* harmony default export */ const ad_hoc_command = ((o, command) => $`
  • ${command.node === o.showform ? ad_hoc_command_form(o, command) : ''}
  • `); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/moderator-tools.js function getRoleHelpText(role) { if (role === 'moderator') { return __("Moderators are privileged users who can change the roles of other users (except those with admin or owner affiliations."); } else if (role === 'participant') { return __("The default role, implies that you can read and write messages."); } else if (role == 'visitor') { return __("Visitors aren't allowed to write messages in a moderated multi-user chat."); } } function getAffiliationHelpText(aff) { if (aff === 'owner') { return __("Owner is the highest affiliation. Owners can modify roles and affiliations of all other users."); } else if (aff === 'admin') { return __("Admin is the 2nd highest affiliation. Admins can modify roles and affiliations of all other users except owners."); } else if (aff === 'outcast') { return __("To ban a user, you give them the affiliation of \"outcast\"."); } } const role_option = o => $` `; const affiliation_option = o => $` `; const tpl_set_role_form = o => { const i18n_change_role = __('Change role'); const i18n_new_role = __('New Role'); const i18n_reason = __('Reason'); return $` `; }; const role_form_toggle = o => $` `; const role_list_item = o => $`
    • JID: ${o.item.jid}
    • Nickname: ${o.item.nick}
    • Role: ${o.item.role} ${o.assignable_roles.length ? role_form_toggle(o) : ''}
      ${o.assignable_roles.length ? tpl_set_role_form(o) : ''}
  • `; const tpl_set_affiliation_form = o => { const i18n_change_affiliation = __('Change affiliation'); const i18n_new_affiliation = __('New affiliation'); const i18n_reason = __('Reason'); return $` `; }; const affiliation_form_toggle = o => $` `; const affiliation_list_item = o => $`
    • JID: ${o.item.jid}
    • Nickname: ${o.item.nick}
    • Affiliation: ${o.item.affiliation} ${o.assignable_affiliations.length ? affiliation_form_toggle(o) : ''}
      ${o.assignable_affiliations.length ? tpl_set_affiliation_form(o) : ''}
  • `; const tpl_navigation = () => $` `; /* harmony default export */ const moderator_tools = (o => { const i18n_affiliation = __('Affiliation'); const i18n_no_users_with_aff = __('No users with that affiliation found.'); const i18n_no_users_with_role = __('No users with that role found.'); const i18n_filter = __('Type here to filter the search results'); const i18n_role = __('Role'); const i18n_show_users = __('Show users'); const i18n_helptext_role = __("Roles are assigned to users to grant or deny them certain abilities in a multi-user chat. " + "They're assigned either explicitly or implicitly as part of an affiliation. " + "A role that's not due to an affiliation, is only valid for the duration of the user's session."); const i18n_helptext_affiliation = __("An affiliation is a long-lived entitlement which typically implies a certain role and which " + "grants privileges and responsibilities. For example admins and owners automatically have the " + "moderator role."); const show_both_tabs = o.queryable_roles.length && o.queryable_affiliations.length; return $` ${o.alert_message ? $`` : ''} ${show_both_tabs ? tpl_navigation() : ''}
    ${o.queryable_affiliations.length ? $`

    ${i18n_helptext_affiliation}

    ${Array.isArray(o.users_with_affiliation) && o.users_with_affiliation.length > 5 ? $`` : ''}
    ${getAffiliationHelpText(o.affiliation) ? $`

    ${getAffiliationHelpText(o.affiliation)}

    ` : ''}
      ${o.loading_users_with_affiliation ? $`
    • ${spinner()}
    • ` : ''} ${Array.isArray(o.users_with_affiliation) && o.users_with_affiliation.length === 0 ? $`
    • ${i18n_no_users_with_aff}
    • ` : ''} ${o.users_with_affiliation instanceof Error ? $`
    • ${o.users_with_affiliation.message}
    • ` : (o.users_with_affiliation || []).map(item => (item.nick || item.jid).match(new RegExp(o.affiliations_filter, 'i')) ? affiliation_list_item(Object.assign({ item }, o)) : '')}
    ` : ''} ${o.queryable_roles.length ? $`

    ${i18n_helptext_role}

    ${Array.isArray(o.users_with_role) && o.users_with_role.length > 5 ? $`` : ''}
    ${getRoleHelpText(o.role) ? $`

    ${getRoleHelpText(o.role)}

    ` : ''}
      ${o.loading_users_with_role ? $`
    • ${spinner()}
    • ` : ''} ${o.users_with_role && o.users_with_role.length === 0 ? $`
    • ${i18n_no_users_with_role}
    • ` : ''} ${(o.users_with_role || []).map(item => item.nick.match(o.roles_filter) ? role_list_item(Object.assign({ item }, o)) : '')}
    ` : ''}
    `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/modtools.js const { Strophe: modtools_Strophe, sizzle: modtools_sizzle, u: modtools_u } = core_converse.env; class ModeratorTools extends CustomElement { static get properties() { return { affiliation: { type: String }, affiliations_filter: { type: String, attribute: false }, alert_message: { type: String, attribute: false }, alert_type: { type: String, attribute: false }, jid: { type: String }, muc: { type: Object, attribute: false }, role: { type: String }, roles_filter: { type: String, attribute: false }, users_with_affiliation: { type: Array, attribute: false }, users_with_role: { type: Array, attribute: false } }; } constructor() { super(); this.affiliation = ''; this.affiliations_filter = ''; this.role = ''; this.roles_filter = ''; } updated(changed) { changed.has('role') && this.onSearchRoleChange(); changed.has('affiliation') && this.onSearchAffiliationChange(); changed.has('jid') && changed.get('jid') && this.initialize(); } async initialize() { this.initialized = getOpenPromise(); const muc = await core_api.rooms.get(this.jid); await muc.initialized; this.muc = muc; this.initialized.resolve(); } render() { var _this$muc; if ((_this$muc = this.muc) !== null && _this$muc !== void 0 && _this$muc.occupants) { const occupant = this.muc.occupants.findWhere({ 'jid': shared_converse.bare_jid }); return moderator_tools({ 'affiliations_filter': this.affiliations_filter, 'alert_message': this.alert_message, 'alert_type': this.alert_type, 'assignAffiliation': ev => this.assignAffiliation(ev), 'assignRole': ev => this.assignRole(ev), 'assignable_affiliations': getAssignableAffiliations(occupant), 'assignable_roles': getAssignableRoles(occupant), 'filterAffiliationResults': ev => this.filterAffiliationResults(ev), 'filterRoleResults': ev => this.filterRoleResults(ev), 'loading_users_with_affiliation': this.loading_users_with_affiliation, 'queryAffiliation': ev => this.queryAffiliation(ev), 'queryRole': ev => this.queryRole(ev), 'queryable_affiliations': AFFILIATIONS.filter(a => !core_api.settings.get('modtools_disable_query').includes(a)), 'queryable_roles': ROLES.filter(a => !core_api.settings.get('modtools_disable_query').includes(a)), 'roles_filter': this.roles_filter, 'switchTab': ev => this.switchTab(ev), 'toggleForm': ev => this.toggleForm(ev), 'users_with_affiliation': this.users_with_affiliation, 'users_with_role': this.users_with_role }); } else { return ''; } } async onSearchAffiliationChange() { if (!this.affiliation) { return; } await this.initialized; this.clearAlert(); this.loading_users_with_affiliation = true; this.users_with_affiliation = null; if (this.shouldFetchAffiliationsList()) { const result = await getAffiliationList(this.affiliation, this.jid); if (result instanceof Error) { this.alert(result.message, 'danger'); this.users_with_affiliation = []; } else { this.users_with_affiliation = result; } } else { this.users_with_affiliation = this.muc.getOccupantsWithAffiliation(this.affiliation); } this.loading_users_with_affiliation = false; } async onSearchRoleChange() { if (!this.role) { return; } await this.initialized; this.clearAlert(); this.users_with_role = this.muc.getOccupantsWithRole(this.role); } shouldFetchAffiliationsList() { const affiliation = this.affiliation; if (affiliation === 'none') { return false; } const auto_fetched_affs = getAutoFetchedAffiliationLists(); if (auto_fetched_affs.includes(affiliation)) { return false; } else { return true; } } toggleForm(ev) { // eslint-disable-line class-methods-use-this ev.stopPropagation(); ev.preventDefault(); const toggle = modtools_u.ancestor(ev.target, '.toggle-form'); const form_class = toggle.getAttribute('data-form'); const form = modtools_u.ancestor(toggle, '.list-group-item').querySelector(`.${form_class}`); if (modtools_u.hasClass('hidden', form)) { modtools_u.removeClass('hidden', form); } else { modtools_u.addClass('hidden', form); } } filterRoleResults(ev) { this.roles_filter = ev.target.value; this.render(); } filterAffiliationResults(ev) { this.affiliations_filter = ev.target.value; } queryRole(ev) { ev.stopPropagation(); ev.preventDefault(); const data = new FormData(ev.target); const role = data.get('role'); this.role = null; this.role = role; } queryAffiliation(ev) { ev.stopPropagation(); ev.preventDefault(); const data = new FormData(ev.target); const affiliation = data.get('affiliation'); this.affiliation = null; this.affiliation = affiliation; } alert(message, type) { this.alert_message = message; this.alert_type = type; } clearAlert() { this.alert_message = undefined; this.alert_type = undefined; } async assignAffiliation(ev) { ev.stopPropagation(); ev.preventDefault(); this.clearAlert(); const data = new FormData(ev.target); const affiliation = data.get('affiliation'); const attrs = { 'jid': data.get('jid'), 'reason': data.get('reason') }; const current_affiliation = this.affiliation; const muc_jid = this.muc.get('jid'); try { await setAffiliation(affiliation, muc_jid, [attrs]); } catch (e) { if (e === null) { this.alert(__('Timeout error while trying to set the affiliation'), 'danger'); } else if (modtools_sizzle(`not-allowed[xmlns="${modtools_Strophe.NS.STANZAS}"]`, e).length) { this.alert(__("Sorry, you're not allowed to make that change"), 'danger'); } else { this.alert(__('Sorry, something went wrong while trying to set the affiliation'), 'danger'); } headless_log.error(e); return; } await this.muc.occupants.fetchMembers(); this.affiliation = null; this.affiliation = current_affiliation; this.alert(__('Affiliation changed'), 'primary'); } assignRole(ev) { ev.stopPropagation(); ev.preventDefault(); this.clearAlert(); const data = new FormData(ev.target); const occupant = this.muc.getOccupant(data.get('jid') || data.get('nick')); const role = data.get('role'); const reason = data.get('reason'); const current_role = this.role; this.muc.setRole(occupant, role, reason, () => { this.alert(__('Role changed'), 'primary'); this.role = null; this.role = current_role; }, e => { if (modtools_sizzle(`not-allowed[xmlns="${modtools_Strophe.NS.STANZAS}"]`, e).length) { this.alert(__("You're not allowed to make that change"), 'danger'); } else { this.alert(__('Sorry, something went wrong while trying to set the role'), 'danger'); if (modtools_u.isErrorObject(e)) { headless_log.error(e); } } }); } } core_api.elements.define('converse-modtools', ModeratorTools); ;// CONCATENATED MODULE: ./src/plugins/muc-views/modals/templates/moderator-tools.js /* harmony default export */ const templates_moderator_tools = (o => { const i18n_moderator_tools = __('Moderator Tools'); return $` `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/modals/moderator-tools.js const ModeratorToolsModal = base.extend({ id: "converse-modtools-modal", persistent: true, initialize(attrs) { this.jid = attrs.jid; this.affiliation = attrs.affiliation; base.prototype.initialize.apply(this, arguments); }, toHTML() { return templates_moderator_tools(this); } }); /* harmony default export */ const modals_moderator_tools = (ModeratorToolsModal); ;// CONCATENATED MODULE: ./src/plugins/muc-views/utils.js const { Strophe: muc_views_utils_Strophe, $iq: muc_views_utils_$iq, sizzle: muc_views_utils_sizzle, u: muc_views_utils_u } = core_converse.env; const COMMAND_TO_AFFILIATION = { 'admin': 'admin', 'ban': 'outcast', 'member': 'member', 'owner': 'owner', 'revoke': 'none' }; const COMMAND_TO_ROLE = { 'deop': 'participant', 'kick': 'none', 'mute': 'visitor', 'op': 'moderator', 'voice': 'participant' }; function utils_clearHistory(jid) { if (shared_converse.router.history.getFragment() === `converse/room?jid=${jid}`) { shared_converse.router.navigate(''); } } async function destroyMUC(model) { const messages = [__('Are you sure you want to destroy this groupchat?')]; let fields = [{ 'name': 'challenge', 'label': __('Please enter the XMPP address of this groupchat to confirm'), 'challenge': model.get('jid'), 'placeholder': __('name@example.org'), 'required': true }, { 'name': 'reason', 'label': __('Optional reason for destroying this groupchat'), 'placeholder': __('Reason') }, { 'name': 'newjid', 'label': __('Optional XMPP address for a new groupchat that replaces this one'), 'placeholder': __('replacement@example.org') }]; try { var _fields$filter$pop, _fields$filter$pop2; fields = await core_api.confirm(__('Confirm'), messages, fields); const reason = (_fields$filter$pop = fields.filter(f => f.name === 'reason').pop()) === null || _fields$filter$pop === void 0 ? void 0 : _fields$filter$pop.value; const newjid = (_fields$filter$pop2 = fields.filter(f => f.name === 'newjid').pop()) === null || _fields$filter$pop2 === void 0 ? void 0 : _fields$filter$pop2.value; return model.sendDestroyIQ(reason, newjid).then(() => model.close()); } catch (e) { headless_log.error(e); } } function getNicknameRequiredTemplate(model) { const jid = model.get('jid'); if (core_api.settings.get('muc_show_logs_before_join')) { return $``; } else { return $``; } } function getChatRoomBodyTemplate(o) { const view = o.model.session.get('view'); const jid = o.model.get('jid'); const RS = core_converse.ROOMSTATUS; const conn_status = o.model.session.get('connection_status'); if (view === core_converse.MUC.VIEWS.CONFIG) { return $``; } else { return $` ${conn_status == RS.PASSWORD_REQUIRED ? $`` : ''} ${conn_status == RS.ENTERED ? $`` : ''} ${conn_status == RS.CONNECTING ? spinner() : ''} ${conn_status == RS.NICKNAME_REQUIRED ? getNicknameRequiredTemplate(o.model) : ''} ${conn_status == RS.DISCONNECTED ? $`` : ''} ${conn_status == RS.BANNED ? $`` : ''} ${conn_status == RS.DESTROYED ? $`` : ''} `; } } function getAutoCompleteListItem(text, input) { input = input.trim(); const element = document.createElement('li'); element.setAttribute('aria-selected', 'false'); if (core_api.settings.get('muc_mention_autocomplete_show_avatar')) { const img = document.createElement('img'); let dataUri = 'data:' + shared_converse.DEFAULT_IMAGE_TYPE + ';base64,' + shared_converse.DEFAULT_IMAGE; if (shared_converse.vcards) { const vcard = shared_converse.vcards.findWhere({ 'nickname': text }); if (vcard) dataUri = 'data:' + vcard.get('image_type') + ';base64,' + vcard.get('image'); } img.setAttribute('src', dataUri); img.setAttribute('width', '22'); img.setAttribute('class', 'avatar avatar-autocomplete'); element.appendChild(img); } const regex = new RegExp('(' + input + ')', 'ig'); const parts = input ? text.split(regex) : [text]; parts.forEach(txt => { if (input && txt.match(regex)) { const match = document.createElement('mark'); match.textContent = txt; element.appendChild(match); } else { element.appendChild(document.createTextNode(txt)); } }); return element; } async function getAutoCompleteList() { const models = [...(await core_api.rooms.get()), ...(await core_api.contacts.get())]; const jids = [...new Set(models.map(o => muc_views_utils_Strophe.getDomainFromJid(o.get('jid'))))]; return jids; } async function fetchCommandForm(command) { const node = command.node; const jid = command.jid; const stanza = muc_views_utils_$iq({ 'type': 'set', 'to': jid }).c('command', { 'xmlns': muc_views_utils_Strophe.NS.ADHOC, 'node': node, 'action': 'execute' }); try { var _sizzle$pop; const iq = await core_api.sendIQ(stanza); const cmd_el = muc_views_utils_sizzle(`command[xmlns="${muc_views_utils_Strophe.NS.ADHOC}"]`, iq).pop(); command.sessionid = cmd_el.getAttribute('sessionid'); command.instructions = (_sizzle$pop = muc_views_utils_sizzle('x[type="form"][xmlns="jabber:x:data"] instructions', cmd_el).pop()) === null || _sizzle$pop === void 0 ? void 0 : _sizzle$pop.textContent; command.fields = muc_views_utils_sizzle('x[type="form"][xmlns="jabber:x:data"] field', cmd_el).map(f => muc_views_utils_u.xForm2TemplateResult(f, cmd_el)); } catch (e) { if (e === null) { headless_log.error(`Error: timeout while trying to execute command for ${jid}`); } else { headless_log.error(`Error while trying to execute command for ${jid}`); headless_log.error(e); } command.fields = []; } } function setRole(muc, command, args) { let required_affiliations = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : []; let required_roles = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : []; const role = COMMAND_TO_ROLE[command]; if (!role) { throw Error(`ChatRoomView#setRole called with invalid command: ${command}`); } if (!muc.verifyAffiliations(required_affiliations) || !muc.verifyRoles(required_roles)) { return false; } if (!muc.validateRoleOrAffiliationChangeArgs(command, args)) { return false; } const nick_or_jid = muc.getNickOrJIDFromCommandArgs(args); if (!nick_or_jid) { return false; } const reason = args.split(nick_or_jid, 2)[1].trim(); // We're guaranteed to have an occupant due to getNickOrJIDFromCommandArgs const occupant = muc.getOccupant(nick_or_jid); muc.setRole(occupant, role, reason, undefined, e => muc.onCommandError(e)); return true; } function verifyAndSetAffiliation(muc, command, args, required_affiliations) { const affiliation = COMMAND_TO_AFFILIATION[command]; if (!affiliation) { throw Error(`verifyAffiliations called with invalid command: ${command}`); } if (!muc.verifyAffiliations(required_affiliations)) { return false; } if (!muc.validateRoleOrAffiliationChangeArgs(command, args)) { return false; } const nick_or_jid = muc.getNickOrJIDFromCommandArgs(args); if (!nick_or_jid) { return false; } let jid; const reason = args.split(nick_or_jid, 2)[1].trim(); const occupant = muc.getOccupant(nick_or_jid); if (occupant) { jid = occupant.get('jid'); } else { if (muc_views_utils_u.isValidJID(nick_or_jid)) { jid = nick_or_jid; } else { const message = __("Couldn't find a participant with that nickname. " + 'They might have left the groupchat.'); muc.createMessage({ message, 'type': 'error' }); return; } } const attrs = { jid, reason }; if (occupant && core_api.settings.get('auto_register_muc_nickname')) { attrs['nick'] = occupant.get('nick'); } setAffiliation(affiliation, muc.get('jid'), [attrs]).then(() => muc.occupants.fetchMembers()).catch(err => muc.onCommandError(err)); } function showModeratorToolsModal(muc, affiliation) { if (!muc.verifyRoles(['moderator'])) { return; } let modal = core_api.modal.get(modals_moderator_tools.id); if (modal) { modal.affiliation = affiliation; modal.render(); } else { modal = core_api.modal.create(modals_moderator_tools, { affiliation, 'jid': muc.get('jid') }); } modal.show(); } function showOccupantModal(ev, occupant) { core_api.modal.show(modals_occupant, { 'model': occupant }, ev); } function parseMessageForMUCCommands(data, handled) { const model = data.model; if (handled || model.get('type') !== shared_converse.CHATROOMS_TYPE || core_api.settings.get('muc_disable_slash_commands') && !Array.isArray(core_api.settings.get('muc_disable_slash_commands'))) { return handled; } let text = data.text; text = text.replace(/^\s*/, ''); const command = (text.match(/^\/([a-zA-Z]*) ?/) || ['']).pop().toLowerCase(); if (!command) { return false; } const args = text.slice(('/' + command).length + 1).trim(); const allowed_commands = model.getAllowedCommands() ?? []; if (command === 'admin' && allowed_commands.includes(command)) { verifyAndSetAffiliation(model, command, args, ['owner']); return true; } else if (command === 'ban' && allowed_commands.includes(command)) { verifyAndSetAffiliation(model, command, args, ['admin', 'owner']); return true; } else if (command === 'modtools' && allowed_commands.includes(command)) { showModeratorToolsModal(model, args); return true; } else if (command === 'deop' && allowed_commands.includes(command)) { // FIXME: /deop only applies to setting a moderators // role to "participant" (which only admin/owner can // do). Moderators can however set non-moderator's role // to participant (e.g. visitor => participant). // Currently we don't distinguish between these two // cases. setRole(model, command, args, ['admin', 'owner']); return true; } else if (command === 'destroy' && allowed_commands.includes(command)) { if (!model.verifyAffiliations(['owner'])) { return true; } destroyMUC(model).catch(e => model.onCommandError(e)); return true; } else if (command === 'help' && allowed_commands.includes(command)) { model.set({ 'show_help_messages': false }, { 'silent': true }); model.set({ 'show_help_messages': true }); return true; } else if (command === 'kick' && allowed_commands.includes(command)) { setRole(model, command, args, [], ['moderator']); return true; } else if (command === 'mute' && allowed_commands.includes(command)) { setRole(model, command, args, [], ['moderator']); return true; } else if (command === 'member' && allowed_commands.includes(command)) { verifyAndSetAffiliation(model, command, args, ['admin', 'owner']); return true; } else if (command === 'nick' && allowed_commands.includes(command)) { if (!model.verifyRoles(['visitor', 'participant', 'moderator'])) { return true; } else if (args.length === 0) { // e.g. Your nickname is "coolguy69" const message = __('Your nickname is "%1$s"', model.get('nick')); model.createMessage({ message, 'type': 'error' }); } else { model.setNickname(args); } return true; } else if (command === 'owner' && allowed_commands.includes(command)) { verifyAndSetAffiliation(model, command, args, ['owner']); return true; } else if (command === 'op' && allowed_commands.includes(command)) { setRole(model, command, args, ['admin', 'owner']); return true; } else if (command === 'register' && allowed_commands.includes(command)) { if (args.length > 1) { model.createMessage({ 'message': __('Error: invalid number of arguments'), 'type': 'error' }); } else { model.registerNickname().then(err_msg => { err_msg && model.createMessage({ 'message': err_msg, 'type': 'error' }); }); } return true; } else if (command === 'revoke' && allowed_commands.includes(command)) { verifyAndSetAffiliation(model, command, args, ['admin', 'owner']); return true; } else if (command === 'topic' && allowed_commands.includes(command) || command === 'subject' && allowed_commands.includes(command)) { model.setSubject(args); return true; } else if (command === 'voice' && allowed_commands.includes(command)) { setRole(model, command, args, [], ['moderator']); return true; } else { return false; } } ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/ad-hoc.js /* harmony default export */ const ad_hoc = (o => { const i18n_choose_service = __('On which entity do you want to run commands?'); const i18n_choose_service_instructions = __('Certain XMPP services and entities allow privileged users to execute ad-hoc commands on them.'); const i18n_commands_found = __('Commands found'); const i18n_fetch_commands = __('List available commands'); const i18n_jid_placeholder = __('XMPP Address'); const i18n_no_commands_found = __('No commands found'); return $` ${o.alert ? $`` : ''}
    ${o.view === 'list-commands' ? $`
    • ${o.commands.length ? i18n_commands_found : i18n_no_commands_found}:
    • ${o.commands.map(cmd => ad_hoc_command(o, cmd))}
    ` : ''}
    `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/adhoc-commands.js const { Strophe: adhoc_commands_Strophe, $iq: adhoc_commands_$iq, sizzle: adhoc_commands_sizzle, u: adhoc_commands_u } = core_converse.env; class AdHocCommands extends CustomElement { static get properties() { return { 'alert': { type: String }, 'alert_type': { type: String }, 'nonce': { type: String }, // Used to force re-rendering 'showform': { type: String }, 'view': { type: String } }; } constructor() { super(); this.view = 'choose-service'; this.showform = ''; this.commands = []; } render() { return ad_hoc({ 'alert': this.alert, 'alert_type': this.alert_type, 'commands': this.commands, 'fetchCommands': ev => this.fetchCommands(ev), 'hideCommandForm': ev => this.hideCommandForm(ev), 'runCommand': ev => this.runCommand(ev), 'showform': this.showform, 'toggleCommandForm': ev => this.toggleCommandForm(ev), 'view': this.view }); } async fetchCommands(ev) { ev.preventDefault(); delete this.alert_type; delete this.alert; const form_data = new FormData(ev.target); const jid = form_data.get('jid').trim(); let supported; try { supported = await core_api.disco.supports(adhoc_commands_Strophe.NS.ADHOC, jid); } catch (e) { headless_log.error(e); } if (supported) { try { this.commands = await core_api.adhoc.getCommands(jid); this.view = 'list-commands'; } catch (e) { headless_log.error(e); this.alert_type = 'danger'; this.alert = __('Sorry, an error occurred while looking for commands on that entity.'); this.commands = []; headless_log.error(e); return; } } else { this.alert_type = 'danger'; this.alert = __("The specified entity doesn't support ad-hoc commands"); } } async toggleCommandForm(ev) { ev.preventDefault(); const node = ev.target.getAttribute('data-command-node'); const cmd = this.commands.filter(c => c.node === node)[0]; this.showform !== node && (await fetchCommandForm(cmd)); this.showform = node; } hideCommandForm(ev) { ev.preventDefault(); this.showform = ''; } async runCommand(ev) { ev.preventDefault(); const form_data = new FormData(ev.target); const jid = form_data.get('command_jid').trim(); const node = form_data.get('command_node').trim(); const cmd = this.commands.filter(c => c.node === node)[0]; cmd.alert = null; this.nonce = adhoc_commands_u.getUniqueId(); const inputs = adhoc_commands_sizzle(':input:not([type=button]):not([type=submit])', ev.target); const config_array = inputs.filter(i => !['command_jid', 'command_node'].includes(i.getAttribute('name'))).map(adhoc_commands_u.webForm2xForm).filter(n => n); const iq = adhoc_commands_$iq({ to: jid, type: "set" }).c("command", { 'sessionid': cmd.sessionid, 'node': cmd.node, 'xmlns': adhoc_commands_Strophe.NS.ADHOC }).c("x", { xmlns: adhoc_commands_Strophe.NS.XFORM, type: "submit" }); config_array.forEach(node => iq.cnode(node).up()); let result; try { result = await core_api.sendIQ(iq); } catch (e) { cmd.alert_type = 'danger'; cmd.alert = __('Sorry, an error occurred while trying to execute the command. See the developer console for details'); headless_log.error('Error while trying to execute an ad-hoc command'); headless_log.error(e); } if (result) { var _result$querySelector; cmd.alert = (_result$querySelector = result.querySelector('note')) === null || _result$querySelector === void 0 ? void 0 : _result$querySelector.textContent; } else { cmd.alert = 'Done'; } cmd.alert_type = 'primary'; this.nonce = adhoc_commands_u.getUniqueId(); } } core_api.elements.define('converse-adhoc-commands', AdHocCommands); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/message-form.js /* harmony default export */ const templates_message_form = (o => { const label_message = o.composing_spoiler ? __('Hidden message') : __('Message'); const label_spoiler_hint = __('Optional hint'); const show_send_button = core_api.settings.get('show_send_button'); return $`
    `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/message-form.js class MUCMessageForm extends MessageForm { async connectedCallback() { super.connectedCallback(); await this.model.initialized; } toHTML() { var _this$querySelector, _this$querySelector2; return templates_message_form(Object.assign(this.model.toJSON(), { 'hint_value': (_this$querySelector = this.querySelector('.spoiler-hint')) === null || _this$querySelector === void 0 ? void 0 : _this$querySelector.value, 'message_value': (_this$querySelector2 = this.querySelector('.chat-textarea')) === null || _this$querySelector2 === void 0 ? void 0 : _this$querySelector2.value, 'onChange': ev => this.model.set({ 'draft': ev.target.value }), 'onDrop': ev => this.onDrop(ev), 'onKeyDown': ev => this.onKeyDown(ev), 'onKeyUp': ev => this.onKeyUp(ev), 'onPaste': ev => this.onPaste(ev), 'scrolled': this.model.ui.get('scrolled'), 'viewUnreadMessages': ev => this.viewUnreadMessages(ev) })); } afterRender() { const entered = this.model.session.get('connection_status') === core_converse.ROOMSTATUS.ENTERED; const can_edit = entered && !(this.model.features.get('moderated') && this.model.getOwnRole() === 'visitor'); if (entered && can_edit) { this.initMentionAutoComplete(); } } initMentionAutoComplete() { this.mention_auto_complete = new shared_converse.AutoComplete(this, { 'auto_first': true, 'auto_evaluate': false, 'min_chars': core_api.settings.get('muc_mention_autocomplete_min_chars'), 'match_current_word': true, 'list': () => this.getAutoCompleteList(), 'filter': core_api.settings.get('muc_mention_autocomplete_filter') == 'contains' ? shared_converse.FILTER_CONTAINS : shared_converse.FILTER_STARTSWITH, 'ac_triggers': ['Tab', '@'], 'include_triggers': [], 'item': getAutoCompleteListItem }); this.mention_auto_complete.on('suggestion-box-selectcomplete', () => this.auto_completing = false); } getAutoCompleteList() { return this.model.getAllKnownNicknames().map(nick => ({ 'label': nick, 'value': `@${nick}` })); } onKeyDown(ev) { if (this.mention_auto_complete.onKeyDown(ev)) { return; } super.onKeyDown(ev); } onKeyUp(ev) { this.mention_auto_complete.evaluate(ev); super.onKeyUp(ev); } } core_api.elements.define('converse-muc-message-form', MUCMessageForm); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/muc-nickname-form.js /* harmony default export */ const muc_nickname_form = (el => { var _el$model, _el$model2, _el$model3; const i18n_nickname = __('Nickname'); const i18n_join = (_el$model = el.model) !== null && _el$model !== void 0 && _el$model.isEntered() ? __('Change nickname') : __('Enter groupchat'); const i18n_heading = core_api.settings.get('muc_show_logs_before_join') ? __('Choose a nickname to enter') : __('Please choose your nickname'); const validation_message = (_el$model2 = el.model) === null || _el$model2 === void 0 ? void 0 : _el$model2.get('nickname_validation_message'); return $`
    el.submitNickname(ev)}>

    ${validation_message}

    `; }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/muc-views/styles/nickname-form.scss var nickname_form = __webpack_require__(6233); ;// CONCATENATED MODULE: ./src/plugins/muc-views/styles/nickname-form.scss var nickname_form_options = {}; nickname_form_options.styleTagTransform = (styleTagTransform_default()); nickname_form_options.setAttributes = (setAttributesWithoutAttributes_default()); nickname_form_options.insert = insertBySelector_default().bind(null, "head"); nickname_form_options.domAPI = (styleDomAPI_default()); nickname_form_options.insertStyleElement = (insertStyleElement_default()); var nickname_form_update = injectStylesIntoStyleTag_default()(nickname_form/* default */.Z, nickname_form_options); /* harmony default export */ const styles_nickname_form = (nickname_form/* default */.Z && nickname_form/* default.locals */.Z.locals ? nickname_form/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/muc-views/nickname-form.js class MUCNicknameForm extends CustomElement { static get properties() { return { 'jid': { type: String } }; } connectedCallback() { super.connectedCallback(); this.model = shared_converse.chatboxes.get(this.jid); } render() { return muc_nickname_form(this); } submitNickname(ev) { ev.preventDefault(); const nick = ev.target.nick.value.trim(); if (!nick) { return; } if (this.model.isEntered()) { this.model.setNickname(nick); this.closeModal(); } else { this.model.join(nick); } } closeModal() { const evt = document.createEvent('Event'); evt.initEvent('hide.bs.modal', true, true); this.dispatchEvent(evt); } } core_api.elements.define('converse-muc-nickname-form', MUCNicknameForm); /* harmony default export */ const muc_views_nickname_form = ((/* unused pure expression or super */ null && (MUCNicknameForm))); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/muc-bottom-panel.js const tpl_can_edit = o => { const unread_msgs = __('You have unread messages'); const message_limit = core_api.settings.get('message_limit'); const show_call_button = core_api.settings.get('visible_toolbar_buttons').call; const show_emoji_button = core_api.settings.get('visible_toolbar_buttons').emoji; const show_send_button = core_api.settings.get('show_send_button'); const show_spoiler_button = core_api.settings.get('visible_toolbar_buttons').spoiler; const show_toolbar = core_api.settings.get('show_toolbar'); return $` ${o.model.ui.get('scrolled') && o.model.get('num_unread') ? $`
    o.viewUnreadMessages(ev)}>▼ ${unread_msgs} ▼
    ` : ''} ${show_toolbar ? $` ` : ''} `; }; /* harmony default export */ const muc_bottom_panel = (o => { const unread_msgs = __('You have unread messages'); const conn_status = o.model.session.get('connection_status'); const i18n_not_allowed = __("You're not allowed to send messages in this room"); if (conn_status === core_converse.ROOMSTATUS.ENTERED) { return $` ${o.model.ui.get('scrolled') && o.model.get('num_unread_general') ? $`
    o.viewUnreadMessages(ev)}>▼ ${unread_msgs} ▼
    ` : ''} ${o.can_edit ? tpl_can_edit(o) : $`${i18n_not_allowed}`}`; } else if (conn_status == core_converse.ROOMSTATUS.NICKNAME_REQUIRED) { if (core_api.settings.get('muc_show_logs_before_join')) { return $` `; } } else { return ''; } }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/muc-views/styles/muc-bottom-panel.scss var styles_muc_bottom_panel = __webpack_require__(6916); ;// CONCATENATED MODULE: ./src/plugins/muc-views/styles/muc-bottom-panel.scss var muc_bottom_panel_options = {}; muc_bottom_panel_options.styleTagTransform = (styleTagTransform_default()); muc_bottom_panel_options.setAttributes = (setAttributesWithoutAttributes_default()); muc_bottom_panel_options.insert = insertBySelector_default().bind(null, "head"); muc_bottom_panel_options.domAPI = (styleDomAPI_default()); muc_bottom_panel_options.insertStyleElement = (insertStyleElement_default()); var muc_bottom_panel_update = injectStylesIntoStyleTag_default()(styles_muc_bottom_panel/* default */.Z, muc_bottom_panel_options); /* harmony default export */ const muc_views_styles_muc_bottom_panel = (styles_muc_bottom_panel/* default */.Z && styles_muc_bottom_panel/* default.locals */.Z.locals ? styles_muc_bottom_panel/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/muc-views/bottom-panel.js function muc_views_bottom_panel_defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } class MUCBottomPanel extends ChatBottomPanel { constructor() { super(...arguments); muc_views_bottom_panel_defineProperty(this, "events", { 'click .hide-occupants': 'hideOccupants', 'click .send-button': 'sendButtonClicked' }); } async initialize() { await super.initialize(); this.listenTo(this.model, 'change:hidden_occupants', this.debouncedRender); this.listenTo(this.model, 'change:num_unread_general', this.debouncedRender); this.listenTo(this.model.features, 'change:moderated', this.debouncedRender); this.listenTo(this.model.occupants, 'add', this.renderIfOwnOccupant); this.listenTo(this.model.occupants, 'change:role', this.renderIfOwnOccupant); this.listenTo(this.model.session, 'change:connection_status', this.debouncedRender); } render() { const entered = this.model.session.get('connection_status') === core_converse.ROOMSTATUS.ENTERED; const can_edit = entered && !(this.model.features.get('moderated') && this.model.getOwnRole() === 'visitor'); x(muc_bottom_panel({ can_edit, entered, 'model': this.model, 'is_groupchat': true, 'viewUnreadMessages': ev => this.viewUnreadMessages(ev) }), this); } renderIfOwnOccupant(o) { o.get('jid') === shared_converse.bare_jid && this.debouncedRender(); } sendButtonClicked(ev) { var _this$querySelector; (_this$querySelector = this.querySelector('converse-muc-message-form')) === null || _this$querySelector === void 0 ? void 0 : _this$querySelector.onFormSubmitted(ev); } hideOccupants(ev) { var _ev$preventDefault, _ev$stopPropagation; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); ev === null || ev === void 0 ? void 0 : (_ev$stopPropagation = ev.stopPropagation) === null || _ev$stopPropagation === void 0 ? void 0 : _ev$stopPropagation.call(ev); this.model.save({ 'hidden_occupants': true }); } } core_api.elements.define('converse-muc-bottom-panel', MUCBottomPanel); ;// CONCATENATED MODULE: ./src/plugins/muc-views/constants.js const PRETTY_CHAT_STATUS = { 'offline': 'Offline', 'unavailable': 'Unavailable', 'xa': 'Extended Away', 'away': 'Away', 'dnd': 'Do not disturb', 'chat': 'Chattty', 'online': 'Online' }; ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/occupant.js const i18n_occupant_hint = o => __('Click to mention %1$s in your message.', o.get('nick')); const occupant_title = o => { const role = o.get('role'); const hint_occupant = i18n_occupant_hint(o); const i18n_moderator_hint = __('This user is a moderator.'); const i18n_participant_hint = __('This user can send messages in this groupchat.'); const i18n_visitor_hint = __('This user can NOT send messages in this groupchat.'); const spaced_jid = o.get('jid') ? `${o.get('jid')} ` : ''; if (role === "moderator") { return `${spaced_jid}${i18n_moderator_hint} ${hint_occupant}`; } else if (role === "participant") { return `${spaced_jid}${i18n_participant_hint} ${hint_occupant}`; } else if (role === "visitor") { return `${spaced_jid}${i18n_visitor_hint} ${hint_occupant}`; } else if (!["visitor", "participant", "moderator"].includes(role)) { return `${spaced_jid}${hint_occupant}`; } }; /* harmony default export */ const muc_views_templates_occupant = ((o, chat) => { var _o$vcard, _o$vcard2; const affiliation = o.get('affiliation'); const hint_show = PRETTY_CHAT_STATUS[o.get('show')]; const i18n_admin = __('Admin'); const i18n_member = __('Member'); const i18n_moderator = __('Moderator'); const i18n_owner = __('Owner'); const i18n_visitor = __('Visitor'); const role = o.get('role'); const show = o.get('show'); let classes, color; if (show === 'online') { [classes, color] = ['fa fa-circle', 'chat-status-online']; } else if (show === 'dnd') { [classes, color] = ['fa fa-minus-circle', 'chat-status-busy']; } else if (show === 'away') { [classes, color] = ['fa fa-circle', 'chat-status-away']; } else { [classes, color] = ['fa fa-circle', 'subdued-color']; } return $`
  • ${o.getDisplayName()} ${affiliation === "owner" ? $`${i18n_owner}` : ''} ${affiliation === "admin" ? $`${i18n_admin}` : ''} ${affiliation === "member" ? $`${i18n_member}` : ''} ${role === "moderator" ? $`${i18n_moderator}` : ''} ${role === "visitor" ? $`${i18n_visitor}` : ''}
  • `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/muc-sidebar.js /* harmony default export */ const muc_sidebar = (o => { const i18n_participants = o.occupants.length === 1 ? __('Participant') : __('Participants'); return $`
    ${o.occupants.length} ${i18n_participants}
      ${o.occupants.map(occ => muc_views_templates_occupant(occ, o))}
    `; }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/shared/styles/status.scss var styles_status = __webpack_require__(9959); ;// CONCATENATED MODULE: ./src/shared/styles/status.scss var status_options = {}; status_options.styleTagTransform = (styleTagTransform_default()); status_options.setAttributes = (setAttributesWithoutAttributes_default()); status_options.insert = insertBySelector_default().bind(null, "head"); status_options.domAPI = (styleDomAPI_default()); status_options.insertStyleElement = (insertStyleElement_default()); var status_update = injectStylesIntoStyleTag_default()(styles_status/* default */.Z, status_options); /* harmony default export */ const shared_styles_status = (styles_status/* default */.Z && styles_status/* default.locals */.Z.locals ? styles_status/* default.locals */.Z.locals : undefined); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/muc-views/styles/muc-occupants.scss var muc_occupants = __webpack_require__(902); ;// CONCATENATED MODULE: ./src/plugins/muc-views/styles/muc-occupants.scss var muc_occupants_options = {}; muc_occupants_options.styleTagTransform = (styleTagTransform_default()); muc_occupants_options.setAttributes = (setAttributesWithoutAttributes_default()); muc_occupants_options.insert = insertBySelector_default().bind(null, "head"); muc_occupants_options.domAPI = (styleDomAPI_default()); muc_occupants_options.insertStyleElement = (insertStyleElement_default()); var muc_occupants_update = injectStylesIntoStyleTag_default()(muc_occupants/* default */.Z, muc_occupants_options); /* harmony default export */ const styles_muc_occupants = (muc_occupants/* default */.Z && muc_occupants/* default.locals */.Z.locals ? muc_occupants/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/muc-views/sidebar.js const { u: sidebar_u } = core_converse.env; class MUCSidebar extends CustomElement { static get properties() { return { jid: { type: String } }; } connectedCallback() { super.connectedCallback(); this.model = shared_converse.chatboxes.get(this.jid); this.listenTo(this.model.occupants, 'add', this.requestUpdate); this.listenTo(this.model.occupants, 'remove', this.requestUpdate); this.listenTo(this.model.occupants, 'change', this.requestUpdate); this.listenTo(this.model.occupants, 'vcard:change', this.requestUpdate); this.listenTo(this.model.occupants, 'vcard:add', this.requestUpdate); this.model.initialized.then(() => this.requestUpdate()); } render() { const tpl = muc_sidebar(Object.assign(this.model.toJSON(), { 'occupants': [...this.model.occupants.models], 'closeSidebar': ev => this.closeSidebar(ev), 'onOccupantClicked': ev => this.onOccupantClicked(ev) })); return tpl; } closeSidebar(ev) { var _ev$preventDefault, _ev$stopPropagation; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); ev === null || ev === void 0 ? void 0 : (_ev$stopPropagation = ev.stopPropagation) === null || _ev$stopPropagation === void 0 ? void 0 : _ev$stopPropagation.call(ev); sidebar_u.safeSave(this.model, { 'hidden_occupants': true }); } onOccupantClicked(ev) { var _ev$preventDefault2; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault2 = ev.preventDefault) === null || _ev$preventDefault2 === void 0 ? void 0 : _ev$preventDefault2.call(ev); const view = shared_converse.chatboxviews.get(this.getAttribute('jid')); view === null || view === void 0 ? void 0 : view.getMessageForm().insertIntoTextArea(`@${ev.target.textContent}`); } } core_api.elements.define('converse-muc-sidebar', MUCSidebar); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/muc-chatarea.js /* harmony default export */ const muc_chatarea = (o => { var _o$model; return $`
    ${(_o$model = o.model) !== null && _o$model !== void 0 && _o$model.get('show_help_messages') ? $`
    ` : ''}
    ${o.model ? $` ` : ''} `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/chatarea.js const { u: chatarea_u } = core_converse.env; class MUCChatArea extends CustomElement { static get properties() { return { jid: { type: String }, show_help_messages: { type: Boolean }, type: { type: String } }; } async initialize() { this.model = await core_api.rooms.get(this.jid); this.listenTo(this.model, 'change:show_help_messages', () => this.requestUpdate()); this.listenTo(this.model, 'change:hidden_occupants', () => this.requestUpdate()); this.listenTo(this.model.session, 'change:connection_status', () => this.requestUpdate()); // Bind so that we can pass it to addEventListener and removeEventListener this.onMouseMove = this._onMouseMove.bind(this); this.onMouseUp = this._onMouseUp.bind(this); this.requestUpdate(); // Make sure we render again after the model has been attached } render() { return muc_chatarea({ 'getHelpMessages': () => this.getHelpMessages(), 'jid': this.jid, 'model': this.model, 'onMousedown': ev => this.onMousedown(ev), 'show_send_button': core_api.settings.get('show_send_button'), 'shouldShowSidebar': () => this.shouldShowSidebar(), 'type': this.type }); } shouldShowSidebar() { return !this.model.get('hidden_occupants') && this.model.session.get('connection_status') === core_converse.ROOMSTATUS.ENTERED; } getHelpMessages() { const setting = core_api.settings.get('muc_disable_slash_commands'); const disabled_commands = Array.isArray(setting) ? setting : []; return [`/admin: ${__("Change user's affiliation to admin")}`, `/ban: ${__('Ban user by changing their affiliation to outcast')}`, `/clear: ${__('Clear the chat area')}`, `/close: ${__('Close this groupchat')}`, `/deop: ${__('Change user role to participant')}`, `/destroy: ${__('Remove this groupchat')}`, `/help: ${__('Show this menu')}`, `/kick: ${__('Kick user from groupchat')}`, `/me: ${__('Write in 3rd person')}`, `/member: ${__('Grant membership to a user')}`, `/modtools: ${__('Opens up the moderator tools GUI')}`, `/mute: ${__("Remove user's ability to post messages")}`, `/nick: ${__('Change your nickname')}`, `/op: ${__('Grant moderator role to user')}`, `/owner: ${__('Grant ownership of this groupchat')}`, `/register: ${__('Register your nickname')}`, `/revoke: ${__("Revoke the user's current affiliation")}`, `/subject: ${__('Set groupchat subject')}`, `/topic: ${__('Set groupchat subject (alias for /subject)')}`, `/voice: ${__('Allow muted user to post messages')}`].filter(line => disabled_commands.every(c => !line.startsWith(c + '<', 9))).filter(line => this.model.getAllowedCommands().some(c => line.startsWith(c + '<', 9))); } onMousedown(ev) { if (chatarea_u.hasClass('dragresize-occupants-left', ev.target)) { this.onStartResizeOccupants(ev); } } onStartResizeOccupants(ev) { this.resizing = true; this.addEventListener('mousemove', this.onMouseMove); this.addEventListener('mouseup', this.onMouseUp); const sidebar_el = this.querySelector('converse-muc-sidebar'); const style = window.getComputedStyle(sidebar_el); this.width = parseInt(style.width.replace(/px$/, ''), 10); this.prev_pageX = ev.pageX; } _onMouseMove(ev) { if (this.resizing) { ev.preventDefault(); const delta = this.prev_pageX - ev.pageX; this.resizeSidebarView(delta, ev.pageX); this.prev_pageX = ev.pageX; } } _onMouseUp(ev) { if (this.resizing) { ev.preventDefault(); this.resizing = false; this.removeEventListener('mousemove', this.onMouseMove); this.removeEventListener('mouseup', this.onMouseUp); const sidebar_el = this.querySelector('converse-muc-sidebar'); const element_position = sidebar_el.getBoundingClientRect(); const occupants_width = this.calculateSidebarWidth(element_position, 0); chatarea_u.safeSave(this.model, { occupants_width }); } } calculateSidebarWidth(element_position, delta) { let occupants_width = element_position.width + delta; const room_width = this.clientWidth; // keeping display in boundaries if (occupants_width < room_width * 0.2) { // set pixel to 20% width occupants_width = room_width * 0.2; this.is_minimum = true; } else if (occupants_width > room_width * 0.75) { // set pixel to 75% width occupants_width = room_width * 0.75; this.is_maximum = true; } else if (room_width - occupants_width < 250) { // resize occupants if chat-area becomes smaller than 250px (min-width property set in css) occupants_width = room_width - 250; this.is_maximum = true; } else { this.is_maximum = false; this.is_minimum = false; } return occupants_width; } resizeSidebarView(delta, current_mouse_position) { const sidebar_el = this.querySelector('converse-muc-sidebar'); const element_position = sidebar_el.getBoundingClientRect(); if (this.is_minimum) { this.is_minimum = element_position.left < current_mouse_position; } else if (this.is_maximum) { this.is_maximum = element_position.left > current_mouse_position; } else { const occupants_width = this.calculateSidebarWidth(element_position, delta); sidebar_el.style.flex = '0 0 ' + occupants_width + 'px'; } } } core_api.elements.define('converse-muc-chatarea', MUCChatArea); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/muc-config-form.js const { sizzle: muc_config_form_sizzle } = core_converse.env; const muc_config_form_u = core_converse.env.utils; /* harmony default export */ const muc_config_form = (o => { const whitelist = core_api.settings.get('roomconfig_whitelist'); const config_stanza = o.model.session.get('config_stanza'); let fields = []; let instructions = ''; let title; if (config_stanza) { var _stanza$querySelector, _stanza$querySelector2; const stanza = muc_config_form_u.toStanza(config_stanza); fields = muc_config_form_sizzle('field', stanza); if (whitelist.length) { fields = fields.filter(f => whitelist.includes(f.getAttribute('var'))); } const password_protected = o.model.features.get('passwordprotected'); const options = { 'new_password': !password_protected, 'fixed_username': o.model.get('jid') }; fields = fields.map(f => muc_config_form_u.xForm2TemplateResult(f, stanza, options)); instructions = (_stanza$querySelector = stanza.querySelector('instructions')) === null || _stanza$querySelector === void 0 ? void 0 : _stanza$querySelector.textContent; title = (_stanza$querySelector2 = stanza.querySelector('title')) === null || _stanza$querySelector2 === void 0 ? void 0 : _stanza$querySelector2.textContent; } else { title = __('Loading configuration form'); } const i18n_save = __('Save'); const i18n_cancel = __('Cancel'); return $`
    ${title} ${title !== instructions ? $`

    ${instructions}

    ` : ''} ${fields.length ? fields : spinner({ 'classes': 'hor_centered' })}
    ${fields.length ? $`
    ` : ''}
    `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/config-form.js const { sizzle: config_form_sizzle } = core_converse.env; const config_form_u = core_converse.env.utils; class MUCConfigForm extends CustomElement { static get properties() { return { 'jid': { type: String } }; } connectedCallback() { super.connectedCallback(); this.model = shared_converse.chatboxes.get(this.jid); this.listenTo(this.model.features, 'change:passwordprotected', this.requestUpdate); this.listenTo(this.model.session, 'change:config_stanza', this.requestUpdate); this.getConfig(); } render() { return muc_config_form({ 'model': this.model, 'closeConfigForm': ev => this.closeForm(ev), 'submitConfigForm': ev => this.submitConfigForm(ev) }); } async getConfig() { const iq = await this.model.fetchRoomConfiguration(); this.model.session.set('config_stanza', iq.outerHTML); } async submitConfigForm(ev) { ev.preventDefault(); const inputs = config_form_sizzle(':input:not([type=button]):not([type=submit])', ev.target); const config_array = inputs.map(config_form_u.webForm2xForm).filter(f => f); try { await this.model.sendConfiguration(config_array); } catch (e) { headless_log.error(e); const message = __("Sorry, an error occurred while trying to submit the config form.") + " " + __("Check your browser's developer console for details."); core_api.alert('error', __('Error'), message); } await this.model.refreshDiscoInfo(); this.closeForm(); } closeForm(ev) { var _ev$preventDefault; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); this.model.session.set('view', null); } } core_api.elements.define('converse-muc-config-form', MUCConfigForm); /* harmony default export */ const config_form = ((/* unused pure expression or super */ null && (MUCConfigForm))); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/muc-destroyed.js const tpl_moved = o => { const i18n_moved = __('The conversation has moved to a new address. Click the link below to enter.'); return $`

    ${i18n_moved}

    `; }; /* harmony default export */ const muc_destroyed = (o => { const i18n_non_existent = __('This groupchat no longer exists'); const i18n_reason = __('The following reason was given: "%1$s"', o.reason || ''); return $`

    ${i18n_non_existent}

    ${o.reason ? $`

    ${i18n_reason}

    ` : ''} ${o.moved_jid ? tpl_moved(o) : ''} `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/destroyed.js class MUCDestroyed extends CustomElement { static get properties() { return { 'jid': { type: String } }; } connectedCallback() { super.connectedCallback(); this.model = shared_converse.chatboxes.get(this.jid); } render() { const reason = this.model.get('destroyed_reason'); const moved_jid = this.model.get('moved_jid'); return muc_destroyed({ moved_jid, reason, 'onSwitch': ev => this.onSwitch(ev) }); } async onSwitch(ev) { ev.preventDefault(); const moved_jid = this.model.get('moved_jid'); const room = await core_api.rooms.get(moved_jid, {}, true); room.maybeShow(true); this.model.destroy(); } } core_api.elements.define('converse-muc-destroyed', MUCDestroyed); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/muc-disconnect.js /* harmony default export */ const muc_disconnect = (messages => { return $`

    ${messages[0]}

    ${messages.slice(1).map(m => $`

    ${m}

    `)}
    `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/disconnected.js class MUCDisconnected extends CustomElement { static get properties() { return { 'jid': { type: String } }; } connectedCallback() { super.connectedCallback(); this.model = shared_converse.chatboxes.get(this.jid); } render() { const message = this.model.session.get('disconnection_message'); if (!message) { return; } const messages = [message]; const actor = this.model.session.get('disconnection_actor'); if (actor) { messages.push(__('This action was done by %1$s.', actor)); } const reason = this.model.session.get('disconnection_reason'); if (reason) { messages.push(__('The reason given is: "%1$s".', reason)); } return muc_disconnect(messages); } } core_api.elements.define('converse-muc-disconnected', MUCDisconnected); ;// CONCATENATED MODULE: ./src/plugins/muc-views/modals/templates/muc-invite.js /* harmony default export */ const muc_invite = (o => { const i18n_invite = __('Invite'); const i18n_invite_heading = __('Invite someone to this groupchat'); const i18n_jid_placeholder = __('user@example.org'); const i18n_error_message = __('Please enter a valid XMPP address'); const i18n_invite_label = __('XMPP Address'); const i18n_reason = __('Optional reason for the invitation'); return $` `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/modals/muc-invite.js const muc_invite_u = core_converse.env.utils; /* harmony default export */ const modals_muc_invite = (base.extend({ id: "muc-invite-modal", initialize() { base.prototype.initialize.apply(this, arguments); this.listenTo(this.model, 'change', this.render); this.initInviteWidget(); }, toHTML() { return muc_invite(Object.assign(this.model.toJSON(), { 'submitInviteForm': ev => this.submitInviteForm(ev) })); }, initInviteWidget() { if (this.invite_auto_complete) { this.invite_auto_complete.destroy(); } const list = shared_converse.roster.map(i => ({ 'label': i.getDisplayName(), 'value': i.get('jid') })); const el = this.el.querySelector('.suggestion-box').parentElement; this.invite_auto_complete = new shared_converse.AutoComplete(el, { 'min_chars': 1, 'list': list }); }, submitInviteForm(ev) { ev.preventDefault(); // TODO: Add support for sending an invite to multiple JIDs const data = new FormData(ev.target); const jid = data.get('invitee_jids'); const reason = data.get('reason'); if (muc_invite_u.isValidJID(jid)) { // TODO: Create and use API here this.chatroomview.model.directInvite(jid, reason); this.modal.hide(); } else { this.model.set({ 'invalid_invite_jid': true }); } } })); ;// CONCATENATED MODULE: ./src/plugins/muc-views/modals/templates/nickname.js /* harmony default export */ const nickname = (modal => { const jid = modal.model.get('jid'); const i18n_heading = __('Change your nickname'); return $` `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/modals/nickname.js /* harmony default export */ const modals_nickname = (base.extend({ id: 'change-nickname-modal', initialize(attrs) { this.model = attrs.model; base.prototype.initialize.apply(this, arguments); }, toHTML() { return nickname(this); } })); ;// CONCATENATED MODULE: ./src/plugins/muc-views/modals/templates/muc-details.js const subject = o => { const i18n_topic = __('Topic'); const i18n_topic_author = __('Topic author'); return $`

    ${i18n_topic}:

    ${i18n_topic_author}: ${o.subject && o.subject.author}

    `; }; /* harmony default export */ const muc_details = (model => { const o = model.toJSON(); const config = model.config.toJSON(); const display_name = __('Groupchat info for %1$s', model.getDisplayName()); const features = model.features.toJSON(); const num_occupants = model.occupants.filter(o => o.get('show') !== 'offline').length; const i18n_address = __('XMPP address'); const i18n_archiving = __('Message archiving'); const i18n_archiving_help = __('Messages are archived on the server'); const i18n_desc = __('Description'); const i18n_features = __('Features'); const i18n_hidden = __('Hidden'); const i18n_hidden_help = __('This groupchat is not publicly searchable'); const i18n_members_help = __('This groupchat is restricted to members only'); const i18n_members_only = __('Members only'); const i18n_moderated = __('Moderated'); const i18n_moderated_help = __('Participants entering this groupchat need to request permission to write'); const i18n_name = __('Name'); const i18n_no_pass_help = __('This groupchat does not require a password upon entry'); const i18n_no_password_required = __('No password required'); const i18n_not_anonymous = __('Not anonymous'); const i18n_not_anonymous_help = __('All other groupchat participants can see your XMPP address'); const i18n_not_moderated = __('Not moderated'); const i18n_not_moderated_help = __('Participants entering this groupchat can write right away'); const i18n_online_users = __('Online users'); const i18n_open = __('Open'); const i18n_open_help = __('Anyone can join this groupchat'); const i18n_password_help = __('This groupchat requires a password before entry'); const i18n_password_protected = __('Password protected'); const i18n_persistent = __('Persistent'); const i18n_persistent_help = __('This groupchat persists even if it\'s unoccupied'); const i18n_public = __('Public'); const i18n_semi_anon = __('Semi-anonymous'); const i18n_semi_anon_help = __('Only moderators can see your XMPP address'); const i18n_temporary = __('Temporary'); const i18n_temporary_help = __('This groupchat will disappear once the last person leaves'); return $` `; }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/muc-views/styles/muc-details.scss var styles_muc_details = __webpack_require__(7140); ;// CONCATENATED MODULE: ./src/plugins/muc-views/styles/muc-details.scss var muc_details_options = {}; muc_details_options.styleTagTransform = (styleTagTransform_default()); muc_details_options.setAttributes = (setAttributesWithoutAttributes_default()); muc_details_options.insert = insertBySelector_default().bind(null, "head"); muc_details_options.domAPI = (styleDomAPI_default()); muc_details_options.insertStyleElement = (insertStyleElement_default()); var muc_details_update = injectStylesIntoStyleTag_default()(styles_muc_details/* default */.Z, muc_details_options); /* harmony default export */ const muc_views_styles_muc_details = (styles_muc_details/* default */.Z && styles_muc_details/* default.locals */.Z.locals ? styles_muc_details/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/muc-views/modals/muc-details.js /* harmony default export */ const modals_muc_details = (base.extend({ id: "muc-details-modal", initialize() { base.prototype.initialize.apply(this, arguments); this.listenTo(this.model, 'change', this.render); this.listenTo(this.model.features, 'change', this.render); this.listenTo(this.model.occupants, 'add', this.render); this.listenTo(this.model.occupants, 'change', this.render); }, toHTML() { return muc_details(this.model); } })); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/shared/components/styles/rich-text.scss var styles_rich_text = __webpack_require__(307); ;// CONCATENATED MODULE: ./src/shared/components/styles/rich-text.scss var rich_text_options = {}; rich_text_options.styleTagTransform = (styleTagTransform_default()); rich_text_options.setAttributes = (setAttributesWithoutAttributes_default()); rich_text_options.insert = insertBySelector_default().bind(null, "head"); rich_text_options.domAPI = (styleDomAPI_default()); rich_text_options.insertStyleElement = (insertStyleElement_default()); var rich_text_update = injectStylesIntoStyleTag_default()(styles_rich_text/* default */.Z, rich_text_options); /* harmony default export */ const components_styles_rich_text = (styles_rich_text/* default */.Z && styles_rich_text/* default.locals */.Z.locals ? styles_rich_text/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/shared/components/rich-text.js /** * The RichText custom element allows you to parse transform text into rich DOM elements. * @example */ class rich_text_RichText extends CustomElement { static get properties() { /** * @typedef { Object } RichTextComponentProperties * @property { Boolean } embed_audio * Whether URLs that point to audio files should render as audio players. * @property { Boolean } embed_videos * Whether URLs that point to video files should render as video players. * @property { Array } mentions - An array of objects representing chat mentions * @property { String } nick - The current user's nickname, relevant for mentions * @property { Number } offset - The text offset, in case this is a nested RichText element. * @property { Function } onImgClick * @property { Function } onImgLoad * @property { Boolean } render_styling * Whether XEP-0393 message styling hints should be rendered * @property { Boolean } show_images * Whether URLs that point to image files should render as images * @property { Boolean } hide_media_urls * If media URLs are rendered as media, then this option determines * whether the original URL is also still shown or not. * Only relevant in conjunction with `show_images`, `embed_audio` and `embed_videos`. * @property { Boolean } show_me_message * Whether text that starts with /me should be rendered in the 3rd person. * @property { String } text - The text that will get transformed. */ return { embed_audio: { type: Boolean }, embed_videos: { type: Boolean }, mentions: { type: Array }, nick: { type: String }, offset: { type: Number }, onImgClick: { type: Function }, onImgLoad: { type: Function }, render_styling: { type: Boolean }, show_images: { type: Boolean }, hide_media_urls: { type: Boolean }, show_me_message: { type: Boolean }, text: { type: String } }; } constructor() { super(); this.embed_audio = false; this.embed_videos = false; this.hide_media_urls = false; this.mentions = []; this.offset = 0; this.render_styling = false; this.show_image_urls = true; this.show_images = false; this.show_me_message = false; } render() { const options = { embed_audio: this.embed_audio, embed_videos: this.embed_videos, hide_media_urls: this.hide_media_urls, mentions: this.mentions, nick: this.nick, onImgClick: this.onImgClick, onImgLoad: this.onImgLoad, render_styling: this.render_styling, show_images: this.show_images, show_me_message: this.show_me_message }; return rich_text(this.text, this.offset, options); } } core_api.elements.define('converse-rich-text', rich_text_RichText); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/muc-head.js /* harmony default export */ const muc_head = (el => { var _el$user_settings, _el$user_settings$get, _el$model$vcard, _el$model$vcard2, _el$model$vcard3; const o = el.model.toJSON(); const subject_hidden = (_el$user_settings = el.user_settings) === null || _el$user_settings === void 0 ? void 0 : (_el$user_settings$get = _el$user_settings.get('mucs_with_hidden_subject', [])) === null || _el$user_settings$get === void 0 ? void 0 : _el$user_settings$get.includes(el.model.get('jid')); const i18n_hide_topic = __('Hide the groupchat topic'); const i18n_bookmarked = __('This groupchat is bookmarked'); const subject = o.subject ? o.subject.text : ''; const show_subject = subject && !subject_hidden; const muc_vcard = (_el$model$vcard = el.model.vcard) === null || _el$model$vcard === void 0 ? void 0 : _el$model$vcard.get('image'); return $`
    ${muc_vcard && muc_vcard !== shared_converse.DEFAULT_IMAGE ? $` ` : ''}
    ${!shared_converse.api.settings.get("singleton") ? $`` : ''}
    ${el.model.getDisplayName()} ${o.bookmarked ? $`` : ''}
    ${until_c(el.getHeadingButtons(subject_hidden).then(buttons => { const dropdown_btns = buttons.filter(b => !b.standalone).map(b => getHeadingDropdownItem(b)); return $` ${buttons.filter(b => b.standalone).reverse().map(b => until_c(getHeadingStandaloneButton(b), ''))} ${dropdown_btns.length ? $`` : ''}`; }), '')}
    ${show_subject ? $`

    ` : ''} `; }); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/muc-views/styles/muc-head.scss var styles_muc_head = __webpack_require__(3288); ;// CONCATENATED MODULE: ./src/plugins/muc-views/styles/muc-head.scss var muc_head_options = {}; muc_head_options.styleTagTransform = (styleTagTransform_default()); muc_head_options.setAttributes = (setAttributesWithoutAttributes_default()); muc_head_options.insert = insertBySelector_default().bind(null, "head"); muc_head_options.domAPI = (styleDomAPI_default()); muc_head_options.insertStyleElement = (insertStyleElement_default()); var muc_head_update = injectStylesIntoStyleTag_default()(styles_muc_head/* default */.Z, muc_head_options); /* harmony default export */ const muc_views_styles_muc_head = (styles_muc_head/* default */.Z && styles_muc_head/* default.locals */.Z.locals ? styles_muc_head/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/muc-views/heading.js class MUCHeading extends CustomElement { async initialize() { this.model = shared_converse.chatboxes.get(this.getAttribute('jid')); this.listenTo(this.model, 'change', () => this.requestUpdate()); this.listenTo(this.model, 'vcard:add', () => this.requestUpdate()); this.listenTo(this.model, 'vcard:change', () => this.requestUpdate()); this.user_settings = await shared_converse.api.user.settings.getModel(); this.listenTo(this.user_settings, 'change:mucs_with_hidden_subject', () => this.requestUpdate()); await this.model.initialized; this.listenTo(this.model.features, 'change:open', () => this.requestUpdate()); this.model.occupants.forEach(o => this.onOccupantAdded(o)); this.listenTo(this.model.occupants, 'add', this.onOccupantAdded); this.listenTo(this.model.occupants, 'change:affiliation', this.onOccupantAffiliationChanged); this.requestUpdate(); } render() { return this.model && this.user_settings ? muc_head(this) : ''; } onOccupantAdded(occupant) { if (occupant.get('jid') === shared_converse.bare_jid) { this.requestUpdate(); } } onOccupantAffiliationChanged(occupant) { if (occupant.get('jid') === shared_converse.bare_jid) { this.requestUpdate(); } } showRoomDetailsModal(ev) { ev.preventDefault(); core_api.modal.show(modals_muc_details, { 'model': this.model }, ev); } showInviteModal(ev) { ev.preventDefault(); core_api.modal.show(modals_muc_invite, { 'model': new Model(), 'chatroomview': this }, ev); } toggleTopic(ev) { var _ev$preventDefault; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); this.model.toggleSubjectHiddenState(); } getAndRenderConfigurationForm() { this.model.session.set('view', core_converse.MUC.VIEWS.CONFIG); } close(ev) { ev.preventDefault(); this.model.close(); } destroy(ev) { ev.preventDefault(); destroyMUC(this.model); } /** * Returns a list of objects which represent buttons for the groupchat header. * @emits _converse#getHeadingButtons */ getHeadingButtons(subject_hidden) { const buttons = []; buttons.push({ 'i18n_text': __('Details'), 'i18n_title': __('Show more information about this groupchat'), 'handler': ev => this.showRoomDetailsModal(ev), 'a_class': 'show-muc-details-modal', 'icon_class': 'fa-info-circle', 'name': 'details' }); if (this.model.getOwnAffiliation() === 'owner') { buttons.push({ 'i18n_text': __('Configure'), 'i18n_title': __('Configure this groupchat'), 'handler': () => this.getAndRenderConfigurationForm(), 'a_class': 'configure-chatroom-button', 'icon_class': 'fa-wrench', 'name': 'configure' }); } buttons.push({ 'i18n_text': __('Nickname'), 'i18n_title': __("Change the nickname you're using in this groupchat"), 'handler': ev => core_api.modal.show(modals_nickname, { 'model': this.model }, ev), 'a_class': 'open-nickname-modal', 'icon_class': 'fa-smile', 'name': 'nickname' }); if (this.model.invitesAllowed()) { buttons.push({ 'i18n_text': __('Invite'), 'i18n_title': __('Invite someone to join this groupchat'), 'handler': ev => this.showInviteModal(ev), 'a_class': 'open-invite-modal', 'icon_class': 'fa-user-plus', 'name': 'invite' }); } const subject = this.model.get('subject'); if (subject && subject.text) { buttons.push({ 'i18n_text': subject_hidden ? __('Show topic') : __('Hide topic'), 'i18n_title': subject_hidden ? __('Show the topic message in the heading') : __('Hide the topic in the heading'), 'handler': ev => this.toggleTopic(ev), 'a_class': 'hide-topic', 'icon_class': 'fa-minus-square', 'name': 'toggle-topic' }); } const conn_status = this.model.session.get('connection_status'); if (conn_status === core_converse.ROOMSTATUS.ENTERED) { const allowed_commands = this.model.getAllowedCommands(); if (allowed_commands.includes('modtools')) { buttons.push({ 'i18n_text': __('Moderate'), 'i18n_title': __('Moderate this groupchat'), 'handler': () => showModeratorToolsModal(this.model), 'a_class': 'moderate-chatroom-button', 'icon_class': 'fa-user-cog', 'name': 'moderate' }); } if (allowed_commands.includes('destroy')) { buttons.push({ 'i18n_text': __('Destroy'), 'i18n_title': __('Remove this groupchat'), 'handler': ev => this.destroy(ev), 'a_class': 'destroy-chatroom-button', 'icon_class': 'fa-trash', 'name': 'destroy' }); } } if (!core_api.settings.get('singleton')) { buttons.push({ 'i18n_text': __('Leave'), 'i18n_title': __('Leave and close this groupchat'), 'handler': async ev => { ev.stopPropagation(); const messages = [__('Are you sure you want to leave this groupchat?')]; const result = await core_api.confirm(__('Confirm'), messages); result && this.close(ev); }, 'a_class': 'close-chatbox-button', 'standalone': core_api.settings.get('view_mode') === 'overlayed', 'icon_class': 'fa-sign-out-alt', 'name': 'signout' }); } const el = shared_converse.chatboxviews.get(this.getAttribute('jid')); if (el) { // This hook is described in src/plugins/chatview/heading.js return shared_converse.api.hook('getHeadingButtons', el, buttons); } else { return Promise.resolve(buttons); // Happens during tests } } } core_api.elements.define('converse-muc-heading', MUCHeading); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/muc-password-form.js /* harmony default export */ const muc_password_form = (o => { const i18n_heading = __('This groupchat requires a password'); const i18n_password = __('Password: '); const i18n_submit = __('Submit'); return $`

    ${o.validation_message}

    `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/password-form.js class MUCPasswordForm extends CustomElement { static get properties() { return { 'jid': { type: String } }; } connectedCallback() { super.connectedCallback(); this.model = shared_converse.chatboxes.get(this.jid); this.listenTo(this.model, 'change:password_validation_message', this.render); this.render(); } render() { return muc_password_form({ 'jid': this.model.get('jid'), 'submitPassword': ev => this.submitPassword(ev), 'validation_message': this.model.get('password_validation_message') }); } submitPassword(ev) { ev.preventDefault(); const password = this.querySelector('input[type=password]').value; this.model.join(this.model.get('nick'), password); this.model.set('password_validation_message', null); } } core_api.elements.define('converse-muc-password-form', MUCPasswordForm); /* harmony default export */ const password_form = ((/* unused pure expression or super */ null && (MUCPasswordForm))); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/muc.js /* harmony default export */ const templates_muc = (o => { return $`
    ${o.model ? $`
    ${getChatRoomBodyTemplate(o)}
    ` : ''}
    `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/muc.js function muc_defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } class MUCView extends BaseChatView { constructor() { super(...arguments); muc_defineProperty(this, "length", 300); muc_defineProperty(this, "is_chatroom", true); } async initialize() { this.model = await core_api.rooms.get(this.jid); shared_converse.chatboxviews.add(this.jid, this); this.setAttribute('id', this.model.get('box_id')); this.listenTo(shared_converse, 'windowStateChanged', this.onWindowStateChanged); this.listenTo(this.model, 'change:composing_spoiler', this.requestUpdateMessageForm); this.listenTo(this.model.session, 'change:connection_status', this.onConnectionStatusChanged); this.listenTo(this.model.session, 'change:view', this.requestUpdate); this.onConnectionStatusChanged(); this.model.maybeShow(); /** * Triggered once a {@link _converse.ChatRoomView} has been opened * @event _converse#chatRoomViewInitialized * @type { _converse.ChatRoomView } * @example _converse.api.listen.on('chatRoomViewInitialized', view => { ... }); */ core_api.trigger('chatRoomViewInitialized', this); } render() { return templates_muc({ 'model': this.model }); } onConnectionStatusChanged() { const conn_status = this.model.session.get('connection_status'); if (conn_status === core_converse.ROOMSTATUS.CONNECTING) { this.model.session.save({ 'disconnection_actor': undefined, 'disconnection_message': undefined, 'disconnection_reason': undefined }); this.model.save({ 'moved_jid': undefined, 'password_validation_message': undefined, 'reason': undefined }); } this.requestUpdate(); } } core_api.elements.define('converse-muc', MUCView); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/muc-views/styles/index.scss var muc_views_styles = __webpack_require__(3076); ;// CONCATENATED MODULE: ./src/plugins/muc-views/styles/index.scss var muc_views_styles_options = {}; muc_views_styles_options.styleTagTransform = (styleTagTransform_default()); muc_views_styles_options.setAttributes = (setAttributesWithoutAttributes_default()); muc_views_styles_options.insert = insertBySelector_default().bind(null, "head"); muc_views_styles_options.domAPI = (styleDomAPI_default()); muc_views_styles_options.insertStyleElement = (insertStyleElement_default()); var muc_views_styles_update = injectStylesIntoStyleTag_default()(muc_views_styles/* default */.Z, muc_views_styles_options); /* harmony default export */ const plugins_muc_views_styles = (muc_views_styles/* default */.Z && muc_views_styles/* default.locals */.Z.locals ? muc_views_styles/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/muc-views/index.js /** * @copyright The Converse.js developers * @description XEP-0045 Multi-User Chat Views * @license Mozilla Public License (MPLv2) */ const { Strophe: muc_views_Strophe } = core_converse.env; core_converse.MUC.VIEWS = { CONFIG: 'config-form' }; core_converse.plugins.add('converse-muc-views', { /* Dependencies are other plugins which might be * overridden or relied upon, and therefore need to be loaded before * this plugin. They are "optional" because they might not be * available, in which case any overrides applicable to them will be * ignored. * * NB: These plugins need to have already been loaded via require.js. * * It's possible to make these dependencies "non-optional". * If the setting "strict_plugin_dependencies" is set to true, * an error will be raised if the plugin is not found. */ dependencies: ['converse-modal', 'converse-controlbox', 'converse-chatview'], initialize() { const { _converse } = this; // Configuration values for this plugin // ==================================== // Refer to docs/source/configuration.rst for explanations of these // configuration settings. core_api.settings.extend({ 'auto_list_rooms': false, 'cache_muc_messages': true, 'locked_muc_nickname': false, 'modtools_disable_query': [], 'muc_disable_slash_commands': false, 'muc_mention_autocomplete_filter': 'contains', 'muc_mention_autocomplete_min_chars': 0, 'muc_mention_autocomplete_show_avatar': true, 'muc_roomid_policy': null, 'muc_roomid_policy_hint': null, 'roomconfig_whitelist': [], 'show_retraction_warning': true, 'visible_toolbar_buttons': { 'toggle_occupants': true } }); _converse.ChatRoomView = MUCView; if (!core_api.settings.get('muc_domain')) { // Use service discovery to get the default MUC domain core_api.listen.on('serviceDiscovered', async feature => { if ((feature === null || feature === void 0 ? void 0 : feature.get('var')) === muc_views_Strophe.NS.MUC) { if (feature.entity.get('jid').includes('@')) { // Ignore full JIDs, we're only looking for a MUC service, not a room return; } const identity = await feature.entity.getIdentity('conference', 'text'); if (identity) { core_api.settings.set('muc_domain', muc_views_Strophe.getDomainFromJid(feature.get('from'))); } } }); } core_api.listen.on('clearsession', () => { const view = _converse.chatboxviews.get('controlbox'); if (view && view.roomspanel) { view.roomspanel.model.destroy(); view.roomspanel.remove(); delete view.roomspanel; } }); core_api.listen.on('chatBoxClosed', model => { if (model.get('type') === _converse.CHATROOMS_TYPE) { utils_clearHistory(model.get('jid')); } }); core_api.listen.on('parseMessageForCommands', parseMessageForMUCCommands); } }); // EXTERNAL MODULE: ./node_modules/favico.js-slevomat/favico.js var favico = __webpack_require__(4023); var favico_default = /*#__PURE__*/__webpack_require__.n(favico); ;// CONCATENATED MODULE: ./src/plugins/notifications/utils.js const { Strophe: notifications_utils_Strophe } = core_converse.env; const supports_html5_notification = ('Notification' in window); core_converse.env.Favico = (favico_default()); let favicon; function isMessageToHiddenChat(attrs) { var _converse$chatboxes$g; return shared_converse.isTestEnv() || (((_converse$chatboxes$g = shared_converse.chatboxes.get(attrs.from)) === null || _converse$chatboxes$g === void 0 ? void 0 : _converse$chatboxes$g.isHidden()) ?? false); } function areDesktopNotificationsEnabled() { return shared_converse.isTestEnv() || supports_html5_notification && core_api.settings.get('show_desktop_notifications') && Notification.permission === 'granted'; } function clearFavicon() { var _navigator$clearAppBa, _navigator; favicon = null; (_navigator$clearAppBa = (_navigator = navigator).clearAppBadge) === null || _navigator$clearAppBa === void 0 ? void 0 : _navigator$clearAppBa.call(_navigator).catch(e => headless_log.error("Could not clear unread count in app badge " + e)); } function updateUnreadFavicon() { if (core_api.settings.get('show_tab_notifications')) { var _navigator$setAppBadg, _navigator2; favicon = favicon ?? new core_converse.env.Favico({ type: 'circle', animation: 'pop' }); const chats = shared_converse.chatboxes.models; const num_unread = chats.reduce((acc, chat) => acc + (chat.get('num_unread') || 0), 0); favicon.badge(num_unread); (_navigator$setAppBadg = (_navigator2 = navigator).setAppBadge) === null || _navigator$setAppBadg === void 0 ? void 0 : _navigator$setAppBadg.call(_navigator2, num_unread).catch(e => headless_log.error("Could set unread count in app badge - " + e)); } } function isReferenced(references, muc_jid, nick) { const check = r => [shared_converse.bare_jid, `${muc_jid}/${nick}`].includes(r.uri.replace(/^xmpp:/, '')); return references.reduce((acc, r) => acc || check(r), false); } /** * Is this a group message for which we should notify the user? * @private * @param { MUCMessageAttributes } attrs */ async function shouldNotifyOfGroupMessage(attrs) { if (!(attrs !== null && attrs !== void 0 && attrs.body) && !(attrs !== null && attrs !== void 0 && attrs.message)) { // attrs.message is used by 'info' messages return false; } const jid = attrs.from; const muc_jid = attrs.from_muc; const notify_all = core_api.settings.get('notify_all_room_messages'); const room = shared_converse.chatboxes.get(muc_jid); const resource = notifications_utils_Strophe.getResourceFromJid(jid); const sender = resource && notifications_utils_Strophe.unescapeNode(resource) || ''; let is_mentioned = false; const nick = room.get('nick'); if (core_api.settings.get('notify_nicknames_without_references')) { is_mentioned = new RegExp(`\\b${nick}\\b`).test(attrs.body); } const is_not_mine = sender !== nick; const should_notify_user = notify_all === true || Array.isArray(notify_all) && notify_all.includes(muc_jid) || isReferenced(attrs.references, muc_jid, nick) || is_mentioned; if (is_not_mine && !!should_notify_user) { /** * *Hook* which allows plugins to run further logic to determine * whether a notification should be sent out for this message. * @event _converse#shouldNotifyOfGroupMessage * @example * api.listen.on('shouldNotifyOfGroupMessage', (should_notify) => { * return should_notify && flurb === floob; * }); */ const should_notify = await core_api.hook('shouldNotifyOfGroupMessage', attrs, true); return should_notify; } return false; } async function shouldNotifyOfInfoMessage(attrs) { if (!attrs.from_muc) { return false; } const room = await core_api.rooms.get(attrs.from_muc); if (!room) { return false; } const nick = room.get('nick'); const muc_jid = attrs.from_muc; const notify_all = core_api.settings.get('notify_all_room_messages'); return notify_all === true || Array.isArray(notify_all) && notify_all.includes(muc_jid) || isReferenced(attrs.references, muc_jid, nick); } /** * @private * @async * @method shouldNotifyOfMessage * @param { MessageData|MUCMessageData } data */ function shouldNotifyOfMessage(data) { const { attrs } = data; if (!attrs || attrs.is_forwarded) { return false; } if (attrs['type'] === 'groupchat') { return shouldNotifyOfGroupMessage(attrs); } else if (attrs['type'] === 'info') { return shouldNotifyOfInfoMessage(attrs); } else if (attrs.is_headline) { // We want to show notifications for headline messages. return isMessageToHiddenChat(attrs); } const is_me = notifications_utils_Strophe.getBareJidFromJid(attrs.from) === shared_converse.bare_jid; return !isEmptyMessage(attrs) && !is_me && (core_api.settings.get('show_desktop_notifications') === 'all' || isMessageToHiddenChat(attrs)); } function showFeedbackNotification(data) { if (data.klass === 'error' || data.klass === 'warn') { const n = new Notification(data.subject, { body: data.message, lang: shared_converse.locale, icon: core_api.settings.get('notification_icon') }); setTimeout(n.close.bind(n), 5000); } } /** * Creates an HTML5 Notification to inform of a change in a * contact's chat state. */ function showChatStateNotification(contact) { var _api$settings$get; if ((_api$settings$get = core_api.settings.get('chatstate_notification_blacklist')) !== null && _api$settings$get !== void 0 && _api$settings$get.includes(contact.jid)) { // Don't notify if the user is being ignored. return; } const chat_state = contact.presence.get('show'); let message = null; if (chat_state === 'offline') { message = __('has gone offline'); } else if (chat_state === 'away') { message = __('has gone away'); } else if (chat_state === 'dnd') { message = __('is busy'); } else if (chat_state === 'online') { message = __('has come online'); } if (message === null) { return; } const n = new Notification(contact.getDisplayName(), { body: message, lang: shared_converse.locale, icon: core_api.settings.get('notification_icon') }); setTimeout(() => n.close(), 5000); } /** * Shows an HTML5 Notification with the passed in message * @private * @param { MessageData|MUCMessageData } data */ function showMessageNotification(data) { const { attrs } = data; if (attrs.is_error) { return; } if (!areDesktopNotificationsEnabled()) { return; } let title, roster_item; const full_from_jid = attrs.from; const from_jid = notifications_utils_Strophe.getBareJidFromJid(full_from_jid); if (attrs.type == 'info') { title = attrs.message; } else if (attrs.type === 'headline') { if (!from_jid.includes('@') || core_api.settings.get('allow_non_roster_messaging')) { title = __('Notification from %1$s', from_jid); } else { return; } } else if (!from_jid.includes('@')) { // workaround for Prosody which doesn't give type "headline" title = __('Notification from %1$s', from_jid); } else if (attrs.type === 'groupchat') { title = __('%1$s says', notifications_utils_Strophe.getResourceFromJid(full_from_jid)); } else { if (shared_converse.roster === undefined) { headless_log.error('Could not send notification, because roster is undefined'); return; } roster_item = shared_converse.roster.get(from_jid); if (roster_item !== undefined) { title = __('%1$s says', roster_item.getDisplayName()); } else { if (core_api.settings.get('allow_non_roster_messaging')) { title = __('%1$s says', from_jid); } else { return; } } } let body; if (attrs.type == 'info') { body = attrs.reason; } else { body = attrs.is_encrypted ? attrs.plaintext : attrs.body; if (!body) { return; } } const n = new Notification(title, { 'body': body, 'lang': shared_converse.locale, 'icon': core_api.settings.get('notification_icon'), 'requireInteraction': !core_api.settings.get('notification_delay') }); if (core_api.settings.get('notification_delay')) { setTimeout(() => n.close(), core_api.settings.get('notification_delay')); } n.onclick = function (event) { event.preventDefault(); window.focus(); const chat = shared_converse.chatboxes.get(from_jid); chat.maybeShow(true); }; } function playSoundNotification() { if (core_api.settings.get('play_sounds') && window.Audio !== undefined) { const audioOgg = new Audio(core_api.settings.get('sounds_path') + 'msg_received.ogg'); const canPlayOgg = audioOgg.canPlayType('audio/ogg'); if (canPlayOgg === 'probably') { return audioOgg.play(); } const audioMp3 = new Audio(core_api.settings.get('sounds_path') + 'msg_received.mp3'); const canPlayMp3 = audioMp3.canPlayType('audio/mp3'); if (canPlayMp3 === 'probably') { audioMp3.play(); } else if (canPlayOgg === 'maybe') { audioOgg.play(); } else if (canPlayMp3 === 'maybe') { audioMp3.play(); } } } /** * Event handler for the on('message') event. Will call methods * to play sounds and show HTML5 notifications. */ async function handleMessageNotification(data) { if (!(await shouldNotifyOfMessage(data))) { return false; } /** * Triggered when a notification (sound or HTML5 notification) for a new * message has will be made. * @event _converse#messageNotification * @type { MessageData|MUCMessageData} * @example _converse.api.listen.on('messageNotification', data => { ... }); */ core_api.trigger('messageNotification', data); playSoundNotification(); showMessageNotification(data); } function handleFeedback(data) { if (areDesktopNotificationsEnabled(true)) { showFeedbackNotification(data); } } /** * Event handler for on('contactPresenceChanged'). * Will show an HTML5 notification to indicate that the chat status has changed. */ function handleChatStateNotification(contact) { if (areDesktopNotificationsEnabled() && core_api.settings.get('show_chat_state_notifications')) { showChatStateNotification(contact); } } function showContactRequestNotification(contact) { const n = new Notification(contact.getDisplayName(), { body: __('wants to be your contact'), lang: shared_converse.locale, icon: core_api.settings.get('notification_icon') }); setTimeout(() => n.close(), 5000); } function handleContactRequestNotification(contact) { if (areDesktopNotificationsEnabled(true)) { showContactRequestNotification(contact); } } function requestPermission() { if (supports_html5_notification && !['denied', 'granted'].includes(Notification.permission)) { // Ask user to enable HTML5 notifications Notification.requestPermission(); } } ;// CONCATENATED MODULE: ./src/plugins/notifications/index.js /** * @module converse-notification * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.plugins.add('converse-notification', { dependencies: ['converse-chatboxes'], initialize() { core_api.settings.extend({ // ^ a list of JIDs to ignore concerning chat state notifications chatstate_notification_blacklist: [], notification_delay: 5000, notification_icon: 'logo/conversejs-filled.svg', notify_all_room_messages: false, notify_nicknames_without_references: false, play_sounds: true, show_chat_state_notifications: false, show_desktop_notifications: true, show_tab_notifications: true, sounds_path: core_api.settings.get('assets_path') + '/sounds/' }); /************************ Event Handlers ************************/ core_api.listen.on('clearSession', clearFavicon); // Needed for tests core_api.waitUntil('chatBoxesInitialized').then(() => shared_converse.chatboxes.on('change:num_unread', updateUnreadFavicon)); core_api.listen.on('pluginsInitialized', function () { // We only register event handlers after all plugins are // registered, because other plugins might override some of our // handlers. core_api.listen.on('contactRequest', handleContactRequestNotification); core_api.listen.on('contactPresenceChanged', handleChatStateNotification); core_api.listen.on('message', handleMessageNotification); core_api.listen.on('feedback', handleFeedback); core_api.listen.on('connected', requestPermission); }); } }); ;// CONCATENATED MODULE: ./src/utils/file.js const MIMETYPES_MAP = { 'aac': 'audio/aac', 'abw': 'application/x-abiword', 'arc': 'application/x-freearc', 'avi': 'video/x-msvideo', 'azw': 'application/vnd.amazon.ebook', 'bin': 'application/octet-stream', 'bmp': 'image/bmp', 'bz': 'application/x-bzip', 'bz2': 'application/x-bzip2', 'cda': 'application/x-cdf', 'csh': 'application/x-csh', 'css': 'text/css', 'csv': 'text/csv', 'doc': 'application/msword', 'docx': 'application/vnd.openxmlformats-officedocument.wordprocessingml.document', 'eot': 'application/vnd.ms-fontobject', 'epub': 'application/epub+zip', 'gif': 'image/gif', 'gz': 'application/gzip', 'htm': 'text/html', 'html': 'text/html', 'ico': 'image/vnd.microsoft.icon', 'ics': 'text/calendar', 'jar': 'application/java-archive', 'jpeg': 'image/jpeg', 'jpg': 'image/jpeg', 'js': 'text/javascript', 'json': 'application/json', 'jsonld': 'application/ld+json', 'm4a': 'audio/mp4', 'mid': 'audio/midi', 'midi': 'audio/midi', 'mjs': 'text/javascript', 'mp3': 'audio/mpeg', 'mp4': 'video/mp4', 'mpeg': 'video/mpeg', 'mpkg': 'application/vnd.apple.installer+xml', 'odp': 'application/vnd.oasis.opendocument.presentation', 'ods': 'application/vnd.oasis.opendocument.spreadsheet', 'odt': 'application/vnd.oasis.opendocument.text', 'oga': 'audio/ogg', 'ogv': 'video/ogg', 'ogx': 'application/ogg', 'opus': 'audio/opus', 'otf': 'font/otf', 'png': 'image/png', 'pdf': 'application/pdf', 'php': 'application/x-httpd-php', 'ppt': 'application/vnd.ms-powerpoint', 'pptx': 'application/vnd.openxmlformats-officedocument.presentationml.presentation', 'rar': 'application/vnd.rar', 'rtf': 'application/rtf', 'sh': 'application/x-sh', 'svg': 'image/svg+xml', 'swf': 'application/x-shockwave-flash', 'tar': 'application/x-tar', 'tif': 'image/tiff', 'tiff': 'image/tiff', 'ts': 'video/mp2t', 'ttf': 'font/ttf', 'txt': 'text/plain', 'vsd': 'application/vnd.visio', 'wav': 'audio/wav', 'weba': 'audio/webm', 'webm': 'video/webm', 'webp': 'image/webp', 'woff': 'font/woff', 'woff2': 'font/woff2', 'xhtml': 'application/xhtml+xml', 'xls': 'application/vnd.ms-excel', 'xlsx': 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet', 'xml': 'text/xml', 'xul': 'application/vnd.mozilla.xul+xml', 'zip': 'application/zip', '3gp': 'video/3gpp', '3g2': 'video/3gpp2', '7z': 'application/x-7z-compressed' }; ;// CONCATENATED MODULE: ./src/plugins/omemo/consts.js const UNDECIDED = 0; const TRUSTED = 1; const UNTRUSTED = -1; const TAG_LENGTH = 128; const KEY_ALGO = { 'name': 'AES-GCM', 'length': 128 }; ;// CONCATENATED MODULE: ./node_modules/lodash-es/concat.js /** * Creates a new array concatenating `array` with any additional arrays * and/or values. * * @static * @memberOf _ * @since 4.0.0 * @category Array * @param {Array} array The array to concatenate. * @param {...*} [values] The values to concatenate. * @returns {Array} Returns the new concatenated array. * @example * * var array = [1]; * var other = _.concat(array, 2, [3], [[4]]); * * console.log(other); * // => [1, 2, 3, [4]] * * console.log(array); * // => [1] */ function concat() { var length = arguments.length; if (!length) { return []; } var args = Array(length - 1), array = arguments[0], index = length; while (index--) { args[index - 1] = arguments[index]; } return _arrayPush(lodash_es_isArray(array) ? _copyArray(array) : [array], _baseFlatten(args, 1)); } /* harmony default export */ const lodash_es_concat = (concat); ;// CONCATENATED MODULE: ./src/headless/utils/arraybuffer.js const { u: arraybuffer_u } = core_converse.env; function appendArrayBuffer(buffer1, buffer2) { const tmp = new Uint8Array(buffer1.byteLength + buffer2.byteLength); tmp.set(new Uint8Array(buffer1), 0); tmp.set(new Uint8Array(buffer2), buffer1.byteLength); return tmp.buffer; } function arrayBufferToHex(ab) { // https://stackoverflow.com/questions/40031688/javascript-arraybuffer-to-hex#40031979 return Array.prototype.map.call(new Uint8Array(ab), x => ('00' + x.toString(16)).slice(-2)).join(''); } function arrayBufferToString(ab) { return new TextDecoder("utf-8").decode(ab); } function stringToArrayBuffer(string) { const bytes = new TextEncoder("utf-8").encode(string); return bytes.buffer; } function arrayBufferToBase64(ab) { return btoa(new Uint8Array(ab).reduce((data, byte) => data + String.fromCharCode(byte), '')); } function base64ToArrayBuffer(b64) { const binary_string = window.atob(b64), len = binary_string.length, bytes = new Uint8Array(len); for (let i = 0; i < len; i++) { bytes[i] = binary_string.charCodeAt(i); } return bytes.buffer; } function hexToArrayBuffer(hex) { const typedArray = new Uint8Array(hex.match(/[\da-f]{2}/gi).map(h => parseInt(h, 16))); return typedArray.buffer; } Object.assign(arraybuffer_u, { arrayBufferToHex, arrayBufferToString, stringToArrayBuffer, arrayBufferToBase64, base64ToArrayBuffer }); ;// CONCATENATED MODULE: ./src/plugins/omemo/utils.js /* global libsignal */ const { Strophe: omemo_utils_Strophe, URI: utils_URI, sizzle: omemo_utils_sizzle, u: omemo_utils_u } = core_converse.env; function formatFingerprint(fp) { fp = fp.replace(/^05/, ''); for (let i = 1; i < 8; i++) { const idx = i * 8 + i - 1; fp = fp.slice(0, idx) + ' ' + fp.slice(idx); } return fp; } function handleMessageSendError(e, chat) { if (e.name === 'IQError') { chat.save('omemo_supported', false); const err_msgs = []; if (omemo_utils_sizzle(`presence-subscription-required[xmlns="${omemo_utils_Strophe.NS.PUBSUB_ERROR}"]`, e.iq).length) { err_msgs.push(__("Sorry, we're unable to send an encrypted message because %1$s " + 'requires you to be subscribed to their presence in order to see their OMEMO information', e.iq.getAttribute('from'))); } else if (omemo_utils_sizzle(`remote-server-not-found[xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"]`, e.iq).length) { err_msgs.push(__("Sorry, we're unable to send an encrypted message because the remote server for %1$s could not be found", e.iq.getAttribute('from'))); } else { err_msgs.push(__('Unable to send an encrypted message due to an unexpected error.')); err_msgs.push(e.iq.outerHTML); } core_api.alert('error', __('Error'), err_msgs); } else if (e.user_facing) { core_api.alert('error', __('Error'), [e.message]); } throw e; } function getOutgoingMessageAttributes(chat, attrs) { if (chat.get('omemo_active') && attrs.body) { attrs['is_encrypted'] = true; attrs['plaintext'] = attrs.body; attrs['body'] = __('This is an OMEMO encrypted message which your client doesn’t seem to support. ' + 'Find more information on https://conversations.im/omemo'); } return attrs; } async function encryptMessage(plaintext) { // The client MUST use fresh, randomly generated key/IV pairs // with AES-128 in Galois/Counter Mode (GCM). // For GCM a 12 byte IV is strongly suggested as other IV lengths // will require additional calculations. In principle any IV size // can be used as long as the IV doesn't ever repeat. NIST however // suggests that only an IV size of 12 bytes needs to be supported // by implementations. // // https://crypto.stackexchange.com/questions/26783/ciphertext-and-tag-size-and-iv-transmission-with-aes-in-gcm-mode const iv = crypto.getRandomValues(new window.Uint8Array(12)); const key = await crypto.subtle.generateKey(KEY_ALGO, true, ['encrypt', 'decrypt']); const algo = { 'name': 'AES-GCM', 'iv': iv, 'tagLength': TAG_LENGTH }; const encrypted = await crypto.subtle.encrypt(algo, key, stringToArrayBuffer(plaintext)); const length = encrypted.byteLength - (128 + 7 >> 3); const ciphertext = encrypted.slice(0, length); const tag = encrypted.slice(length); const exported_key = await crypto.subtle.exportKey('raw', key); return { 'key': exported_key, 'tag': tag, 'key_and_tag': appendArrayBuffer(exported_key, tag), 'payload': arrayBufferToBase64(ciphertext), 'iv': arrayBufferToBase64(iv) }; } async function decryptMessage(obj) { const key_obj = await crypto.subtle.importKey('raw', obj.key, KEY_ALGO, true, ['encrypt', 'decrypt']); const cipher = appendArrayBuffer(base64ToArrayBuffer(obj.payload), obj.tag); const algo = { 'name': 'AES-GCM', 'iv': base64ToArrayBuffer(obj.iv), 'tagLength': TAG_LENGTH }; return arrayBufferToString(await crypto.subtle.decrypt(algo, key_obj, cipher)); } async function encryptFile(file) { const iv = crypto.getRandomValues(new Uint8Array(12)); const key = await crypto.subtle.generateKey({ name: 'AES-GCM', length: 256 }, true, ['encrypt', 'decrypt']); const encrypted = await crypto.subtle.encrypt({ name: 'AES-GCM', iv }, key, await file.arrayBuffer()); const exported_key = await window.crypto.subtle.exportKey('raw', key); const encrypted_file = new File([encrypted], file.name, { type: file.type, lastModified: file.lastModified }); encrypted_file.xep454_ivkey = arrayBufferToHex(iv) + arrayBufferToHex(exported_key); return encrypted_file; } function setEncryptedFileURL(message, attrs) { const url = attrs.oob_url.replace(/^https?:/, 'aesgcm:') + '#' + message.file.xep454_ivkey; return Object.assign(attrs, { 'oob_url': null, // Since only the body gets encrypted, we don't set the oob_url 'message': url, 'body': url }); } async function decryptFile(iv, key, cipher) { const key_obj = await crypto.subtle.importKey('raw', hexToArrayBuffer(key), 'AES-GCM', false, ['decrypt']); const algo = { 'name': 'AES-GCM', 'iv': hexToArrayBuffer(iv) }; return crypto.subtle.decrypt(algo, key_obj, cipher); } async function downloadFile(url) { let response; try { response = await fetch(url); } catch (e) { headless_log.error(`${e.name}: Failed to download encrypted media: ${url}`); headless_log.error(e); return null; } if (response.status >= 200 && response.status < 400) { return response.arrayBuffer(); } } async function getAndDecryptFile(uri) { var _uri$filename; const hash = uri.hash().slice(1); const protocol = window.location.hostname === 'localhost' ? 'http' : 'https'; const http_url = uri.toString().replace(/^aesgcm/, protocol); const cipher = await downloadFile(http_url); if (cipher === null) { headless_log.error(`Could not decrypt file ${uri.toString()} since it could not be downloaded`); return null; } const iv = hash.slice(0, 24); const key = hash.slice(24); let content; try { content = await decryptFile(iv, key, cipher); } catch (e) { headless_log.error(`Could not decrypt file ${uri.toString()}`); headless_log.error(e); return null; } const [filename, extension] = (_uri$filename = uri.filename()) === null || _uri$filename === void 0 ? void 0 : _uri$filename.split('.'); const mimetype = MIMETYPES_MAP[extension]; try { const file = new File([content], filename, { 'type': mimetype }); return URL.createObjectURL(file); } catch (e) { headless_log.error(`Could not decrypt file ${uri.toString()}`); headless_log.error(e); return null; } } function getTemplateForObjectURL(uri, obj_url, richtext) { const file_url = uri.toString(); if (obj_url === null) { return file_url; } if (isImageURL(file_url)) { return src_templates_image({ 'src': obj_url, 'onClick': richtext.onImgClick, 'onLoad': richtext.onImgLoad }); } else if (isAudioURL(file_url)) { return audio(obj_url); } else if (isVideoURL(file_url)) { return video(obj_url); } else { return file(obj_url, uri.filename()); } } function addEncryptedFiles(text, offset, richtext) { const objs = []; try { const parse_options = { 'start': /\b(aesgcm:\/\/)/gi }; utils_URI.withinString(text, (url, start, end) => { objs.push({ url, start, end }); return url; }, parse_options); } catch (error) { headless_log.debug(error); return; } objs.forEach(o => { const uri = getURI(text.slice(o.start, o.end)); const promise = getAndDecryptFile(uri).then(obj_url => getTemplateForObjectURL(uri, obj_url, richtext)); const template = $`${until_c(promise, '')}`; richtext.addTemplateResult(o.start + offset, o.end + offset, template); }); } function handleEncryptedFiles(richtext) { if (!shared_converse.config.get('trusted')) { return; } richtext.addAnnotations((text, offset) => addEncryptedFiles(text, offset, richtext)); } /** * Hook handler for { @link parseMessage } and { @link parseMUCMessage }, which * parses the passed in `message` stanza for OMEMO attributes and then sets * them on the attrs object. * @param { XMLElement } stanza - The message stanza * @param { (MUCMessageAttributes|MessageAttributes) } attrs * @returns (MUCMessageAttributes|MessageAttributes) */ async function parseEncryptedMessage(stanza, attrs) { var _api$omemo; if (core_api.settings.get('clear_cache_on_logout') || !attrs.is_encrypted || attrs.encryption_namespace !== omemo_utils_Strophe.NS.OMEMO) { return attrs; } const encrypted_el = omemo_utils_sizzle(`encrypted[xmlns="${omemo_utils_Strophe.NS.OMEMO}"]`, stanza).pop(); const header = encrypted_el.querySelector('header'); attrs.encrypted = { 'device_id': header.getAttribute('sid') }; const device_id = await ((_api$omemo = core_api.omemo) === null || _api$omemo === void 0 ? void 0 : _api$omemo.getDeviceID()); const key = device_id && omemo_utils_sizzle(`key[rid="${device_id}"]`, encrypted_el).pop(); if (key) { var _encrypted_el$querySe; Object.assign(attrs.encrypted, { 'iv': header.querySelector('iv').textContent, 'key': key.textContent, 'payload': ((_encrypted_el$querySe = encrypted_el.querySelector('payload')) === null || _encrypted_el$querySe === void 0 ? void 0 : _encrypted_el$querySe.textContent) || null, 'prekey': ['true', '1'].includes(key.getAttribute('prekey')) }); } else { return Object.assign(attrs, { 'error_condition': 'not-encrypted-for-this-device', 'error_type': 'Decryption', 'is_ephemeral': true, 'is_error': true, 'type': 'error' }); } // https://xmpp.org/extensions/xep-0384.html#usecases-receiving if (attrs.encrypted.prekey === true) { return decryptPrekeyWhisperMessage(attrs); } else { return decryptWhisperMessage(attrs); } } function utils_onChatBoxesInitialized() { shared_converse.chatboxes.on('add', chatbox => { checkOMEMOSupported(chatbox); if (chatbox.get('type') === shared_converse.CHATROOMS_TYPE) { chatbox.occupants.on('add', o => onOccupantAdded(chatbox, o)); chatbox.features.on('change', () => checkOMEMOSupported(chatbox)); } }); } function onChatInitialized(el) { el.listenTo(el.model.messages, 'add', message => { if (message.get('is_encrypted') && !message.get('is_error')) { el.model.save('omemo_supported', true); } }); el.listenTo(el.model, 'change:omemo_supported', () => { if (!el.model.get('omemo_supported') && el.model.get('omemo_active')) { el.model.set('omemo_active', false); } else { var _el$querySelector; // Manually trigger an update, setting omemo_active to // false above will automatically trigger one. (_el$querySelector = el.querySelector('converse-chat-toolbar')) === null || _el$querySelector === void 0 ? void 0 : _el$querySelector.requestUpdate(); } }); el.listenTo(el.model, 'change:omemo_active', () => { el.querySelector('converse-chat-toolbar').requestUpdate(); }); } function getSessionCipher(jid, id) { const address = new libsignal.SignalProtocolAddress(jid, id); return new window.libsignal.SessionCipher(shared_converse.omemo_store, address); } function getJIDForDecryption(attrs) { const from_jid = attrs.from_muc ? attrs.from_real_jid : attrs.from; if (!from_jid) { Object.assign(attrs, { 'error_text': __("Sorry, could not decrypt a received OMEMO " + "message because we don't have the XMPP address for that user."), 'error_type': 'Decryption', 'is_ephemeral': true, 'is_error': true, 'type': 'error' }); throw new Error("Could not find JID to decrypt OMEMO message for"); } return from_jid; } async function handleDecryptedWhisperMessage(attrs, key_and_tag) { const from_jid = getJIDForDecryption(attrs); const devicelist = await core_api.omemo.devicelists.get(from_jid, true); const encrypted = attrs.encrypted; let device = devicelist.devices.get(encrypted.device_id); if (!device) { device = await devicelist.devices.create({ 'id': encrypted.device_id, 'jid': from_jid }, { 'promise': true }); } if (encrypted.payload) { const key = key_and_tag.slice(0, 16); const tag = key_and_tag.slice(16); const result = await omemo.decryptMessage(Object.assign(encrypted, { 'key': key, 'tag': tag })); device.save('active', true); return result; } } function getDecryptionErrorAttributes(e) { return { 'error_text': __('Sorry, could not decrypt a received OMEMO message due to an error.') + ` ${e.name} ${e.message}`, 'error_condition': e.name, 'error_message': e.message, 'error_type': 'Decryption', 'is_ephemeral': true, 'is_error': true, 'type': 'error' }; } async function decryptPrekeyWhisperMessage(attrs) { const from_jid = getJIDForDecryption(attrs); const session_cipher = getSessionCipher(from_jid, parseInt(attrs.encrypted.device_id, 10)); const key = base64ToArrayBuffer(attrs.encrypted.key); let key_and_tag; try { key_and_tag = await session_cipher.decryptPreKeyWhisperMessage(key, 'binary'); } catch (e) { // TODO from the XEP: // There are various reasons why decryption of an // OMEMOKeyExchange or an OMEMOAuthenticatedMessage // could fail. One reason is if the message was // received twice and already decrypted once, in this // case the client MUST ignore the decryption failure // and not show any warnings/errors. In all other cases // of decryption failure, clients SHOULD respond by // forcibly doing a new key exchange and sending a new // OMEMOKeyExchange with a potentially empty SCE // payload. By building a new session with the original // sender this way, the invalid session of the original // sender will get overwritten with this newly created, // valid session. headless_log.error(`${e.name} ${e.message}`); return Object.assign(attrs, getDecryptionErrorAttributes(e)); } // TODO from the XEP: // When a client receives the first message for a given // ratchet key with a counter of 53 or higher, it MUST send // a heartbeat message. Heartbeat messages are normal OMEMO // encrypted messages where the SCE payload does not include // any elements. These heartbeat messages cause the ratchet // to forward, thus consequent messages will have the // counter restarted from 0. try { const plaintext = await handleDecryptedWhisperMessage(attrs, key_and_tag); await shared_converse.omemo_store.generateMissingPreKeys(); await shared_converse.omemo_store.publishBundle(); if (plaintext) { return Object.assign(attrs, { 'plaintext': plaintext }); } else { return Object.assign(attrs, { 'is_only_key': true }); } } catch (e) { headless_log.error(`${e.name} ${e.message}`); return Object.assign(attrs, getDecryptionErrorAttributes(e)); } } async function decryptWhisperMessage(attrs) { const from_jid = getJIDForDecryption(attrs); const session_cipher = getSessionCipher(from_jid, parseInt(attrs.encrypted.device_id, 10)); const key = base64ToArrayBuffer(attrs.encrypted.key); try { const key_and_tag = await session_cipher.decryptWhisperMessage(key, 'binary'); const plaintext = await handleDecryptedWhisperMessage(attrs, key_and_tag); return Object.assign(attrs, { 'plaintext': plaintext }); } catch (e) { headless_log.error(`${e.name} ${e.message}`); return Object.assign(attrs, getDecryptionErrorAttributes(e)); } } function addKeysToMessageStanza(stanza, dicts, iv) { for (const i in dicts) { if (Object.prototype.hasOwnProperty.call(dicts, i)) { const payload = dicts[i].payload; const device = dicts[i].device; const prekey = 3 == parseInt(payload.type, 10); stanza.c('key', { 'rid': device.get('id') }).t(btoa(payload.body)); if (prekey) { stanza.attrs({ 'prekey': prekey }); } stanza.up(); if (i == dicts.length - 1) { stanza.c('iv').t(iv).up().up(); } } } return Promise.resolve(stanza); } /** * Given an XML element representing a user's OMEMO bundle, parse it * and return a map. */ function parseBundle(bundle_el) { const signed_prekey_public_el = bundle_el.querySelector('signedPreKeyPublic'); const signed_prekey_signature_el = bundle_el.querySelector('signedPreKeySignature'); const prekeys = omemo_utils_sizzle(`prekeys > preKeyPublic`, bundle_el).map(el => ({ 'id': parseInt(el.getAttribute('preKeyId'), 10), 'key': el.textContent })); return { 'identity_key': bundle_el.querySelector('identityKey').textContent.trim(), 'signed_prekey': { 'id': parseInt(signed_prekey_public_el.getAttribute('signedPreKeyId'), 10), 'public_key': signed_prekey_public_el.textContent, 'signature': signed_prekey_signature_el.textContent }, 'prekeys': prekeys }; } async function generateFingerprint(device) { var _device$get; if ((_device$get = device.get('bundle')) !== null && _device$get !== void 0 && _device$get.fingerprint) { return; } const bundle = await device.getBundle(); bundle['fingerprint'] = arrayBufferToHex(base64ToArrayBuffer(bundle['identity_key'])); device.save('bundle', bundle); device.trigger('change:bundle'); // Doesn't get triggered automatically due to pass-by-reference } async function getDevicesForContact(jid) { await core_api.waitUntil('OMEMOInitialized'); const devicelist = await core_api.omemo.devicelists.get(jid, true); await devicelist.fetchDevices(); return devicelist.devices; } async function generateDeviceID() { /* Generates a device ID, making sure that it's unique */ const devicelist = await core_api.omemo.devicelists.get(shared_converse.bare_jid, true); const existing_ids = devicelist.devices.pluck('id'); let device_id = libsignal.KeyHelper.generateRegistrationId(); // Before publishing a freshly generated device id for the first time, // a device MUST check whether that device id already exists, and if so, generate a new one. let i = 0; while (existing_ids.includes(device_id)) { device_id = libsignal.KeyHelper.generateRegistrationId(); i++; if (i === 10) { throw new Error('Unable to generate a unique device ID'); } } return device_id.toString(); } async function buildSession(device) { const address = new libsignal.SignalProtocolAddress(device.get('jid'), device.get('id')); const sessionBuilder = new libsignal.SessionBuilder(shared_converse.omemo_store, address); const prekey = device.getRandomPreKey(); const bundle = await device.getBundle(); return sessionBuilder.processPreKey({ 'registrationId': parseInt(device.get('id'), 10), 'identityKey': base64ToArrayBuffer(bundle.identity_key), 'signedPreKey': { 'keyId': bundle.signed_prekey.id, // 'publicKey': base64ToArrayBuffer(bundle.signed_prekey.public_key), 'signature': base64ToArrayBuffer(bundle.signed_prekey.signature) }, 'preKey': { 'keyId': prekey.id, // 'publicKey': base64ToArrayBuffer(prekey.key) } }); } async function getSession(device) { if (!device.get('bundle')) { headless_log.error(`Could not build an OMEMO session for device ${device.get('id')} because we don't have its bundle`); return null; } const address = new libsignal.SignalProtocolAddress(device.get('jid'), device.get('id')); const session = await shared_converse.omemo_store.loadSession(address.toString()); if (session) { return session; } else { try { const session = await buildSession(device); return session; } catch (e) { headless_log.error(`Could not build an OMEMO session for device ${device.get('id')}`); headless_log.error(e); return null; } } } async function updateBundleFromStanza(stanza) { const items_el = omemo_utils_sizzle(`items`, stanza).pop(); if (!items_el || !items_el.getAttribute('node').startsWith(omemo_utils_Strophe.NS.OMEMO_BUNDLES)) { return; } const device_id = items_el.getAttribute('node').split(':')[1]; const jid = stanza.getAttribute('from'); const bundle_el = omemo_utils_sizzle(`item > bundle`, items_el).pop(); const devicelist = await core_api.omemo.devicelists.get(jid, true); const device = devicelist.devices.get(device_id) || devicelist.devices.create({ 'id': device_id, jid }); device.save({ 'bundle': parseBundle(bundle_el) }); } async function updateDevicesFromStanza(stanza) { const items_el = omemo_utils_sizzle(`items[node="${omemo_utils_Strophe.NS.OMEMO_DEVICELIST}"]`, stanza).pop(); if (!items_el) { return; } const device_selector = `item list[xmlns="${omemo_utils_Strophe.NS.OMEMO}"] device`; const device_ids = omemo_utils_sizzle(device_selector, items_el).map(d => d.getAttribute('id')); const jid = stanza.getAttribute('from'); const devicelist = await core_api.omemo.devicelists.get(jid, true); const devices = devicelist.devices; const removed_ids = lodash_es_difference(devices.pluck('id'), device_ids); removed_ids.forEach(id => { if (jid === shared_converse.bare_jid && id === shared_converse.omemo_store.get('device_id')) { return; // We don't set the current device as inactive } devices.get(id).save('active', false); }); device_ids.forEach(device_id => { const device = devices.get(device_id); if (device) { device.save('active', true); } else { devices.create({ 'id': device_id, 'jid': jid }); } }); if (omemo_utils_u.isSameBareJID(jid, shared_converse.bare_jid)) { // Make sure our own device is on the list // (i.e. if it was removed, add it again). devicelist.publishCurrentDevice(device_ids); } } function registerPEPPushHandler() { // Add a handler for devices pushed from other connected clients shared_converse.connection.addHandler(async message => { try { if (omemo_utils_sizzle(`event[xmlns="${omemo_utils_Strophe.NS.PUBSUB}#event"]`, message).length) { await core_api.waitUntil('OMEMOInitialized'); await updateDevicesFromStanza(message); await updateBundleFromStanza(message); } } catch (e) { headless_log.error(e.message); } return true; }, null, 'message', 'headline'); } async function restoreOMEMOSession() { if (shared_converse.omemo_store === undefined) { const id = `converse.omemosession-${shared_converse.bare_jid}`; shared_converse.omemo_store = new shared_converse.OMEMOStore({ id }); initStorage(shared_converse.omemo_store, id); } await shared_converse.omemo_store.fetchSession(); } async function fetchDeviceLists() { shared_converse.devicelists = new shared_converse.DeviceLists(); const id = `converse.devicelists-${shared_converse.bare_jid}`; initStorage(shared_converse.devicelists, id); await new Promise(resolve => { shared_converse.devicelists.fetch({ 'success': resolve, 'error': (_m, e) => { headless_log.error(e); resolve(); } }); }); // Call API method to wait for our own device list to be fetched from the // server or to be created. If we have no pre-existing OMEMO session, this // will cause a new device and bundle to be generated and published. await core_api.omemo.devicelists.get(shared_converse.bare_jid, true); } async function initOMEMO(reconnecting) { if (reconnecting) { return; } if (!shared_converse.config.get('trusted') || core_api.settings.get('clear_cache_on_logout')) { headless_log.warn('Not initializing OMEMO, since this browser is not trusted or clear_cache_on_logout is set to true'); return; } try { await fetchDeviceLists(); await restoreOMEMOSession(); await shared_converse.omemo_store.publishBundle(); } catch (e) { headless_log.error('Could not initialize OMEMO support'); headless_log.error(e); return; } /** * Triggered once OMEMO support has been initialized * @event _converse#OMEMOInitialized * @example _converse.api.listen.on('OMEMOInitialized', () => { ... }); */ core_api.trigger('OMEMOInitialized'); } async function onOccupantAdded(chatroom, occupant) { if (occupant.isSelf() || !chatroom.features.get('nonanonymous') || !chatroom.features.get('membersonly')) { return; } if (chatroom.get('omemo_active')) { const supported = await shared_converse.contactHasOMEMOSupport(occupant.get('jid')); if (!supported) { chatroom.createMessage({ 'message': __("%1$s doesn't appear to have a client that supports OMEMO. " + 'Encrypted chat will no longer be possible in this grouchat.', occupant.get('nick')), 'type': 'error' }); chatroom.save({ 'omemo_active': false, 'omemo_supported': false }); } } } async function checkOMEMOSupported(chatbox) { let supported; if (chatbox.get('type') === shared_converse.CHATROOMS_TYPE) { await core_api.waitUntil('OMEMOInitialized'); supported = chatbox.features.get('nonanonymous') && chatbox.features.get('membersonly'); } else if (chatbox.get('type') === shared_converse.PRIVATE_CHAT_TYPE) { supported = await shared_converse.contactHasOMEMOSupport(chatbox.get('jid')); } chatbox.set('omemo_supported', supported); if (supported && core_api.settings.get('omemo_default')) { chatbox.set('omemo_active', true); } } function toggleOMEMO(ev) { ev.stopPropagation(); ev.preventDefault(); const toolbar_el = omemo_utils_u.ancestor(ev.target, 'converse-chat-toolbar'); if (!toolbar_el.model.get('omemo_supported')) { let messages; if (toolbar_el.model.get('type') === shared_converse.CHATROOMS_TYPE) { messages = [__('Cannot use end-to-end encryption in this groupchat, ' + 'either the groupchat has some anonymity or not all participants support OMEMO.')]; } else { messages = [__("Cannot use end-to-end encryption because %1$s uses a client that doesn't support OMEMO.", toolbar_el.model.contact.getDisplayName())]; } return core_api.alert('error', __('Error'), messages); } toolbar_el.model.save({ 'omemo_active': !toolbar_el.model.get('omemo_active') }); } function getOMEMOToolbarButton(toolbar_el, buttons) { const model = toolbar_el.model; const is_muc = model.get('type') === shared_converse.CHATROOMS_TYPE; let title; if (model.get('omemo_supported')) { const i18n_plaintext = __('Messages are being sent in plaintext'); const i18n_encrypted = __('Messages are sent encrypted'); title = model.get('omemo_active') ? i18n_encrypted : i18n_plaintext; } else if (is_muc) { title = __('This groupchat needs to be members-only and non-anonymous in ' + 'order to support OMEMO encrypted messages'); } else { title = __('OMEMO encryption is not supported'); } let color; if (model.get('omemo_supported')) { if (model.get('omemo_active')) { color = is_muc ? `var(--muc-color)` : `var(--chat-toolbar-btn-color)`; } else { color = `var(--error-color)`; } } else { color = `var(--muc-toolbar-btn-disabled-color)`; } buttons.push($` `); return buttons; } async function getBundlesAndBuildSessions(chatbox) { const no_devices_err = __('Sorry, no devices found to which we can send an OMEMO encrypted message.'); let devices; if (chatbox.get('type') === shared_converse.CHATROOMS_TYPE) { const collections = await Promise.all(chatbox.occupants.map(o => getDevicesForContact(o.get('jid')))); devices = collections.reduce((a, b) => lodash_es_concat(a, b.models), []); } else if (chatbox.get('type') === shared_converse.PRIVATE_CHAT_TYPE) { const their_devices = await getDevicesForContact(chatbox.get('jid')); if (their_devices.length === 0) { const err = new Error(no_devices_err); err.user_facing = true; throw err; } const own_list = await core_api.omemo.devicelists.get(shared_converse.bare_jid); const own_devices = own_list.devices; devices = [...own_devices.models, ...their_devices.models]; } // Filter out our own device const id = shared_converse.omemo_store.get('device_id'); devices = devices.filter(d => d.get('id') !== id); // Fetch bundles if necessary await Promise.all(devices.map(d => d.getBundle())); const sessions = devices.filter(d => d).map(d => getSession(d)); await Promise.all(sessions); if (sessions.includes(null)) { // We couldn't build a session for certain devices. devices = devices.filter(d => sessions[devices.indexOf(d)]); if (devices.length === 0) { const err = new Error(no_devices_err); err.user_facing = true; throw err; } } return devices; } function encryptKey(key_and_tag, device) { return getSessionCipher(device.get('jid'), device.get('id')).encrypt(key_and_tag).then(payload => ({ 'payload': payload, 'device': device })); } async function createOMEMOMessageStanza(chat, data) { let { stanza } = data; const { message } = data; if (!message.get('is_encrypted')) { return data; } if (!message.get('body')) { throw new Error('No message body to encrypt!'); } const devices = await getBundlesAndBuildSessions(chat); // An encrypted header is added to the message for // each device that is supposed to receive it. // These headers simply contain the key that the // payload message is encrypted with, // and they are separately encrypted using the // session corresponding to the counterpart device. stanza.c('encrypted', { 'xmlns': omemo_utils_Strophe.NS.OMEMO }).c('header', { 'sid': shared_converse.omemo_store.get('device_id') }); const { key_and_tag, iv, payload } = await omemo.encryptMessage(message.get('plaintext')); // The 16 bytes key and the GCM authentication tag (The tag // SHOULD have at least 128 bit) are concatenated and for each // intended recipient device, i.e. both own devices as well as // devices associated with the contact, the result of this // concatenation is encrypted using the corresponding // long-standing SignalProtocol session. const dicts = await Promise.all(devices.filter(device => device.get('trusted') != UNTRUSTED && device.get('active')).map(device => encryptKey(key_and_tag, device))); stanza = await addKeysToMessageStanza(stanza, dicts, iv); stanza.c('payload').t(payload).up().up(); stanza.c('store', { 'xmlns': omemo_utils_Strophe.NS.HINTS }).up(); stanza.c('encryption', { 'xmlns': omemo_utils_Strophe.NS.EME, namespace: omemo_utils_Strophe.NS.OMEMO }); return { message, stanza }; } const omemo = { decryptMessage, encryptMessage, formatFingerprint }; ;// CONCATENATED MODULE: ./src/plugins/omemo/templates/fingerprints.js const device_fingerprint = (el, device) => { const i18n_trusted = __('Trusted'); const i18n_untrusted = __('Untrusted'); if (device.get('bundle') && device.get('bundle').fingerprint) { return $`
  • ${formatFingerprint(device.get('bundle').fingerprint)}
  • `; } else { return ''; } }; /* harmony default export */ const fingerprints = (el => { const i18n_fingerprints = __('OMEMO Fingerprints'); const i18n_no_devices = __("No OMEMO-enabled devices found"); const devices = el.devicelist.devices; return $`
    • ${i18n_fingerprints}
    • ${devices.length ? devices.map(device => device_fingerprint(el, device)) : $`
    • ${i18n_no_devices}
    • `}
    `; }); ;// CONCATENATED MODULE: ./src/plugins/omemo/fingerprints.js class Fingerprints extends CustomElement { static get properties() { return { 'jid': { type: String } }; } async initialize() { this.devicelist = await core_api.omemo.devicelists.get(this.jid, true); this.listenTo(this.devicelist.devices, 'change:bundle', this.requestUpdate); this.listenTo(this.devicelist.devices, 'change:trusted', this.requestUpdate); this.listenTo(this.devicelist.devices, 'remove', this.requestUpdate); this.listenTo(this.devicelist.devices, 'add', this.requestUpdate); this.listenTo(this.devicelist.devices, 'reset', this.requestUpdate); this.requestUpdate(); } render() { return this.devicelist ? fingerprints(this) : ''; } toggleDeviceTrust(ev) { const radio = ev.target; const device = this.devicelist.devices.get(radio.getAttribute('name')); device.save('trusted', parseInt(radio.value, 10)); } } core_api.elements.define('converse-omemo-fingerprints', Fingerprints); ;// CONCATENATED MODULE: ./src/plugins/omemo/templates/profile.js const fingerprint = el => $` ${formatFingerprint(el.current_device.get('bundle').fingerprint)}`; const device_with_fingerprint = el => { const i18n_fingerprint_checkbox_label = __('Checkbox for selecting the following fingerprint'); return $`
  • `; }; const device_without_fingerprint = el => { const i18n_device_without_fingerprint = __('Device without a fingerprint'); const i18n_fingerprint_checkbox_label = __('Checkbox for selecting the following device'); return $`
  • `; }; const device_item = el => $` ${el.device.get('bundle') && el.device.get('bundle').fingerprint ? device_with_fingerprint(el) : device_without_fingerprint(el)} `; const device_list = el => { var _el$other_devices; const i18n_other_devices = __('Other OMEMO-enabled devices'); const i18n_other_devices_label = __('Checkbox to select fingerprints of all other OMEMO devices'); const i18n_remove_devices = __('Remove checked devices and close'); const i18n_select_all = __('Select all'); return $`
    • ${(_el$other_devices = el.other_devices) === null || _el$other_devices === void 0 ? void 0 : _el$other_devices.map(device => device_item(Object.assign({ device }, el)))}
    `; }; /* harmony default export */ const profile = (el => { var _el$other_devices2; const i18n_fingerprint = __("This device's OMEMO fingerprint"); const i18n_generate = __('Generate new keys and fingerprint'); return $`
    • ${i18n_fingerprint}
    • ${el.current_device && el.current_device.get('bundle') && el.current_device.get('bundle').fingerprint ? fingerprint(el) : spinner()}
    ${(_el$other_devices2 = el.other_devices) !== null && _el$other_devices2 !== void 0 && _el$other_devices2.length ? device_list(el) : ''}
    `; }); ;// CONCATENATED MODULE: ./src/plugins/omemo/profile.js const { Strophe: profile_Strophe, sizzle: profile_sizzle, u: profile_u } = core_converse.env; class Profile extends CustomElement { async initialize() { this.devicelist = await core_api.omemo.devicelists.get(shared_converse.bare_jid, true); await this.setAttributes(); this.listenTo(this.devicelist.devices, 'change:bundle', () => this.requestUpdate()); this.listenTo(this.devicelist.devices, 'reset', () => this.requestUpdate()); this.listenTo(this.devicelist.devices, 'reset', () => this.requestUpdate()); this.listenTo(this.devicelist.devices, 'remove', () => this.requestUpdate()); this.listenTo(this.devicelist.devices, 'add', () => this.requestUpdate()); this.requestUpdate(); } async setAttributes() { this.device_id = await core_api.omemo.getDeviceID(); this.current_device = this.devicelist.devices.get(this.device_id); this.other_devices = this.devicelist.devices.filter(d => d.get('id') !== this.device_id); } render() { return this.devicelist ? profile(this) : ''; } selectAll(ev) { // eslint-disable-line class-methods-use-this let sibling = profile_u.ancestor(ev.target, 'li'); while (sibling) { sibling.querySelector('input[type="checkbox"]').checked = ev.target.checked; sibling = sibling.nextElementSibling; } } async removeSelectedFingerprints(ev) { ev.preventDefault(); ev.stopPropagation(); ev.target.querySelector('.select-all').checked = false; const device_ids = profile_sizzle('.fingerprint-removal-item input[type="checkbox"]:checked', ev.target).map(c => c.value); try { await this.devicelist.removeOwnDevices(device_ids); } catch (err) { headless_log.error(err); shared_converse.api.alert(profile_Strophe.LogLevel.ERROR, __('Error'), [__('Sorry, an error occurred while trying to remove the devices.')]); } await this.setAttributes(); this.requestUpdate(); } async generateOMEMODeviceBundle(ev) { ev.preventDefault(); if (confirm(__('Are you sure you want to generate new OMEMO keys? ' + 'This will remove your old keys and all previously encrypted messages will no longer be decryptable on this device.'))) { await core_api.omemo.bundle.generate(); await this.setAttributes(); this.requestUpdate(); } } } core_api.elements.define('converse-omemo-profile', Profile); ;// CONCATENATED MODULE: ./src/modals/templates/user-settings.js const user_settings_tpl_navigation = o => { const i18n_about = __('About'); const i18n_commands = __('Commands'); return $` `; }; /* harmony default export */ const templates_user_settings = (o => { const i18n_modal_title = __('Settings'); const first_subtitle = __('%1$s Open Source %2$s XMPP chat client brought to you by %3$s Opkode %2$s', '', '', ''); const second_subtitle = __('%1$s Translate %2$s it into your own language', '', ''); const show_client_info = core_api.settings.get('show_client_info'); const allow_adhoc_commands = core_api.settings.get('allow_adhoc_commands'); const show_both_tabs = show_client_info && allow_adhoc_commands; return $` `; }); ;// CONCATENATED MODULE: ./src/modals/user-settings.js let user_settings_converse; /* harmony default export */ const modals_user_settings = (base.extend({ id: "converse-client-info-modal", initialize(settings) { user_settings_converse = settings._converse; base.prototype.initialize.apply(this, arguments); }, toHTML() { return templates_user_settings(Object.assign(this.model.toJSON(), this.model.vcard.toJSON(), { 'version_name': user_settings_converse.VERSION_NAME })); } })); ;// CONCATENATED MODULE: ./src/plugins/profile/utils.js function getPrettyStatus(stat) { 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'); } else { return __(stat) || __('online'); } } ;// CONCATENATED MODULE: ./src/plugins/profile/templates/profile.js function tpl_signout(o) { const i18n_logout = __('Log out'); return $` `; } function tpl_user_settings_button(o) { const i18n_details = __('Show details about this chat client'); return $` `; } /* harmony default export */ const templates_profile = (el => { var _el$model$vcard, _el$model$vcard2, _el$model$vcard3; const chat_status = el.model.get('status') || 'offline'; const fullname = ((_el$model$vcard = el.model.vcard) === null || _el$model$vcard === void 0 ? void 0 : _el$model$vcard.get('fullname')) || shared_converse.bare_jid; const status_message = el.model.get('status_message') || __("I am %1$s", getPrettyStatus(chat_status)); const i18n_change_status = __('Click to change your chat status'); const show_settings_button = core_api.settings.get('show_client_info') || core_api.settings.get('allow_adhoc_commands'); let classes, color; if (chat_status === 'online') { [classes, color] = ['fa fa-circle chat-status', 'chat-status-online']; } else if (chat_status === 'dnd') { [classes, color] = ['fa fa-minus-circle chat-status', 'chat-status-busy']; } else if (chat_status === 'away') { [classes, color] = ['fa fa-circle chat-status', 'chat-status-away']; } else { [classes, color] = ['fa fa-circle chat-status', 'subdued-color']; } return $`
    ${fullname} ${show_settings_button ? tpl_user_settings_button(el) : ''} ${core_api.settings.get('allow_logout') ? tpl_signout(el) : ''}
    `; }); ;// CONCATENATED MODULE: ./src/plugins/profile/statusview.js class statusview_Profile extends CustomElement { initialize() { this.model = shared_converse.xmppstatus; this.listenTo(this.model, "change", this.requestUpdate); this.listenTo(this.model, "vcard:add", this.requestUpdate); this.listenTo(this.model, "vcard:change", this.requestUpdate); } render() { return templates_profile(this); } showProfileModal(ev) { ev === null || ev === void 0 ? void 0 : ev.preventDefault(); core_api.modal.show(shared_converse.ProfileModal, { model: this.model }, ev); } showStatusChangeModal(ev) { ev === null || ev === void 0 ? void 0 : ev.preventDefault(); core_api.modal.show(shared_converse.ChatStatusModal, { model: this.model }, ev); } showUserSettingsModal(ev) { ev === null || ev === void 0 ? void 0 : ev.preventDefault(); core_api.modal.show(modals_user_settings, { model: this.model, _converse: shared_converse }, ev); } logout(ev) { // eslint-disable-line class-methods-use-this ev === null || ev === void 0 ? void 0 : ev.preventDefault(); const result = confirm(__("Are you sure you want to log out?")); if (result === true) { core_api.user.logout(); } } } core_api.elements.define('converse-user-profile', statusview_Profile); ;// CONCATENATED MODULE: ./src/plugins/profile/templates/chat-status-modal.js /* harmony default export */ const chat_status_modal = (o => $` `); ;// CONCATENATED MODULE: ./src/plugins/profile/modals/chat-status.js const chat_status_u = core_converse.env.utils; const ChatStatusModal = base.extend({ id: "modal-status-change", events: { "submit form#set-xmpp-status": "onFormSubmitted", "click .clear-input": "clearStatusMessage" }, toHTML() { return chat_status_modal(Object.assign(this.model.toJSON(), this.model.vcard.toJSON(), { 'label_away': __('Away'), 'label_busy': __('Busy'), 'label_cancel': __('Cancel'), 'label_close': __('Close'), 'label_custom_status': __('Custom status'), 'label_offline': __('Offline'), 'label_online': __('Online'), 'label_save': __('Save'), 'label_xa': __('Away for long'), 'modal_title': __('Change chat status'), 'placeholder_status_message': __('Personal status message') })); }, afterRender() { this.el.addEventListener('shown.bs.modal', () => { this.el.querySelector('input[name="status_message"]').focus(); }, false); }, clearStatusMessage(ev) { if (ev && ev.preventDefault) { ev.preventDefault(); chat_status_u.hideElement(this.el.querySelector('.clear-input')); } const roster_filter = this.el.querySelector('input[name="status_message"]'); roster_filter.value = ''; }, onFormSubmitted(ev) { ev.preventDefault(); const data = new FormData(ev.target); this.model.save({ 'status_message': data.get('status_message'), 'status': data.get('chat_status') }); this.modal.hide(); } }); shared_converse.ChatStatusModal = ChatStatusModal; /* harmony default export */ const chat_status = ((/* unused pure expression or super */ null && (ChatStatusModal))); ;// CONCATENATED MODULE: ./src/shared/components/image-picker.js const i18n_profile_picture = __('Your profile picture'); class ImagePicker extends CustomElement { static get properties() { return { 'height': { type: Number }, 'data': { type: Object }, 'width': { type: Number } }; } render() { return $` `; } openFileSelection(ev) { ev.preventDefault(); this.querySelector('input[type="file"]').click(); } updateFilePreview(ev) { const file = ev.target.files[0]; const reader = new FileReader(); reader.onloadend = () => { this.data = { 'data_uri': reader.result, 'image_type': file.type }; }; reader.readAsDataURL(file); } } core_api.elements.define('converse-image-picker', ImagePicker); ;// CONCATENATED MODULE: ./src/plugins/profile/templates/profile_modal.js const omemo_page = () => $`
    `; /* harmony default export */ const profile_modal = (o => { var _converse$pluggable$p, _converse$pluggable$p2; const heading_profile = __('Your Profile'); const i18n_email = __('Email'); const i18n_fullname = __('Full Name'); const i18n_jid = __('XMPP Address'); const i18n_nickname = __('Nickname'); const i18n_role = __('Role'); const i18n_save = __('Save and close'); const i18n_role_help = __('Use commas to separate multiple roles. Your roles are shown next to your name on your chat messages.'); const i18n_url = __('URL'); const i18n_omemo = __('OMEMO'); const i18n_profile = __('Profile'); const navigation = $``; return $` `; }); // EXTERNAL MODULE: ./node_modules/client-compress/dist/index.js var dist = __webpack_require__(577); var dist_default = /*#__PURE__*/__webpack_require__.n(dist); ;// CONCATENATED MODULE: ./src/plugins/profile/modals/profile.js const { sizzle: modals_profile_sizzle } = core_converse.env; const profile_options = { targetSize: 0.1, quality: 0.75, maxWidth: 256, maxHeight: 256 }; const compress = new (dist_default())(profile_options); const ProfileModal = base.extend({ id: "user-profile-modal", events: { 'submit .profile-form': 'onFormSubmitted' }, initialize() { this.listenTo(this.model, 'change', this.render); base.prototype.initialize.apply(this, arguments); /** * Triggered when the _converse.ProfileModal has been created and initialized. * @event _converse#profileModalInitialized * @type { _converse.XMPPStatus } * @example _converse.api.listen.on('profileModalInitialized', status => { ... }); */ core_api.trigger('profileModalInitialized', this.model); }, toHTML() { return profile_modal(Object.assign(this.model.toJSON(), this.model.vcard.toJSON(), { 'view': this })); }, afterRender() { this.tabs = modals_profile_sizzle('.nav-item .nav-link', this.el).map(e => new (bootstrap_native_default()).Tab(e)); }, async setVCard(data) { try { await core_api.vcard.set(shared_converse.bare_jid, data); } catch (err) { headless_log.fatal(err); this.alert([__("Sorry, an error happened while trying to save your profile data."), __("You can check your browser's developer console for any error output.")].join(" ")); return; } this.modal.hide(); }, onFormSubmitted(ev) { ev.preventDefault(); const reader = new FileReader(); const form_data = new FormData(ev.target); const image_file = form_data.get('image'); const data = { 'fn': form_data.get('fn'), 'nickname': form_data.get('nickname'), 'role': form_data.get('role'), 'email': form_data.get('email'), 'url': form_data.get('url') }; if (!image_file.size) { Object.assign(data, { 'image': this.model.vcard.get('image'), 'image_type': this.model.vcard.get('image_type') }); this.setVCard(data); } else { const files = [image_file]; compress.compress(files).then(conversions => { const { photo } = conversions[0]; reader.onloadend = () => { Object.assign(data, { 'image': btoa(reader.result), 'image_type': image_file.type }); this.setVCard(data); }; reader.readAsBinaryString(photo.data); }); } } }); shared_converse.ProfileModal = ProfileModal; /* harmony default export */ const modals_profile = ((/* unused pure expression or super */ null && (ProfileModal))); ;// CONCATENATED MODULE: ./src/plugins/profile/index.js /** * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.plugins.add('converse-profile', { dependencies: ["converse-status", "converse-modal", "converse-vcard", "converse-chatboxviews"], initialize() { core_api.settings.extend({ 'allow_adhoc_commands': true, 'show_client_info': true }); } }); ;// CONCATENATED MODULE: ./src/plugins/omemo/mixins/converse.js const ConverseMixins = { generateFingerprints: async function (jid) { const devices = await getDevicesForContact(jid); return Promise.all(devices.map(d => generateFingerprint(d))); }, getDeviceForContact: function (jid, device_id) { return getDevicesForContact(jid).then(devices => devices.get(device_id)); }, contactHasOMEMOSupport: async function (jid) { /* Checks whether the contact advertises any OMEMO-compatible devices. */ const devices = await getDevicesForContact(jid); return devices.length > 0; } }; /* harmony default export */ const mixins_converse = (ConverseMixins); ;// CONCATENATED MODULE: ./src/plugins/omemo/errors.js class IQError extends Error { constructor(message, iq) { super(message, iq); this.name = 'IQError'; this.iq = iq; } } ;// CONCATENATED MODULE: ./src/plugins/omemo/device.js const { Strophe: device_Strophe, sizzle: device_sizzle, u: device_u, $iq: device_$iq } = core_converse.env; /** * @class * @namespace _converse.Device * @memberOf _converse */ const Device = Model.extend({ defaults: { 'trusted': UNDECIDED, 'active': true }, getRandomPreKey() { // XXX: assumes that the bundle has already been fetched const bundle = this.get('bundle'); return bundle.prekeys[device_u.getRandomInt(bundle.prekeys.length)]; }, async fetchBundleFromServer() { const stanza = device_$iq({ 'type': 'get', 'from': shared_converse.bare_jid, 'to': this.get('jid') }).c('pubsub', { 'xmlns': device_Strophe.NS.PUBSUB }).c('items', { 'node': `${device_Strophe.NS.OMEMO_BUNDLES}:${this.get('id')}` }); let iq; try { iq = await core_api.sendIQ(stanza); } catch (iq) { headless_log.error(`Could not fetch bundle for device ${this.get('id')} from ${this.get('jid')}`); headless_log.error(iq); return null; } if (iq.querySelector('error')) { throw new IQError('Could not fetch bundle', iq); } const publish_el = device_sizzle(`items[node="${device_Strophe.NS.OMEMO_BUNDLES}:${this.get('id')}"]`, iq).pop(); const bundle_el = device_sizzle(`bundle[xmlns="${device_Strophe.NS.OMEMO}"]`, publish_el).pop(); const bundle = parseBundle(bundle_el); this.save('bundle', bundle); return bundle; }, /** * Fetch and save the bundle information associated with * this device, if the information is not cached already. * @method _converse.Device#getBundle */ getBundle() { if (this.get('bundle')) { return Promise.resolve(this.get('bundle'), this); } else { return this.fetchBundleFromServer(); } } }); /* harmony default export */ const device = (Device); ;// CONCATENATED MODULE: ./src/plugins/omemo/devicelist.js const { Strophe: devicelist_Strophe, $build: devicelist_$build, $iq: devicelist_$iq, sizzle: devicelist_sizzle } = core_converse.env; /** * @class * @namespace _converse.DeviceList * @memberOf _converse */ const DeviceList = Model.extend({ idAttribute: 'jid', async initialize() { this.initialized = getOpenPromise(); await this.initDevices(); this.initialized.resolve(); }, initDevices() { this.devices = new shared_converse.Devices(); const id = `converse.devicelist-${shared_converse.bare_jid}-${this.get('jid')}`; initStorage(this.devices, id); return this.fetchDevices(); }, async onDevicesFound(collection) { if (collection.length === 0) { let ids = []; try { ids = await this.fetchDevicesFromServer(); } catch (e) { if (e === null) { headless_log.error(`Timeout error while fetching devices for ${this.get('jid')}`); } else { headless_log.error(`Could not fetch devices for ${this.get('jid')}`); headless_log.error(e); } this.destroy(); } if (this.get('jid') === shared_converse.bare_jid) { this.publishCurrentDevice(ids); } } }, fetchDevices() { if (this._devices_promise === undefined) { this._devices_promise = new Promise(resolve => { this.devices.fetch({ 'success': c => resolve(this.onDevicesFound(c)), 'error': (_, e) => { headless_log.error(e); resolve(); } }); }); } return this._devices_promise; }, async getOwnDeviceId() { let device_id = shared_converse.omemo_store.get('device_id'); if (!this.devices.get(device_id)) { // Generate a new bundle if we cannot find our device await shared_converse.omemo_store.generateBundle(); device_id = shared_converse.omemo_store.get('device_id'); } return device_id; }, async publishCurrentDevice(device_ids) { if (this.get('jid') !== shared_converse.bare_jid) { return; // We only publish for ourselves. } await restoreOMEMOSession(); if (!shared_converse.omemo_store) { // Happens during tests. The connection gets torn down // before publishCurrentDevice has time to finish. headless_log.warn('publishCurrentDevice: omemo_store is not defined, likely a timing issue'); return; } if (!device_ids.includes(await this.getOwnDeviceId())) { return this.publishDevices(); } }, async fetchDevicesFromServer() { const stanza = devicelist_$iq({ 'type': 'get', 'from': shared_converse.bare_jid, 'to': this.get('jid') }).c('pubsub', { 'xmlns': devicelist_Strophe.NS.PUBSUB }).c('items', { 'node': devicelist_Strophe.NS.OMEMO_DEVICELIST }); const iq = await core_api.sendIQ(stanza); const selector = `list[xmlns="${devicelist_Strophe.NS.OMEMO}"] device`; const device_ids = devicelist_sizzle(selector, iq).map(d => d.getAttribute('id')); const jid = this.get('jid'); return Promise.all(device_ids.map(id => this.devices.create({ id, jid }, { 'promise': true }))); }, /** * Send an IQ stanza to the current user's "devices" PEP node to * ensure that all devices are published for potential chat partners to see. * See: https://xmpp.org/extensions/xep-0384.html#usecases-announcing */ publishDevices() { const item = devicelist_$build('item', { 'id': 'current' }).c('list', { 'xmlns': devicelist_Strophe.NS.OMEMO }); this.devices.filter(d => d.get('active')).forEach(d => item.c('device', { 'id': d.get('id') }).up()); const options = { 'pubsub#access_model': 'open' }; return core_api.pubsub.publish(null, devicelist_Strophe.NS.OMEMO_DEVICELIST, item, options, false); }, async removeOwnDevices(device_ids) { if (this.get('jid') !== shared_converse.bare_jid) { throw new Error("Cannot remove devices from someone else's device list"); } await Promise.all(device_ids.map(id => this.devices.get(id)).map(d => new Promise(resolve => d.destroy({ 'success': resolve, 'error': (_, e) => { headless_log.error(e); resolve(); } })))); return this.publishDevices(); } }); /* harmony default export */ const devicelist = (DeviceList); ;// CONCATENATED MODULE: ./src/plugins/omemo/devicelists.js /** * @class * @namespace _converse.DeviceLists * @memberOf _converse */ const DeviceLists = Collection.extend({ model: devicelist }); /* harmony default export */ const devicelists = (DeviceLists); ;// CONCATENATED MODULE: ./src/plugins/omemo/devices.js /* harmony default export */ const devices = (Collection.extend({ model: device })); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_baseRange.js /* Built-in method references for those with the same name as other `lodash` methods. */ var nativeCeil = Math.ceil, _baseRange_nativeMax = Math.max; /** * The base implementation of `_.range` and `_.rangeRight` which doesn't * coerce arguments. * * @private * @param {number} start The start of the range. * @param {number} end The end of the range. * @param {number} step The value to increment or decrement by. * @param {boolean} [fromRight] Specify iterating from right to left. * @returns {Array} Returns the range of numbers. */ function baseRange(start, end, step, fromRight) { var index = -1, length = _baseRange_nativeMax(nativeCeil((end - start) / (step || 1)), 0), result = Array(length); while (length--) { result[fromRight ? length : ++index] = start; start += step; } return result; } /* harmony default export */ const _baseRange = (baseRange); ;// CONCATENATED MODULE: ./node_modules/lodash-es/_createRange.js /** * Creates a `_.range` or `_.rangeRight` function. * * @private * @param {boolean} [fromRight] Specify iterating from right to left. * @returns {Function} Returns the new range function. */ function createRange(fromRight) { return function(start, end, step) { if (step && typeof step != 'number' && _isIterateeCall(start, end, step)) { end = step = undefined; } // Ensure the sign of `-0` is preserved. start = lodash_es_toFinite(start); if (end === undefined) { end = start; start = 0; } else { end = lodash_es_toFinite(end); } step = step === undefined ? (start < end ? 1 : -1) : lodash_es_toFinite(step); return _baseRange(start, end, step, fromRight); }; } /* harmony default export */ const _createRange = (createRange); ;// CONCATENATED MODULE: ./node_modules/lodash-es/range.js /** * Creates an array of numbers (positive and/or negative) progressing from * `start` up to, but not including, `end`. A step of `-1` is used if a negative * `start` is specified without an `end` or `step`. If `end` is not specified, * it's set to `start` with `start` then set to `0`. * * **Note:** JavaScript follows the IEEE-754 standard for resolving * floating-point values which can produce unexpected results. * * @static * @since 0.1.0 * @memberOf _ * @category Util * @param {number} [start=0] The start of the range. * @param {number} end The end of the range. * @param {number} [step=1] The value to increment or decrement by. * @returns {Array} Returns the range of numbers. * @see _.inRange, _.rangeRight * @example * * _.range(4); * // => [0, 1, 2, 3] * * _.range(-4); * // => [0, -1, -2, -3] * * _.range(1, 5); * // => [1, 2, 3, 4] * * _.range(0, 20, 5); * // => [0, 5, 10, 15] * * _.range(0, -4, -1); * // => [0, -1, -2, -3] * * _.range(1, 4, 0); * // => [1, 1, 1] * * _.range(0); * // => [] */ var range = _createRange(); /* harmony default export */ const lodash_es_range = (range); ;// CONCATENATED MODULE: ./src/plugins/omemo/store.js /* global libsignal */ const { Strophe: store_Strophe, $build: store_$build, u: store_u } = core_converse.env; const OMEMOStore = Model.extend({ Direction: { SENDING: 1, RECEIVING: 2 }, getIdentityKeyPair() { const keypair = this.get('identity_keypair'); return Promise.resolve({ 'privKey': store_u.base64ToArrayBuffer(keypair.privKey), 'pubKey': store_u.base64ToArrayBuffer(keypair.pubKey) }); }, getLocalRegistrationId() { return Promise.resolve(parseInt(this.get('device_id'), 10)); }, isTrustedIdentity(identifier, identity_key, direction) { // eslint-disable-line no-unused-vars if (identifier === null || identifier === undefined) { throw new Error("Can't check identity key for invalid key"); } if (!(identity_key instanceof ArrayBuffer)) { throw new Error('Expected identity_key to be an ArrayBuffer'); } const trusted = this.get('identity_key' + identifier); if (trusted === undefined) { return Promise.resolve(true); } return Promise.resolve(store_u.arrayBufferToBase64(identity_key) === trusted); }, loadIdentityKey(identifier) { if (identifier === null || identifier === undefined) { throw new Error("Can't load identity_key for invalid identifier"); } return Promise.resolve(store_u.base64ToArrayBuffer(this.get('identity_key' + identifier))); }, saveIdentity(identifier, identity_key) { if (identifier === null || identifier === undefined) { throw new Error("Can't save identity_key for invalid identifier"); } const address = new libsignal.SignalProtocolAddress.fromString(identifier); const existing = this.get('identity_key' + address.getName()); const b64_idkey = store_u.arrayBufferToBase64(identity_key); this.save('identity_key' + address.getName(), b64_idkey); if (existing && b64_idkey !== existing) { return Promise.resolve(true); } else { return Promise.resolve(false); } }, getPreKeys() { return this.get('prekeys') || {}; }, loadPreKey(key_id) { const res = this.getPreKeys()[key_id]; if (res) { return Promise.resolve({ 'privKey': store_u.base64ToArrayBuffer(res.privKey), 'pubKey': store_u.base64ToArrayBuffer(res.pubKey) }); } return Promise.resolve(); }, storePreKey(key_id, key_pair) { const prekey = {}; prekey[key_id] = { 'pubKey': store_u.arrayBufferToBase64(key_pair.pubKey), 'privKey': store_u.arrayBufferToBase64(key_pair.privKey) }; this.save('prekeys', Object.assign(this.getPreKeys(), prekey)); return Promise.resolve(); }, removePreKey(key_id) { this.save('prekeys', lodash_es_omit(this.getPreKeys(), key_id)); return Promise.resolve(); }, loadSignedPreKey(keyId) { // eslint-disable-line no-unused-vars const res = this.get('signed_prekey'); if (res) { return Promise.resolve({ 'privKey': store_u.base64ToArrayBuffer(res.privKey), 'pubKey': store_u.base64ToArrayBuffer(res.pubKey) }); } return Promise.resolve(); }, storeSignedPreKey(spk) { if (typeof spk !== 'object') { // XXX: We've changed the signature of this method from the // example given in InMemorySignalProtocolStore. // Should be fine because the libsignal code doesn't // actually call this method. throw new Error('storeSignedPreKey: expected an object'); } this.save('signed_prekey', { 'id': spk.keyId, 'privKey': store_u.arrayBufferToBase64(spk.keyPair.privKey), 'pubKey': store_u.arrayBufferToBase64(spk.keyPair.pubKey), // XXX: The InMemorySignalProtocolStore does not pass // in or store the signature, but we need it when we // publish out bundle and this method isn't called from // within libsignal code, so we modify it to also store // the signature. 'signature': store_u.arrayBufferToBase64(spk.signature) }); return Promise.resolve(); }, removeSignedPreKey(key_id) { if (this.get('signed_prekey')['id'] === key_id) { this.unset('signed_prekey'); this.save(); } return Promise.resolve(); }, loadSession(identifier) { return Promise.resolve(this.get('session' + identifier)); }, storeSession(identifier, record) { return Promise.resolve(this.save('session' + identifier, record)); }, removeSession(identifier) { return Promise.resolve(this.unset('session' + identifier)); }, removeAllSessions(identifier) { const keys = Object.keys(this.attributes).filter(key => key.startsWith('session' + identifier) ? key : false); const attrs = {}; keys.forEach(key => { attrs[key] = undefined; }); this.save(attrs); return Promise.resolve(); }, publishBundle() { const signed_prekey = this.get('signed_prekey'); const node = `${store_Strophe.NS.OMEMO_BUNDLES}:${this.get('device_id')}`; const item = store_$build('item').c('bundle', { 'xmlns': store_Strophe.NS.OMEMO }).c('signedPreKeyPublic', { 'signedPreKeyId': signed_prekey.id }).t(signed_prekey.pubKey).up().c('signedPreKeySignature').t(signed_prekey.signature).up().c('identityKey').t(this.get('identity_keypair').pubKey).up().c('prekeys'); Object.values(this.get('prekeys')).forEach((prekey, id) => item.c('preKeyPublic', { 'preKeyId': id }).t(prekey.pubKey).up()); const options = { 'pubsub#access_model': 'open' }; return core_api.pubsub.publish(null, node, item, options, false); }, async generateMissingPreKeys() { const missing_keys = lodash_es_difference(lodash_es_invokeMap(lodash_es_range(0, shared_converse.NUM_PREKEYS), Number.prototype.toString), Object.keys(this.getPreKeys())); if (missing_keys.length < 1) { headless_log.warn('No missing prekeys to generate for our own device'); return Promise.resolve(); } const keys = await Promise.all(missing_keys.map(id => libsignal.KeyHelper.generatePreKey(parseInt(id, 10)))); keys.forEach(k => this.storePreKey(k.keyId, k.keyPair)); const marshalled_keys = Object.keys(this.getPreKeys()).map(k => ({ 'id': k.keyId, 'key': store_u.arrayBufferToBase64(k.pubKey) })); const devicelist = await core_api.omemo.devicelists.get(shared_converse.bare_jid); const device = devicelist.devices.get(this.get('device_id')); const bundle = await device.getBundle(); device.save('bundle', Object.assign(bundle, { 'prekeys': marshalled_keys })); }, /** * Generate the data used by the X3DH key agreement protocol * that can be used to build a session with a device. */ async generateBundle() { // The first thing that needs to happen if a client wants to // start using OMEMO is they need to generate an IdentityKey // and a Device ID. The IdentityKey is a Curve25519 [6] // public/private Key pair. The Device ID is a randomly // generated integer between 1 and 2^31 - 1. const identity_keypair = await libsignal.KeyHelper.generateIdentityKeyPair(); const bundle = {}; const identity_key = store_u.arrayBufferToBase64(identity_keypair.pubKey); const device_id = await generateDeviceID(); bundle['identity_key'] = identity_key; bundle['device_id'] = device_id; this.save({ 'device_id': device_id, 'identity_keypair': { 'privKey': store_u.arrayBufferToBase64(identity_keypair.privKey), 'pubKey': identity_key }, 'identity_key': identity_key }); const signed_prekey = await libsignal.KeyHelper.generateSignedPreKey(identity_keypair, 0); this.storeSignedPreKey(signed_prekey); bundle['signed_prekey'] = { 'id': signed_prekey.keyId, 'public_key': store_u.arrayBufferToBase64(signed_prekey.keyPair.pubKey), 'signature': store_u.arrayBufferToBase64(signed_prekey.signature) }; const keys = await Promise.all(lodash_es_range(0, shared_converse.NUM_PREKEYS).map(id => libsignal.KeyHelper.generatePreKey(id))); keys.forEach(k => this.storePreKey(k.keyId, k.keyPair)); const devicelist = await core_api.omemo.devicelists.get(shared_converse.bare_jid); const device = await devicelist.devices.create({ 'id': bundle.device_id, 'jid': shared_converse.bare_jid }, { 'promise': true }); const marshalled_keys = keys.map(k => ({ 'id': k.keyId, 'key': store_u.arrayBufferToBase64(k.keyPair.pubKey) })); bundle['prekeys'] = marshalled_keys; device.save('bundle', bundle); }, fetchSession() { if (this._setup_promise === undefined) { this._setup_promise = new Promise((resolve, reject) => { this.fetch({ 'success': () => { if (!this.get('device_id')) { this.generateBundle().then(resolve).catch(reject); } else { resolve(); } }, 'error': (model, resp) => { headless_log.warn("Could not fetch OMEMO session from cache, we'll generate a new one."); headless_log.warn(resp); this.generateBundle().then(resolve).catch(reject); } }); }); } return this._setup_promise; } }); /* harmony default export */ const store = (OMEMOStore); ;// CONCATENATED MODULE: ./src/plugins/omemo/api.js /* harmony default export */ const omemo_api = ({ /** * The "omemo" namespace groups methods relevant to OMEMO * encryption. * * @namespace _converse.api.omemo * @memberOf _converse.api */ 'omemo': { /** * Returns the device ID of the current device. */ async getDeviceID() { await core_api.waitUntil('OMEMOInitialized'); return shared_converse.omemo_store.get('device_id'); }, /** * The "devicelists" namespace groups methods related to OMEMO device lists * * @namespace _converse.api.omemo.devicelists * @memberOf _converse.api.omemo */ 'devicelists': { /** * Returns the {@link _converse.DeviceList} for a particular JID. * The device list will be created if it doesn't exist already. * @method _converse.api.omemo.devicelists.get * @param { String } jid - The Jabber ID for which the device list will be returned. * @param { bool } create=false - Set to `true` if the device list * should be created if it cannot be found. */ async get(jid) { let create = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false; const list = shared_converse.devicelists.get(jid) || (create ? shared_converse.devicelists.create({ jid }) : null); await (list === null || list === void 0 ? void 0 : list.initialized); return list; } }, /** * The "bundle" namespace groups methods relevant to the user's * OMEMO bundle. * * @namespace _converse.api.omemo.bundle * @memberOf _converse.api.omemo */ 'bundle': { /** * Lets you generate a new OMEMO device bundle * * @method _converse.api.omemo.bundle.generate * @returns {promise} Promise which resolves once we have a result from the server. */ 'generate': async () => { await core_api.waitUntil('OMEMOInitialized'); // Remove current device const devicelist = await core_api.omemo.devicelists.get(shared_converse.bare_jid); const device_id = shared_converse.omemo_store.get('device_id'); if (device_id) { const device = devicelist.devices.get(device_id); shared_converse.omemo_store.unset(device_id); if (device) { await new Promise(done => device.destroy({ 'success': done, 'error': done })); } devicelist.devices.trigger('remove'); } // Generate new device bundle and publish // https://xmpp.org/extensions/attic/xep-0384-0.3.0.html#usecases-announcing await shared_converse.omemo_store.generateBundle(); await devicelist.publishDevices(); const device = devicelist.devices.get(shared_converse.omemo_store.get('device_id')); const fp = generateFingerprint(device); await shared_converse.omemo_store.publishBundle(); return fp; } } } }); ;// CONCATENATED MODULE: ./src/plugins/omemo/index.js /** * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) */ const { Strophe: omemo_Strophe } = core_converse.env; core_converse.env.omemo = omemo; omemo_Strophe.addNamespace('OMEMO_DEVICELIST', omemo_Strophe.NS.OMEMO + '.devicelist'); omemo_Strophe.addNamespace('OMEMO_VERIFICATION', omemo_Strophe.NS.OMEMO + '.verification'); omemo_Strophe.addNamespace('OMEMO_WHITELISTED', omemo_Strophe.NS.OMEMO + '.whitelisted'); omemo_Strophe.addNamespace('OMEMO_BUNDLES', omemo_Strophe.NS.OMEMO + '.bundles'); core_converse.plugins.add('converse-omemo', { enabled(_converse) { return window.libsignal && _converse.config.get('trusted') && !core_api.settings.get('clear_cache_on_logout') && !_converse.api.settings.get('blacklisted_plugins').includes('converse-omemo'); }, dependencies: ['converse-chatview', 'converse-pubsub'], initialize() { core_api.settings.extend({ 'omemo_default': false }); core_api.promises.add(['OMEMOInitialized']); shared_converse.NUM_PREKEYS = 100; // Set here so that tests can override Object.assign(shared_converse, mixins_converse); Object.assign(shared_converse.api, omemo_api); shared_converse.OMEMOStore = store; shared_converse.Device = device; shared_converse.Devices = devices; shared_converse.DeviceList = devicelist; shared_converse.DeviceLists = devicelists; /******************** Event Handlers ********************/ core_api.waitUntil('chatBoxesInitialized').then(utils_onChatBoxesInitialized); core_api.listen.on('getOutgoingMessageAttributes', getOutgoingMessageAttributes); core_api.listen.on('createMessageStanza', async (chat, data) => { try { data = await createOMEMOMessageStanza(chat, data); } catch (e) { handleMessageSendError(e, chat); } return data; }); core_api.listen.on('afterFileUploaded', (msg, attrs) => msg.file.xep454_ivkey ? setEncryptedFileURL(msg, attrs) : attrs); core_api.listen.on('beforeFileUpload', (chat, file) => chat.get('omemo_active') ? encryptFile(file) : file); core_api.listen.on('parseMessage', parseEncryptedMessage); core_api.listen.on('parseMUCMessage', parseEncryptedMessage); core_api.listen.on('chatBoxViewInitialized', onChatInitialized); core_api.listen.on('chatRoomViewInitialized', onChatInitialized); core_api.listen.on('connected', registerPEPPushHandler); core_api.listen.on('getToolbarButtons', getOMEMOToolbarButton); core_api.listen.on('statusInitialized', initOMEMO); core_api.listen.on('addClientFeatures', () => core_api.disco.own.features.add(`${omemo_Strophe.NS.OMEMO_DEVICELIST}+notify`)); core_api.listen.on('afterMessageBodyTransformed', handleEncryptedFiles); core_api.listen.on('userDetailsModalInitialized', contact => { const jid = contact.get('jid'); shared_converse.generateFingerprints(jid).catch(e => headless_log.error(e)); }); core_api.listen.on('profileModalInitialized', () => { shared_converse.generateFingerprints(shared_converse.bare_jid).catch(e => headless_log.error(e)); }); core_api.listen.on('clearSession', () => { delete shared_converse.omemo_store; if (shared_converse.shouldClearCache() && shared_converse.devicelists) { shared_converse.devicelists.clearStore(); delete shared_converse.devicelists; } }); } }); ;// CONCATENATED MODULE: ./src/plugins/push/utils.js const { Strophe: push_utils_Strophe, $iq: push_utils_$iq } = core_converse.env; async function disablePushAppServer(domain, push_app_server) { if (!push_app_server.jid) { return; } if (!(await core_api.disco.supports(push_utils_Strophe.NS.PUSH, domain || shared_converse.bare_jid))) { headless_log.warn(`Not disabling push app server "${push_app_server.jid}", no disco support from your server.`); return; } const stanza = push_utils_$iq({ 'type': 'set' }); if (domain !== shared_converse.bare_jid) { stanza.attrs({ 'to': domain }); } stanza.c('disable', { 'xmlns': push_utils_Strophe.NS.PUSH, 'jid': push_app_server.jid }); if (push_app_server.node) { stanza.attrs({ 'node': push_app_server.node }); } core_api.sendIQ(stanza).catch(e => { headless_log.error(`Could not disable push app server for ${push_app_server.jid}`); headless_log.error(e); }); } async function enablePushAppServer(domain, push_app_server) { if (!push_app_server.jid || !push_app_server.node) { return; } const identity = await core_api.disco.getIdentity('pubsub', 'push', push_app_server.jid); if (!identity) { return headless_log.warn(`Not enabling push the service "${push_app_server.jid}", it doesn't have the right disco identtiy.`); } const result = await Promise.all([core_api.disco.supports(push_utils_Strophe.NS.PUSH, push_app_server.jid), core_api.disco.supports(push_utils_Strophe.NS.PUSH, domain)]); if (!result[0] && !result[1]) { headless_log.warn(`Not enabling push app server "${push_app_server.jid}", no disco support from your server.`); return; } const stanza = push_utils_$iq({ 'type': 'set' }); if (domain !== shared_converse.bare_jid) { stanza.attrs({ 'to': domain }); } stanza.c('enable', { 'xmlns': push_utils_Strophe.NS.PUSH, 'jid': push_app_server.jid, 'node': push_app_server.node }); if (push_app_server.secret) { stanza.c('x', { 'xmlns': push_utils_Strophe.NS.XFORM, 'type': 'submit' }).c('field', { 'var': 'FORM_TYPE' }).c('value').t(`${push_utils_Strophe.NS.PUBSUB}#publish-options`).up().up().c('field', { 'var': 'secret' }).c('value').t(push_app_server.secret); } return core_api.sendIQ(stanza); } async function enablePush(domain) { domain = domain || shared_converse.bare_jid; const push_enabled = shared_converse.session.get('push_enabled') || []; if (push_enabled.includes(domain)) { return; } const enabled_services = core_api.settings.get('push_app_servers').filter(s => !s.disable); const disabled_services = core_api.settings.get('push_app_servers').filter(s => s.disable); const enabled = enabled_services.map(s => enablePushAppServer(domain, s)); const disabled = disabled_services.map(s => disablePushAppServer(domain, s)); try { await Promise.all(enabled.concat(disabled)); } catch (e) { headless_log.error('Could not enable or disable push App Server'); if (e) headless_log.error(e); } finally { push_enabled.push(domain); } shared_converse.session.save('push_enabled', push_enabled); } function onChatBoxAdded(model) { if (model.get('type') == shared_converse.CHATROOMS_TYPE) { enablePush(push_utils_Strophe.getDomainFromJid(model.get('jid'))); } } ;// CONCATENATED MODULE: ./src/plugins/push/index.js /** * @description * Converse.js plugin which add support for registering * an "App Server" as defined in XEP-0357 * @copyright 2021, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ const { Strophe: push_Strophe } = core_converse.env; push_Strophe.addNamespace('PUSH', 'urn:xmpp:push:0'); core_converse.plugins.add('converse-push', { initialize() { /* The initialize function gets called as soon as the plugin is * loaded by converse.js's plugin machinery. */ core_api.settings.extend({ 'push_app_servers': [], 'enable_muc_push': false }); core_api.listen.on('statusInitialized', () => enablePush()); if (core_api.settings.get('enable_muc_push')) { core_api.listen.on('chatBoxesInitialized', () => shared_converse.chatboxes.on('add', onChatBoxAdded)); } } }); ;// CONCATENATED MODULE: ./src/plugins/register/templates/registration_form.js /* harmony default export */ const registration_form = (o => { const i18n_choose_provider = __('Choose a different provider'); const i18n_has_account = __('Already have a chat account?'); const i18n_legend = __('Account Registration:'); const i18n_login = __('Log in here'); const i18n_register = __('Register'); const registration_domain = core_api.settings.get('registration_domain'); return $`
    ${i18n_legend} ${o.domain}

    ${o.title}

    ${o.instructions}

    ${o.form_fields}
    ${o.fields ? $` ` : ''} ${registration_domain ? '' : $` `}

    ${i18n_has_account}

    `; }); ;// CONCATENATED MODULE: ./src/plugins/register/templates/register_panel.js const tpl_form_request = () => { const default_domain = core_api.settings.get('registration_domain'); const i18n_fetch_form = __("Hold tight, we're fetching the registration form…"); const i18n_cancel = __('Cancel'); return $`
    ${spinner({ 'classes': 'hor_centered' })}

    ${i18n_fetch_form}

    ${default_domain ? '' : $` `}
    `; }; const tpl_domain_input = () => { const domain_placeholder = core_api.settings.get('domain_placeholder'); const i18n_providers = __('Tip: A list of public XMPP providers is available'); const i18n_providers_link = __('here'); const href_providers = core_api.settings.get('providers_link'); return $`

    ${i18n_providers} ${i18n_providers_link}.

    `; }; const tpl_fetch_form_buttons = () => { const i18n_register = __('Fetch registration form'); const i18n_existing_account = __('Already have a chat account?'); const i18n_login = __('Log in here'); return $`

    ${i18n_existing_account}

    `; }; const tpl_choose_provider = () => { const default_domain = core_api.settings.get('registration_domain'); const i18n_create_account = __('Create your account'); const i18n_choose_provider = __('Please enter the XMPP provider to register with:'); return $`
    ${i18n_create_account}
    ${default_domain ? default_domain : tpl_domain_input()}
    ${default_domain ? '' : tpl_fetch_form_buttons()}
    `; }; const CHOOSE_PROVIDER = 0; const FETCHING_FORM = 1; const REGISTRATION_FORM = 2; /* harmony default export */ const register_panel = (o => { return $` ${o.model.get('registration_status') === CHOOSE_PROVIDER ? tpl_choose_provider() : ''} ${o.model.get('registration_status') === FETCHING_FORM ? tpl_form_request(o) : ''} ${o.model.get('registration_status') === REGISTRATION_FORM ? registration_form(o) : ''} `; }); ;// CONCATENATED MODULE: ./src/plugins/register/panel.js function register_panel_defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } // Strophe methods for building stanzas const { Strophe: panel_Strophe, sizzle: panel_sizzle, $iq: panel_$iq } = core_converse.env; const panel_u = core_converse.env.utils; const panel_CHOOSE_PROVIDER = 0; const panel_FETCHING_FORM = 1; const panel_REGISTRATION_FORM = 2; /** * @class * @namespace _converse.RegisterPanel * @memberOf _converse */ class RegisterPanel extends ElementView { constructor() { super(...arguments); register_panel_defineProperty(this, "id", "converse-register-panel"); register_panel_defineProperty(this, "className", 'controlbox-pane fade-in'); register_panel_defineProperty(this, "events", { 'submit form#converse-register': 'onFormSubmission', 'click .button-cancel': 'renderProviderChoiceForm' }); } initialize() { this.reset(); const controlbox = shared_converse.chatboxes.get('controlbox'); this.model = controlbox; this.listenTo(shared_converse, 'connectionInitialized', this.registerHooks); this.listenTo(this.model, 'change:registration_status', this.render); const domain = core_api.settings.get('registration_domain'); if (domain) { this.fetchRegistrationForm(domain); } else { this.model.set('registration_status', panel_CHOOSE_PROVIDER); } } render() { x(register_panel({ 'domain': this.domain, 'fields': this.fields, 'form_fields': this.form_fields, 'instructions': this.instructions, 'model': this.model, 'title': this.title }), this); } /** * Hook into Strophe's _connect_cb, so that we can send an IQ * requesting the registration fields. */ registerHooks() { const conn = shared_converse.connection; const connect_cb = conn._connect_cb.bind(conn); conn._connect_cb = (req, callback, raw) => { if (!this._registering) { connect_cb(req, callback, raw); } else { if (this.getRegistrationFields(req, callback)) { this._registering = false; } } }; } connectedCallback() { super.connectedCallback(); this.render(); } /** * Send an IQ stanza to the XMPP server asking for the registration fields. * @private * @method _converse.RegisterPanel#getRegistrationFields * @param { Strophe.Request } req - The current request * @param { Function } callback - The callback function */ getRegistrationFields(req, _callback) { const conn = shared_converse.connection; conn.connected = true; const body = conn._proto._reqToData(req); if (!body) { return; } if (conn._proto._connect_cb(body) === panel_Strophe.Status.CONNFAIL) { this.showValidationError(__("Sorry, we're unable to connect to your chosen provider.")); return false; } const register = body.getElementsByTagName("register"); const mechanisms = body.getElementsByTagName("mechanism"); if (register.length === 0 && mechanisms.length === 0) { conn._proto._no_auth_received(_callback); return false; } if (register.length === 0) { conn._changeConnectStatus(panel_Strophe.Status.REGIFAIL); this.showValidationError(__("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); const stanza = panel_$iq({ type: "get" }).c("query", { xmlns: panel_Strophe.NS.REGISTER }).tree(); stanza.setAttribute("id", conn.getUniqueId("sendIQ")); conn.send(stanza); conn.connected = false; return true; } /** * Handler for {@link _converse.RegisterPanel#getRegistrationFields} * @private * @method _converse.RegisterPanel#onRegistrationFields * @param { XMLElement } stanza - The query stanza. */ onRegistrationFields(stanza) { if (stanza.getAttribute("type") === "error") { shared_converse.connection._changeConnectStatus(panel_Strophe.Status.REGIFAIL, __('Something went wrong while establishing a connection with "%1$s". ' + 'Are you sure it exists?', this.domain)); return false; } if (stanza.getElementsByTagName("query").length !== 1) { shared_converse.connection._changeConnectStatus(panel_Strophe.Status.REGIFAIL, "unknown"); return false; } this.setFields(stanza); if (this.model.get('registration_status') === panel_FETCHING_FORM) { this.renderRegistrationForm(stanza); } return false; } reset(settings) { const defaults = { fields: {}, urls: [], title: "", instructions: "", registered: false, _registering: false, domain: null, form_type: null }; Object.assign(this, defaults); if (settings) { Object.assign(this, lodash_es_pick(settings, Object.keys(defaults))); } } /** * Event handler when the #converse-register form is submitted. * Depending on the available input fields, we delegate to other methods. * @private * @param { Event } ev */ onFormSubmission(ev) { if (ev && ev.preventDefault) { ev.preventDefault(); } if (ev.target.querySelector('input[name=domain]') === null) { this.submitRegistrationForm(ev.target); } else { this.onProviderChosen(ev.target); } } /** * Callback method that gets called when the user has chosen an XMPP provider * @private * @method _converse.RegisterPanel#onProviderChosen * @param { HTMLElement } form - The form that was submitted */ onProviderChosen(form) { const domain_input = form.querySelector('input[name=domain]'), domain = domain_input === null || domain_input === void 0 ? void 0 : domain_input.value; if (!domain) { // TODO: add validation message domain_input.classList.add('error'); return; } form.querySelector('input[type=submit]').classList.add('hidden'); this.fetchRegistrationForm(domain.trim()); } /** * Fetch a registration form from the requested domain * @private * @method _converse.RegisterPanel#fetchRegistrationForm * @param { String } domain_name - XMPP server domain */ async fetchRegistrationForm(domain_name) { var _converse$connection; this.model.set('registration_status', panel_FETCHING_FORM); this.reset({ 'domain': panel_Strophe.getDomainFromJid(domain_name), '_registering': true }); await shared_converse.initConnection(this.domain); // When testing, the test tears down before the async function // above finishes. So we use optional chaining here (_converse$connection = shared_converse.connection) === null || _converse$connection === void 0 ? void 0 : _converse$connection.connect(this.domain, "", status => this.onConnectStatusChanged(status)); return false; } giveFeedback(message, klass) { let feedback = this.querySelector('.reg-feedback'); if (feedback !== null) { feedback.parentNode.removeChild(feedback); } const form = this.querySelector('form'); form.insertAdjacentHTML('afterbegin', ''); feedback = form.querySelector('.reg-feedback'); feedback.textContent = message; if (klass) { feedback.classList.add(klass); } } showSpinner() { const form = this.querySelector('form'); x(spinner(), form); return this; } /** * Callback function called by Strophe whenever the connection status changes. * Passed to Strophe specifically during a registration attempt. * @private * @method _converse.RegisterPanel#onConnectStatusChanged * @param { integer } status_code - The Strophe.Status status code */ onConnectStatusChanged(status_code) { headless_log.debug('converse-register: onConnectStatusChanged'); if ([panel_Strophe.Status.DISCONNECTED, panel_Strophe.Status.CONNFAIL, panel_Strophe.Status.REGIFAIL, panel_Strophe.Status.NOTACCEPTABLE, panel_Strophe.Status.CONFLICT].includes(status_code)) { headless_log.error(`Problem during registration: Strophe.Status is ${shared_converse.CONNECTION_STATUS[status_code]}`); this.abortRegistration(); } else if (status_code === panel_Strophe.Status.REGISTERED) { headless_log.debug("Registered successfully."); shared_converse.connection.reset(); this.showSpinner(); if (["converse/login", "converse/register"].includes(shared_converse.router.history.getFragment())) { shared_converse.router.navigate('', { 'replace': true }); } if (this.fields.password && this.fields.username) { // automatically log the user in shared_converse.connection.connect(this.fields.username.toLowerCase() + '@' + this.domain.toLowerCase(), this.fields.password, shared_converse.onConnectStatusChanged); this.giveFeedback(__('Now logging you in'), 'info'); } else { shared_converse.giveFeedback(__('Registered successfully')); } this.reset(); } } getLegacyFormFields() { const input_fields = Object.keys(this.fields).map(key => { if (key === "username") { return form_username({ 'domain': ` @${this.domain}`, 'name': key, 'type': "text", 'label': key, 'value': '', 'required': true }); } else { return form_input({ 'label': key, 'name': key, 'placeholder': key, 'required': true, 'type': key === 'password' || key === 'email' ? key : "text", 'value': '' }); } }); const urls = this.urls.map(u => form_url({ 'label': '', 'value': u })); return [...input_fields, ...urls]; } getFormFields(stanza) { if (this.form_type === 'xform') { return Array.from(stanza.querySelectorAll('field')).map(field => utils_form.xForm2TemplateResult(field, stanza, { 'domain': this.domain })); } else { return this.getLegacyFormFields(); } } /** * Renders the registration form based on the XForm fields * received from the XMPP server. * @private * @method _converse.RegisterPanel#renderRegistrationForm * @param { XMLElement } stanza - The IQ stanza received from the XMPP server. */ renderRegistrationForm(stanza) { this.form_fields = this.getFormFields(stanza); this.model.set('registration_status', panel_REGISTRATION_FORM); } showValidationError(message) { const form = this.querySelector('form'); let flash = form.querySelector('.form-errors'); if (flash === null) { flash = ''; const instructions = form.querySelector('p.instructions'); if (instructions === null) { form.insertAdjacentHTML('afterbegin', flash); } else { instructions.insertAdjacentHTML('afterend', flash); } flash = form.querySelector('.form-errors'); } else { flash.innerHTML = ''; } flash.insertAdjacentHTML('beforeend', '

    ' + message + '

    '); flash.classList.remove('hidden'); } /** * Report back to the user any error messages received from the * XMPP server after attempted registration. * @private * @method _converse.RegisterPanel#reportErrors * @param { XMLElement } stanza - The IQ stanza received from the XMPP server */ reportErrors(stanza) { const errors = stanza.querySelectorAll('error'); errors.forEach(e => this.showValidationError(e.textContent)); if (!errors.length) { const message = __('The provider rejected your registration attempt. ' + 'Please check the values you entered for correctness.'); this.showValidationError(message); } } renderProviderChoiceForm(ev) { if (ev && ev.preventDefault) { ev.preventDefault(); } shared_converse.connection._proto._abortAllRequests(); shared_converse.connection.reset(); this.render(); } abortRegistration() { shared_converse.connection._proto._abortAllRequests(); shared_converse.connection.reset(); if ([panel_FETCHING_FORM, panel_REGISTRATION_FORM].includes(this.model.get('registration_status'))) { if (core_api.settings.get('registration_domain')) { this.fetchRegistrationForm(core_api.settings.get('registration_domain')); } } else { this.render(); } } /** * Handler, when the user submits the registration form. * Provides form error feedback or starts the registration process. * @private * @method _converse.RegisterPanel#submitRegistrationForm * @param { HTMLElement } form - The HTML form that was submitted */ submitRegistrationForm(form) { const has_empty_inputs = Array.from(this.querySelectorAll('input.required')).reduce((result, input) => { if (input.value === '') { input.classList.add('error'); return result + 1; } return result; }, 0); if (has_empty_inputs) { return; } const inputs = panel_sizzle(':input:not([type=button]):not([type=submit])', form); const iq = panel_$iq({ 'type': 'set', 'id': panel_u.getUniqueId() }).c("query", { xmlns: panel_Strophe.NS.REGISTER }); if (this.form_type === 'xform') { iq.c("x", { xmlns: panel_Strophe.NS.XFORM, type: 'submit' }); const xml_nodes = inputs.map(i => utils_form.webForm2xForm(i)).filter(n => n); xml_nodes.forEach(n => iq.cnode(n).up()); } else { inputs.forEach(input => iq.c(input.getAttribute('name'), {}, input.value)); } shared_converse.connection._addSysHandler(this._onRegisterIQ.bind(this), null, "iq", null, null); shared_converse.connection.send(iq); this.setFields(iq.tree()); } /* Stores the values that will be sent to the XMPP server during attempted registration. * @private * @method _converse.RegisterPanel#setFields * @param { XMLElement } stanza - the IQ stanza that will be sent to the XMPP server. */ setFields(stanza) { const query = stanza.querySelector('query'); const xform = panel_sizzle(`x[xmlns="${panel_Strophe.NS.XFORM}"]`, query); if (xform.length > 0) { this._setFieldsFromXForm(xform.pop()); } else { this._setFieldsFromLegacy(query); } } _setFieldsFromLegacy(query) { [].forEach.call(query.children, field => { if (field.tagName.toLowerCase() === 'instructions') { this.instructions = panel_Strophe.getText(field); return; } else if (field.tagName.toLowerCase() === 'x') { if (field.getAttribute('xmlns') === 'jabber:x:oob') { this.urls.concat(panel_sizzle('url', field).map(u => u.textContent)); } return; } this.fields[field.tagName.toLowerCase()] = panel_Strophe.getText(field); }); this.form_type = 'legacy'; } _setFieldsFromXForm(xform) { var _xform$querySelector, _xform$querySelector2; this.title = (_xform$querySelector = xform.querySelector('title')) === null || _xform$querySelector === void 0 ? void 0 : _xform$querySelector.textContent; this.instructions = (_xform$querySelector2 = xform.querySelector('instructions')) === null || _xform$querySelector2 === void 0 ? void 0 : _xform$querySelector2.textContent; xform.querySelectorAll('field').forEach(field => { const _var = field.getAttribute('var'); if (_var) { var _field$querySelector; this.fields[_var.toLowerCase()] = ((_field$querySelector = field.querySelector('value')) === null || _field$querySelector === void 0 ? void 0 : _field$querySelector.textContent) ?? ''; } else { // TODO: other option seems to be type="fixed" headless_log.warn("Found field we couldn't parse"); } }); this.form_type = 'xform'; } /** * Callback method that gets called when a return IQ stanza * is received from the XMPP server, after attempting to * register a new user. * @private * @method _converse.RegisterPanel#reportErrors * @param { XMLElement } stanza - The IQ stanza. */ _onRegisterIQ(stanza) { if (stanza.getAttribute("type") === "error") { headless_log.error("Registration failed."); this.reportErrors(stanza); let error = stanza.getElementsByTagName("error"); if (error.length !== 1) { shared_converse.connection._changeConnectStatus(panel_Strophe.Status.REGIFAIL, "unknown"); return false; } error = error[0].firstElementChild.tagName.toLowerCase(); if (error === 'conflict') { shared_converse.connection._changeConnectStatus(panel_Strophe.Status.CONFLICT, error); } else if (error === 'not-acceptable') { shared_converse.connection._changeConnectStatus(panel_Strophe.Status.NOTACCEPTABLE, error); } else { shared_converse.connection._changeConnectStatus(panel_Strophe.Status.REGIFAIL, error); } } else { shared_converse.connection._changeConnectStatus(panel_Strophe.Status.REGISTERED, null); } return false; } } core_api.elements.define('converse-register-panel', RegisterPanel); ;// CONCATENATED MODULE: ./src/plugins/register/index.js /** * @module converse-register * @description * This is a Converse.js plugin which add support for in-band registration * as specified in XEP-0077. * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ // Strophe methods for building stanzas const { Strophe: register_Strophe } = core_converse.env; // Add Strophe Namespaces register_Strophe.addNamespace('REGISTER', 'jabber:iq:register'); // Add Strophe Statuses const register_i = Object.keys(register_Strophe.Status).reduce((max, k) => Math.max(max, register_Strophe.Status[k]), 0); register_Strophe.Status.REGIFAIL = register_i + 1; register_Strophe.Status.REGISTERED = register_i + 2; register_Strophe.Status.CONFLICT = register_i + 3; register_Strophe.Status.NOTACCEPTABLE = register_i + 5; core_converse.plugins.add('converse-register', { dependencies: ['converse-controlbox'], enabled() { return true; }, initialize() { shared_converse.CONNECTION_STATUS[register_Strophe.Status.REGIFAIL] = 'REGIFAIL'; shared_converse.CONNECTION_STATUS[register_Strophe.Status.REGISTERED] = 'REGISTERED'; shared_converse.CONNECTION_STATUS[register_Strophe.Status.CONFLICT] = 'CONFLICT'; shared_converse.CONNECTION_STATUS[register_Strophe.Status.NOTACCEPTABLE] = 'NOTACCEPTABLE'; core_api.settings.extend({ 'allow_registration': true, 'domain_placeholder': __(' e.g. conversejs.org'), // Placeholder text shown in the domain input on the registration form 'providers_link': 'https://compliance.conversations.im/', // Link to XMPP providers shown on registration page 'registration_domain': '' }); async function setActiveForm(value) { await core_api.waitUntil('controlBoxInitialized'); const controlbox = shared_converse.chatboxes.get('controlbox'); controlbox.set({ 'active-form': value }); } shared_converse.router.route('converse/login', () => setActiveForm('login')); shared_converse.router.route('converse/register', () => setActiveForm('register')); core_api.listen.on('controlBoxInitialized', view => { view.model.on('change:active-form', view.showLoginOrRegisterForm, view); }); } }); ;// CONCATENATED MODULE: ./src/plugins/roomslist/model.js const { Strophe: roomslist_model_Strophe } = core_converse.env; const RoomsListModel = Model.extend({ defaults: function () { return { 'muc_domain': core_api.settings.get('muc_domain'), 'nick': shared_converse.getDefaultMUCNickname(), 'toggle-state': shared_converse.OPENED }; }, initialize() { core_api.settings.listen.on('change:muc_domain', muc_domain => this.setDomain(muc_domain)); }, setDomain(jid) { if (!core_api.settings.get('locked_muc_domain')) { this.save('muc_domain', roomslist_model_Strophe.getDomainFromJid(jid)); } } }); /* harmony default export */ const roomslist_model = (RoomsListModel); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/add-muc.js const nickname_input = o => { const i18n_nickname = __('Nickname'); const i18n_required_field = __('This field is required'); return $`
    `; }; /* harmony default export */ const add_muc = (o => { const i18n_join = __('Join'); const i18n_enter = __('Enter a new Groupchat'); return $` `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/modals/add-muc.js const add_muc_u = core_converse.env.utils; const { Strophe: add_muc_Strophe } = core_converse.env; /* harmony default export */ const modals_add_muc = (base.extend({ persistent: true, id: 'add-chatroom-modal', events: { 'submit form.add-chatroom': 'openChatRoom', 'keyup .roomjid-input': 'checkRoomidPolicy', 'change .roomjid-input': 'checkRoomidPolicy' }, initialize() { base.prototype.initialize.apply(this, arguments); this.listenTo(this.model, 'change:muc_domain', this.render); this.muc_roomid_policy_error_msg = null; }, toHTML() { let placeholder = ''; if (!core_api.settings.get('locked_muc_domain')) { const muc_domain = this.model.get('muc_domain') || core_api.settings.get('muc_domain'); placeholder = muc_domain ? `name@${muc_domain}` : __('name@conference.example.org'); } return add_muc(Object.assign(this.model.toJSON(), { '_converse': shared_converse, 'label_room_address': core_api.settings.get('muc_domain') ? __('Groupchat name') : __('Groupchat address'), 'chatroom_placeholder': placeholder, 'muc_roomid_policy_error_msg': this.muc_roomid_policy_error_msg, 'muc_roomid_policy_hint': core_api.settings.get('muc_roomid_policy_hint') })); }, afterRender() { this.el.addEventListener('shown.bs.modal', () => { this.el.querySelector('input[name="chatroom"]').focus(); }, false); }, parseRoomDataFromEvent(form) { const data = new FormData(form); const jid = data.get('chatroom'); let nick; if (core_api.settings.get('locked_muc_nickname')) { nick = shared_converse.getDefaultMUCNickname(); if (!nick) { throw new Error("Using locked_muc_nickname but no nickname found!"); } } else { nick = data.get('nickname').trim(); } return { 'jid': jid, 'nick': nick }; }, openChatRoom(ev) { ev.preventDefault(); const data = this.parseRoomDataFromEvent(ev.target); if (data.nick === "") { // Make sure defaults apply if no nick is provided. data.nick = undefined; } let jid; if (core_api.settings.get('locked_muc_domain') || core_api.settings.get('muc_domain') && !add_muc_u.isValidJID(data.jid)) { jid = `${add_muc_Strophe.escapeNode(data.jid)}@${core_api.settings.get('muc_domain')}`; } else { jid = data.jid; this.model.setDomain(jid); } core_api.rooms.open(jid, Object.assign(data, { jid }), true); this.modal.hide(); ev.target.reset(); }, checkRoomidPolicy() { if (core_api.settings.get('muc_roomid_policy') && core_api.settings.get('muc_domain')) { let jid = this.el.querySelector('.roomjid-input').value; if (core_api.settings.get('locked_muc_domain') || !add_muc_u.isValidJID(jid)) { jid = `${add_muc_Strophe.escapeNode(jid)}@${core_api.settings.get('muc_domain')}`; } const roomid = add_muc_Strophe.getNodeFromJid(jid); const roomdomain = add_muc_Strophe.getDomainFromJid(jid); if (core_api.settings.get('muc_domain') !== roomdomain || core_api.settings.get('muc_roomid_policy').test(roomid)) { this.muc_roomid_policy_error_msg = null; } else { this.muc_roomid_policy_error_msg = __('Groupchat id is invalid.'); } this.render(); } } })); ;// CONCATENATED MODULE: ./node_modules/lodash-es/head.js /** * Gets the first element of `array`. * * @static * @memberOf _ * @since 0.1.0 * @alias first * @category Array * @param {Array} array The array to query. * @returns {*} Returns the first element of `array`. * @example * * _.head([1, 2, 3]); * // => 1 * * _.head([]); * // => undefined */ function head(array) { return (array && array.length) ? array[0] : undefined; } /* harmony default export */ const lodash_es_head = (head); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/muc-list.js const muc_list_form = o => { const i18n_query = __('Show groupchats'); const i18n_server_address = __('Server address'); return $`
    `; }; const tpl_item = (o, item) => { const i18n_info_title = __('Show more information on this groupchat'); const i18n_open_title = __('Click to open this groupchat'); return $`
  • `; }; /* harmony default export */ const muc_list = (o => { const i18n_list_chatrooms = __('Query for Groupchats'); return $` `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/templates/muc-description.js /* harmony default export */ const muc_description = (o => { const i18n_desc = __('Description:'); const i18n_jid = __('Groupchat XMPP Address:'); const i18n_occ = __('Participants:'); const i18n_features = __('Features:'); const i18n_requires_auth = __('Requires authentication'); const i18n_hidden = __('Hidden'); const i18n_requires_invite = __('Requires an invitation'); const i18n_moderated = __('Moderated'); const i18n_non_anon = __('Non-anonymous'); const i18n_open_room = __('Open'); const i18n_permanent_room = __('Permanent'); const i18n_public = __('Public'); const i18n_semi_anon = __('Semi-anonymous'); const i18n_temp_room = __('Temporary'); const i18n_unmoderated = __('Unmoderated'); return $`

    ${i18n_jid} ${o.jid}

    ${i18n_desc} ${o.desc}

    ${i18n_occ} ${o.occ}

    ${i18n_features}

      ${o.passwordprotected ? $`
    • ${i18n_requires_auth}
    • ` : ''} ${o.hidden ? $`
    • ${i18n_hidden}
    • ` : ''} ${o.membersonly ? $`
    • ${i18n_requires_invite}
    • ` : ''} ${o.moderated ? $`
    • ${i18n_moderated}
    • ` : ''} ${o.nonanonymous ? $`
    • ${i18n_non_anon}
    • ` : ''} ${o.open ? $`
    • ${i18n_open_room}
    • ` : ''} ${o.persistent ? $`
    • ${i18n_permanent_room}
    • ` : ''} ${o.publicroom ? $`
    • ${i18n_public}
    • ` : ''} ${o.semianonymous ? $`
    • ${i18n_semi_anon}
    • ` : ''} ${o.temporary ? $`
    • ${i18n_temp_room}
    • ` : ''} ${o.unmoderated ? $`
    • ${i18n_unmoderated}
    • ` : ''}

    `; }); ;// CONCATENATED MODULE: ./src/plugins/muc-views/modals/muc-list.js const { Strophe: muc_list_Strophe, $iq: muc_list_$iq, sizzle: muc_list_sizzle } = core_converse.env; const muc_list_u = core_converse.env.utils; /* Insert groupchat info (based on returned #disco IQ stanza) * @function insertRoomInfo * @param { HTMLElement } el - The HTML DOM element that contains the info. * @param { XMLElement } stanza - The IQ stanza containing the groupchat info. */ function insertRoomInfo(el, stanza) { var _head, _head2; // All MUC features found here: https://xmpp.org/registrar/disco-features.html el.querySelector('span.spinner').remove(); el.querySelector('a.room-info').classList.add('selected'); el.insertAdjacentHTML('beforeEnd', muc_list_u.getElementFromTemplateResult(muc_description({ 'jid': stanza.getAttribute('from'), 'desc': (_head = lodash_es_head(muc_list_sizzle('field[var="muc#roominfo_description"] value', stanza))) === null || _head === void 0 ? void 0 : _head.textContent, 'occ': (_head2 = lodash_es_head(muc_list_sizzle('field[var="muc#roominfo_occupants"] value', stanza))) === null || _head2 === void 0 ? void 0 : _head2.textContent, 'hidden': muc_list_sizzle('feature[var="muc_hidden"]', stanza).length, 'membersonly': muc_list_sizzle('feature[var="muc_membersonly"]', stanza).length, 'moderated': muc_list_sizzle('feature[var="muc_moderated"]', stanza).length, 'nonanonymous': muc_list_sizzle('feature[var="muc_nonanonymous"]', stanza).length, 'open': muc_list_sizzle('feature[var="muc_open"]', stanza).length, 'passwordprotected': muc_list_sizzle('feature[var="muc_passwordprotected"]', stanza).length, 'persistent': muc_list_sizzle('feature[var="muc_persistent"]', stanza).length, 'publicroom': muc_list_sizzle('feature[var="muc_publicroom"]', stanza).length, 'semianonymous': muc_list_sizzle('feature[var="muc_semianonymous"]', stanza).length, 'temporary': muc_list_sizzle('feature[var="muc_temporary"]', stanza).length, 'unmoderated': muc_list_sizzle('feature[var="muc_unmoderated"]', stanza).length }))); } /** * Show/hide extra information about a groupchat in a listing. * @function toggleRoomInfo * @param { Event } */ function toggleRoomInfo(ev) { const parent_el = muc_list_u.ancestor(ev.target, '.room-item'); const div_el = parent_el.querySelector('div.room-info'); if (div_el) { muc_list_u.slideIn(div_el).then(muc_list_u.removeElement); parent_el.querySelector('a.room-info').classList.remove('selected'); } else { parent_el.insertAdjacentElement('beforeend', muc_list_u.getElementFromTemplateResult(spinner())); core_api.disco.info(ev.target.getAttribute('data-room-jid'), null).then(stanza => insertRoomInfo(parent_el, stanza)).catch(e => headless_log.error(e)); } } /* harmony default export */ const modals_muc_list = (base.extend({ id: "muc-list-modal", persistent: true, initialize() { this.items = []; this.loading_items = false; base.prototype.initialize.apply(this, arguments); this.listenTo(this.model, 'change:muc_domain', this.onDomainChange); this.listenTo(this.model, 'change:feedback_text', () => this.render()); this.el.addEventListener('shown.bs.modal', () => core_api.settings.get('locked_muc_domain') ? this.updateRoomsList() : this.el.querySelector('input[name="server"]').focus()); this.model.save('feedback_text', ''); }, toHTML() { return muc_list(Object.assign(this.model.toJSON(), { 'show_form': !core_api.settings.get('locked_muc_domain'), 'server_placeholder': this.model.get('muc_domain') || __('conference.example.org'), 'items': this.items, 'loading_items': this.loading_items, 'openRoom': ev => this.openRoom(ev), 'setDomainFromEvent': ev => this.setDomainFromEvent(ev), 'submitForm': ev => this.showRooms(ev), 'toggleRoomInfo': ev => this.toggleRoomInfo(ev) })); }, openRoom(ev) { ev.preventDefault(); const jid = ev.target.getAttribute('data-room-jid'); const name = ev.target.getAttribute('data-room-name'); this.modal.hide(); core_api.rooms.open(jid, { 'name': name }, true); }, toggleRoomInfo(ev) { ev.preventDefault(); toggleRoomInfo(ev); }, onDomainChange() { core_api.settings.get('auto_list_rooms') && this.updateRoomsList(); }, /** * Handle the IQ stanza returned from the server, containing * all its public groupchats. * @private * @method _converse.ChatRoomView#onRoomsFound * @param { HTMLElement } iq */ onRoomsFound(iq) { this.loading_items = false; const rooms = iq ? muc_list_sizzle('query item', iq) : []; if (rooms.length) { this.model.set({ 'feedback_text': __('Groupchats found') }, { 'silent': true }); this.items = rooms.map(getAttributes); } else { this.items = []; this.model.set({ 'feedback_text': __('No groupchats found') }, { 'silent': true }); } this.render(); return true; }, /** * Send an IQ stanza to the server asking for all groupchats * @private * @method _converse.ChatRoomView#updateRoomsList */ updateRoomsList() { const iq = muc_list_$iq({ 'to': this.model.get('muc_domain'), 'from': shared_converse.connection.jid, 'type': "get" }).c("query", { xmlns: muc_list_Strophe.NS.DISCO_ITEMS }); core_api.sendIQ(iq).then(iq => this.onRoomsFound(iq)).catch(() => this.onRoomsFound()); }, showRooms(ev) { ev.preventDefault(); this.loading_items = true; this.render(); const data = new FormData(ev.target); this.model.setDomain(data.get('server')); this.updateRoomsList(); }, setDomainFromEvent(ev) { this.model.setDomain(ev.target.value); }, setNick(ev) { this.model.save({ nick: ev.target.value }); } })); ;// CONCATENATED MODULE: ./src/plugins/roomslist/templates/roomslist.js const bookmark = o => { const i18n_add_bookmark = __('Bookmark this groupchat'); const i18n_remove_bookmark = __('Unbookmark this groupchat'); if (o.bookmarked) { return $` `; } else { return $` `; } }; const unread_indicator = o => $`${o.room.get('num_unread')}`; const activity_indicator = () => $``; const room_item = o => { const i18n_leave_room = __('Leave this groupchat'); const has_unread_msgs = o.room.get('num_unread_general') || o.room.get('has_activity'); return $`
    ${o.room.get('num_unread') ? unread_indicator(o) : o.room.get('has_activity') ? activity_indicator(o) : ''} ${o.room.getDisplayName()} ${core_api.settings.get('allow_bookmarks') ? bookmark(o) : ''}
    `; }; /* harmony default export */ const roomslist = (o => { const i18n_desc_rooms = __('Click to toggle the list of open groupchats'); const i18n_heading_chatrooms = __('Groupchats'); const i18n_title_list_rooms = __('Query for groupchats'); const i18n_title_new_room = __('Add a new groupchat'); return $`
    ${__('Open Groupchats')}
    ${o.rooms.map(room => room_item(Object.assign({ room }, o)))}
    `; }); ;// CONCATENATED MODULE: ./src/plugins/roomslist/view.js const { Strophe: view_Strophe, u: view_u } = core_converse.env; class RoomsList extends CustomElement { initialize() { const id = `converse.roomspanel${shared_converse.bare_jid}`; this.model = new roomslist_model({ id }); initStorage(this.model, id); this.model.fetch(); this.listenTo(shared_converse.chatboxes, 'add', this.renderIfChatRoom); this.listenTo(shared_converse.chatboxes, 'remove', this.renderIfChatRoom); this.listenTo(shared_converse.chatboxes, 'destroy', this.renderIfChatRoom); this.listenTo(shared_converse.chatboxes, 'change', this.renderIfRelevantChange); this.requestUpdate(); } renderIfChatRoom(model) { view_u.isChatRoom(model) && this.requestUpdate(); } renderIfRelevantChange(model) { const attrs = ['bookmarked', 'hidden', 'name', 'num_unread', 'num_unread_general', 'has_activity']; const changed = model.changed || {}; if (view_u.isChatRoom(model) && Object.keys(changed).filter(m => attrs.includes(m)).length) { this.requestUpdate(); } } render() { return roomslist({ 'addBookmark': ev => this.addBookmark(ev), 'allow_bookmarks': core_api.settings.get('allow_bookmarks') && shared_converse.bookmarks, 'closeRoom': ev => this.closeRoom(ev), 'collapsed': this.model.get('toggle-state') !== shared_converse.OPENED, 'currently_open': room => isUniView() && !room.get('hidden'), 'model': this.model, 'openRoom': ev => this.openRoom(ev), 'removeBookmark': ev => this.removeBookmark(ev), 'rooms': shared_converse.chatboxes.filter(m => m.get('type') === shared_converse.CHATROOMS_TYPE), 'showRoomDetailsModal': ev => this.showRoomDetailsModal(ev), 'toggleRoomsList': ev => this.toggleRoomsList(ev), 'toggle_state': this.model.get('toggle-state') }); } showRoomDetailsModal(ev) { // eslint-disable-line class-methods-use-this const jid = ev.target.getAttribute('data-room-jid'); const room = shared_converse.chatboxes.get(jid); ev.preventDefault(); core_api.modal.show(modals_muc_details, { 'model': room }, ev); } async openRoom(ev) { // eslint-disable-line class-methods-use-this ev.preventDefault(); const name = ev.target.textContent; const jid = ev.target.getAttribute('data-room-jid'); const data = { 'name': name || view_Strophe.unescapeNode(view_Strophe.getNodeFromJid(jid)) || jid }; await core_api.rooms.open(jid, data, true); } async closeRoom(ev) { // eslint-disable-line class-methods-use-this ev.preventDefault(); const name = ev.target.getAttribute('data-room-name'); if (confirm(__("Are you sure you want to leave the groupchat %1$s?", name))) { const jid = ev.target.getAttribute('data-room-jid'); const room = await core_api.rooms.get(jid); room.close(); } } removeBookmark(ev) { // eslint-disable-line class-methods-use-this shared_converse.removeBookmarkViaEvent(ev); } addBookmark(ev) { // eslint-disable-line class-methods-use-this shared_converse.addBookmarkViaEvent(ev); } toggleRoomsList(ev) { var _ev$preventDefault; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); const icon_el = ev.target.matches('.fa') ? ev.target : ev.target.querySelector('.fa'); if (icon_el.classList.contains("fa-caret-down")) { view_u.slideIn(this.querySelector('.open-rooms-list')).then(() => { this.model.save({ 'toggle-state': shared_converse.CLOSED }); icon_el.classList.remove("fa-caret-down"); icon_el.classList.add("fa-caret-right"); }); } else { view_u.slideOut(this.querySelector('.open-rooms-list')).then(() => { this.model.save({ 'toggle-state': shared_converse.OPENED }); icon_el.classList.remove("fa-caret-right"); icon_el.classList.add("fa-caret-down"); }); } } } core_api.elements.define('converse-rooms-list', RoomsList); ;// CONCATENATED MODULE: ./src/plugins/roomslist/index.js /** * @description * Converse.js plugin which shows a list of currently open * rooms in the "Rooms Panel" of the ControlBox. * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.plugins.add('converse-roomslist', { dependencies: ["converse-singleton", "converse-controlbox", "converse-muc", "converse-bookmarks"], initialize() {} }); ;// CONCATENATED MODULE: ./src/shared/templates/icons.js /* harmony default export */ const icons = (() => $` `); ;// CONCATENATED MODULE: ./src/shared/components/font-awesome.js class FontAwesome extends CustomElement { render() { // eslint-disable-line class-methods-use-this return icons(); } } window.customElements.define('converse-fontawesome', FontAwesome); ;// CONCATENATED MODULE: ./src/plugins/rootview/templates/root.js /* harmony default export */ const templates_root = (() => { const extra_classes = core_api.settings.get('singleton') ? ['converse-singleton'] : []; extra_classes.push(`converse-${core_api.settings.get('view_mode')}`); return $`
    `; }); ;// CONCATENATED MODULE: ./src/plugins/rootview/utils.js function getTheme() { if (window.matchMedia('(prefers-color-scheme: dark)').matches) { return core_api.settings.get('dark_theme'); } else { return core_api.settings.get('theme'); } } function ensureElement() { if (!core_api.settings.get('auto_insert')) { return; } const root = core_api.settings.get('root'); if (!root.querySelector('converse-root')) { const el = document.createElement('converse-root'); const body = root.querySelector('body'); if (body) { body.appendChild(el); } else { root.appendChild(el); // Perhaps inside a web component? } } } // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/rootview/styles/root.scss var styles_root = __webpack_require__(1513); ;// CONCATENATED MODULE: ./src/plugins/rootview/styles/root.scss var root_options = {}; root_options.styleTagTransform = (styleTagTransform_default()); root_options.setAttributes = (setAttributesWithoutAttributes_default()); root_options.insert = insertBySelector_default().bind(null, "head"); root_options.domAPI = (styleDomAPI_default()); root_options.insertStyleElement = (insertStyleElement_default()); var root_update = injectStylesIntoStyleTag_default()(styles_root/* default */.Z, root_options); /* harmony default export */ const rootview_styles_root = (styles_root/* default */.Z && styles_root/* default.locals */.Z.locals ? styles_root/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/rootview/root.js /** * `converse-root` is an optional custom element which can be used to * declaratively insert the Converse UI into the DOM. * * It can be inserted into the DOM before or after Converse has loaded or been * initialized. */ class ConverseRoot extends CustomElement { render() { // eslint-disable-line class-methods-use-this return templates_root(); } initialize() { this.setAttribute('id', 'conversejs'); this.setClasses(); const settings = getAppSettings(); this.listenTo(settings, 'change:view_mode', () => this.setClasses()); this.listenTo(settings, 'change:singleton', () => this.setClasses()); window.matchMedia('(prefers-color-scheme: dark)').addListener(() => this.setClasses()); window.matchMedia('(prefers-color-scheme: light)').addListener(() => this.setClasses()); } setClasses() { this.className = ""; this.classList.add('conversejs'); this.classList.add(`converse-${core_api.settings.get('view_mode')}`); this.classList.add(`theme-${getTheme()}`); this.requestUpdate(); } } ;// CONCATENATED MODULE: ./src/plugins/rootview/index.js core_converse.plugins.add('converse-rootview', { initialize() { // Configuration values for this plugin // ==================================== // Refer to docs/source/configuration.rst for explanations of these // configuration settings. core_api.settings.extend({ 'auto_insert': true, 'theme': 'classic', 'dark_theme': 'dracula' }); core_api.listen.on('chatBoxesInitialized', ensureElement); // Only define the element now, otherwise it it's already in the DOM // before `converse.initialized` has been called it will render too // early. core_api.elements.define('converse-root', ConverseRoot); } }); ;// CONCATENATED MODULE: ./src/modals/templates/add-contact.js /* harmony default export */ const add_contact = (o => { const i18n_contact_placeholder = __('name@example.org'); const i18n_add = __('Add'); const i18n_error_message = __('Please enter a valid XMPP address'); const i18n_new_contact = __('Add a Contact'); const i18n_xmpp_address = __('XMPP Address'); const i18n_nickname = __('Nickname'); return $` `; }); ;// CONCATENATED MODULE: ./src/modals/add-contact.js const { Strophe: add_contact_Strophe } = core_converse.env; const add_contact_u = core_converse.env.utils; const AddContactModal = base.extend({ id: "add-contact-modal", events: { 'submit form': 'addContactFromForm' }, initialize() { base.prototype.initialize.apply(this, arguments); this.listenTo(this.model, 'change', this.render); }, toHTML() { const label_nickname = core_api.settings.get('xhr_user_search_url') ? __('Contact name') : __('Optional nickname'); return add_contact(Object.assign(this.model.toJSON(), { _converse: shared_converse, label_nickname })); }, afterRender() { if (typeof core_api.settings.get('xhr_user_search_url') === 'string') { this.initXHRAutoComplete(); } else { this.initJIDAutoComplete(); } const jid_input = this.el.querySelector('input[name="jid"]'); this.el.addEventListener('shown.bs.modal', () => jid_input.focus(), false); }, initJIDAutoComplete() { if (!core_api.settings.get('autocomplete_add_contact')) { return; } const el = this.el.querySelector('.suggestion-box__jid').parentElement; this.jid_auto_complete = new shared_converse.AutoComplete(el, { 'data': (text, input) => `${input.slice(0, input.indexOf("@"))}@${text}`, 'filter': shared_converse.FILTER_STARTSWITH, 'list': [...new Set(shared_converse.roster.map(item => add_contact_Strophe.getDomainFromJid(item.get('jid'))))] }); }, initXHRAutoComplete() { if (!core_api.settings.get('autocomplete_add_contact')) { return this.initXHRFetch(); } const el = this.el.querySelector('.suggestion-box__name').parentElement; this.name_auto_complete = new shared_converse.AutoComplete(el, { 'auto_evaluate': false, 'filter': shared_converse.FILTER_STARTSWITH, 'list': [] }); const xhr = new window.XMLHttpRequest(); // `open` must be called after `onload` for mock/testing purposes. xhr.onload = () => { if (xhr.responseText) { const r = xhr.responseText; this.name_auto_complete.list = JSON.parse(r).map(i => ({ 'label': i.fullname || i.jid, 'value': i.jid })); this.name_auto_complete.auto_completing = true; this.name_auto_complete.evaluate(); } }; const input_el = this.el.querySelector('input[name="name"]'); input_el.addEventListener('input', lodash_es_debounce(() => { xhr.open("GET", `${core_api.settings.get('xhr_user_search_url')}q=${encodeURIComponent(input_el.value)}`, true); xhr.send(); }, 300)); this.name_auto_complete.on('suggestion-box-selectcomplete', ev => { this.el.querySelector('input[name="name"]').value = ev.text.label; this.el.querySelector('input[name="jid"]').value = ev.text.value; }); }, initXHRFetch() { this.xhr = new window.XMLHttpRequest(); this.xhr.onload = () => { if (this.xhr.responseText) { const r = this.xhr.responseText; const list = JSON.parse(r).map(i => ({ 'label': i.fullname || i.jid, 'value': i.jid })); if (list.length !== 1) { const el = this.el.querySelector('.invalid-feedback'); el.textContent = __('Sorry, could not find a contact with that name'); add_contact_u.addClass('d-block', el); return; } const jid = list[0].value; if (this.validateSubmission(jid)) { const form = this.el.querySelector('form'); const name = list[0].label; this.afterSubmission(form, jid, name); } } }; }, validateSubmission(jid) { const el = this.el.querySelector('.invalid-feedback'); if (!jid || lodash_es_compact(jid.split('@')).length < 2) { add_contact_u.addClass('is-invalid', this.el.querySelector('input[name="jid"]')); add_contact_u.addClass('d-block', el); return false; } else if (shared_converse.roster.get(add_contact_Strophe.getBareJidFromJid(jid))) { el.textContent = __('This contact has already been added'); add_contact_u.addClass('d-block', el); return false; } add_contact_u.removeClass('d-block', el); return true; }, afterSubmission(form, jid, name) { shared_converse.roster.addAndSubscribe(jid, name); this.model.clear(); this.modal.hide(); }, addContactFromForm(ev) { ev.preventDefault(); const data = new FormData(ev.target), jid = (data.get('jid') || '').trim(); if (!jid && typeof core_api.settings.get('xhr_user_search_url') === 'string') { const input_el = this.el.querySelector('input[name="name"]'); this.xhr.open("GET", `${core_api.settings.get('xhr_user_search_url')}q=${encodeURIComponent(input_el.value)}`, true); this.xhr.send(); return; } if (this.validateSubmission(jid)) { this.afterSubmission(ev.target, jid, data.get('name')); } } }); shared_converse.AddContactModal = AddContactModal; /* harmony default export */ const modals_add_contact = ((/* unused pure expression or super */ null && (AddContactModal))); ;// CONCATENATED MODULE: ./src/plugins/rosterview/utils.js function highlightRosterItem(chatbox) { var _converse$roster, _converse$roster$get; (_converse$roster = shared_converse.roster) === null || _converse$roster === void 0 ? void 0 : (_converse$roster$get = _converse$roster.get(chatbox.get('jid'))) === null || _converse$roster$get === void 0 ? void 0 : _converse$roster$get.trigger('highlight'); } function toggleGroup(ev, name) { var _ev$preventDefault; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); const collapsed = shared_converse.roster.state.get('collapsed_groups'); if (collapsed.includes(name)) { shared_converse.roster.state.save('collapsed_groups', collapsed.filter(n => n !== name)); } else { shared_converse.roster.state.save('collapsed_groups', [...collapsed, name]); } } function isContactFiltered(contact, groupname) { const filter = shared_converse.roster_filter; const type = filter.get('filter_type'); const q = type === 'state' ? filter.get('chat_state').toLowerCase() : filter.get('filter_text').toLowerCase(); if (!q) return false; if (type === 'state') { const sticky_groups = [shared_converse.HEADER_REQUESTING_CONTACTS, shared_converse.HEADER_UNREAD]; if (sticky_groups.includes(groupname)) { // When filtering by chat state, we still want to // show sticky groups, even though they don't // match the state in question. return false; } else if (q === 'unread_messages') { return contact.get('num_unread') === 0; } else if (q === 'online') { return ["offline", "unavailable"].includes(contact.presence.get('show')); } else { return !contact.presence.get('show').includes(q); } } else if (type === 'contacts') { return !contact.getFilterCriteria().includes(q); } } function shouldShowContact(contact, groupname) { const chat_status = contact.presence.get('show'); if (core_api.settings.get('hide_offline_users') && chat_status === 'offline') { // If pending or requesting, show if (contact.get('ask') === 'subscribe' || contact.get('subscription') === 'from' || contact.get('requesting') === true) { return !isContactFiltered(contact, groupname); } return false; } return !isContactFiltered(contact, groupname); } function shouldShowGroup(group) { const filter = shared_converse.roster_filter; const type = filter.get('filter_type'); if (type === 'groups') { var _filter$get; const q = (_filter$get = filter.get('filter_text')) === null || _filter$get === void 0 ? void 0 : _filter$get.toLowerCase(); if (!q) { return true; } if (!group.toLowerCase().includes(q)) { return false; } } return true; } ;// CONCATENATED MODULE: ./src/plugins/rosterview/templates/group.js const { u: group_u } = core_converse.env; function renderContact(contact) { const jid = contact.get('jid'); const extra_classes = []; if (isUniView()) { const chatbox = shared_converse.chatboxes.get(jid); if (chatbox && !chatbox.get('hidden')) { extra_classes.push('open'); } } const ask = contact.get('ask'); const requesting = contact.get('requesting'); const subscription = contact.get('subscription'); if (ask === 'subscribe' || subscription === 'from') { /* ask === 'subscribe' * Means we have asked to subscribe to them. * * subscription === 'from' * They are subscribed to us, 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. */ extra_classes.push('pending-xmpp-contact'); } else if (requesting === true) { extra_classes.push('requesting-xmpp-contact'); } else if (subscription === 'both' || subscription === 'to' || group_u.isSameBareJID(jid, shared_converse.connection.jid)) { extra_classes.push('current-xmpp-contact'); extra_classes.push(subscription); extra_classes.push(contact.presence.get('show')); } return $`
  • `; } /* harmony default export */ const group = (o => { const i18n_title = __('Click to hide these contacts'); const collapsed = shared_converse.roster.state.get('collapsed_groups'); return $`
    toggleGroup(ev, o.name)}> ${o.name}
      ${o.contacts.map(renderContact)}
    `; }); ;// CONCATENATED MODULE: ./src/plugins/rosterview/templates/roster.js function populateContactsMap(contacts_map, contact) { if (contact.get('ask') === 'subscribe') { const name = shared_converse.HEADER_PENDING_CONTACTS; contacts_map[name] ? contacts_map[name].push(contact) : contacts_map[name] = [contact]; } else if (contact.get('requesting')) { const name = shared_converse.HEADER_REQUESTING_CONTACTS; contacts_map[name] ? contacts_map[name].push(contact) : contacts_map[name] = [contact]; } else { let contact_groups; if (core_api.settings.get('roster_groups')) { contact_groups = contact.get('groups'); contact_groups = contact_groups.length === 0 ? [shared_converse.HEADER_UNGROUPED] : contact_groups; } else { contact_groups = [shared_converse.HEADER_CURRENT_CONTACTS]; } for (const name of contact_groups) { contacts_map[name] ? contacts_map[name].push(contact) : contacts_map[name] = [contact]; } } if (contact.get('num_unread')) { const name = shared_converse.HEADER_UNREAD; contacts_map[name] ? contacts_map[name].push(contact) : contacts_map[name] = [contact]; } return contacts_map; } /* harmony default export */ const roster = (el => { const i18n_heading_contacts = __('Contacts'); const i18n_title_add_contact = __('Add a contact'); const i18n_title_sync_contacts = __('Re-sync your contacts'); const roster = shared_converse.roster || []; const contacts_map = roster.reduce((acc, contact) => populateContactsMap(acc, contact), {}); const groupnames = Object.keys(contacts_map).filter(shouldShowGroup); groupnames.sort(groupsComparator); return $`
    ${repeat_c(groupnames, n => n, name => { const contacts = contacts_map[name].filter(c => shouldShowContact(c, name)); contacts.sort(contactsComparator); if (contacts.length) { return group({ 'contacts': contacts, 'name': name }); } else { return ''; } })}
    `; }); ;// CONCATENATED MODULE: ./src/plugins/rosterview/rosterview.js /** * @class * @namespace _converse.RosterView * @memberOf _converse */ class RosterView extends CustomElement { async initialize() { await core_api.waitUntil('rosterInitialized'); this.listenTo(shared_converse, 'rosterContactsFetched', this.requestUpdate); this.listenTo(shared_converse.presences, 'change:show', this.requestUpdate); this.listenTo(shared_converse.roster, 'add', this.requestUpdate); this.listenTo(shared_converse.roster, 'destroy', this.requestUpdate); this.listenTo(shared_converse.roster, 'remove', this.requestUpdate); this.listenTo(shared_converse.roster, 'change', this.requestUpdate); this.listenTo(shared_converse.roster.state, 'change', this.requestUpdate); /** * Triggered once the _converse.RosterView instance has been created and initialized. * @event _converse#rosterViewInitialized * @example _converse.api.listen.on('rosterViewInitialized', () => { ... }); */ core_api.trigger('rosterViewInitialized'); } firstUpdated() { this.listenToRosterFilter(); } render() { return roster(this); } listenToRosterFilter() { this.filter_view = this.querySelector('converse-roster-filter'); this.filter_view.addEventListener('update', () => this.requestUpdate()); } showAddContactModal(ev) { // eslint-disable-line class-methods-use-this core_api.modal.show(shared_converse.AddContactModal, { 'model': new Model() }, ev); } async syncContacts(ev) { // eslint-disable-line class-methods-use-this ev.preventDefault(); this.syncing_contacts = true; this.requestUpdate(); shared_converse.roster.data.save('version', null); await shared_converse.roster.fetchFromServer(); core_api.user.presence.send(); this.syncing_contacts = false; this.requestUpdate(); } } core_api.elements.define('converse-roster', RosterView); ;// CONCATENATED MODULE: ./src/plugins/rosterview/templates/pending_contact.js const tpl_pending_contact = o => $`${o.display_name}`; /* harmony default export */ const pending_contact = (o => { const i18n_remove = __('Click to remove %1$s as a contact', o.display_name); return $` ${core_api.settings.get('allow_chat_pending_contacts') ? $`${tpl_pending_contact(o)}` : tpl_pending_contact(o)} `; }); ;// CONCATENATED MODULE: ./src/plugins/rosterview/templates/requesting_contact.js const tpl_requesting_contact = o => $`${o.display_name}`; /* harmony default export */ const requesting_contact = (o => $` ${core_api.settings.get('allow_chat_pending_contacts') ? $`${tpl_requesting_contact(o)}` : tpl_requesting_contact(o)} `); ;// CONCATENATED MODULE: ./src/plugins/rosterview/constants.js const 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') }; ;// CONCATENATED MODULE: ./src/plugins/rosterview/templates/roster_item.js const tpl_remove_link = (el, item) => { const display_name = item.getDisplayName(); const i18n_remove = __('Click to remove %1$s as a contact', display_name); return $` `; }; /* harmony default export */ const roster_item = ((el, item) => { var _el$model$vcard, _el$model$vcard2; const show = item.presence.get('show') || 'offline'; let classes, color; if (show === 'online') { [classes, color] = ['fa fa-circle', 'chat-status-online']; } else if (show === 'dnd') { [classes, color] = ['fa fa-minus-circle', 'chat-status-busy']; } else if (show === 'away') { [classes, color] = ['fa fa-circle', 'chat-status-away']; } else { [classes, color] = ['fa fa-circle', 'subdued-color']; } const desc_status = STATUSES[show]; const num_unread = item.get('num_unread') || 0; const display_name = item.getDisplayName(); const i18n_chat = __('Click to chat with %1$s (XMPP address: %2$s)', display_name, el.model.get('jid')); return $` ${num_unread ? $`${num_unread}` : ''} ${display_name} ${core_api.settings.get('allow_contact_removal') ? tpl_remove_link(el, item) : ''}`; }); ;// CONCATENATED MODULE: ./src/plugins/rosterview/contactview.js const contactview_u = core_converse.env.utils; class contactview_RosterContact extends CustomElement { static get properties() { return { model: { type: Object } }; } initialize() { this.listenTo(this.model, "change", () => this.requestUpdate()); this.listenTo(this.model, "highlight", () => this.requestUpdate()); this.listenTo(this.model, 'vcard:add', () => this.requestUpdate()); this.listenTo(this.model, 'vcard:change', () => this.requestUpdate()); } render() { const ask = this.model.get('ask'); const requesting = this.model.get('requesting'); const subscription = this.model.get('subscription'); const jid = this.model.get('jid'); if (ask === 'subscribe' || subscription === 'from') { /* 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. */ const display_name = this.model.getDisplayName(); return pending_contact(Object.assign(this.model.toJSON(), { display_name, 'openChat': ev => this.openChat(ev), 'removeContact': ev => this.removeContact(ev) })); } else if (requesting === true) { const display_name = this.model.getDisplayName(); return requesting_contact(Object.assign(this.model.toJSON(), { display_name, 'openChat': ev => this.openChat(ev), 'acceptRequest': ev => this.acceptRequest(ev), 'declineRequest': ev => this.declineRequest(ev), 'desc_accept': __("Click to accept the contact request from %1$s", display_name), 'desc_decline': __("Click to decline the contact request from %1$s", display_name), 'allow_chat_pending_contacts': core_api.settings.get('allow_chat_pending_contacts') })); } else if (subscription === 'both' || subscription === 'to' || contactview_u.isSameBareJID(jid, shared_converse.connection.jid)) { return this.renderRosterItem(this.model); } } renderRosterItem(item) { return roster_item(this, item); } openChat(ev) { var _ev$preventDefault; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault = ev.preventDefault) === null || _ev$preventDefault === void 0 ? void 0 : _ev$preventDefault.call(ev); this.model.openChat(); } removeContact(ev) { var _ev$preventDefault2; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault2 = ev.preventDefault) === null || _ev$preventDefault2 === void 0 ? void 0 : _ev$preventDefault2.call(ev); if (!core_api.settings.get('allow_contact_removal')) { return; } if (!confirm(__("Are you sure you want to remove this contact?"))) { return; } try { this.model.removeFromRoster(); if (this.model.collection) { // The model might have already been removed as // result of a roster push. this.model.destroy(); } } catch (e) { headless_log.error(e); core_api.alert('error', __('Error'), [__('Sorry, there was an error while trying to remove %1$s as a contact.', this.model.getDisplayName())]); } } async acceptRequest(ev) { var _ev$preventDefault3; ev === null || ev === void 0 ? void 0 : (_ev$preventDefault3 = ev.preventDefault) === null || _ev$preventDefault3 === void 0 ? void 0 : _ev$preventDefault3.call(ev); await shared_converse.roster.sendContactAddIQ(this.model.get('jid'), this.model.getFullname(), []); this.model.authorize().subscribe(); } declineRequest(ev) { if (ev && ev.preventDefault) { ev.preventDefault(); } const result = confirm(__("Are you sure you want to decline this contact request?")); if (result === true) { this.model.unauthorize().destroy(); } return this; } } core_api.elements.define('converse-roster-contact', contactview_RosterContact); ;// CONCATENATED MODULE: ./src/plugins/rosterview/templates/roster_filter.js /* harmony default export */ const roster_filter = (o => { const i18n_placeholder = __('Filter'); const title_contact_filter = __('Filter by contact name'); const title_group_filter = __('Filter by group name'); const title_status_filter = __('Filter by status'); const label_any = __('Any'); const label_unread_messages = __('Unread'); const label_online = __('Online'); const label_chatty = __('Chatty'); const label_busy = __('Busy'); const label_away = __('Away'); const label_xa = __('Extended Away'); const label_offline = __('Offline'); return $`
    `; }); ;// CONCATENATED MODULE: ./src/plugins/rosterview/filterview.js class RosterFilterView extends CustomElement { async initialize() { await core_api.waitUntil('rosterInitialized'); this.model = shared_converse.roster_filter; this.liveFilter = lodash_es_debounce(() => { this.model.save({ 'filter_text': this.querySelector('.roster-filter').value }); }, 250); this.listenTo(shared_converse, 'rosterContactsFetched', () => this.requestUpdate()); this.listenTo(shared_converse.presences, 'change:show', () => this.requestUpdate()); this.listenTo(shared_converse.roster, "add", () => this.requestUpdate()); this.listenTo(shared_converse.roster, "destroy", () => this.requestUpdate()); this.listenTo(shared_converse.roster, "remove", () => this.requestUpdate()); this.listenTo(this.model, 'change', this.dispatchUpdateEvent); this.listenTo(this.model, 'change', () => this.requestUpdate()); this.requestUpdate(); } render() { return this.model ? roster_filter(Object.assign(this.model.toJSON(), { visible: this.shouldBeVisible(), changeChatStateFilter: ev => this.changeChatStateFilter(ev), changeTypeFilter: ev => this.changeTypeFilter(ev), clearFilter: ev => this.clearFilter(ev), liveFilter: ev => this.liveFilter(ev), submitFilter: ev => this.submitFilter(ev) })) : ''; } dispatchUpdateEvent() { this.dispatchEvent(new CustomEvent('update', { 'detail': this.model.changed })); } changeChatStateFilter(ev) { ev && ev.preventDefault(); this.model.save({ 'chat_state': this.querySelector('.state-type').value }); } changeTypeFilter(ev) { var _ancestor; ev && ev.preventDefault(); const type = ((_ancestor = ancestor(ev.target, 'converse-icon')) === null || _ancestor === void 0 ? void 0 : _ancestor.dataset.type) || 'contacts'; if (type === 'state') { this.model.save({ 'filter_type': type, 'chat_state': this.querySelector('.state-type').value }); } else { this.model.save({ 'filter_type': type, 'filter_text': this.querySelector('.roster-filter').value }); } } submitFilter(ev) { ev && ev.preventDefault(); this.liveFilter(); } /** * Returns true if the filter is enabled (i.e. if the user * has added values to the filter). * @private * @method _converse.RosterFilterView#isActive */ isActive() { return this.model.get('filter_type') === 'state' || this.model.get('filter_text'); } shouldBeVisible() { var _converse$roster; return ((_converse$roster = shared_converse.roster) === null || _converse$roster === void 0 ? void 0 : _converse$roster.length) >= 5 || this.isActive(); } clearFilter(ev) { ev && ev.preventDefault(); this.model.save({ 'filter_text': '' }); } } core_api.elements.define('converse-roster-filter', RosterFilterView); // EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[2].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[2].use[3]!./node_modules/mini-css-extract-plugin/dist/loader.js!./node_modules/css-loader/dist/cjs.js??ruleSet[1].rules[5].use[1]!./node_modules/postcss-loader/dist/cjs.js!./node_modules/sass-loader/dist/cjs.js??ruleSet[1].rules[5].use[3]!./src/plugins/rosterview/styles/roster.scss var styles_roster = __webpack_require__(1984); ;// CONCATENATED MODULE: ./src/plugins/rosterview/styles/roster.scss var roster_options = {}; roster_options.styleTagTransform = (styleTagTransform_default()); roster_options.setAttributes = (setAttributesWithoutAttributes_default()); roster_options.insert = insertBySelector_default().bind(null, "head"); roster_options.domAPI = (styleDomAPI_default()); roster_options.insertStyleElement = (insertStyleElement_default()); var roster_update = injectStylesIntoStyleTag_default()(styles_roster/* default */.Z, roster_options); /* harmony default export */ const rosterview_styles_roster = (styles_roster/* default */.Z && styles_roster/* default.locals */.Z.locals ? styles_roster/* default.locals */.Z.locals : undefined); ;// CONCATENATED MODULE: ./src/plugins/rosterview/index.js /** * @copyright 2022, the Converse.js contributors * @license Mozilla Public License (MPLv2) */ core_converse.plugins.add('converse-rosterview', { dependencies: ["converse-roster", "converse-modal", "converse-chatboxviews"], initialize() { core_api.settings.extend({ 'autocomplete_add_contact': true, 'allow_chat_pending_contacts': true, 'allow_contact_removal': true, 'hide_offline_users': false, 'roster_groups': true, 'xhr_user_search_url': null }); core_api.promises.add('rosterViewInitialized'); shared_converse.RosterFilter = RosterFilter; shared_converse.RosterFilterView = RosterFilterView; shared_converse.RosterContactView = contactview_RosterContact; /* -------- Event Handlers ----------- */ core_api.listen.on('chatBoxesInitialized', () => { shared_converse.chatboxes.on('destroy', chatbox => highlightRosterItem(chatbox)); shared_converse.chatboxes.on('change:hidden', chatbox => highlightRosterItem(chatbox)); }); core_api.listen.on('afterTearDown', () => { var _converse$rotergroups; return (_converse$rotergroups = shared_converse.rotergroups) === null || _converse$rotergroups === void 0 ? void 0 : _converse$rotergroups.off().reset(); }); } }); ;// CONCATENATED MODULE: ./src/converse.js /** * @description Converse.js (A browser based XMPP chat client) * @copyright 2021, The Converse developers * @license Mozilla Public License (MPLv2) */ /* START: Removable plugins * ------------------------ * Any of the following plugin imports may be removed if the plugin is not needed */ // Views for XEP-0048 Bookmarks // Renders standalone chat boxes for single user chat // The control box // Allows chat boxes to be resized by dragging them // Allows chat boxes to be minimized // Views related to MUC // XEP-0357 Push Notifications // XEP-0077 In-band registration // Show currently open chat rooms /* END: Removable components */ shared_converse.CustomElement = CustomElement; const initialize = core_converse.initialize; core_converse.initialize = function (settings, callback) { if (Array.isArray(settings.whitelisted_plugins)) { settings.whitelisted_plugins = settings.whitelisted_plugins.concat(VIEW_PLUGINS); } else { settings.whitelisted_plugins = VIEW_PLUGINS; } return initialize(settings, callback); }; /* harmony default export */ const src_converse = (core_converse); /***/ }), /***/ 6151: /***/ (function(module, exports, __webpack_require__) { var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_DEFINE_RESULT__;// Native Javascript for Bootstrap 4 v2.0.27 | © dnp_theme | MIT-License (function (root, factory) { if (true) { // AMD support: !(__WEBPACK_AMD_DEFINE_ARRAY__ = [], __WEBPACK_AMD_DEFINE_FACTORY__ = (factory), __WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ? (__WEBPACK_AMD_DEFINE_FACTORY__.apply(exports, __WEBPACK_AMD_DEFINE_ARRAY__)) : __WEBPACK_AMD_DEFINE_FACTORY__), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__)); } else { var bsn; } }(this, function () { /* Native Javascript for Bootstrap 4 | Internal Utility Functions ----------------------------------------------------------------*/ "use strict"; // globals var globalObject = typeof __webpack_require__.g !== 'undefined' ? __webpack_require__.g : this||window, DOC = document, HTML = DOC.documentElement, body = 'body', // allow the library to be used in // Native Javascript for Bootstrap Global Object BSN = globalObject.BSN = {}, supports = BSN.supports = [], // function toggle attributes dataToggle = 'data-toggle', dataDismiss = 'data-dismiss', dataSpy = 'data-spy', dataRide = 'data-ride', // components stringAlert = 'Alert', stringButton = 'Button', stringCarousel = 'Carousel', stringCollapse = 'Collapse', stringDropdown = 'Dropdown', stringModal = 'Modal', stringPopover = 'Popover', stringScrollSpy = 'ScrollSpy', stringTab = 'Tab', stringTooltip = 'Tooltip', stringToast = 'Toast', // options DATA API dataAutohide = 'data-autohide', databackdrop = 'data-backdrop', dataKeyboard = 'data-keyboard', dataTarget = 'data-target', dataInterval = 'data-interval', dataHeight = 'data-height', dataPause = 'data-pause', dataTitle = 'data-title', dataOriginalTitle = 'data-original-title', dataDismissible = 'data-dismissible', dataTrigger = 'data-trigger', dataAnimation = 'data-animation', dataContainer = 'data-container', dataPlacement = 'data-placement', dataDelay = 'data-delay', // option keys backdrop = 'backdrop', keyboard = 'keyboard', delay = 'delay', content = 'content', target = 'target', currentTarget = 'currentTarget', interval = 'interval', pause = 'pause', animation = 'animation', placement = 'placement', container = 'container', // box model offsetTop = 'offsetTop', offsetBottom = 'offsetBottom', offsetLeft = 'offsetLeft', scrollTop = 'scrollTop', scrollLeft = 'scrollLeft', clientWidth = 'clientWidth', clientHeight = 'clientHeight', offsetWidth = 'offsetWidth', offsetHeight = 'offsetHeight', innerWidth = 'innerWidth', innerHeight = 'innerHeight', scrollHeight = 'scrollHeight', scrollWidth = 'scrollWidth', height = 'height', // aria ariaExpanded = 'aria-expanded', ariaHidden = 'aria-hidden', ariaSelected = 'aria-selected', // event names clickEvent = 'click', focusEvent = 'focus', hoverEvent = 'hover', keydownEvent = 'keydown', keyupEvent = 'keyup', resizeEvent = 'resize', // passive scrollEvent = 'scroll', // passive mouseHover = ('onmouseleave' in DOC) ? [ 'mouseenter', 'mouseleave'] : [ 'mouseover', 'mouseout' ], // touch since 2.0.26 touchEvents = { start: 'touchstart', end: 'touchend', move:'touchmove' }, // passive // originalEvents showEvent = 'show', shownEvent = 'shown', hideEvent = 'hide', hiddenEvent = 'hidden', closeEvent = 'close', closedEvent = 'closed', slidEvent = 'slid', slideEvent = 'slide', changeEvent = 'change', // other getAttribute = 'getAttribute', setAttribute = 'setAttribute', hasAttribute = 'hasAttribute', createElement = 'createElement', appendChild = 'appendChild', innerHTML = 'innerHTML', getElementsByTagName = 'getElementsByTagName', preventDefault = 'preventDefault', getBoundingClientRect = 'getBoundingClientRect', querySelectorAll = 'querySelectorAll', getElementsByCLASSNAME = 'getElementsByClassName', getComputedStyle = 'getComputedStyle', indexOf = 'indexOf', parentNode = 'parentNode', length = 'length', toLowerCase = 'toLowerCase', Transition = 'Transition', Duration = 'Duration', Webkit = 'Webkit', style = 'style', push = 'push', tabindex = 'tabindex', contains = 'contains', active = 'active', showClass = 'show', collapsing = 'collapsing', disabled = 'disabled', loading = 'loading', left = 'left', right = 'right', top = 'top', bottom = 'bottom', // tooltip / popover tipPositions = /\b(top|bottom|left|right)+/, // modal modalOverlay = 0, fixedTop = 'fixed-top', fixedBottom = 'fixed-bottom', // transitionEnd since 2.0.4 supportTransitions = Webkit+Transition in HTML[style] || Transition[toLowerCase]() in HTML[style], transitionEndEvent = Webkit+Transition in HTML[style] ? Webkit[toLowerCase]()+Transition+'End' : Transition[toLowerCase]()+'end', transitionDuration = Webkit+Duration in HTML[style] ? Webkit[toLowerCase]()+Transition+Duration : Transition[toLowerCase]()+Duration, // set new focus element since 2.0.3 setFocus = function(element){ element.focus ? element.focus() : element.setActive(); }, // class manipulation, since 2.0.0 requires polyfill.js addClass = function(element,classNAME) { element.classList.add(classNAME); }, removeClass = function(element,classNAME) { element.classList.remove(classNAME); }, hasClass = function(element,classNAME){ // since 2.0.0 return element.classList[contains](classNAME); }, // selection methods getElementsByClassName = function(element,classNAME) { // returns Array return [].slice.call(element[getElementsByCLASSNAME]( classNAME )); }, queryElement = function (selector, parent) { var lookUp = parent ? parent : DOC; return typeof selector === 'object' ? selector : lookUp.querySelector(selector); }, getClosest = function (element, selector) { //element is the element and selector is for the closest parent element to find // source http://gomakethings.com/climbing-up-and-down-the-dom-tree-with-vanilla-javascript/ var firstChar = selector.charAt(0), selectorSubstring = selector.substr(1); if ( firstChar === '.' ) {// If selector is a class for ( ; element && element !== DOC; element = element[parentNode] ) { // Get closest match if ( queryElement(selector,element[parentNode]) !== null && hasClass(element,selectorSubstring) ) { return element; } } } else if ( firstChar === '#' ) { // If selector is an ID for ( ; element && element !== DOC; element = element[parentNode] ) { // Get closest match if ( element.id === selectorSubstring ) { return element; } } } return false; }, // event attach jQuery style / trigger since 1.2.0 on = function (element, event, handler, options) { options = options || false; element.addEventListener(event, handler, options); }, off = function(element, event, handler, options) { options = options || false; element.removeEventListener(event, handler, options); }, one = function (element, event, handler, options) { // one since 2.0.4 on(element, event, function handlerWrapper(e){ handler(e); off(element, event, handlerWrapper, options); }, options); }, // determine support for passive events supportPassive = (function(){ // Test via a getter in the options object to see if the passive property is accessed var result = false; try { var opts = Object.defineProperty({}, 'passive', { get: function() { result = true; } }); one(globalObject, 'testPassive', null, opts); } catch (e) {} return result; }()), // event options // https://github.com/WICG/EventListenerOptions/blob/gh-pages/explainer.md#feature-detection passiveHandler = supportPassive ? { passive: true } : false, // transitions getTransitionDurationFromElement = function(element) { var duration = supportTransitions ? globalObject[getComputedStyle](element)[transitionDuration] : 0; duration = parseFloat(duration); duration = typeof duration === 'number' && !isNaN(duration) ? duration * 1000 : 0; return duration; // we take a short offset to make sure we fire on the next frame after animation }, emulateTransitionEnd = function(element,handler){ // emulateTransitionEnd since 2.0.4 var called = 0, duration = getTransitionDurationFromElement(element); duration ? one(element, transitionEndEvent, function(e){ !called && handler(e), called = 1; }) : setTimeout(function() { !called && handler(), called = 1; }, 17); }, bootstrapCustomEvent = function (eventName, componentName, related) { var OriginalCustomEvent = new CustomEvent( eventName + '.bs.' + componentName); OriginalCustomEvent.relatedTarget = related; this.dispatchEvent(OriginalCustomEvent); }, // tooltip / popover stuff getScroll = function() { // also Affix and ScrollSpy uses it return { y : globalObject.pageYOffset || HTML[scrollTop], x : globalObject.pageXOffset || HTML[scrollLeft] } }, styleTip = function(link,element,position,parent) { // both popovers and tooltips (target,tooltip,placement,elementToAppendTo) var elementDimensions = { w : element[offsetWidth], h: element[offsetHeight] }, windowWidth = (HTML[clientWidth] || DOC[body][clientWidth]), windowHeight = (HTML[clientHeight] || DOC[body][clientHeight]), rect = link[getBoundingClientRect](), scroll = parent === DOC[body] ? getScroll() : { x: parent[offsetLeft] + parent[scrollLeft], y: parent[offsetTop] + parent[scrollTop] }, linkDimensions = { w: rect[right] - rect[left], h: rect[bottom] - rect[top] }, isPopover = hasClass(element,'popover'), topPosition, leftPosition, arrow = queryElement('.arrow',element), arrowTop, arrowLeft, arrowWidth, arrowHeight, halfTopExceed = rect[top] + linkDimensions.h/2 - elementDimensions.h/2 < 0, halfLeftExceed = rect[left] + linkDimensions.w/2 - elementDimensions.w/2 < 0, halfRightExceed = rect[left] + elementDimensions.w/2 + linkDimensions.w/2 >= windowWidth, halfBottomExceed = rect[top] + elementDimensions.h/2 + linkDimensions.h/2 >= windowHeight, topExceed = rect[top] - elementDimensions.h < 0, leftExceed = rect[left] - elementDimensions.w < 0, bottomExceed = rect[top] + elementDimensions.h + linkDimensions.h >= windowHeight, rightExceed = rect[left] + elementDimensions.w + linkDimensions.w >= windowWidth; // recompute position position = (position === left || position === right) && leftExceed && rightExceed ? top : position; // first, when both left and right limits are exceeded, we fall back to top|bottom position = position === top && topExceed ? bottom : position; position = position === bottom && bottomExceed ? top : position; position = position === left && leftExceed ? right : position; position = position === right && rightExceed ? left : position; // update tooltip/popover class element.className[indexOf](position) === -1 && (element.className = element.className.replace(tipPositions,position)); // we check the computed width & height and update here arrowWidth = arrow[offsetWidth]; arrowHeight = arrow[offsetHeight]; // apply styling to tooltip or popover if ( position === left || position === right ) { // secondary|side positions if ( position === left ) { // LEFT leftPosition = rect[left] + scroll.x - elementDimensions.w - ( isPopover ? arrowWidth : 0 ); } else { // RIGHT leftPosition = rect[left] + scroll.x + linkDimensions.w; } // adjust top and arrow if (halfTopExceed) { topPosition = rect[top] + scroll.y; arrowTop = linkDimensions.h/2 - arrowWidth; } else if (halfBottomExceed) { topPosition = rect[top] + scroll.y - elementDimensions.h + linkDimensions.h; arrowTop = elementDimensions.h - linkDimensions.h/2 - arrowWidth; } else { topPosition = rect[top] + scroll.y - elementDimensions.h/2 + linkDimensions.h/2; arrowTop = elementDimensions.h/2 - (isPopover ? arrowHeight*0.9 : arrowHeight/2); } } else if ( position === top || position === bottom ) { // primary|vertical positions if ( position === top) { // TOP topPosition = rect[top] + scroll.y - elementDimensions.h - ( isPopover ? arrowHeight : 0 ); } else { // BOTTOM topPosition = rect[top] + scroll.y + linkDimensions.h; } // adjust left | right and also the arrow if (halfLeftExceed) { leftPosition = 0; arrowLeft = rect[left] + linkDimensions.w/2 - arrowWidth; } else if (halfRightExceed) { leftPosition = windowWidth - elementDimensions.w*1.01; arrowLeft = elementDimensions.w - ( windowWidth - rect[left] ) + linkDimensions.w/2 - arrowWidth/2; } else { leftPosition = rect[left] + scroll.x - elementDimensions.w/2 + linkDimensions.w/2; arrowLeft = elementDimensions.w/2 - ( isPopover ? arrowWidth : arrowWidth/2 ); } } // apply style to tooltip/popover and its arrow element[style][top] = topPosition + 'px'; element[style][left] = leftPosition + 'px'; arrowTop && (arrow[style][top] = arrowTop + 'px'); arrowLeft && (arrow[style][left] = arrowLeft + 'px'); }; BSN.version = '2.0.27'; /* Native Javascript for Bootstrap 4 | Alert -------------------------------------------*/ // ALERT DEFINITION // ================ var Alert = function( element ) { // initialization element element = queryElement(element); // bind, target alert, duration and stuff var self = this, component = 'alert', alert = getClosest(element,'.'+component), triggerHandler = function(){ hasClass(alert,'fade') ? emulateTransitionEnd(alert,transitionEndHandler) : transitionEndHandler(); }, // handlers clickHandler = function(e){ alert = getClosest(e[target],'.'+component); element = queryElement('['+dataDismiss+'="'+component+'"]',alert); element && alert && (element === e[target] || element[contains](e[target])) && self.close(); }, transitionEndHandler = function(){ bootstrapCustomEvent.call(alert, closedEvent, component); off(element, clickEvent, clickHandler); // detach it's listener alert[parentNode].removeChild(alert); }; // public method this.close = function() { if ( alert && element && hasClass(alert,showClass) ) { bootstrapCustomEvent.call(alert, closeEvent, component); removeClass(alert,showClass); alert && triggerHandler(); } }; // init if ( !(stringAlert in element ) ) { // prevent adding event handlers twice on(element, clickEvent, clickHandler); } element[stringAlert] = self; }; // ALERT DATA API // ============== supports[push]([stringAlert, Alert, '['+dataDismiss+'="alert"]']); /* Native Javascript for Bootstrap 4 | Button ---------------------------------------------*/ // BUTTON DEFINITION // =================== var Button = function( element ) { // initialization element element = queryElement(element); // constant var toggled = false, // toggled makes sure to prevent triggering twice the change.bs.button events // strings component = 'button', checked = 'checked', LABEL = 'LABEL', INPUT = 'INPUT', // private methods keyHandler = function(e){ var key = e.which || e.keyCode; key === 32 && e[target] === DOC.activeElement && toggle(e); }, preventScroll = function(e){ var key = e.which || e.keyCode; key === 32 && e[preventDefault](); }, toggle = function(e) { var label = e[target].tagName === LABEL ? e[target] : e[target][parentNode].tagName === LABEL ? e[target][parentNode] : null; // the .btn label if ( !label ) return; //react if a label or its immediate child is clicked var labels = getElementsByClassName(label[parentNode],'btn'), // all the button group buttons input = label[getElementsByTagName](INPUT)[0]; if ( !input ) return; // return if no input found // manage the dom manipulation if ( input.type === 'checkbox' ) { //checkboxes if ( !input[checked] ) { addClass(label,active); input[getAttribute](checked); input[setAttribute](checked,checked); input[checked] = true; } else { removeClass(label,active); input[getAttribute](checked); input.removeAttribute(checked); input[checked] = false; } if (!toggled) { // prevent triggering the event twice toggled = true; bootstrapCustomEvent.call(input, changeEvent, component); //trigger the change for the input bootstrapCustomEvent.call(element, changeEvent, component); //trigger the change for the btn-group } } if ( input.type === 'radio' && !toggled ) { // radio buttons // don't trigger if already active (the OR condition is a hack to check if the buttons were selected with key press and NOT mouse click) if ( !input[checked] || (e.screenX === 0 && e.screenY == 0) ) { addClass(label,active); addClass(label,focusEvent); input[setAttribute](checked,checked); input[checked] = true; bootstrapCustomEvent.call(input, changeEvent, component); //trigger the change for the input bootstrapCustomEvent.call(element, changeEvent, component); //trigger the change for the btn-group toggled = true; for (var i = 0, ll = labels[length]; i1?idx-1:0) : key === 40 ? (idx 1 ) { activeTab = activeTabs[activeTabs[length]-1]; } return activeTab; }, getActiveContent = function() { return queryElement(getActiveTab()[getAttribute]('href')); }, // handler clickHandler = function(e) { e[preventDefault](); next = e[currentTarget]; !tabs[isAnimating] && !hasClass(next,active) && self.show(); }; // public method this.show = function() { // the tab we clicked is now the next tab next = next || element; nextContent = queryElement(next[getAttribute]('href')); //this is the actual object, the next tab content to activate activeTab = getActiveTab(); activeContent = getActiveContent(); tabs[isAnimating] = true; removeClass(activeTab,active); activeTab[setAttribute](ariaSelected,'false'); addClass(next,active); next[setAttribute](ariaSelected,'true'); if ( dropdown ) { if ( !hasClass(element[parentNode],'dropdown-menu') ) { if (hasClass(dropdown,active)) removeClass(dropdown,active); } else { if (!hasClass(dropdown,active)) addClass(dropdown,active); } } bootstrapCustomEvent.call(activeTab, hideEvent, component, next); if (hasClass(activeContent, 'fade')) { removeClass(activeContent,showClass); emulateTransitionEnd(activeContent, triggerHide); } else { triggerHide(); } }; // init if ( !(stringTab in element) ) { // prevent adding event handlers twice on(element, clickEvent, clickHandler); } if (self[height]) { tabsContentContainer = getActiveContent()[parentNode]; } element[stringTab] = self; }; // TAB DATA API // ============ supports[push]( [ stringTab, Tab, '['+dataToggle+'="tab"]' ] ); /* Native Javascript for Bootstrap | Initialize Data API --------------------------------------------------------*/ var initializeDataAPI = function( constructor, collection ){ for (var i=0, l=collection[length]; i { !function(t,e){ true?module.exports=e():0}(window,function(){return function(t){var e={};function r(n){if(e[n])return e[n].exports;var i=e[n]={i:n,l:!1,exports:{}};return t[n].call(i.exports,i,i.exports,r),i.l=!0,i.exports}return r.m=t,r.c=e,r.d=function(t,e,n){r.o(t,e)||Object.defineProperty(t,e,{configurable:!1,enumerable:!0,get:n})},r.r=function(t){Object.defineProperty(t,"__esModule",{value:!0})},r.n=function(t){var e=t&&t.__esModule?function(){return t.default}:function(){return t};return r.d(e,"a",e),e},r.o=function(t,e){return Object.prototype.hasOwnProperty.call(t,e)},r.p="",r(r.s=7)}([function(t,e,r){"use strict";Object.defineProperty(e,"__esModule",{value:!0});e.loadImageElement=function(t,e){return new Promise(function(r,n){t.addEventListener("load",function(){r(t)},!1),t.addEventListener("error",function(t){n(t)},!1),t.src=e})},e.resize=function(t,e,r,n){if(!r&&!n)return{currentWidth:t,currentHeight:e};var i=t/e,o=void 0,a=void 0;return i>r/n?a=(o=Math.min(t,r))/i:o=(a=Math.min(e,n))*i,{width:o,height:a}}},function(t,e,r){"use strict";Object.defineProperty(e,"__esModule",{value:!0});e.base64ToFile=function(t){for(var e=arguments.length>1&&void 0!==arguments[1]?arguments[1]:"image/jpeg",r=window.atob(t),n=[],i=0;i8)return o.drawImage(t,0,0,i.width,i.height),i;switch(n>4&&(i.width=r,i.height=e),n){case 2:o.translate(e,0),o.scale(-1,1);break;case 3:o.translate(e,r),o.rotate(Math.PI);break;case 4:o.translate(0,r),o.scale(1,-1);break;case 5:o.rotate(.5*Math.PI),o.scale(1,-1);break;case 6:o.rotate(.5*Math.PI),o.translate(0,-r);break;case 7:o.rotate(.5*Math.PI),o.translate(e,-r),o.scale(-1,1);break;case 8:o.rotate(-.5*Math.PI),o.translate(-e,0)}return n>4?o.drawImage(t,0,0,i.height,i.width):o.drawImage(t,0,0,i.width,i.height),i},e.canvasToBlob=function(t,e){return new Promise(function(r,n){t.toBlob(function(t){r(t)},"image/jpeg",e)})},e.size=function(t){return{kB:.001*t,MB:1e-6*t}},e.blobToBase64=function(t){return new Promise(function(e,r){var n=new window.FileReader;n.addEventListener("load",function(t){e(t.target.result)},!1),n.addEventListener("error",function(t){r(t)},!1),n.readAsDataURL(t)})}},function(t,e,r){t.exports=r(6)},function(t,e,r){"use strict";Object.defineProperty(e,"__esModule",{value:!0});e.extractOrientation=function(t){return new Promise(function(e,r){var n=new window.FileReader;n.onload=function(t){var r=new DataView(t.target.result);65496!==r.getUint16(0,!1)&&e(-2);for(var n=r.byteLength,i=2;i=0;--o){var a=this.tryEntries[o],u=a.completion;if("root"===a.tryLoc)return n("end");if(a.tryLoc<=this.prev){var s=i.call(a,"catchLoc"),c=i.call(a,"finallyLoc");if(s&&c){if(this.prev=0;--r){var n=this.tryEntries[r];if(n.tryLoc<=this.prev&&i.call(n,"finallyLoc")&&this.prev=0;--e){var r=this.tryEntries[e];if(r.finallyLoc===t)return this.complete(r.completion,r.afterLoc),P(r),A}},catch:function(t){for(var e=this.tryEntries.length-1;e>=0;--e){var r=this.tryEntries[e];if(r.tryLoc===t){var n=r.completion;if("throw"===n.type){var i=n.arg;P(r)}return i}}throw new Error("illegal catch attempt")},delegateYield:function(t,e,n){return this.delegate={iterator:z(t),resultName:e,nextLoc:n},"next"===this.method&&(this.arg=r),A}}}function w(t,e,r,n){var i=e&&e.prototype instanceof x?e:x,o=Object.create(i.prototype),a=new O(n||[]);return o._invoke=function(t,e,r){var n=l;return function(i,o){if(n===p)throw new Error("Generator is already running");if(n===d){if("throw"===i)throw o;return j()}for(r.method=i,r.arg=o;;){var a=r.delegate;if(a){var u=k(a,r);if(u){if(u===A)continue;return u}}if("next"===r.method)r.sent=r._sent=r.arg;else if("throw"===r.method){if(n===l)throw n=d,r.arg;r.dispatchException(r.arg)}else"return"===r.method&&r.abrupt("return",r.arg);n=p;var s=E(t,e,r);if("normal"===s.type){if(n=r.done?d:h,s.arg===A)continue;return{value:s.arg,done:r.done}}"throw"===s.type&&(n=d,r.method="throw",r.arg=s.arg)}}}(t,r,a),o}function E(t,e,r){try{return{type:"normal",arg:t.call(e,r)}}catch(t){return{type:"throw",arg:t}}}function x(){}function b(){}function B(){}function Q(t){["next","throw","return"].forEach(function(e){t[e]=function(t){return this._invoke(e,t)}})}function _(t){var e;this._invoke=function(r,n){function o(){return new Promise(function(e,o){!function e(r,n,o,a){var u=E(t[r],t,n);if("throw"!==u.type){var s=u.arg,c=s.value;return c&&"object"==typeof c&&i.call(c,"__await")?Promise.resolve(c.__await).then(function(t){e("next",t,o,a)},function(t){e("throw",t,o,a)}):Promise.resolve(c).then(function(t){s.value=t,o(s)},a)}a(u.arg)}(r,n,e,o)})}return e=e?e.then(o,o):o()}}function k(t,e){var n=t.iterator[e.method];if(n===r){if(e.delegate=null,"throw"===e.method){if(t.iterator.return&&(e.method="return",e.arg=r,k(t,e),"throw"===e.method))return A;e.method="throw",e.arg=new TypeError("The iterator does not provide a 'throw' method")}return A}var i=E(n,t.iterator,e.arg);if("throw"===i.type)return e.method="throw",e.arg=i.arg,e.delegate=null,A;var o=i.arg;return o?o.done?(e[t.resultName]=o.value,e.next=t.nextLoc,"return"!==e.method&&(e.method="next",e.arg=r),e.delegate=null,A):o:(e.method="throw",e.arg=new TypeError("iterator result is not an object"),e.delegate=null,A)}function L(t){var e={tryLoc:t[0]};1 in t&&(e.catchLoc=t[1]),2 in t&&(e.finallyLoc=t[2],e.afterLoc=t[3]),this.tryEntries.push(e)}function P(t){var e=t.completion||{};e.type="normal",delete e.arg,t.completion=e}function O(t){this.tryEntries=[{tryLoc:"root"}],t.forEach(L,this),this.reset(!0)}function z(t){if(t){var e=t[a];if(e)return e.call(t);if("function"==typeof t.next)return t;if(!isNaN(t.length)){var n=-1,o=function e(){for(;++n=0,o=i&&n.regeneratorRuntime;if(n.regeneratorRuntime=void 0,t.exports=r(5),i)n.regeneratorRuntime=o;else try{delete n.regeneratorRuntime}catch(t){n.regeneratorRuntime=void 0}},function(t,e,r){"use strict";Object.defineProperty(e,"__esModule",{value:!0});var n=s(r(2)),i=function(){function t(t,e){for(var r=0;r0&&void 0!==arguments[0]?arguments[0]:{};!function(t,e){if(!(t instanceof e))throw new TypeError("Cannot call a class as a function")}(this,t),this.setOptions(e)}return i(t,[{key:"setOptions",value:function(t){var e={targetSize:1/0,quality:.75,minQuality:.5,qualityStepSize:.1,maxWidth:1920,maxHeight:1920,resize:!0,throwIfSizeNotReached:!1,autoRotate:!0},r=new Proxy(t,{get:function(t,r){return r in t?t[r]:e[r]}});this.options=r}},{key:"_compressFile",value:function(){var t=c(n.default.mark(function t(e){var r,i;return n.default.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:return r=new u.default(e),(i={}).start=window.performance.now(),i.quality=this.options.quality,i.startType=r.type,t.next=7,r.load();case 7:return t.next=9,this._compressImage(r,i);case 9:return t.abrupt("return",t.sent);case 10:case"end":return t.stop()}},t,this)}));return function(e){return t.apply(this,arguments)}}()},{key:"_compressImage",value:function(){var t=c(n.default.mark(function t(e,r){var i,u,s,c,f;return n.default.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:return r.startWidth=e.width,r.startHeight=e.height,i=void 0,u=void 0,this.options.resize?(s=(0,a.resize)(e.width,e.height,this.options.maxWidth,this.options.maxHeight),i=s.width,u=s.height):(i=e.width,u=e.height),r.endWidth=i,r.endHeight=u,c=this.doAutoRotation?void 0:1,f=e.getCanvas(i,u,c),r.iterations=0,r.startSizeMB=o.size(e.size).MB,t.next=12,this._loopCompression(f,e,r);case 12:return r.endSizeMB=o.size(e.size).MB,r.sizeReducedInPercent=(r.startSizeMB-r.endSizeMB)/r.startSizeMB*100,r.end=window.performance.now(),r.elapsedTimeInSeconds=(r.end-r.start)/1e3,r.endType=e.type,t.abrupt("return",{photo:e,info:r});case 18:case"end":return t.stop()}},t,this)}));return function(e,r){return t.apply(this,arguments)}}()},{key:"_loopCompression",value:function(){var t=c(n.default.mark(function t(e,r,i){var a;return n.default.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:return i.iterations++,t.t0=r,t.next=4,o.canvasToBlob(e,i.quality);case 4:if(t.t1=t.sent,t.t0.setData.call(t.t0,t.t1),1==i.iterations&&(r.width=i.endWidth,r.height=i.endHeight),!(o.size(r.size).MB>this.options.targetSize)){t.next=24;break}if(!(i.quality.toFixed(10)-.1 { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 6931: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 352: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 7802: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 5599: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 1875: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 2791: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 5956: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 9679: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 4915: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 1638: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 3076: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 6916: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 7140: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 3288: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 902: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 6233: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 1513: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 1984: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 1833: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 4921: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 5222: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 8054: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 8125: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 4166: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 6278: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 9523: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 6176: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 5046: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 7631: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 4903: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 6450: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 307: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 9537: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 9959: /***/ ((module, __webpack_exports__, __webpack_require__) => { "use strict"; /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__) /* harmony export */ }); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(7537); /* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(3645); /* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__); // Imports var ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default())); // Module ___CSS_LOADER_EXPORT___.push([module.id, "", "",{"version":3,"sources":[],"names":[],"mappings":"","sourceRoot":""}]); // Exports /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___); /***/ }), /***/ 3645: /***/ ((module) => { "use strict"; /* MIT License http://www.opensource.org/licenses/mit-license.php Author Tobias Koppers @sokra */ module.exports = function (cssWithMappingToString) { var list = []; // return the list of modules as css string list.toString = function toString() { return this.map(function (item) { var content = ""; var needLayer = typeof item[5] !== "undefined"; if (item[4]) { content += "@supports (".concat(item[4], ") {"); } if (item[2]) { content += "@media ".concat(item[2], " {"); } if (needLayer) { content += "@layer".concat(item[5].length > 0 ? " ".concat(item[5]) : "", " {"); } content += cssWithMappingToString(item); if (needLayer) { content += "}"; } if (item[2]) { content += "}"; } if (item[4]) { content += "}"; } return content; }).join(""); }; // import a list of modules into the list list.i = function i(modules, media, dedupe, supports, layer) { if (typeof modules === "string") { modules = [[null, modules, undefined]]; } var alreadyImportedModules = {}; if (dedupe) { for (var k = 0; k < this.length; k++) { var id = this[k][0]; if (id != null) { alreadyImportedModules[id] = true; } } } for (var _k = 0; _k < modules.length; _k++) { var item = [].concat(modules[_k]); if (dedupe && alreadyImportedModules[item[0]]) { continue; } if (typeof layer !== "undefined") { if (typeof item[5] === "undefined") { item[5] = layer; } else { item[1] = "@layer".concat(item[5].length > 0 ? " ".concat(item[5]) : "", " {").concat(item[1], "}"); item[5] = layer; } } if (media) { if (!item[2]) { item[2] = media; } else { item[1] = "@media ".concat(item[2], " {").concat(item[1], "}"); item[2] = media; } } if (supports) { if (!item[4]) { item[4] = "".concat(supports); } else { item[1] = "@supports (".concat(item[4], ") {").concat(item[1], "}"); item[4] = supports; } } list.push(item); } }; return list; }; /***/ }), /***/ 7537: /***/ ((module) => { "use strict"; module.exports = function (item) { var content = item[1]; var cssMapping = item[3]; if (!cssMapping) { return content; } if (typeof btoa === "function") { var base64 = btoa(unescape(encodeURIComponent(JSON.stringify(cssMapping)))); var data = "sourceMappingURL=data:application/json;charset=utf-8;base64,".concat(base64); var sourceMapping = "/*# ".concat(data, " */"); var sourceURLs = cssMapping.sources.map(function (source) { return "/*# sourceURL=".concat(cssMapping.sourceRoot || "").concat(source, " */"); }); return [content].concat(sourceURLs).concat([sourceMapping]).join("\n"); } return [content].join("\n"); }; /***/ }), /***/ 7484: /***/ (function(module) { !function(t,e){ true?module.exports=e():0}(this,(function(){"use strict";var t=1e3,e=6e4,n=36e5,r="millisecond",i="second",s="minute",u="hour",a="day",o="week",f="month",h="quarter",c="year",d="date",$="Invalid Date",l=/^(\d{4})[-/]?(\d{1,2})?[-/]?(\d{0,2})[Tt\s]*(\d{1,2})?:?(\d{1,2})?:?(\d{1,2})?[.:]?(\d+)?$/,y=/\[([^\]]+)]|Y{1,4}|M{1,4}|D{1,2}|d{1,4}|H{1,2}|h{1,2}|a|A|m{1,2}|s{1,2}|Z{1,2}|SSS/g,M={name:"en",weekdays:"Sunday_Monday_Tuesday_Wednesday_Thursday_Friday_Saturday".split("_"),months:"January_February_March_April_May_June_July_August_September_October_November_December".split("_")},m=function(t,e,n){var r=String(t);return!r||r.length>=e?t:""+Array(e+1-r.length).join(n)+t},g={s:m,z:function(t){var e=-t.utcOffset(),n=Math.abs(e),r=Math.floor(n/60),i=n%60;return(e<=0?"+":"-")+m(r,2,"0")+":"+m(i,2,"0")},m:function t(e,n){if(e.date()1)return t(u[0])}else{var a=e.name;D[a]=e,i=a}return!r&&i&&(v=i),i||!r&&v},w=function(t,e){if(p(t))return t.clone();var n="object"==typeof e?e:{};return n.date=t,n.args=arguments,new _(n)},O=g;O.l=S,O.i=p,O.w=function(t,e){return w(t,{locale:e.$L,utc:e.$u,x:e.$x,$offset:e.$offset})};var _=function(){function M(t){this.$L=S(t.locale,null,!0),this.parse(t)}var m=M.prototype;return m.parse=function(t){this.$d=function(t){var e=t.date,n=t.utc;if(null===e)return new Date(NaN);if(O.u(e))return new Date;if(e instanceof Date)return new Date(e);if("string"==typeof e&&!/Z$/i.test(e)){var r=e.match(l);if(r){var i=r[2]-1||0,s=(r[7]||"0").substring(0,3);return n?new Date(Date.UTC(r[1],i,r[3]||1,r[4]||0,r[5]||0,r[6]||0,s)):new Date(r[1],i,r[3]||1,r[4]||0,r[5]||0,r[6]||0,s)}}return new Date(e)}(t),this.$x=t.x||{},this.init()},m.init=function(){var t=this.$d;this.$y=t.getFullYear(),this.$M=t.getMonth(),this.$D=t.getDate(),this.$W=t.getDay(),this.$H=t.getHours(),this.$m=t.getMinutes(),this.$s=t.getSeconds(),this.$ms=t.getMilliseconds()},m.$utils=function(){return O},m.isValid=function(){return!(this.$d.toString()===$)},m.isSame=function(t,e){var n=w(t);return this.startOf(e)<=n&&n<=this.endOf(e)},m.isAfter=function(t,e){return w(t) { var map = { "./af.js": [ 5903, 9210 ], "./am.js": [ 9911, 5073 ], "./ar-dz.js": [ 7200, 9406 ], "./ar-iq.js": [ 7719, 2990 ], "./ar-kw.js": [ 2376, 9897 ], "./ar-ly.js": [ 8540, 3521 ], "./ar-ma.js": [ 6817, 5313 ], "./ar-sa.js": [ 1573, 485 ], "./ar-tn.js": [ 9339, 8040 ], "./ar.js": [ 3939, 6755 ], "./az.js": [ 8092, 4963 ], "./be.js": [ 504, 9478 ], "./bg.js": [ 9091, 578 ], "./bi.js": [ 9149, 2984 ], "./bm.js": [ 5287, 2263 ], "./bn-bd.js": [ 4067, 1351 ], "./bn.js": [ 5254, 280 ], "./bo.js": [ 2502, 9950 ], "./br.js": [ 8864, 760 ], "./bs.js": [ 4502, 9833 ], "./ca.js": [ 3646, 102 ], "./cs.js": [ 8507, 7400 ], "./cv.js": [ 6636, 4481 ], "./cy.js": [ 8792, 6740 ], "./da.js": [ 7427, 2548 ], "./de-at.js": [ 3237, 7175 ], "./de-ch.js": [ 6148, 1679 ], "./de.js": [ 790, 52 ], "./dv.js": [ 1794, 5569 ], "./el.js": [ 5423, 1606 ], "./en-au.js": [ 5109, 5485 ], "./en-ca.js": [ 5105, 4035 ], "./en-gb.js": [ 9517, 6031 ], "./en-ie.js": [ 758, 8129 ], "./en-il.js": [ 5805, 3463 ], "./en-in.js": [ 8529, 6898 ], "./en-nz.js": [ 302, 8547 ], "./en-sg.js": [ 5941, 1735 ], "./en-tt.js": [ 6183, 6105 ], "./en.js": [ 5054, 535 ], "./eo.js": [ 4990, 5121 ], "./es-do.js": [ 3864, 8758 ], "./es-mx.js": [ 7118, 7416 ], "./es-pr.js": [ 3521, 911 ], "./es-us.js": [ 6165, 3208 ], "./es.js": [ 7763, 3411 ], "./et.js": [ 9670, 4153 ], "./eu.js": [ 6629, 1396 ], "./fa.js": [ 6953, 5544 ], "./fi.js": [ 7822, 2130 ], "./fo.js": [ 9197, 8745 ], "./fr-ca.js": [ 7989, 7363 ], "./fr-ch.js": [ 4254, 7952 ], "./fr.js": [ 6023, 1910 ], "./fy.js": [ 3220, 6376 ], "./ga.js": [ 7467, 688 ], "./gd.js": [ 4855, 5050 ], "./gl.js": [ 229, 5818 ], "./gom-latn.js": [ 6312, 825 ], "./gu.js": [ 7632, 3623 ], "./he.js": [ 5418, 9372 ], "./hi.js": [ 7573, 8010 ], "./hr.js": [ 6257, 7419 ], "./ht.js": [ 8889, 5822 ], "./hu.js": [ 8562, 8214 ], "./hy-am.js": [ 8242, 5407 ], "./id.js": [ 3783, 9513 ], "./is.js": [ 8980, 1194 ], "./it-ch.js": [ 3706, 6010 ], "./it.js": [ 5551, 1880 ], "./ja.js": [ 6831, 1107 ], "./jv.js": [ 2641, 4305 ], "./ka.js": [ 6622, 5186 ], "./kk.js": [ 2921, 5206 ], "./km.js": [ 5567, 2475 ], "./kn.js": [ 1113, 7523 ], "./ko.js": [ 9132, 3446 ], "./ku.js": [ 4888, 7024 ], "./ky.js": [ 466, 5055 ], "./lb.js": [ 1796, 5215 ], "./lo.js": [ 8894, 1204 ], "./lt.js": [ 8768, 7899 ], "./lv.js": [ 953, 631 ], "./me.js": [ 8066, 145 ], "./mi.js": [ 8602, 7454 ], "./mk.js": [ 1560, 4951 ], "./ml.js": [ 4017, 7679 ], "./mn.js": [ 4717, 8618 ], "./mr.js": [ 5473, 5600 ], "./ms-my.js": [ 7387, 882 ], "./ms.js": [ 5742, 9095 ], "./mt.js": [ 8477, 9665 ], "./my.js": [ 2966, 5166 ], "./nb.js": [ 9682, 646 ], "./ne.js": [ 4149, 9030 ], "./nl-be.js": [ 7496, 3155 ], "./nl.js": [ 9182, 1520 ], "./nn.js": [ 2722, 7050 ], "./oc-lnc.js": [ 6159, 7203 ], "./pa-in.js": [ 5914, 5850 ], "./pl.js": [ 1987, 1211 ], "./pt-br.js": [ 7548, 5274 ], "./pt.js": [ 5001, 265 ], "./rn.js": [ 123, 4678 ], "./ro.js": [ 8146, 8022 ], "./ru.js": [ 600, 559 ], "./rw.js": [ 6509, 3221 ], "./sd.js": [ 5437, 1298 ], "./se.js": [ 772, 1942 ], "./si.js": [ 7109, 9333 ], "./sk.js": [ 5627, 6783 ], "./sl.js": [ 2544, 9625 ], "./sq.js": [ 8341, 8603 ], "./sr-cyrl.js": [ 7101, 3435 ], "./sr.js": [ 617, 7390 ], "./ss.js": [ 4127, 9238 ], "./sv-fi.js": [ 6421, 9997 ], "./sv.js": [ 1876, 9652 ], "./sw.js": [ 2030, 9733 ], "./ta.js": [ 5596, 7645 ], "./te.js": [ 5159, 7714 ], "./tet.js": [ 9157, 555 ], "./tg.js": [ 9928, 2446 ], "./th.js": [ 2019, 1729 ], "./tk.js": [ 5817, 5256 ], "./tl-ph.js": [ 6513, 9443 ], "./tlh.js": [ 7296, 2814 ], "./tr.js": [ 3035, 8665 ], "./tzl.js": [ 7797, 2843 ], "./tzm-latn.js": [ 261, 3933 ], "./tzm.js": [ 4722, 4342 ], "./ug-cn.js": [ 313, 6890 ], "./uk.js": [ 4144, 1619 ], "./ur.js": [ 2957, 9568 ], "./uz-latn.js": [ 8727, 1110 ], "./uz.js": [ 7486, 3153 ], "./vi.js": [ 7553, 8073 ], "./x-pseudo.js": [ 5321, 4423 ], "./yo.js": [ 4724, 8692 ], "./zh-cn.js": [ 3852, 9630 ], "./zh-hk.js": [ 2390, 3755 ], "./zh-tw.js": [ 3901, 6776 ], "./zh.js": [ 2009, 8458 ] }; function webpackAsyncContext(req) { if(!__webpack_require__.o(map, req)) { return Promise.resolve().then(() => { var e = new Error("Cannot find module '" + req + "'"); e.code = 'MODULE_NOT_FOUND'; throw e; }); } var ids = map[req], id = ids[0]; return __webpack_require__.e(ids[1]).then(() => { return __webpack_require__.t(id, 7 | 16); }); } webpackAsyncContext.keys = () => (Object.keys(map)); webpackAsyncContext.id = 9434; module.exports = webpackAsyncContext; /***/ }), /***/ 8734: /***/ (function(module) { !function(e,t){ true?module.exports=t():0}(this,(function(){"use strict";return function(e,t,r){var n=t.prototype,s=n.format;r.en.ordinal=function(e){var t=["th","st","nd","rd"],r=e%100;return"["+e+(t[(r-20)%10]||t[r]||t[0])+"]"},n.format=function(e){var t=this,r=this.$locale();if(!this.isValid())return s.bind(this)(e);var n=this.$utils(),a=(e||"YYYY-MM-DDTHH:mm:ssZ").replace(/\[([^\]]+)]|Q|wo|ww|w|WW|W|zzz|z|gggg|GGGG|Do|X|x|k{1,2}|S/g,(function(e){switch(e){case"Q":return Math.ceil((t.$M+1)/3);case"Do":return r.ordinal(t.$D);case"gggg":return t.weekYear();case"GGGG":return t.isoWeekYear();case"wo":return r.ordinal(t.week(),"W");case"w":case"ww":return n.s(t.week(),"w"===e?1:2,"0");case"W":case"WW":return n.s(t.isoWeek(),"W"===e?1:2,"0");case"k":case"kk":return n.s(String(0===t.$H?24:t.$H),"k"===e?1:2,"0");case"X":return Math.floor(t.$d.getTime()/1e3);case"x":return t.$d.getTime();case"z":return"["+t.offsetName()+"]";case"zzz":return"["+t.offsetName("long")+"]";default:return e}}));return s.bind(this)(a)}}})); /***/ }), /***/ 7856: /***/ (function(module) { /*! @license DOMPurify 2.3.6 | (c) Cure53 and other contributors | Released under the Apache license 2.0 and Mozilla Public License 2.0 | github.com/cure53/DOMPurify/blob/2.3.6/LICENSE */ (function (global, factory) { true ? module.exports = factory() : 0; }(this, function () { 'use strict'; function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } } var hasOwnProperty = Object.hasOwnProperty, setPrototypeOf = Object.setPrototypeOf, isFrozen = Object.isFrozen, getPrototypeOf = Object.getPrototypeOf, getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor; var freeze = Object.freeze, seal = Object.seal, create = Object.create; // eslint-disable-line import/no-mutable-exports var _ref = typeof Reflect !== 'undefined' && Reflect, apply = _ref.apply, construct = _ref.construct; if (!apply) { apply = function apply(fun, thisValue, args) { return fun.apply(thisValue, args); }; } if (!freeze) { freeze = function freeze(x) { return x; }; } if (!seal) { seal = function seal(x) { return x; }; } if (!construct) { construct = function construct(Func, args) { return new (Function.prototype.bind.apply(Func, [null].concat(_toConsumableArray(args))))(); }; } var arrayForEach = unapply(Array.prototype.forEach); var arrayPop = unapply(Array.prototype.pop); var arrayPush = unapply(Array.prototype.push); var stringToLowerCase = unapply(String.prototype.toLowerCase); var stringMatch = unapply(String.prototype.match); var stringReplace = unapply(String.prototype.replace); var stringIndexOf = unapply(String.prototype.indexOf); var stringTrim = unapply(String.prototype.trim); var regExpTest = unapply(RegExp.prototype.test); var typeErrorCreate = unconstruct(TypeError); function unapply(func) { return function (thisArg) { for (var _len = arguments.length, args = Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) { args[_key - 1] = arguments[_key]; } return apply(func, thisArg, args); }; } function unconstruct(func) { return function () { for (var _len2 = arguments.length, args = Array(_len2), _key2 = 0; _key2 < _len2; _key2++) { args[_key2] = arguments[_key2]; } return construct(func, args); }; } /* Add properties to a lookup table */ function addToSet(set, array) { if (setPrototypeOf) { // Make 'in' and truthy checks like Boolean(set.constructor) // independent of any properties defined on Object.prototype. // Prevent prototype setters from intercepting set as a this value. setPrototypeOf(set, null); } var l = array.length; while (l--) { var element = array[l]; if (typeof element === 'string') { var lcElement = stringToLowerCase(element); if (lcElement !== element) { // Config presets (e.g. tags.js, attrs.js) are immutable. if (!isFrozen(array)) { array[l] = lcElement; } element = lcElement; } } set[element] = true; } return set; } /* Shallow clone an object */ function clone(object) { var newObject = create(null); var property = void 0; for (property in object) { if (apply(hasOwnProperty, object, [property])) { newObject[property] = object[property]; } } return newObject; } /* IE10 doesn't support __lookupGetter__ so lets' * simulate it. It also automatically checks * if the prop is function or getter and behaves * accordingly. */ function lookupGetter(object, prop) { while (object !== null) { var desc = getOwnPropertyDescriptor(object, prop); if (desc) { if (desc.get) { return unapply(desc.get); } if (typeof desc.value === 'function') { return unapply(desc.value); } } object = getPrototypeOf(object); } function fallbackValue(element) { console.warn('fallback value for', element); return null; } return fallbackValue; } var html = freeze(['a', 'abbr', 'acronym', 'address', 'area', 'article', 'aside', 'audio', 'b', 'bdi', 'bdo', 'big', 'blink', 'blockquote', 'body', 'br', 'button', 'canvas', 'caption', 'center', 'cite', 'code', 'col', 'colgroup', 'content', 'data', 'datalist', 'dd', 'decorator', 'del', 'details', 'dfn', 'dialog', 'dir', 'div', 'dl', 'dt', 'element', 'em', 'fieldset', 'figcaption', 'figure', 'font', 'footer', 'form', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'head', 'header', 'hgroup', 'hr', 'html', 'i', 'img', 'input', 'ins', 'kbd', 'label', 'legend', 'li', 'main', 'map', 'mark', 'marquee', 'menu', 'menuitem', 'meter', 'nav', 'nobr', 'ol', 'optgroup', 'option', 'output', 'p', 'picture', 'pre', 'progress', 'q', 'rp', 'rt', 'ruby', 's', 'samp', 'section', 'select', 'shadow', 'small', 'source', 'spacer', 'span', 'strike', 'strong', 'style', 'sub', 'summary', 'sup', 'table', 'tbody', 'td', 'template', 'textarea', 'tfoot', 'th', 'thead', 'time', 'tr', 'track', 'tt', 'u', 'ul', 'var', 'video', 'wbr']); // SVG var svg = freeze(['svg', 'a', 'altglyph', 'altglyphdef', 'altglyphitem', 'animatecolor', 'animatemotion', 'animatetransform', 'circle', 'clippath', 'defs', 'desc', 'ellipse', 'filter', 'font', 'g', 'glyph', 'glyphref', 'hkern', 'image', 'line', 'lineargradient', 'marker', 'mask', 'metadata', 'mpath', 'path', 'pattern', 'polygon', 'polyline', 'radialgradient', 'rect', 'stop', 'style', 'switch', 'symbol', 'text', 'textpath', 'title', 'tref', 'tspan', 'view', 'vkern']); var svgFilters = freeze(['feBlend', 'feColorMatrix', 'feComponentTransfer', 'feComposite', 'feConvolveMatrix', 'feDiffuseLighting', 'feDisplacementMap', 'feDistantLight', 'feFlood', 'feFuncA', 'feFuncB', 'feFuncG', 'feFuncR', 'feGaussianBlur', 'feImage', 'feMerge', 'feMergeNode', 'feMorphology', 'feOffset', 'fePointLight', 'feSpecularLighting', 'feSpotLight', 'feTile', 'feTurbulence']); // List of SVG elements that are disallowed by default. // We still need to know them so that we can do namespace // checks properly in case one wants to add them to // allow-list. var svgDisallowed = freeze(['animate', 'color-profile', 'cursor', 'discard', 'fedropshadow', 'font-face', 'font-face-format', 'font-face-name', 'font-face-src', 'font-face-uri', 'foreignobject', 'hatch', 'hatchpath', 'mesh', 'meshgradient', 'meshpatch', 'meshrow', 'missing-glyph', 'script', 'set', 'solidcolor', 'unknown', 'use']); var mathMl = freeze(['math', 'menclose', 'merror', 'mfenced', 'mfrac', 'mglyph', 'mi', 'mlabeledtr', 'mmultiscripts', 'mn', 'mo', 'mover', 'mpadded', 'mphantom', 'mroot', 'mrow', 'ms', 'mspace', 'msqrt', 'mstyle', 'msub', 'msup', 'msubsup', 'mtable', 'mtd', 'mtext', 'mtr', 'munder', 'munderover']); // Similarly to SVG, we want to know all MathML elements, // even those that we disallow by default. var mathMlDisallowed = freeze(['maction', 'maligngroup', 'malignmark', 'mlongdiv', 'mscarries', 'mscarry', 'msgroup', 'mstack', 'msline', 'msrow', 'semantics', 'annotation', 'annotation-xml', 'mprescripts', 'none']); var text = freeze(['#text']); var html$1 = freeze(['accept', 'action', 'align', 'alt', 'autocapitalize', 'autocomplete', 'autopictureinpicture', 'autoplay', 'background', 'bgcolor', 'border', 'capture', 'cellpadding', 'cellspacing', 'checked', 'cite', 'class', 'clear', 'color', 'cols', 'colspan', 'controls', 'controlslist', 'coords', 'crossorigin', 'datetime', 'decoding', 'default', 'dir', 'disabled', 'disablepictureinpicture', 'disableremoteplayback', 'download', 'draggable', 'enctype', 'enterkeyhint', 'face', 'for', 'headers', 'height', 'hidden', 'high', 'href', 'hreflang', 'id', 'inputmode', 'integrity', 'ismap', 'kind', 'label', 'lang', 'list', 'loading', 'loop', 'low', 'max', 'maxlength', 'media', 'method', 'min', 'minlength', 'multiple', 'muted', 'name', 'nonce', 'noshade', 'novalidate', 'nowrap', 'open', 'optimum', 'pattern', 'placeholder', 'playsinline', 'poster', 'preload', 'pubdate', 'radiogroup', 'readonly', 'rel', 'required', 'rev', 'reversed', 'role', 'rows', 'rowspan', 'spellcheck', 'scope', 'selected', 'shape', 'size', 'sizes', 'span', 'srclang', 'start', 'src', 'srcset', 'step', 'style', 'summary', 'tabindex', 'title', 'translate', 'type', 'usemap', 'valign', 'value', 'width', 'xmlns', 'slot']); var svg$1 = freeze(['accent-height', 'accumulate', 'additive', 'alignment-baseline', 'ascent', 'attributename', 'attributetype', 'azimuth', 'basefrequency', 'baseline-shift', 'begin', 'bias', 'by', 'class', 'clip', 'clippathunits', 'clip-path', 'clip-rule', 'color', 'color-interpolation', 'color-interpolation-filters', 'color-profile', 'color-rendering', 'cx', 'cy', 'd', 'dx', 'dy', 'diffuseconstant', 'direction', 'display', 'divisor', 'dur', 'edgemode', 'elevation', 'end', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'filterunits', 'flood-color', 'flood-opacity', 'font-family', 'font-size', 'font-size-adjust', 'font-stretch', 'font-style', 'font-variant', 'font-weight', 'fx', 'fy', 'g1', 'g2', 'glyph-name', 'glyphref', 'gradientunits', 'gradienttransform', 'height', 'href', 'id', 'image-rendering', 'in', 'in2', 'k', 'k1', 'k2', 'k3', 'k4', 'kerning', 'keypoints', 'keysplines', 'keytimes', 'lang', 'lengthadjust', 'letter-spacing', 'kernelmatrix', 'kernelunitlength', 'lighting-color', 'local', 'marker-end', 'marker-mid', 'marker-start', 'markerheight', 'markerunits', 'markerwidth', 'maskcontentunits', 'maskunits', 'max', 'mask', 'media', 'method', 'mode', 'min', 'name', 'numoctaves', 'offset', 'operator', 'opacity', 'order', 'orient', 'orientation', 'origin', 'overflow', 'paint-order', 'path', 'pathlength', 'patterncontentunits', 'patterntransform', 'patternunits', 'points', 'preservealpha', 'preserveaspectratio', 'primitiveunits', 'r', 'rx', 'ry', 'radius', 'refx', 'refy', 'repeatcount', 'repeatdur', 'restart', 'result', 'rotate', 'scale', 'seed', 'shape-rendering', 'specularconstant', 'specularexponent', 'spreadmethod', 'startoffset', 'stddeviation', 'stitchtiles', 'stop-color', 'stop-opacity', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke', 'stroke-width', 'style', 'surfacescale', 'systemlanguage', 'tabindex', 'targetx', 'targety', 'transform', 'transform-origin', 'text-anchor', 'text-decoration', 'text-rendering', 'textlength', 'type', 'u1', 'u2', 'unicode', 'values', 'viewbox', 'visibility', 'version', 'vert-adv-y', 'vert-origin-x', 'vert-origin-y', 'width', 'word-spacing', 'wrap', 'writing-mode', 'xchannelselector', 'ychannelselector', 'x', 'x1', 'x2', 'xmlns', 'y', 'y1', 'y2', 'z', 'zoomandpan']); var mathMl$1 = freeze(['accent', 'accentunder', 'align', 'bevelled', 'close', 'columnsalign', 'columnlines', 'columnspan', 'denomalign', 'depth', 'dir', 'display', 'displaystyle', 'encoding', 'fence', 'frame', 'height', 'href', 'id', 'largeop', 'length', 'linethickness', 'lspace', 'lquote', 'mathbackground', 'mathcolor', 'mathsize', 'mathvariant', 'maxsize', 'minsize', 'movablelimits', 'notation', 'numalign', 'open', 'rowalign', 'rowlines', 'rowspacing', 'rowspan', 'rspace', 'rquote', 'scriptlevel', 'scriptminsize', 'scriptsizemultiplier', 'selection', 'separator', 'separators', 'stretchy', 'subscriptshift', 'supscriptshift', 'symmetric', 'voffset', 'width', 'xmlns']); var xml = freeze(['xlink:href', 'xml:id', 'xlink:title', 'xml:space', 'xmlns:xlink']); // eslint-disable-next-line unicorn/better-regex var MUSTACHE_EXPR = seal(/\{\{[\s\S]*|[\s\S]*\}\}/gm); // Specify template detection regex for SAFE_FOR_TEMPLATES mode var ERB_EXPR = seal(/<%[\s\S]*|[\s\S]*%>/gm); var DATA_ATTR = seal(/^data-[\-\w.\u00B7-\uFFFF]/); // eslint-disable-line no-useless-escape var ARIA_ATTR = seal(/^aria-[\-\w]+$/); // eslint-disable-line no-useless-escape var IS_ALLOWED_URI = seal(/^(?:(?:(?:f|ht)tps?|mailto|tel|callto|cid|xmpp):|[^a-z]|[a-z+.\-]+(?:[^a-z+.\-:]|$))/i // eslint-disable-line no-useless-escape ); var IS_SCRIPT_OR_DATA = seal(/^(?:\w+script|data):/i); var ATTR_WHITESPACE = seal(/[\u0000-\u0020\u00A0\u1680\u180E\u2000-\u2029\u205F\u3000]/g // eslint-disable-line no-control-regex ); var DOCTYPE_NAME = seal(/^html$/i); 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; }; function _toConsumableArray$1(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } } var getGlobal = function getGlobal() { return typeof window === 'undefined' ? null : window; }; /** * Creates a no-op policy for internal use only. * Don't export this function outside this module! * @param {?TrustedTypePolicyFactory} trustedTypes The policy factory. * @param {Document} document The document object (to determine policy name suffix) * @return {?TrustedTypePolicy} The policy created (or null, if Trusted Types * are not supported). */ var _createTrustedTypesPolicy = function _createTrustedTypesPolicy(trustedTypes, document) { if ((typeof trustedTypes === 'undefined' ? 'undefined' : _typeof(trustedTypes)) !== 'object' || typeof trustedTypes.createPolicy !== 'function') { return null; } // Allow the callers to control the unique policy name // by adding a data-tt-policy-suffix to the script element with the DOMPurify. // Policy creation with duplicate names throws in Trusted Types. var suffix = null; var ATTR_NAME = 'data-tt-policy-suffix'; if (document.currentScript && document.currentScript.hasAttribute(ATTR_NAME)) { suffix = document.currentScript.getAttribute(ATTR_NAME); } var policyName = 'dompurify' + (suffix ? '#' + suffix : ''); try { return trustedTypes.createPolicy(policyName, { createHTML: function createHTML(html$$1) { return html$$1; } }); } catch (_) { // Policy creation failed (most likely another DOMPurify script has // already run). Skip creating the policy, as this will only cause errors // if TT are enforced. console.warn('TrustedTypes policy ' + policyName + ' could not be created.'); return null; } }; function createDOMPurify() { var window = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : getGlobal(); var DOMPurify = function DOMPurify(root) { return createDOMPurify(root); }; /** * Version label, exposed for easier checks * if DOMPurify is up to date or not */ DOMPurify.version = '2.3.6'; /** * Array of elements that DOMPurify removed during sanitation. * Empty if nothing was removed. */ DOMPurify.removed = []; if (!window || !window.document || window.document.nodeType !== 9) { // Not running in a browser, provide a factory function // so that you can pass your own Window DOMPurify.isSupported = false; return DOMPurify; } var originalDocument = window.document; var document = window.document; var DocumentFragment = window.DocumentFragment, HTMLTemplateElement = window.HTMLTemplateElement, Node = window.Node, Element = window.Element, NodeFilter = window.NodeFilter, _window$NamedNodeMap = window.NamedNodeMap, NamedNodeMap = _window$NamedNodeMap === undefined ? window.NamedNodeMap || window.MozNamedAttrMap : _window$NamedNodeMap, HTMLFormElement = window.HTMLFormElement, DOMParser = window.DOMParser, trustedTypes = window.trustedTypes; var ElementPrototype = Element.prototype; var cloneNode = lookupGetter(ElementPrototype, 'cloneNode'); var getNextSibling = lookupGetter(ElementPrototype, 'nextSibling'); var getChildNodes = lookupGetter(ElementPrototype, 'childNodes'); var getParentNode = lookupGetter(ElementPrototype, 'parentNode'); // As per issue #47, the web-components registry is inherited by a // new document created via createHTMLDocument. As per the spec // (http://w3c.github.io/webcomponents/spec/custom/#creating-and-passing-registries) // a new empty registry is used when creating a template contents owner // document, so we use that as our parent document to ensure nothing // is inherited. if (typeof HTMLTemplateElement === 'function') { var template = document.createElement('template'); if (template.content && template.content.ownerDocument) { document = template.content.ownerDocument; } } var trustedTypesPolicy = _createTrustedTypesPolicy(trustedTypes, originalDocument); var emptyHTML = trustedTypesPolicy ? trustedTypesPolicy.createHTML('') : ''; var _document = document, implementation = _document.implementation, createNodeIterator = _document.createNodeIterator, createDocumentFragment = _document.createDocumentFragment, getElementsByTagName = _document.getElementsByTagName; var importNode = originalDocument.importNode; var documentMode = {}; try { documentMode = clone(document).documentMode ? document.documentMode : {}; } catch (_) {} var hooks = {}; /** * Expose whether this browser supports running the full DOMPurify. */ DOMPurify.isSupported = typeof getParentNode === 'function' && implementation && typeof implementation.createHTMLDocument !== 'undefined' && documentMode !== 9; var MUSTACHE_EXPR$$1 = MUSTACHE_EXPR, ERB_EXPR$$1 = ERB_EXPR, DATA_ATTR$$1 = DATA_ATTR, ARIA_ATTR$$1 = ARIA_ATTR, IS_SCRIPT_OR_DATA$$1 = IS_SCRIPT_OR_DATA, ATTR_WHITESPACE$$1 = ATTR_WHITESPACE; var IS_ALLOWED_URI$$1 = IS_ALLOWED_URI; /** * We consider the elements and attributes below to be safe. Ideally * don't add any new ones but feel free to remove unwanted ones. */ /* allowed element names */ var ALLOWED_TAGS = null; var DEFAULT_ALLOWED_TAGS = addToSet({}, [].concat(_toConsumableArray$1(html), _toConsumableArray$1(svg), _toConsumableArray$1(svgFilters), _toConsumableArray$1(mathMl), _toConsumableArray$1(text))); /* Allowed attribute names */ var ALLOWED_ATTR = null; var DEFAULT_ALLOWED_ATTR = addToSet({}, [].concat(_toConsumableArray$1(html$1), _toConsumableArray$1(svg$1), _toConsumableArray$1(mathMl$1), _toConsumableArray$1(xml))); /* * Configure how DOMPUrify should handle custom elements and their attributes as well as customized built-in elements. * @property {RegExp|Function|null} tagNameCheck one of [null, regexPattern, predicate]. Default: `null` (disallow any custom elements) * @property {RegExp|Function|null} attributeNameCheck one of [null, regexPattern, predicate]. Default: `null` (disallow any attributes not on the allow list) * @property {boolean} allowCustomizedBuiltInElements allow custom elements derived from built-ins if they pass CUSTOM_ELEMENT_HANDLING.tagNameCheck. Default: `false`. */ var CUSTOM_ELEMENT_HANDLING = Object.seal(Object.create(null, { tagNameCheck: { writable: true, configurable: false, enumerable: true, value: null }, attributeNameCheck: { writable: true, configurable: false, enumerable: true, value: null }, allowCustomizedBuiltInElements: { writable: true, configurable: false, enumerable: true, value: false } })); /* Explicitly forbidden tags (overrides ALLOWED_TAGS/ADD_TAGS) */ var FORBID_TAGS = null; /* Explicitly forbidden attributes (overrides ALLOWED_ATTR/ADD_ATTR) */ var FORBID_ATTR = null; /* Decide if ARIA attributes are okay */ var ALLOW_ARIA_ATTR = true; /* Decide if custom data attributes are okay */ var ALLOW_DATA_ATTR = true; /* Decide if unknown protocols are okay */ var ALLOW_UNKNOWN_PROTOCOLS = false; /* Output should be safe for common template engines. * This means, DOMPurify removes data attributes, mustaches and ERB */ var SAFE_FOR_TEMPLATES = false; /* Decide if document with ... should be returned */ var WHOLE_DOCUMENT = false; /* Track whether config is already set on this instance of DOMPurify. */ var SET_CONFIG = false; /* Decide if all elements (e.g. style, script) must be children of * document.body. By default, browsers might move them to document.head */ var FORCE_BODY = false; /* Decide if a DOM `HTMLBodyElement` should be returned, instead of a html * string (or a TrustedHTML object if Trusted Types are supported). * If `WHOLE_DOCUMENT` is enabled a `HTMLHtmlElement` will be returned instead */ var RETURN_DOM = false; /* Decide if a DOM `DocumentFragment` should be returned, instead of a html * string (or a TrustedHTML object if Trusted Types are supported) */ var RETURN_DOM_FRAGMENT = false; /* Try to return a Trusted Type object instead of a string, return a string in * case Trusted Types are not supported */ var RETURN_TRUSTED_TYPE = false; /* Output should be free from DOM clobbering attacks? */ var SANITIZE_DOM = true; /* Keep element content when removing element? */ var KEEP_CONTENT = true; /* If a `Node` is passed to sanitize(), then performs sanitization in-place instead * of importing it into a new Document and returning a sanitized copy */ var IN_PLACE = false; /* Allow usage of profiles like html, svg and mathMl */ var USE_PROFILES = {}; /* Tags to ignore content of when KEEP_CONTENT is true */ var FORBID_CONTENTS = null; var DEFAULT_FORBID_CONTENTS = addToSet({}, ['annotation-xml', 'audio', 'colgroup', 'desc', 'foreignobject', 'head', 'iframe', 'math', 'mi', 'mn', 'mo', 'ms', 'mtext', 'noembed', 'noframes', 'noscript', 'plaintext', 'script', 'style', 'svg', 'template', 'thead', 'title', 'video', 'xmp']); /* Tags that are safe for data: URIs */ var DATA_URI_TAGS = null; var DEFAULT_DATA_URI_TAGS = addToSet({}, ['audio', 'video', 'img', 'source', 'image', 'track']); /* Attributes safe for values like "javascript:" */ var URI_SAFE_ATTRIBUTES = null; var DEFAULT_URI_SAFE_ATTRIBUTES = addToSet({}, ['alt', 'class', 'for', 'id', 'label', 'name', 'pattern', 'placeholder', 'role', 'summary', 'title', 'value', 'style', 'xmlns']); var MATHML_NAMESPACE = 'http://www.w3.org/1998/Math/MathML'; var SVG_NAMESPACE = 'http://www.w3.org/2000/svg'; var HTML_NAMESPACE = 'http://www.w3.org/1999/xhtml'; /* Document namespace */ var NAMESPACE = HTML_NAMESPACE; var IS_EMPTY_INPUT = false; /* Parsing of strict XHTML documents */ var PARSER_MEDIA_TYPE = void 0; var SUPPORTED_PARSER_MEDIA_TYPES = ['application/xhtml+xml', 'text/html']; var DEFAULT_PARSER_MEDIA_TYPE = 'text/html'; var transformCaseFunc = void 0; /* Keep a reference to config to pass to hooks */ var CONFIG = null; /* Ideally, do not touch anything below this line */ /* ______________________________________________ */ var formElement = document.createElement('form'); var isRegexOrFunction = function isRegexOrFunction(testValue) { return testValue instanceof RegExp || testValue instanceof Function; }; /** * _parseConfig * * @param {Object} cfg optional config literal */ // eslint-disable-next-line complexity var _parseConfig = function _parseConfig(cfg) { if (CONFIG && CONFIG === cfg) { return; } /* Shield configuration object from tampering */ if (!cfg || (typeof cfg === 'undefined' ? 'undefined' : _typeof(cfg)) !== 'object') { cfg = {}; } /* Shield configuration object from prototype pollution */ cfg = clone(cfg); /* Set configuration parameters */ ALLOWED_TAGS = 'ALLOWED_TAGS' in cfg ? addToSet({}, cfg.ALLOWED_TAGS) : DEFAULT_ALLOWED_TAGS; ALLOWED_ATTR = 'ALLOWED_ATTR' in cfg ? addToSet({}, cfg.ALLOWED_ATTR) : DEFAULT_ALLOWED_ATTR; URI_SAFE_ATTRIBUTES = 'ADD_URI_SAFE_ATTR' in cfg ? addToSet(clone(DEFAULT_URI_SAFE_ATTRIBUTES), cfg.ADD_URI_SAFE_ATTR) : DEFAULT_URI_SAFE_ATTRIBUTES; DATA_URI_TAGS = 'ADD_DATA_URI_TAGS' in cfg ? addToSet(clone(DEFAULT_DATA_URI_TAGS), cfg.ADD_DATA_URI_TAGS) : DEFAULT_DATA_URI_TAGS; FORBID_CONTENTS = 'FORBID_CONTENTS' in cfg ? addToSet({}, cfg.FORBID_CONTENTS) : DEFAULT_FORBID_CONTENTS; FORBID_TAGS = 'FORBID_TAGS' in cfg ? addToSet({}, cfg.FORBID_TAGS) : {}; FORBID_ATTR = 'FORBID_ATTR' in cfg ? addToSet({}, cfg.FORBID_ATTR) : {}; USE_PROFILES = 'USE_PROFILES' in cfg ? cfg.USE_PROFILES : false; ALLOW_ARIA_ATTR = cfg.ALLOW_ARIA_ATTR !== false; // Default true ALLOW_DATA_ATTR = cfg.ALLOW_DATA_ATTR !== false; // Default true ALLOW_UNKNOWN_PROTOCOLS = cfg.ALLOW_UNKNOWN_PROTOCOLS || false; // Default false SAFE_FOR_TEMPLATES = cfg.SAFE_FOR_TEMPLATES || false; // Default false WHOLE_DOCUMENT = cfg.WHOLE_DOCUMENT || false; // Default false RETURN_DOM = cfg.RETURN_DOM || false; // Default false RETURN_DOM_FRAGMENT = cfg.RETURN_DOM_FRAGMENT || false; // Default false RETURN_TRUSTED_TYPE = cfg.RETURN_TRUSTED_TYPE || false; // Default false FORCE_BODY = cfg.FORCE_BODY || false; // Default false SANITIZE_DOM = cfg.SANITIZE_DOM !== false; // Default true KEEP_CONTENT = cfg.KEEP_CONTENT !== false; // Default true IN_PLACE = cfg.IN_PLACE || false; // Default false IS_ALLOWED_URI$$1 = cfg.ALLOWED_URI_REGEXP || IS_ALLOWED_URI$$1; NAMESPACE = cfg.NAMESPACE || HTML_NAMESPACE; if (cfg.CUSTOM_ELEMENT_HANDLING && isRegexOrFunction(cfg.CUSTOM_ELEMENT_HANDLING.tagNameCheck)) { CUSTOM_ELEMENT_HANDLING.tagNameCheck = cfg.CUSTOM_ELEMENT_HANDLING.tagNameCheck; } if (cfg.CUSTOM_ELEMENT_HANDLING && isRegexOrFunction(cfg.CUSTOM_ELEMENT_HANDLING.attributeNameCheck)) { CUSTOM_ELEMENT_HANDLING.attributeNameCheck = cfg.CUSTOM_ELEMENT_HANDLING.attributeNameCheck; } if (cfg.CUSTOM_ELEMENT_HANDLING && typeof cfg.CUSTOM_ELEMENT_HANDLING.allowCustomizedBuiltInElements === 'boolean') { CUSTOM_ELEMENT_HANDLING.allowCustomizedBuiltInElements = cfg.CUSTOM_ELEMENT_HANDLING.allowCustomizedBuiltInElements; } PARSER_MEDIA_TYPE = // eslint-disable-next-line unicorn/prefer-includes SUPPORTED_PARSER_MEDIA_TYPES.indexOf(cfg.PARSER_MEDIA_TYPE) === -1 ? PARSER_MEDIA_TYPE = DEFAULT_PARSER_MEDIA_TYPE : PARSER_MEDIA_TYPE = cfg.PARSER_MEDIA_TYPE; // HTML tags and attributes are not case-sensitive, converting to lowercase. Keeping XHTML as is. transformCaseFunc = PARSER_MEDIA_TYPE === 'application/xhtml+xml' ? function (x) { return x; } : stringToLowerCase; if (SAFE_FOR_TEMPLATES) { ALLOW_DATA_ATTR = false; } if (RETURN_DOM_FRAGMENT) { RETURN_DOM = true; } /* Parse profile info */ if (USE_PROFILES) { ALLOWED_TAGS = addToSet({}, [].concat(_toConsumableArray$1(text))); ALLOWED_ATTR = []; if (USE_PROFILES.html === true) { addToSet(ALLOWED_TAGS, html); addToSet(ALLOWED_ATTR, html$1); } if (USE_PROFILES.svg === true) { addToSet(ALLOWED_TAGS, svg); addToSet(ALLOWED_ATTR, svg$1); addToSet(ALLOWED_ATTR, xml); } if (USE_PROFILES.svgFilters === true) { addToSet(ALLOWED_TAGS, svgFilters); addToSet(ALLOWED_ATTR, svg$1); addToSet(ALLOWED_ATTR, xml); } if (USE_PROFILES.mathMl === true) { addToSet(ALLOWED_TAGS, mathMl); addToSet(ALLOWED_ATTR, mathMl$1); addToSet(ALLOWED_ATTR, xml); } } /* Merge configuration parameters */ if (cfg.ADD_TAGS) { if (ALLOWED_TAGS === DEFAULT_ALLOWED_TAGS) { ALLOWED_TAGS = clone(ALLOWED_TAGS); } addToSet(ALLOWED_TAGS, cfg.ADD_TAGS); } if (cfg.ADD_ATTR) { if (ALLOWED_ATTR === DEFAULT_ALLOWED_ATTR) { ALLOWED_ATTR = clone(ALLOWED_ATTR); } addToSet(ALLOWED_ATTR, cfg.ADD_ATTR); } if (cfg.ADD_URI_SAFE_ATTR) { addToSet(URI_SAFE_ATTRIBUTES, cfg.ADD_URI_SAFE_ATTR); } if (cfg.FORBID_CONTENTS) { if (FORBID_CONTENTS === DEFAULT_FORBID_CONTENTS) { FORBID_CONTENTS = clone(FORBID_CONTENTS); } addToSet(FORBID_CONTENTS, cfg.FORBID_CONTENTS); } /* Add #text in case KEEP_CONTENT is set to true */ if (KEEP_CONTENT) { ALLOWED_TAGS['#text'] = true; } /* Add html, head and body to ALLOWED_TAGS in case WHOLE_DOCUMENT is true */ if (WHOLE_DOCUMENT) { addToSet(ALLOWED_TAGS, ['html', 'head', 'body']); } /* Add tbody to ALLOWED_TAGS in case tables are permitted, see #286, #365 */ if (ALLOWED_TAGS.table) { addToSet(ALLOWED_TAGS, ['tbody']); delete FORBID_TAGS.tbody; } // Prevent further manipulation of configuration. // Not available in IE8, Safari 5, etc. if (freeze) { freeze(cfg); } CONFIG = cfg; }; var MATHML_TEXT_INTEGRATION_POINTS = addToSet({}, ['mi', 'mo', 'mn', 'ms', 'mtext']); var HTML_INTEGRATION_POINTS = addToSet({}, ['foreignobject', 'desc', 'title', 'annotation-xml']); /* Keep track of all possible SVG and MathML tags * so that we can perform the namespace checks * correctly. */ var ALL_SVG_TAGS = addToSet({}, svg); addToSet(ALL_SVG_TAGS, svgFilters); addToSet(ALL_SVG_TAGS, svgDisallowed); var ALL_MATHML_TAGS = addToSet({}, mathMl); addToSet(ALL_MATHML_TAGS, mathMlDisallowed); /** * * * @param {Element} element a DOM element whose namespace is being checked * @returns {boolean} Return false if the element has a * namespace that a spec-compliant parser would never * return. Return true otherwise. */ var _checkValidNamespace = function _checkValidNamespace(element) { var parent = getParentNode(element); // In JSDOM, if we're inside shadow DOM, then parentNode // can be null. We just simulate parent in this case. if (!parent || !parent.tagName) { parent = { namespaceURI: HTML_NAMESPACE, tagName: 'template' }; } var tagName = stringToLowerCase(element.tagName); var parentTagName = stringToLowerCase(parent.tagName); if (element.namespaceURI === SVG_NAMESPACE) { // The only way to switch from HTML namespace to SVG // is via . If it happens via any other tag, then // it should be killed. if (parent.namespaceURI === HTML_NAMESPACE) { return tagName === 'svg'; } // The only way to switch from MathML to SVG is via // svg if parent is either or MathML // text integration points. if (parent.namespaceURI === MATHML_NAMESPACE) { return tagName === 'svg' && (parentTagName === 'annotation-xml' || MATHML_TEXT_INTEGRATION_POINTS[parentTagName]); } // We only allow elements that are defined in SVG // spec. All others are disallowed in SVG namespace. return Boolean(ALL_SVG_TAGS[tagName]); } if (element.namespaceURI === MATHML_NAMESPACE) { // The only way to switch from HTML namespace to MathML // is via . If it happens via any other tag, then // it should be killed. if (parent.namespaceURI === HTML_NAMESPACE) { return tagName === 'math'; } // The only way to switch from SVG to MathML is via // and HTML integration points if (parent.namespaceURI === SVG_NAMESPACE) { return tagName === 'math' && HTML_INTEGRATION_POINTS[parentTagName]; } // We only allow elements that are defined in MathML // spec. All others are disallowed in MathML namespace. return Boolean(ALL_MATHML_TAGS[tagName]); } if (element.namespaceURI === HTML_NAMESPACE) { // The only way to switch from SVG to HTML is via // HTML integration points, and from MathML to HTML // is via MathML text integration points if (parent.namespaceURI === SVG_NAMESPACE && !HTML_INTEGRATION_POINTS[parentTagName]) { return false; } if (parent.namespaceURI === MATHML_NAMESPACE && !MATHML_TEXT_INTEGRATION_POINTS[parentTagName]) { return false; } // Certain elements are allowed in both SVG and HTML // namespace. We need to specify them explicitly // so that they don't get erronously deleted from // HTML namespace. var commonSvgAndHTMLElements = addToSet({}, ['title', 'style', 'font', 'a', 'script']); // We disallow tags that are specific for MathML // or SVG and should never appear in HTML namespace return !ALL_MATHML_TAGS[tagName] && (commonSvgAndHTMLElements[tagName] || !ALL_SVG_TAGS[tagName]); } // The code should never reach this place (this means // that the element somehow got namespace that is not // HTML, SVG or MathML). Return false just in case. return false; }; /** * _forceRemove * * @param {Node} node a DOM node */ var _forceRemove = function _forceRemove(node) { arrayPush(DOMPurify.removed, { element: node }); try { // eslint-disable-next-line unicorn/prefer-dom-node-remove node.parentNode.removeChild(node); } catch (_) { try { node.outerHTML = emptyHTML; } catch (_) { node.remove(); } } }; /** * _removeAttribute * * @param {String} name an Attribute name * @param {Node} node a DOM node */ var _removeAttribute = function _removeAttribute(name, node) { try { arrayPush(DOMPurify.removed, { attribute: node.getAttributeNode(name), from: node }); } catch (_) { arrayPush(DOMPurify.removed, { attribute: null, from: node }); } node.removeAttribute(name); // We void attribute values for unremovable "is"" attributes if (name === 'is' && !ALLOWED_ATTR[name]) { if (RETURN_DOM || RETURN_DOM_FRAGMENT) { try { _forceRemove(node); } catch (_) {} } else { try { node.setAttribute(name, ''); } catch (_) {} } } }; /** * _initDocument * * @param {String} dirty a string of dirty markup * @return {Document} a DOM, filled with the dirty markup */ var _initDocument = function _initDocument(dirty) { /* Create a HTML document */ var doc = void 0; var leadingWhitespace = void 0; if (FORCE_BODY) { dirty = '' + dirty; } else { /* If FORCE_BODY isn't used, leading whitespace needs to be preserved manually */ var matches = stringMatch(dirty, /^[\r\n\t ]+/); leadingWhitespace = matches && matches[0]; } if (PARSER_MEDIA_TYPE === 'application/xhtml+xml') { // Root of XHTML doc must contain xmlns declaration (see https://www.w3.org/TR/xhtml1/normative.html#strict) dirty = '' + dirty + ''; } var dirtyPayload = trustedTypesPolicy ? trustedTypesPolicy.createHTML(dirty) : dirty; /* * Use the DOMParser API by default, fallback later if needs be * DOMParser not work for svg when has multiple root element. */ if (NAMESPACE === HTML_NAMESPACE) { try { doc = new DOMParser().parseFromString(dirtyPayload, PARSER_MEDIA_TYPE); } catch (_) {} } /* Use createHTMLDocument in case DOMParser is not available */ if (!doc || !doc.documentElement) { doc = implementation.createDocument(NAMESPACE, 'template', null); try { doc.documentElement.innerHTML = IS_EMPTY_INPUT ? '' : dirtyPayload; } catch (_) { // Syntax error if dirtyPayload is invalid xml } } var body = doc.body || doc.documentElement; if (dirty && leadingWhitespace) { body.insertBefore(document.createTextNode(leadingWhitespace), body.childNodes[0] || null); } /* Work on whole document or just its body */ if (NAMESPACE === HTML_NAMESPACE) { return getElementsByTagName.call(doc, WHOLE_DOCUMENT ? 'html' : 'body')[0]; } return WHOLE_DOCUMENT ? doc.documentElement : body; }; /** * _createIterator * * @param {Document} root document/fragment to create iterator for * @return {Iterator} iterator instance */ var _createIterator = function _createIterator(root) { return createNodeIterator.call(root.ownerDocument || root, root, // eslint-disable-next-line no-bitwise NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT | NodeFilter.SHOW_TEXT, null, false); }; /** * _isClobbered * * @param {Node} elm element to check for clobbering attacks * @return {Boolean} true if clobbered, false if safe */ var _isClobbered = function _isClobbered(elm) { return elm instanceof HTMLFormElement && (typeof elm.nodeName !== 'string' || typeof elm.textContent !== 'string' || typeof elm.removeChild !== 'function' || !(elm.attributes instanceof NamedNodeMap) || typeof elm.removeAttribute !== 'function' || typeof elm.setAttribute !== 'function' || typeof elm.namespaceURI !== 'string' || typeof elm.insertBefore !== 'function'); }; /** * _isNode * * @param {Node} obj object to check whether it's a DOM node * @return {Boolean} true is object is a DOM node */ var _isNode = function _isNode(object) { return (typeof Node === 'undefined' ? 'undefined' : _typeof(Node)) === 'object' ? object instanceof Node : object && (typeof object === 'undefined' ? 'undefined' : _typeof(object)) === 'object' && typeof object.nodeType === 'number' && typeof object.nodeName === 'string'; }; /** * _executeHook * Execute user configurable hooks * * @param {String} entryPoint Name of the hook's entry point * @param {Node} currentNode node to work on with the hook * @param {Object} data additional hook parameters */ var _executeHook = function _executeHook(entryPoint, currentNode, data) { if (!hooks[entryPoint]) { return; } arrayForEach(hooks[entryPoint], function (hook) { hook.call(DOMPurify, currentNode, data, CONFIG); }); }; /** * _sanitizeElements * * @protect nodeName * @protect textContent * @protect removeChild * * @param {Node} currentNode to check for permission to exist * @return {Boolean} true if node was killed, false if left alive */ var _sanitizeElements = function _sanitizeElements(currentNode) { var content = void 0; /* Execute a hook if present */ _executeHook('beforeSanitizeElements', currentNode, null); /* Check if element is clobbered or can clobber */ if (_isClobbered(currentNode)) { _forceRemove(currentNode); return true; } /* Check if tagname contains Unicode */ if (stringMatch(currentNode.nodeName, /[\u0080-\uFFFF]/)) { _forceRemove(currentNode); return true; } /* Now let's check the element's type and name */ var tagName = transformCaseFunc(currentNode.nodeName); /* Execute a hook if present */ _executeHook('uponSanitizeElement', currentNode, { tagName: tagName, allowedTags: ALLOWED_TAGS }); /* Detect mXSS attempts abusing namespace confusion */ if (!_isNode(currentNode.firstElementChild) && (!_isNode(currentNode.content) || !_isNode(currentNode.content.firstElementChild)) && regExpTest(/<[/\w]/g, currentNode.innerHTML) && regExpTest(/<[/\w]/g, currentNode.textContent)) { _forceRemove(currentNode); return true; } /* Mitigate a problem with templates inside select */ if (tagName === 'select' && regExpTest(/