{ "version": 3, "sources": ["../javascripts/application/debug.js", "../javascripts/shared/definitions.js", "../javascripts/shared/objects.js", "../javascripts/shared/decode.js", "javascripts-shared-assets.js", "../javascripts/shared/assets.js", "../javascripts/shared/base-class.js", "../javascripts/application/setup.js", "../../../node_modules/jquery/dist/jquery.js", "../../../node_modules/select2/dist/js/select2.js", "../javascripts/shared/time.js", "../javascripts/shared/strings.js", "../../../node_modules/@rails/actioncable/app/assets/javascripts/actioncable.esm.js", "../javascripts/shared/cable-consumer.js", "../../../node_modules/namespace-emitter/index.js", "../../../node_modules/lodash/isObject.js", "../../../node_modules/lodash/_freeGlobal.js", "../../../node_modules/lodash/_root.js", "../../../node_modules/lodash/now.js", "../../../node_modules/lodash/_trimmedEndIndex.js", "../../../node_modules/lodash/_baseTrim.js", "../../../node_modules/lodash/_Symbol.js", "../../../node_modules/lodash/_getRawTag.js", "../../../node_modules/lodash/_objectToString.js", "../../../node_modules/lodash/_baseGetTag.js", "../../../node_modules/lodash/isObjectLike.js", "../../../node_modules/lodash/isSymbol.js", "../../../node_modules/lodash/toNumber.js", "../../../node_modules/lodash/debounce.js", "../../../node_modules/lodash/throttle.js", "../../../node_modules/@transloadit/prettier-bytes/src/prettierBytes.ts", "../../../node_modules/wildcard/index.js", "../../../node_modules/mime-match/index.js", "../../../node_modules/turbolinks/dist/turbolinks.js", "../javascripts/application.js", "../javascripts/vendor/jquery.js", "../javascripts/vendor/rails.js", "../../../node_modules/@rails/ujs/app/assets/javascripts/rails-ujs.esm.js", "../javascripts/vendor/select2.js", "../javascripts/vendor/capybara-lockstep.js", "../javascripts/feature/advanced-search.js", "../javascripts/shared/arrays.js", "../javascripts/shared/events.js", "../javascripts/shared/form.js", "../javascripts/shared/random.js", "../javascripts/shared/url.js", "../javascripts/shared/accessibility.js", "../javascripts/shared/css.js", "../javascripts/shared/keyboard.js", "../javascripts/shared/html.js", "../javascripts/shared/menu.js", "../javascripts/feature/client-debug.js", "../javascripts/feature/download.js", "../javascripts/shared/xhr.js", "../javascripts/shared/http.js", "../javascripts/feature/flash.js", "../javascripts/shared/flash.js", "../javascripts/shared/unicode.js", "../javascripts/feature/fonts.js", "../javascripts/feature/help.js", "../javascripts/feature/iframe.js", "../javascripts/feature/images.js", "../javascripts/feature/model-form.js", "../javascripts/shared/controller.js", "../javascripts/shared/lookup-modal.js", "../javascripts/channels/lookup-channel.js", "../javascripts/shared/cable-channel.js", "../javascripts/shared/api.js", "../javascripts/shared/channel-request.js", "../javascripts/shared/channel-response.js", "../javascripts/shared/lookup-request.js", "../javascripts/shared/lookup-response.js", "../javascripts/shared/json.js", "../javascripts/shared/modal-dialog.js", "../javascripts/shared/modal-base.js", "../javascripts/shared/modal-hooks.js", "../javascripts/shared/callbacks.js", "../javascripts/shared/nav-group.js", "../javascripts/shared/exceptions.js", "../javascripts/shared/math.js", "../javascripts/shared/search-in-progress.js", "../javascripts/shared/overlay.js", "../javascripts/shared/uploader.js", "../javascripts/vendor/uppy.js", "../../../node_modules/@uppy/core/lib/index.js", "../../../node_modules/@uppy/utils/lib/Translator.js", "../../../node_modules/@uppy/core/lib/Uppy.js", "../../../node_modules/nanoid/non-secure/index.js", "../../../node_modules/@uppy/store-default/lib/index.js", "../../../node_modules/@uppy/utils/lib/getFileNameAndExtension.js", "../../../node_modules/@uppy/utils/lib/mimeTypes.js", "../../../node_modules/@uppy/utils/lib/getFileType.js", "../../../node_modules/@uppy/utils/lib/generateFileID.js", "../../../node_modules/@uppy/core/lib/supportsUploadProgress.js", "../../../node_modules/@uppy/core/lib/getFileName.js", "../../../node_modules/@uppy/utils/lib/getTimeStamp.js", "../../../node_modules/@uppy/core/lib/loggers.js", "../../../node_modules/@uppy/core/lib/Restricter.js", "../../../node_modules/@uppy/core/lib/locale.js", "../../../node_modules/preact/src/constants.js", "../../../node_modules/preact/src/util.js", "../../../node_modules/preact/src/options.js", "../../../node_modules/preact/src/create-element.js", "../../../node_modules/preact/src/component.js", "../../../node_modules/preact/src/diff/props.js", "../../../node_modules/preact/src/create-context.js", "../../../node_modules/preact/src/diff/children.js", "../../../node_modules/preact/src/diff/index.js", "../../../node_modules/preact/src/render.js", "../../../node_modules/preact/src/clone-element.js", "../../../node_modules/preact/src/diff/catch-error.js", "../../../node_modules/@uppy/utils/lib/isDOMElement.js", "../../../node_modules/@uppy/utils/lib/findDOMElement.js", "../../../node_modules/@uppy/utils/lib/getTextDirection.js", "../../../node_modules/@uppy/core/lib/BasePlugin.js", "../../../node_modules/@uppy/core/lib/UIPlugin.js", "../../../node_modules/@uppy/utils/lib/toArray.js", "../../../node_modules/@uppy/file-input/lib/locale.js", "../../../node_modules/@uppy/file-input/lib/FileInput.js", "../../../node_modules/@uppy/informer/lib/FadeIn.js", "../../../node_modules/@uppy/informer/lib/TransitionGroup.js", "../../../node_modules/@uppy/informer/lib/Informer.js", "../../../node_modules/@uppy/progress-bar/lib/ProgressBar.js", "../../../node_modules/@uppy/core/lib/EventManager.js", "../../../node_modules/@uppy/utils/lib/RateLimitedQueue.js", "../../../node_modules/@uppy/utils/lib/NetworkError.js", "../../../node_modules/@uppy/utils/lib/isNetworkError.js", "../../../node_modules/@uppy/utils/lib/ProgressTimeout.js", "../../../node_modules/@uppy/utils/lib/fetcher.js", "../../../node_modules/@uppy/utils/lib/fileFilters.js", "../../../node_modules/@uppy/utils/lib/getAllowedMetaFields.js", "../../../node_modules/@uppy/xhr-upload/lib/locale.js", "../../../node_modules/@uppy/xhr-upload/lib/index.js", "../javascripts/feature/panel.js", "../javascripts/feature/records.js", "../javascripts/feature/scroll.js", "../javascripts/feature/search-analysis.js", "../javascripts/shared/color.js", "../javascripts/shared/database.js", "../javascripts/shared/queue.js", "../javascripts/shared/session-state.js", "../javascripts/feature/search-in-progress.js", "../javascripts/feature/session.js", "../javascripts/shared/browser.js", "../javascripts/feature/skip-nav.js", "../javascripts/feature/table.js", "../javascripts/controllers/manifest-edit.js", "../javascripts/shared/field.js", "../javascripts/shared/inline-popup.js", "../javascripts/shared/grids.js", "../javascripts/shared/manifests.js", "../javascripts/controllers/manifest-remit.js", "../javascripts/shared/submit-modal.js", "../javascripts/channels/submit-channel.js", "../javascripts/shared/submit-response.js", "../javascripts/shared/submit-request.js", "../javascripts/controllers/search.js", "../javascripts/controllers/tool.js", "../javascripts/tool/bibliographic-lookup.js", "../javascripts/tool/math-detective.js", "../javascripts/shared/image.js", "../javascripts/controllers/user_sessions.js", "../javascripts/controllers/generic.js", "../javascripts/shared/analytics.js", "../javascripts/application/start.js", "../javascripts/vendor/turbolinks.js"], "sourceRoot": "https://raw.githubusercontent.com/uvalib/emma/master/app/assets/javascripts/", "sourcesContent": ["// app/assets/javascripts/application/debug.js\n//\n// Application-wide debugging control.\n//\n// NOTE: It is assumed that this file is always imported in each JavaScript\n// source file, particularly to expose the @typedefs in \"types.js\".\n\n\nimport \"./types\"\n\n\n/**\n * Default setting for emitting console log output as each module file is read.\n *\n * @type {boolean}\n */\nconst FILE_DEBUG = true;\n\n// noinspection FunctionNamingConventionJS, JSUnusedGlobalSymbols\n/**\n * An instance of this class is assigned to window.APP_DEBUG to allow\n * per-module control of debug output from the console.

\n *\n * _**Turning debugging on or off:**_\n * ```\n * window.APP_DEBUG.on(module_name);\n * window.APP_DEBUG.off(module_name);\n * ```\n * persists between pages via localStorage.

\n *\n * _**Resetting debug status for a module:**_\n * ```\n * window.APP_DEBUG.reset(module_name);\n * ```\n * removes it from localStorage so that the default setting for the module is\n * honored.

\n */\nexport class AppDebug {\n\n static CLASS_NAME = \"AppDebug\";\n\n // ========================================================================\n // Type definitions\n // ========================================================================\n\n /**\n * @typedef {boolean|\"true\"|\"false\"} BooleanValue\n */\n\n /**\n * @typedef {object} LoggingFunctions\n *\n * @property {function():boolean} debugging\n * @property {function():string} logPrefix\n * @property {function(...*):undefined} error\n * @property {function(...*):undefined} warn\n * @property {function(...*):undefined} log\n * @property {function(...*):undefined} info\n * @property {function(...*):undefined} debug\n */\n\n // ========================================================================\n // Constants\n // ========================================================================\n\n static STORE_KEY = \"DEBUG\";\n static GLOBAL_STORE_KEY = this.STORE_KEY;\n static STORE_KEY_PREFIX = `${this.STORE_KEY}/`;\n\n /**\n * Console string substitution sequence.\n *\n * @see https://developer.mozilla.org/en-US/docs/Web/API/console\n *\n * @type {RegExp}\n */\n static CONSOLE_FMT_MATCH = /%[cdfioOs]|%(\\d+(\\.\\d+)?|\\.\\d+)[dfi]/;\n\n /**\n * For use with {@link consoleFmt}.\n *\n * @type {Object.}\n */\n static CONSOLE_BOX_FMT = {\n display: \"inline-block\",\n padding: \"0 0.25em\",\n margin: \"-0.375em 0\",\n \"line-height\": \"1.75\",\n \"font-size\": \"larger\",\n color: \"white\",\n background: \"red\",\n };\n\n /**\n * Right-aligned width of the leading module name in console output.\n *\n * @type {number}\n */\n static MOD_ALIGN = 15;\n\n // ========================================================================\n // Constructor\n // ========================================================================\n\n /**\n * Create a new instance, by default initializing based on the dynamic\n * settings injected into `` so that debugging is off by default when\n * deployed and on by default otherwise.\n *\n * @param {BooleanValue} [setting]\n */\n constructor(setting) {\n let active = setting;\n if (typeof active === \"undefined\") {\n // noinspection JSUnresolvedVariable\n const settings = window.ASSET_OVERRIDES?.OverrideScriptSettings;\n active = settings?.APP_DEBUG;\n if (typeof active === \"undefined\") {\n const deployed = settings ? settings.DEPLOYED : true;\n const in_test = (settings?.RAILS_ENV === \"test\");\n active = !deployed && !in_test;\n }\n }\n this.active = active;\n }\n\n // ========================================================================\n // Class methods - internal\n // ========================================================================\n\n /**\n * Store true/false setting.\n *\n * @param {string} key\n * @param {BooleanValue} value\n *\n * @returns {\"true\"|\"false\"}\n * @protected\n */\n static _set(key, value) {\n let setting = value.toString();\n try {\n localStorage.setItem(key, setting);\n } catch (error) {\n console.warn(`${this.CLASS_NAME}: ${key}: ${error}`);\n setting = \"false\";\n }\n return setting;\n }\n\n /**\n * Retrieve a true/false setting.\n *\n * @param {string} key\n *\n * @returns {\"true\"|\"false\"|undefined}\n * @protected\n */\n static _get(key) {\n return localStorage.getItem(key) || undefined;\n }\n\n /**\n * Reset a true/false setting.\n *\n * @param {string} key\n *\n * @returns {true}\n * @protected\n */\n static _clear(key) {\n localStorage.removeItem(key);\n return true;\n }\n\n /**\n * Translate a module name into a storage key.\n *\n * @param {string} mod\n *\n * @returns {string}\n * @protected\n */\n static _keyFor(mod) {\n const prefix = this.STORE_KEY_PREFIX;\n return mod.startsWith(prefix) ? mod : `${prefix}${mod}`;\n }\n\n // ========================================================================\n // Class properties\n // ========================================================================\n\n /**\n * Get global debugging status.\n *\n * @returns {boolean}\n */\n static get active() {\n return this._get(this.GLOBAL_STORE_KEY) === \"true\";\n }\n\n /**\n * Set global debugging status.\n *\n * @param {BooleanValue} setting\n */\n static set active(setting) {\n const unset = (typeof setting === \"undefined\") || (setting === null);\n const value = unset ? false : setting;\n this._set(this.GLOBAL_STORE_KEY, value);\n }\n\n /**\n * Currently persisted storage keys.\n *\n * @returns {string[]}\n */\n static get statusKeys() {\n const keys = [];\n for (let i = 0; i < localStorage.length; i++) {\n const key = localStorage.key(i);\n if (key?.startsWith(this.STORE_KEY_PREFIX)) { keys.push(key) }\n }\n return keys;\n }\n\n // ========================================================================\n // Class methods\n // ========================================================================\n\n /**\n * Get the debug setting for the indicated module.\n *\n * @param {string} mod\n *\n * @returns {\"true\"|\"false\"|undefined}\n */\n static get(mod) {\n const key = this._keyFor(mod);\n return this._get(key);\n }\n\n /**\n * Set debugging on/off for the indicated module, overriding its internal\n * default if one was given.\n *\n * @param {string} mod\n * @param {BooleanValue} [active] Default: *true*.\n *\n * @returns {\"true\"|\"false\"}\n */\n static set(mod, active) {\n const key = this._keyFor(mod);\n const val = (typeof active === \"undefined\") || active;\n return this._set(key, val);\n }\n\n /**\n * Restore the initial self-defined debug setting for the indicated\n * module.\n *\n * @param {string} mod\n *\n * @returns {true}\n */\n static reset(mod) {\n const key = this._keyFor(mod);\n return this._clear(key);\n }\n\n /**\n * Turn on debugging for the indicated module, overriding its internal\n * default if one was given.\n *\n * @param {string} mod\n *\n * @returns {\"true\"|\"false\"}\n */\n static on(mod) {\n if (!this.active) { console.warn(\"NOTE: window.APP_DEBUG not active\") }\n return this.set(mod, true);\n }\n\n /**\n * Turn off debugging for the indicated module, overriding its\n * internal default if one was given.\n *\n * @param {string} mod\n *\n * @returns {\"true\"|\"false\"}\n */\n static off(mod) {\n return this.set(mod, false);\n }\n\n /**\n * Indicate the persisted debugging status for the given module.\n *\n * @param {string} mod\n *\n * @returns {boolean|undefined}\n */\n static isActive(mod) {\n if (!this.active) { return false }\n const active = this.get(mod);\n if (typeof active !== \"undefined\") { return active === \"true\" }\n }\n\n /**\n * Indicate the current debugging status for the given module or report\n * the given default if no status is persisted.\n *\n * @param {string} mod\n * @param {boolean} [mod_default]\n *\n * @returns {boolean}\n */\n static activeFor(mod, mod_default) {\n const active = this.isActive(mod);\n if (typeof active === \"undefined\") {\n return mod_default || false;\n } else {\n return active;\n }\n }\n\n /**\n * Reset all debug settings by removing all localStorage entries starting\n * with {@link STORE_KEY_PREFIX}.\n *\n * @returns {true}\n */\n static resetAll() {\n this.statusKeys.forEach(key => this._clear(key));\n return true;\n }\n\n // ========================================================================\n // Class methods - console output\n // ========================================================================\n\n /**\n * Resolve console arguments to support formatting substitutions.\n *\n * @param {...*} args\n *\n * @returns {*[]}\n */\n static consoleArgs(...args) {\n let parts = Array.isArray(args[0]) ? args.shift() : [];\n parts = [...parts, ...args];\n\n // Formatting only applies to the first argument.\n if (typeof parts[0] !== \"string\") { return parts }\n\n // First argument already has a substitution sequence.\n if (parts[0].match(this.CONSOLE_FMT_MATCH)) { return parts }\n\n // Look for a substitution sequence in a later argument.\n let found = -1;\n const leader = [];\n $.each(parts, (idx, part) => {\n if (typeof part === \"string\") {\n if (part.match(this.CONSOLE_FMT_MATCH)) {\n found = idx;\n }\n leader.push(part);\n }\n return (found < 0); // break loop if found\n });\n\n if (found >= 0) {\n return [leader.join(\" \"), ...parts.slice(found+1)];\n } else {\n return parts;\n }\n }\n\n /**\n * Generate console log arguments for formatted log output.\n *\n * @param {string} text\n * @param {string|string[]|Object.} css\n * @param {...*} args\n *\n * @returns {[string, string, ...*]}\n */\n static consoleFmt(text, css, ...args) {\n let styles;\n if (Array.isArray(css)) {\n styles = css.map(v => v.replace(/[;\\s]+$/, \"\"));\n } else if (typeof css === \"object\") {\n styles = Object.entries(css).map(([k,v]) => `${k}: ${v}`);\n } else if (typeof css === \"string\") {\n styles = css.split(/;+\\s+/);\n } else {\n return [text, ...args];\n }\n return [`%c${text}`, styles.join(\"; \"), ...args];\n }\n\n /**\n * Console log output for indicating when a module file is read.\n *\n * @param {string} path\n * @param {string} [name]\n * @param {boolean} [debug]\n * @param {...*} args\n */\n static file(path, name, debug, ...args) {\n const mod = \"FILE\";\n if (this.activeFor(mod, FILE_DEBUG)) {\n const msg = this.consoleFmt(mod, this.CONSOLE_BOX_FMT, path);\n if (name !== undefined) { msg.push(\"|\", name) }\n if (debug !== undefined) { msg.push(\"|\", \"DEBUG =\", debug) }\n console.log(...msg, ...args);\n }\n }\n\n // ========================================================================\n // Class methods\n // ========================================================================\n\n /**\n * Generate console logging functions for use in the executing module.\n *\n * @param {string} mod\n * @param {boolean} dbg\n *\n * @returns {LoggingFunctions}\n */\n static consoleLogging(mod, dbg) {\n function debugging() { return AppDebug.activeFor(mod, dbg) }\n function logPrefix() { return `${mod.padEnd(AppDebug.MOD_ALIGN)} -` }\n function error(...arg) { console.error(..._prefix(arg)) }\n function warn(...arg) { console.warn(..._prefix(arg)) }\n function log(...arg) { console.log(..._prefix(arg)) }\n function info(...a) { debugging() && console.info(..._prefix(a)) }\n function debug(...a) { debugging() && console.debug(..._prefix(a)) }\n function _prefix(args) {\n const start = Array.isArray(args[0]) ? args.shift() : [];\n return AppDebug.consoleArgs(logPrefix(), ...start, ...args);\n }\n // noinspection JSValidateTypes\n return { debugging, logPrefix, error, warn, log, info, debug };\n }\n\n // ========================================================================\n // Methods - internal\n // ========================================================================\n\n _set(key, value) { return this.constructor._set(key, value) }\n _get(key) { return this.constructor._get(key) }\n _clear(key) { return this.constructor._clear(key) }\n _keyFor(mod) { return this.constructor._keyFor(mod) }\n\n // ========================================================================\n // Properties\n // ========================================================================\n\n get active() { return this.constructor.active }\n set active(setting) { this.constructor.active = setting }\n\n // ========================================================================\n // Methods\n // ========================================================================\n\n get(mod) { return this.constructor.get(mod) }\n set(mod, active) { return this.constructor.set(mod, active) }\n reset(mod) { return this.constructor.reset(mod) }\n on(mod) { return this.constructor.on(mod) }\n off(mod) { return this.constructor.off(mod) }\n isActive(mod) { return this.constructor.isActive(mod) }\n activeFor(mod, def) { return this.constructor.activeFor(mod, def) }\n resetAll() { return this.constructor.resetAll() }\n consoleFmt(...args) { return this.constructor.consoleFmt(...args) }\n file(name, ...args) { return this.constructor.file(name, ...args) }\n}\n\nwindow.APP_DEBUG ||= new AppDebug();\n\nAppDebug.file(\"application/debug\");\n", "// app/assets/javascripts/shared/definitions.js\n\n\nimport { AppDebug } from \"../application/debug\";\n\n\nAppDebug.file(\"shared/definitions\");\n\n// ============================================================================\n// Functions - Element values\n// ============================================================================\n\n/**\n * Indicate whether the item is not undefined.\n *\n * @param {*} item\n *\n * @returns {boolean}\n */\nexport function isDefined(item) {\n return typeof item !== \"undefined\";\n}\n\n/**\n * Indicate whether the item is not defined.\n *\n * @param {*} item\n *\n * @returns {boolean}\n */\nexport function notDefined(item) {\n return typeof item === \"undefined\";\n}\n\n/**\n * Indicate whether the item does not contain a value.\n *\n * @param {*} item\n *\n * @returns {boolean}\n */\nexport function isEmpty(item) {\n switch (true) {\n case (item === false): return false;\n case (item === 0): return false;\n case (!item): return true;\n case isDefined(item?.size): return !item.size;\n case isDefined(item?.length): return !item.length;\n case (typeof item === \"object\"): return !Object.keys(item).length;\n default: return false;\n }\n}\n\n/**\n * Indicate whether the item contains a value.\n *\n * @param {*} item\n *\n * @returns {boolean}\n */\nexport function notEmpty(item) {\n return !isEmpty(item);\n}\n\n/**\n * Indicate whether the item does not contain a value.\n *\n * @param {*} item\n *\n * @returns {boolean}\n */\nexport function isMissing(item) {\n return isEmpty(item);\n}\n\n/**\n * Indicate whether the item contains a value.\n *\n * @param {*} item\n *\n * @returns {boolean}\n */\nexport function isPresent(item) {\n return notEmpty(item);\n}\n\n/**\n * Return the item itself if it is not empty or undefined otherwise.\n *\n * @template T\n *\n * @param {T} item\n *\n * @returns {T|undefined}\n */\nexport function presence(item) {\n return isEmpty(item) ? undefined : item;\n}\n", "// app/assets/javascripts/shared/objects.js\n\n\nimport { AppDebug } from \"../application/debug\";\nimport { isDefined, isPresent } from \"./definitions\";\n\n\nAppDebug.file(\"shared/objects\");\n\n// ============================================================================\n// Functions\n// ============================================================================\n\n/**\n * Generate an object from JSON (used in place of {@link JSON.parse}).\n *\n * @param {*} item\n * @param {string|false|null} [caller] Null or false for no diagnostic message\n * @param {function} [reviver] To {@link JSON.parse}.\n *\n * @returns {object|undefined}\n */\nexport function fromJSON(item, caller, reviver) {\n const func = isDefined(caller) ? caller : \"fromJSON\";\n const type = item && typeof(item);\n let result;\n if (type === \"object\") {\n result = item;\n if (reviver) {\n console.warn(`${func}: reviver not called for item =`, item);\n }\n } else if (type === \"string\") {\n try {\n result = reviver ? JSON.parse(item, reviver) : JSON.parse(item);\n } catch (error) {\n func && console.warn(`${func}: ${error} - item:\\n`, item);\n }\n } else if (item) {\n console.warn(`${func}: unexpected type \"${type}\" for`, item);\n }\n return result;\n}\n\n/**\n * Generate a copy of the item without blank elements.\n *\n * @template T\n *\n * @param {T} item\n * @param {boolean} [trim] If *false*, don't trim white space.\n *\n * @returns {T}\n */\nexport function compact(item, trim) {\n if (item instanceof Node) {\n return item;\n\n } else if (item instanceof jQuery) {\n return item;\n\n } else if (typeof item === \"string\") {\n return (trim === false) ? `${item}` : item.trim();\n\n } else if (Array.isArray(item)) {\n return item.map(v => compact(v, trim)).filter(v => isPresent(v));\n\n } else if (isObject(item)) {\n const pr = objectEntries(item).map(([k,v]) => [k, compact(v, trim)]);\n return Object.fromEntries(pr.filter(([_,v]) => isPresent(v)));\n\n } else {\n return item;\n }\n}\n\n/**\n * Make a completely frozen copy of an item.\n *\n * @template T\n *\n * @param {T} item Source item (which will be unaffected).\n *\n * @returns {T} An immutable copy of **item**.\n */\nexport function deepFreeze(item) {\n let new_item;\n if (Array.isArray(item)) {\n new_item = item.map(v => deepFreeze(v));\n\n } else if (isObject(item)) {\n const kv = objectEntries(item).filter(([k,v]) => [k, deepFreeze(v)]);\n new_item = Object.fromEntries(kv);\n\n } else {\n new_item = item;\n }\n return Object.freeze(new_item);\n}\n\n/**\n * Make a deep copy of the given item.\n *\n * @template T\n *\n * @param {T} item\n *\n * @returns {T}\n */\nexport function deepDup(item) {\n return dup(item, true);\n}\n\n/**\n * Make a duplicate of the given item.\n *\n * @template T\n *\n * @param {T} item\n * @param {boolean} [deep] If *true*, make a deep copy.\n *\n * @returns {T}\n */\nexport function dup(item, deep) {\n if (typeof item === \"string\") { return `${item}` }\n if (typeof item !== \"object\") { return item }\n if (item === null) { return item }\n if (typeof item.toObject === \"function\") { return item.toObject() }\n const array = Array.isArray(item);\n if (!deep) { return array ? [...item] : { ...item } }\n if (array) { return item.map(v => dup(v, deep)) }\n const entries = Object.entries(item).map(([k,v]) => [k, dup(v, deep)]);\n return Object.fromEntries(entries);\n}\n\n// ============================================================================\n// Functions - returning Object\n// ============================================================================\n\n/**\n * Indicate whether the item is an actual object (and not null or an array).\n *\n * @param {*} item\n * @param {boolean} [or_array] If *true*, return *true* for an array.\n *\n * @returns {boolean}\n */\nexport function isObject(item, or_array) {\n return !!item && (typeof item === \"object\") &&\n (!!or_array || !Array.isArray(item));\n}\n\n/**\n * If **item** is a class instance return the value of its toObject() method\n * (if it exists) or a copy of the item otherwise.\n *\n * @param {*} item\n *\n * @returns {object|undefined}\n */\nexport function asObject(item) {\n if (isDefined(item?.toObject)) {\n return item.toObject();\n } else if (isObject(item)) {\n return { ...item };\n }\n}\n\n/**\n * Make a duplicate of the given object.\n *\n * @param {object|undefined} item\n * @param {boolean} [shallow] If *true*, make a shallow copy.\n *\n * @returns {object}\n */\nexport function dupObject(item, shallow) {\n return isObject(item, true) ? dup(item, !shallow) : {};\n}\n\n/**\n * Create an Object from array of key-value pairs, or from an array of keys\n * and a mapper function which returns a value for the given key.

\n *\n * If **arg** is an object or array of key-value pairs, **map_fn** is optional.\n * By default, **map_fn** is assumed to return a key-value pair; if\n * **pair_out** is set to *false* then **map_fn** is assumed to emit a value\n * only.

\n *\n * If **arg** is an array of single values, **map_fn** is required.\n * By default, **map_fn** is assumed to emit a value only; if **pair_out** is\n * set to *true* then **map_fn** is assumed to return a key-value pair.

\n *\n * Invalid pairs elements are silently discarded.\n *\n * @param {string[]|[string,*][]|object} arg\n * @param {function(string)|function(string,*)} [map_fn]\n * @param {boolean} [pair_out]\n *\n * @returns {object}\n */\nexport function toObject(arg, map_fn, pair_out) {\n const func = \"toObject\";\n const src = asObject(arg) || arg;\n const object = isObject(src);\n const mapper = (typeof map_fn === \"function\") ? map_fn : undefined;\n const arity = mapper?.length || -1;\n let result;\n if (!object && !Array.isArray(src)) {\n console.warn(`${func}: not an array or object:`, src);\n } else if (mapper && (arity < 1)) {\n console.error(`${func}: mapper must take 1 or 2 args:`, mapper);\n } else {\n const pair = (kv) => Array.isArray(kv) && (kv.length === 2) && !!kv[0];\n const in1 = (arity === 1);\n const in2 = (arity >= 2);\n const out1 = (pair_out === false);\n const out2 = (pair_out === true);\n let fn = undefined;\n if (object || pair(src[0])) {\n fn ||= in2 && out1 && (([k,v]) => [k, mapper(k,v)]);\n fn ||= in1 && !out2 && (([k,_]) => [k, mapper(k)]);\n fn ||= in2 && (([k,v]) => mapper(k,v));\n fn ||= in1 && (([k,_]) => mapper(k));\n fn ||= (([k,v]) => [k, v]);\n } else {\n fn ||= out2 && mapper;\n fn ||= in2 && ((k,idx) => [idx, mapper(k,idx)]);\n fn ||= in1 && ((k) => [k, mapper(k)]);\n fn ||= ((k) => [k, k]);\n }\n const src_pairs = object ? Object.entries(src) : src;\n const dst_pairs = src_pairs.map(fn).filter(pair);\n result = Object.fromEntries(dst_pairs);\n }\n return result || {};\n}\n\n/**\n * Create a new object with keys/values swapped.\n *\n * @param {object} obj\n *\n * @returns {object}\n */\nexport function invert(obj) {\n return toObject(obj, (k,v) => [v,k]);\n}\n\n// ============================================================================\n// Functions - returning Array\n// ============================================================================\n\n/**\n * Transform an object into an array of key-value pairs.\n *\n * @param {object} item\n *\n * @returns {[string, any][]}\n */\nexport function objectEntries(item) {\n return Object.entries(asObject(item) || {});\n}\n\n// ============================================================================\n// Functions - other\n// ============================================================================\n\n/**\n * Indicate whether two objects are effectively the same.\n *\n * @param {array|object|any} item1\n * @param {array|object|any} item2\n *\n * @returns {boolean}\n */\nexport function equivalent(item1, item2) {\n let result;\n if (isObject(item1) && isObject(item2)) {\n const keys1 = Object.keys(item1);\n const keys2 = Object.keys(item2);\n if ((result = equivalent(keys1, keys2))) {\n $.each(item1, (key, value1) => { // continue while equivalent\n return !(result &&= equivalent(value1, item2[key]));\n });\n }\n } else if (Array.isArray(item1) && Array.isArray(item2)) {\n if ((result = (item1.length === item2.length))) {\n const array1 = [...item1].sort();\n const array2 = [...item2].sort();\n $.each(array1, (idx, value1) => { // continue while equivalent\n return !(result &&= equivalent(value1, array2[idx]));\n });\n }\n } else {\n result = (item1 === item2);\n }\n return result;\n}\n\n/**\n * Indicate whether the object has an entry with the given key.\n *\n * @param {object} obj\n * @param {string|number} key\n *\n * @returns {boolean}\n */\nexport function hasKey(obj, key) {\n return !!obj && !!key && Object.hasOwn(obj, key);\n}\n\n/**\n * Remove the **key** entry from **obj** and return its value.\n *\n * @param {object} obj\n * @param {*} key\n *\n * @returns {*|undefined}\n */\nexport function remove(obj, key) {\n let value;\n if (hasKey(obj, key)) {\n value = obj[key];\n delete obj[key];\n }\n return value;\n}\n", "// app/assets/javascripts/shared/decode.js\n//\n// noinspection JSUnusedGlobalSymbols\n\n\nimport { AppDebug } from \"../application/debug\";\nimport { isDefined } from \"./definitions\";\nimport { fromJSON, isObject } from \"./objects\";\n\n\nAppDebug.file(\"shared/decode\");\n\n// ============================================================================\n// Functions - Type conversions\n//\n// These are primarily for *.js.erb files to allow values that will be inserted\n// by ERB pre-processing to be expressed as a string, so that the Javascript\n// has valid syntax even before pre-processing.\n// ============================================================================\n\n/**\n * Convert a string prepared with \"#js\" to an array or object.\n *\n * @param {string} arg\n *\n * @returns {array|object|undefined}\n *\n * @see file:config/boot.rb \"#js\"\n */\nexport function decodeJSON(arg) {\n const func = \"decodeJSON\";\n const str = arg.replaceAll(/\\n/g, \"\\\\n\");\n return fromJSON(str, func, (k, v) => {\n const encoded = (typeof v === \"string\") && v.includes(\"%5C\");\n return encoded ? decodeURIComponent(v).replaceAll(/\\\\\"/g, '\"') : v;\n });\n}\n\n/**\n * Interpret a string as an Array definition.\n *\n * @param {*} arg\n * @param {string} [separator]\n *\n * @returns {array}\n */\nexport function decodeArray(arg, separator = \",\") {\n let s;\n switch (true) {\n case (!arg): return [];\n case Array.isArray(arg): return arg;\n case (typeof arg === \"object\"): return Object.values(arg);\n case (typeof arg !== \"string\"): return [arg];\n case (s = arg.trim()).startsWith(\"[\"): return decodeJSON(s);\n }\n return s.split(separator).map(v => v.trim());\n}\n\n/**\n * Interpret a string as an Object definition.\n *\n * @param {*} arg\n *\n * @returns {object}\n */\nexport function decodeObject(arg) {\n const result = (typeof arg === \"string\") ? decodeJSON(arg) : arg;\n return isObject(result) ? result : {};\n}\n\n/**\n * Interpret a string as a boolean value.\n *\n * @param {*} arg\n *\n * @returns {boolean}\n */\nexport function decodeBoolean(arg) {\n switch (typeof arg) {\n case \"boolean\": return arg;\n case \"string\": return arg.toLowerCase() === \"true\";\n default: return isDefined(arg);\n }\n}\n\n/**\n * Interpret a string as a integer value.\n *\n * @param {*} arg\n *\n * @returns {number}\n */\nexport function decodeInteger(arg) {\n switch (typeof arg) {\n case \"number\": return arg;\n case \"string\": return Math.max(0, parseInt(arg));\n default: return 0;\n }\n}\n", "// app/assets/builds/javascripts-shared-assets.js\n// app/assets/javascripts/shared/assets.js.erb\n//\n// This file holds all of the constants that rely on ERB processing, which\n// allows shared settings to be maintained in the Rails application.\n//\n// The \"emma:assets:erb\" Rake task processes this file through ERB in order to\n// provide current values from the Ruby-on-Rails source code, except for\n// Image_placeholder_asset, which is set through a `