{"version":3,"file":"frappe-web.min.js","sources":["../../../apps/frappe/frappe/public/js/frappe/class.js","../../../apps/frappe/frappe/public/js/frappe/polyfill.js","../../../apps/frappe/frappe/public/js/lib/md5.min.js","../../../apps/frappe/frappe/public/js/frappe/provide.js","../../../apps/frappe/frappe/public/js/frappe/format.js","../../../apps/frappe/frappe/public/js/frappe/utils/datatype.js","../../../apps/frappe/frappe/public/js/frappe/utils/number_format.js","../../../apps/frappe/node_modules/fast-deep-equal/index.js","../../../apps/frappe/frappe/public/js/frappe/utils/utils.js","../../../apps/frappe/frappe/public/js/frappe/utils/common.js","../../../apps/frappe/frappe/public/js/frappe/form/layout.js","../../../apps/frappe/frappe/public/js/frappe/ui/field_group.js","../../../apps/frappe/frappe/public/js/frappe/dom.js","../../../apps/frappe/frappe/public/js/frappe/ui/dialog.js","../../../apps/frappe/frappe/public/js/frappe/ui/messages.js","../../../apps/frappe/frappe/public/js/frappe/translate.js","../../../apps/frappe/frappe/public/js/frappe/utils/pretty_date.js","../../../apps/frappe/frappe/public/js/frappe/microtemplate.js","../../../apps/frappe/frappe/public/js/frappe/query_string.js","../../../apps/frappe/frappe/public/js/frappe/file_uploader/ProgressRing.vue?rollup-plugin-vue=script.js","../../../apps/frappe/frappe/public/js/frappe/file_uploader/ProgressRing.vue","../../../apps/frappe/frappe/public/js/frappe/file_uploader/FilePreview.vue?rollup-plugin-vue=script.js","../../../apps/frappe/frappe/public/js/frappe/file_uploader/FilePreview.vue","../../../apps/frappe/frappe/public/js/frappe/file_uploader/TreeNode.vue?rollup-plugin-vue=script.js","../../../apps/frappe/frappe/public/js/frappe/file_uploader/TreeNode.vue","../../../apps/frappe/frappe/public/js/frappe/file_uploader/FileBrowser.vue?rollup-plugin-vue=script.js","../../../apps/frappe/frappe/public/js/frappe/file_uploader/FileBrowser.vue","../../../apps/frappe/frappe/public/js/frappe/file_uploader/WebLink.vue?rollup-plugin-vue=script.js","../../../apps/frappe/frappe/public/js/frappe/file_uploader/WebLink.vue","../../../apps/frappe/frappe/public/js/integrations/google_drive_picker.js","../../../apps/frappe/frappe/public/js/frappe/file_uploader/FileUploader.vue?rollup-plugin-vue=script.js","../../../apps/frappe/frappe/public/js/frappe/file_uploader/FileUploader.vue","../../../apps/frappe/frappe/public/js/frappe/file_uploader/index.js","../../../apps/frappe/frappe/public/js/frappe/upload.js","../../../apps/frappe/frappe/public/js/frappe/model/meta.js","../../../apps/frappe/frappe/public/js/frappe/model/model.js","../../../apps/frappe/frappe/public/js/frappe/model/perm.js","../../../apps/frappe/node_modules/highlight.js/lib/core.js","../../../apps/frappe/node_modules/highlight.js/lib/languages/javascript.js","../../../apps/frappe/node_modules/highlight.js/lib/languages/python.js","../../../apps/frappe/node_modules/highlight.js/lib/languages/xml.js","../../../apps/frappe/node_modules/highlight.js/lib/languages/django.js","../../../apps/frappe/node_modules/highlight.js/lib/languages/bash.js","../../../apps/frappe/node_modules/highlight.js/lib/languages/css.js","../../../apps/frappe/node_modules/highlight.js/lib/languages/markdown.js","../../../apps/frappe/node_modules/highlight.js/lib/languages/diff.js","../../../apps/frappe/node_modules/highlight.js/lib/languages/json.js","../../../apps/frappe/node_modules/highlight.js/lib/languages/less.js","../../../apps/frappe/node_modules/highlight.js/lib/languages/nginx.js","../../../apps/frappe/node_modules/highlight.js/lib/languages/scss.js","../../../apps/frappe/node_modules/highlight.js/lib/languages/shell.js","../../../apps/frappe/node_modules/highlight.js/lib/languages/sql.js","../../../apps/frappe/frappe/website/js/syntax_highlight.js","../../../apps/frappe/frappe/website/js/website.js","../../../apps/frappe/frappe/public/js/frappe/socketio_client.js"],"sourcesContent":["// Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors\n// MIT License. See license.txt\n\n/*\n\nInheritence \"Class\"\n-------------------\nsee: http://ejohn.org/blog/simple-javascript-inheritance/\nTo subclass, use:\n\n\tvar MyClass = Class.extend({\n\t\tinit: function\n\t})\n\n*/\n// https://stackoverflow.com/a/15052240/5353542\n\n/* Simple JavaScript Inheritance for ES 5.1\n * based on http://ejohn.org/blog/simple-javascript-inheritance/\n * (inspired by base2 and Prototype)\n * MIT Licensed.\n */\n(function(global) {\n\t\"use strict\";\n\tvar fnTest = /xyz/.test(function(){xyz;}) ? /\\b_super\\b/ : /.*/;\n\n\t// The base Class implementation (does nothing)\n\tfunction Class(){}\n\n\t// Create a new Class that inherits from this class\n\tClass.extend = function(props) {\n\t var _super = this.prototype;\n\n\t // Set up the prototype to inherit from the base class\n\t // (but without running the init constructor)\n\t var proto = Object.create(_super);\n\n\t // Copy the properties over onto the new prototype\n\t for (var name in props) {\n\t\t// Check if we're overwriting an existing function\n\t\tproto[name] = typeof props[name] === \"function\" &&\n\t\t typeof _super[name] == \"function\" && fnTest.test(props[name])\n\t\t ? (function(name, fn){\n\t\t\t return function() {\n\t\t\t\tvar tmp = this._super;\n\n\t\t\t\t// Add a new ._super() method that is the same method\n\t\t\t\t// but on the super-class\n\t\t\t\tthis._super = _super[name];\n\n\t\t\t\t// The method only need to be bound temporarily, so we\n\t\t\t\t// remove it when we're done executing\n\t\t\t\tvar ret = fn.apply(this, arguments);\n\t\t\t\tthis._super = tmp;\n\n\t\t\t\treturn ret;\n\t\t\t };\n\t\t\t})(name, props[name])\n\t\t : props[name];\n\t }\n\n\t // The new constructor\n\t var newClass = typeof proto.init === \"function\"\n\t\t? proto.hasOwnProperty(\"init\")\n\t\t ? proto.init // All construction is actually done in the init method\n\t\t : function SubClass(){ _super.init.apply(this, arguments); }\n\t\t: function EmptyClass(){};\n\n\t // Populate our constructed prototype object\n\t newClass.prototype = proto;\n\n\t // Enforce the constructor to be what we expect\n\t proto.constructor = newClass;\n\n\t // And make this class extendable\n\t newClass.extend = Class.extend;\n\n\t return newClass;\n\t};\n\n\t// export\n\tglobal.Class = Class;\n })(this);","// String.prototype.includes polyfill\n// https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Global_Objects/String/includes\nif (!String.prototype.includes) {\n\tString.prototype.includes = function(search, start) {\n\t\t'use strict';\n\t\tif (typeof start !== 'number') {\n\t\t\tstart = 0;\n\t\t}\n\t\tif (start + search.length > this.length) {\n\t\t\treturn false;\n\t\t} else {\n\t\t\treturn this.indexOf(search, start) !== -1;\n\t\t}\n\t};\n}\n// Array.prototype.includes polyfill\n// https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Global_Objects/Array/includes\nif (!Array.prototype.includes) {\n\tObject.defineProperty(Array.prototype, 'includes', {\n\t\tvalue: function(searchElement, fromIndex) {\n\t\t\tif (this == null) {\n\t\t\t\tthrow new TypeError('\"this\" is null or not defined');\n\t\t\t}\n\t\t\tvar o = Object(this);\n\t\t\tvar len = o.length >>> 0;\n\t\t\tif (len === 0) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t\tvar n = fromIndex | 0;\n\t\t\tvar k = Math.max(n >= 0 ? n : len - Math.abs(n), 0);\n\t\t\twhile (k < len) {\n\t\t\t\tif (o[k] === searchElement) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t\tk++;\n\t\t\t}\n\t\t\treturn false;\n\t\t}\n\t});\n}\n\n\nif (typeof String.prototype.trimLeft !== \"function\") {\n\tString.prototype.trimLeft = function() {\n\t\treturn this.replace(/^\\s+/, \"\");\n\t};\n}\nif (typeof String.prototype.trimRight !== \"function\") {\n\tString.prototype.trimRight = function() {\n\t\treturn this.replace(/\\s+$/, \"\");\n\t};\n}\n\n// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/assign\nif (typeof Object.assign != 'function') {\n\t// Must be writable: true, enumerable: false, configurable: true\n\tObject.defineProperty(Object, \"assign\", {\n\t\tvalue: function assign(target) { // .length of function is 2\n\t\t\t'use strict';\n\t\t\tif (target == null) { // TypeError if undefined or null\n\t\t\t\tthrow new TypeError('Cannot convert undefined or null to object');\n\t\t\t}\n\n\t\t\tvar to = Object(target);\n\n\t\t\tfor (var index = 1; index < arguments.length; index++) {\n\t\t\t\tvar nextSource = arguments[index];\n\n\t\t\t\tif (nextSource != null) { // Skip over if undefined or null\n\t\t\t\t\tfor (var nextKey in nextSource) {\n\t\t\t\t\t\t// Avoid bugs when hasOwnProperty is shadowed\n\t\t\t\t\t\tif (Object.prototype.hasOwnProperty.call(nextSource, nextKey)) {\n\t\t\t\t\t\t\tto[nextKey] = nextSource[nextKey];\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t\treturn to;\n\t\t},\n\t\twritable: true,\n\t\tconfigurable: true\n\t});\n}\n","!function(a){\"use strict\";function b(a,b){var c=(65535&a)+(65535&b),d=(a>>16)+(b>>16)+(c>>16);return d<<16|65535&c}function c(a,b){return a<>>32-b}function d(a,d,e,f,g,h){return b(c(b(b(d,a),b(f,h)),g),e)}function e(a,b,c,e,f,g,h){return d(b&c|~b&e,a,b,f,g,h)}function f(a,b,c,e,f,g,h){return d(b&e|c&~e,a,b,f,g,h)}function g(a,b,c,e,f,g,h){return d(b^c^e,a,b,f,g,h)}function h(a,b,c,e,f,g,h){return d(c^(b|~e),a,b,f,g,h)}function i(a,c){a[c>>5]|=128<>>9<<4)+14]=c;var d,i,j,k,l,m=1732584193,n=-271733879,o=-1732584194,p=271733878;for(d=0;d>5]>>>b%32&255);return c}function k(a){var b,c=[];for(c[(a.length>>2)-1]=void 0,b=0;b>5]|=(255&a.charCodeAt(b/8))<16&&(e=i(e,8*a.length)),c=0;16>c;c+=1)f[c]=909522486^e[c],g[c]=1549556828^e[c];return d=i(f.concat(k(b)),512+8*b.length),j(i(g.concat(d),640))}function n(a){var b,c,d=\"0123456789abcdef\",e=\"\";for(c=0;c>>4&15)+d.charAt(15&b);return e}function o(a){return unescape(encodeURIComponent(a))}function p(a){return l(o(a))}function q(a){return n(p(a))}function r(a,b){return m(o(a),o(b))}function s(a,b){return n(r(a,b))}function t(a,b,c){return b?c?r(b,a):s(b,a):c?p(a):q(a)}\"function\"==typeof define&&define.amd?define(function(){return t}):a.md5=t}(this);","// Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors\n// MIT License. See license.txt\n\n// provide a namespace\nif(!window.frappe)\n\twindow.frappe = {};\n\nfrappe.provide = function(namespace) {\n\t// docs: create a namespace //\n\tvar nsl = namespace.split('.');\n\tvar parent = window;\n\tfor(var i=0; i= 0; i--) {\n\t\t\tvar l = replace_all(str, info.group_sep, \"\").length;\n\t\t\tif (format == \"#,##,###.##\" && str.indexOf(\",\") != -1) { // INR\n\t\t\t\tgroup_position = 2;\n\t\t\t\tl += 1;\n\t\t\t}\n\n\t\t\tstr += integer.charAt(i);\n\n\t\t\tif (l && !((l + 1) % group_position) && i != 0) {\n\t\t\t\tstr += info.group_sep;\n\t\t\t}\n\t\t}\n\t\tpart[0] = str.split(\"\").reverse().join(\"\");\n\t}\n\tif (part[0] + \"\" == \"\") {\n\t\tpart[0] = \"0\";\n\t}\n\n\t// join decimal\n\tpart[1] = (part[1] && info.decimal_str) ? (info.decimal_str + part[1]) : \"\";\n\n\t// join\n\treturn (is_negative ? \"-\" : \"\") + part[0] + part[1];\n};\n\nfunction format_currency(v, currency, decimals) {\n\tvar format = get_number_format(currency);\n\tvar symbol = get_currency_symbol(currency);\n\tif(decimals === undefined) {\n\t\tdecimals = frappe.boot.sysdefaults.currency_precision || null;\n\t}\n\n\tif (symbol)\n\t\treturn symbol + \" \" + format_number(v, format, decimals);\n\telse\n\t\treturn format_number(v, format, decimals);\n}\n\nfunction get_currency_symbol(currency) {\n\tif (frappe.boot) {\n\t\tif (frappe.boot.sysdefaults && frappe.boot.sysdefaults.hide_currency_symbol == \"Yes\")\n\t\t\treturn null;\n\n\t\tif (!currency)\n\t\t\tcurrency = frappe.boot.sysdefaults.currency;\n\n\t\treturn frappe.model.get_value(\":Currency\", currency, \"symbol\") || currency;\n\t} else {\n\t\t// load in template\n\t\treturn frappe.currency_symbols[currency];\n\t}\n}\n\nfunction get_number_format(currency) {\n\treturn (frappe.boot && frappe.boot.sysdefaults && frappe.boot.sysdefaults.number_format) || \"#,###.##\";\n}\n\nfunction get_number_format_info(format) {\n\tvar info = frappe.number_format_info[format];\n\n\tif (!info) {\n\t\tinfo = { decimal_str: \".\", group_sep: \",\" };\n\t}\n\n\t// get the precision from the number format\n\tinfo.precision = format.split(info.decimal_str).slice(1)[0].length;\n\n\treturn info;\n}\n\nfunction _round(num, precision) {\n\tvar is_negative = num < 0 ? true : false;\n\tvar d = cint(precision);\n\tvar m = Math.pow(10, d);\n\tvar n = +(d ? Math.abs(num) * m : Math.abs(num)).toFixed(8); // Avoid rounding errors\n\tvar i = Math.floor(n), f = n - i;\n\tvar r = ((!precision && f == 0.5) ? ((i % 2 == 0) ? i : i + 1) : Math.round(n));\n\tr = d ? r / m : r;\n\treturn is_negative ? -r : r;\n\n}\n\nfunction roundNumber(num, precision) {\n\t// backward compatibility\n\treturn _round(num, precision);\n}\n\nfunction precision(fieldname, doc) {\n\tif (cur_frm) {\n\t\tif (!doc) doc = cur_frm.doc;\n\t\tvar df = frappe.meta.get_docfield(doc.doctype, fieldname, doc.parent || doc.name);\n\t\tif (!df) console.log(fieldname + \": could not find docfield in method precision()\");\n\t\treturn frappe.meta.get_field_precision(df, doc);\n\t} else {\n\t\treturn frappe.boot.sysdefaults.float_precision\n\t}\n}\n\nfunction in_list(list, item) {\n\treturn list.includes(item);\n}\n\nfunction remainder(numerator, denominator, precision) {\n\tprecision = cint(precision);\n\tvar multiplier = Math.pow(10, precision);\n\tif (precision) {\n\t\tvar _remainder = ((numerator * multiplier) % (denominator * multiplier)) / multiplier;\n\t} else {\n\t\tvar _remainder = numerator % denominator;\n\t}\n\n\treturn flt(_remainder, precision);\n}\n\nfunction round_based_on_smallest_currency_fraction(value, currency, precision) {\n\tvar smallest_currency_fraction_value = flt(frappe.model.get_value(\":Currency\",\n\t\tcurrency, \"smallest_currency_fraction_value\"))\n\n\tif (smallest_currency_fraction_value) {\n\t\tvar remainder_val = remainder(value, smallest_currency_fraction_value, precision);\n\t\tif (remainder_val > (smallest_currency_fraction_value / 2)) {\n\t\t\tvalue += (smallest_currency_fraction_value - remainder_val);\n\t\t} else {\n\t\t\tvalue -= remainder_val;\n\t\t}\n\t} else {\n\t\tvalue = _round(value);\n\t}\n\treturn value;\n}\n\nfunction fmt_money(v, format){\n\t// deprecated!\n\t// for backward compatibility\n\treturn format_currency(v, format);\n}\n\n\nObject.assign(window, {\n\tflt,\n\tcint,\n\tstrip_number_groups,\n\tformat_currency,\n\tfmt_money,\n\tget_currency_symbol,\n\tget_number_format,\n\tget_number_format_info,\n\t_round,\n\troundNumber,\n\tprecision,\n\tremainder,\n\tround_based_on_smallest_currency_fraction,\n\tin_list\n});","'use strict';\n\nvar isArray = Array.isArray;\nvar keyList = Object.keys;\nvar hasProp = Object.prototype.hasOwnProperty;\n\nmodule.exports = function equal(a, b) {\n if (a === b) return true;\n\n if (a && b && typeof a == 'object' && typeof b == 'object') {\n var arrA = isArray(a)\n , arrB = isArray(b)\n , i\n , length\n , key;\n\n if (arrA && arrB) {\n length = a.length;\n if (length != b.length) return false;\n for (i = length; i-- !== 0;)\n if (!equal(a[i], b[i])) return false;\n return true;\n }\n\n if (arrA != arrB) return false;\n\n var dateA = a instanceof Date\n , dateB = b instanceof Date;\n if (dateA != dateB) return false;\n if (dateA && dateB) return a.getTime() == b.getTime();\n\n var regexpA = a instanceof RegExp\n , regexpB = b instanceof RegExp;\n if (regexpA != regexpB) return false;\n if (regexpA && regexpB) return a.toString() == b.toString();\n\n var keys = keyList(a);\n length = keys.length;\n\n if (length !== keyList(b).length)\n return false;\n\n for (i = length; i-- !== 0;)\n if (!hasProp.call(b, keys[i])) return false;\n\n for (i = length; i-- !== 0;) {\n key = keys[i];\n if (!equal(a[key], b[key])) return false;\n }\n\n return true;\n }\n\n return a!==a && b!==b;\n};\n","// Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors\n// MIT License. See license.txt\n\nimport deep_equal from \"fast-deep-equal\";\n\nfrappe.provide(\"frappe.utils\");\n\n// Array de duplicate\nif (!Array.prototype.uniqBy) {\n\tObject.defineProperty(Array.prototype, 'uniqBy', {\n\t\tvalue: function (key) {\n\t\t\tvar seen = {};\n\t\t\treturn this.filter(function (item) {\n\t\t\t\tvar k = key(item);\n\t\t\t\treturn k in seen ? false : (seen[k] = true);\n\t\t\t});\n\t\t}\n\t});\n\tObject.defineProperty(Array.prototype, 'move', {\n\t\tvalue: function(from, to) {\n\t\t\tthis.splice(to, 0, this.splice(from, 1)[0]);\n\t\t}\n\t});\n}\n\n// Python's dict.setdefault ported for JS objects\nObject.defineProperty(Object.prototype, \"setDefault\", {\n\tvalue: function(key, default_value) {\n\t\tif (!(key in this)) this[key] = default_value;\n\t\treturn this[key];\n\t},\n\twritable: true\n});\n\n// Pluralize\nString.prototype.plural = function(revert) {\n\tconst plural = {\n\t\t\"(quiz)$\": \"$1zes\",\n\t\t\"^(ox)$\": \"$1en\",\n\t\t\"([m|l])ouse$\": \"$1ice\",\n\t\t\"(matr|vert|ind)ix|ex$\": \"$1ices\",\n\t\t\"(x|ch|ss|sh)$\": \"$1es\",\n\t\t\"([^aeiouy]|qu)y$\": \"$1ies\",\n\t\t\"(hive)$\": \"$1s\",\n\t\t\"(?:([^f])fe|([lr])f)$\": \"$1$2ves\",\n\t\t\"(shea|lea|loa|thie)f$\": \"$1ves\",\n\t\tsis$: \"ses\",\n\t\t\"([ti])um$\": \"$1a\",\n\t\t\"(tomat|potat|ech|her|vet)o$\": \"$1oes\",\n\t\t\"(bu)s$\": \"$1ses\",\n\t\t\"(alias)$\": \"$1es\",\n\t\t\"(octop)us$\": \"$1i\",\n\t\t\"(ax|test)is$\": \"$1es\",\n\t\t\"(us)$\": \"$1es\",\n\t\t\"([^s]+)$\": \"$1s\",\n\t};\n\n\tconst singular = {\n\t\t\"(quiz)zes$\": \"$1\",\n\t\t\"(matr)ices$\": \"$1ix\",\n\t\t\"(vert|ind)ices$\": \"$1ex\",\n\t\t\"^(ox)en$\": \"$1\",\n\t\t\"(alias)es$\": \"$1\",\n\t\t\"(octop|vir)i$\": \"$1us\",\n\t\t\"(cris|ax|test)es$\": \"$1is\",\n\t\t\"(shoe)s$\": \"$1\",\n\t\t\"(o)es$\": \"$1\",\n\t\t\"(bus)es$\": \"$1\",\n\t\t\"([m|l])ice$\": \"$1ouse\",\n\t\t\"(x|ch|ss|sh)es$\": \"$1\",\n\t\t\"(m)ovies$\": \"$1ovie\",\n\t\t\"(s)eries$\": \"$1eries\",\n\t\t\"([^aeiouy]|qu)ies$\": \"$1y\",\n\t\t\"([lr])ves$\": \"$1f\",\n\t\t\"(tive)s$\": \"$1\",\n\t\t\"(hive)s$\": \"$1\",\n\t\t\"(li|wi|kni)ves$\": \"$1fe\",\n\t\t\"(shea|loa|lea|thie)ves$\": \"$1f\",\n\t\t\"(^analy)ses$\": \"$1sis\",\n\t\t\"((a)naly|(b)a|(d)iagno|(p)arenthe|(p)rogno|(s)ynop|(t)he)ses$\":\n\t\t\t\"$1$2sis\",\n\t\t\"([ti])a$\": \"$1um\",\n\t\t\"(n)ews$\": \"$1ews\",\n\t\t\"(h|bl)ouses$\": \"$1ouse\",\n\t\t\"(corpse)s$\": \"$1\",\n\t\t\"(us)es$\": \"$1\",\n\t\ts$: \"\",\n\t};\n\n\tconst irregular = {\n\t\tmove: \"moves\",\n\t\tfoot: \"feet\",\n\t\tgoose: \"geese\",\n\t\tsex: \"sexes\",\n\t\tchild: \"children\",\n\t\tman: \"men\",\n\t\ttooth: \"teeth\",\n\t\tperson: \"people\",\n\t};\n\n\tconst uncountable = [\n\t\t\"sheep\",\n\t\t\"fish\",\n\t\t\"deer\",\n\t\t\"moose\",\n\t\t\"series\",\n\t\t\"species\",\n\t\t\"money\",\n\t\t\"rice\",\n\t\t\"information\",\n\t\t\"equipment\",\n\t];\n\n\t// save some time in the case that singular and plural are the same\n\tif (uncountable.indexOf(this.toLowerCase()) >= 0) return this;\n\n\t// check for irregular forms\n\tlet word;\n\tlet pattern;\n\tlet replace;\n\tfor (word in irregular) {\n\t\tif (revert) {\n\t\t\tpattern = new RegExp(irregular[word] + \"$\", \"i\");\n\t\t\treplace = word;\n\t\t} else {\n\t\t\tpattern = new RegExp(word + \"$\", \"i\");\n\t\t\treplace = irregular[word];\n\t\t}\n\t\tif (pattern.test(this)) return this.replace(pattern, replace);\n\t}\n\n\tlet array;\n\tif (revert) array = singular;\n\telse array = plural;\n\n\t// check for matches using regular expressions\n\tlet reg;\n\tfor (reg in array) {\n\t\tpattern = new RegExp(reg, \"i\");\n\n\t\tif (pattern.test(this)) return this.replace(pattern, array[reg]);\n\t}\n\n\treturn this;\n};\n\nObject.assign(frappe.utils, {\n\tget_random: function(len) {\n\t\tvar text = \"\";\n\t\tvar possible = \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\";\n\n\t\tfor ( var i=0; i < len; i++ )\n\t\t\ttext += possible.charAt(Math.floor(Math.random() * possible.length));\n\n\t\treturn text;\n\t},\n\tget_file_link: function(filename) {\n\t\tfilename = cstr(filename);\n\t\tif (frappe.utils.is_url(filename)) {\n\t\t\treturn filename;\n\t\t} else if (filename.indexOf(\"/\")===-1) {\n\t\t\treturn \"files/\" + filename;\n\t\t} else {\n\t\t\treturn filename;\n\t\t}\n\t},\n\treplace_newlines(t) {\n\t\treturn t?t.replace(/\\n/g, '
'):'';\n\t},\n\tis_html: function(txt) {\n\t\tif (!txt) return false;\n\n\t\tif (txt.indexOf(\"
\")==-1 && txt.indexOf(\"= 768;\n\t},\n\tis_md: function() {\n\t\treturn $(document).width() < 1199 && $(document).width() >= 991;\n\t},\n\tis_json: function(str) {\n\t\ttry {\n\t\t\tJSON.parse(str);\n\t\t} catch (e) {\n\t\t\treturn false;\n\t\t}\n\t\treturn true;\n\t},\n\tstrip_whitespace: function(html) {\n\t\treturn (html || \"\").replace(/

\\s*<\\/p>/g, \"\").replace(/
(\\s*
\\s*)+/g, \"

\");\n\t},\n\tencode_tags: function(html) {\n\t\tvar tagsToReplace = {\n\t\t\t'&': '&',\n\t\t\t'<': '<',\n\t\t\t'>': '>'\n\t\t};\n\n\t\tfunction replaceTag(tag) {\n\t\t\treturn tagsToReplace[tag] || tag;\n\t\t}\n\n\t\treturn html.replace(/[&<>]/g, replaceTag);\n\t},\n\tstrip_original_content: function(txt) {\n\t\tvar out = [],\n\t\t\tpart = [],\n\t\t\tnewline = txt.indexOf(\"
\")===-1 ? \"\\n\" : \"
\";\n\n\t\t$.each(txt.split(newline), function(i, t) {\n\t\t\tvar tt = strip(t);\n\t\t\tif (tt && (tt.substr(0, 1)===\">\" || tt.substr(0, 4)===\">\")) {\n\t\t\t\tpart.push(t);\n\t\t\t} else {\n\t\t\t\tout.concat(part);\n\t\t\t\tout.push(t);\n\t\t\t\tpart = [];\n\t\t\t}\n\t\t});\n\t\treturn out.join(newline);\n\t},\n\n\n\tescape_html: function(txt) {\n\t\tlet escape_html_mapping = {\n\t\t\t'&': '&',\n\t\t\t'<': '<',\n\t\t\t'>': '>',\n\t\t\t'\"': '"',\n\t\t\t\"'\": ''',\n\t\t\t'/': '/',\n\t\t\t'`': '`',\n\t\t\t'=': '='\n\t\t};\n\n\t\treturn String(txt).replace(/[&<>\"'`=/]/g, function(char) {\n\t\t\treturn escape_html_mapping[char];\n\t\t});\n\t},\n\n\thtml2text: function(html) {\n\t\tlet d = document.createElement('div');\n\t\td.innerHTML = html;\n\t\treturn d.textContent;\n\t},\n\n\tis_url: function(txt) {\n\t\treturn txt.toLowerCase().substr(0, 7)=='http://'\n\t\t\t|| txt.toLowerCase().substr(0, 8)=='https://';\n\t},\n\tto_title_case: function(string, with_space=false) {\n\t\tlet titlecased_string = string.toLowerCase().replace(/(?:^|[\\s-/])\\w/g, function(match) {\n\t\t\treturn match.toUpperCase();\n\t\t});\n\n\t\tlet replace_with = with_space ? ' ' : '';\n\n\t\treturn titlecased_string.replace(/-|_/g, replace_with);\n\t},\n\ttoggle_blockquote: function(txt) {\n\t\tif (!txt) return txt;\n\n\t\tvar content = $(\"

\").html(txt);\n\t\tcontent.find(\"blockquote\").parent(\"blockquote\").addClass(\"hidden\")\n\t\t\t.before('

\\\n\t\t\t\t\t• • • \\\n\t\t\t\t

');\n\t\treturn content.html();\n\t},\n\tscroll_to: function(element, animate=true, additional_offset, element_to_be_scrolled) {\n\t\telement_to_be_scrolled = element_to_be_scrolled || $(\"html, body\");\n\t\tlet scroll_top = 0;\n\t\tif (element) {\n\t\t\t// If a number is passed, just subtract the offset,\n\t\t\t// otherwise calculate scroll position from element\n\t\t\tscroll_top = typeof element == \"number\"\n\t\t\t\t? element - cint(additional_offset)\n\t\t\t\t: this.get_scroll_position(element, additional_offset);\n\t\t}\n\n\t\tif (scroll_top < 0) {\n\t\t\tscroll_top = 0;\n\t\t}\n\n\t\t// already there\n\t\tif (scroll_top == element_to_be_scrolled.scrollTop()) {\n\t\t\treturn;\n\t\t}\n\n\t\tif (animate) {\n\t\t\telement_to_be_scrolled.animate({ scrollTop: scroll_top });\n\t\t} else {\n\t\t\telement_to_be_scrolled.scrollTop(scroll_top);\n\t\t}\n\n\t},\n\tget_scroll_position: function(element, additional_offset) {\n\t\tlet header_offset = $(\".navbar\").height() + $(\".page-head:visible\").height();\n\t\tlet scroll_top = $(element).offset().top - header_offset - cint(additional_offset);\n\t\treturn scroll_top;\n\t},\n\tfilter_dict: function(dict, filters) {\n\t\tvar ret = [];\n\t\tif (typeof filters=='string') {\n\t\t\treturn [dict[filters]];\n\t\t}\n\t\t$.each(dict, function(i, d) {\n\t\t\tfor (var key in filters) {\n\t\t\t\tif ($.isArray(filters[key])) {\n\t\t\t\t\tif (filters[key][0]==\"in\") {\n\t\t\t\t\t\tif (filters[key][1].indexOf(d[key])==-1)\n\t\t\t\t\t\t\treturn;\n\t\t\t\t\t} else if (filters[key][0]==\"not in\") {\n\t\t\t\t\t\tif (filters[key][1].indexOf(d[key])!=-1)\n\t\t\t\t\t\t\treturn;\n\t\t\t\t\t} else if (filters[key][0]==\"<\") {\n\t\t\t\t\t\tif (!(d[key] < filters[key])) return;\n\t\t\t\t\t} else if (filters[key][0]==\"<=\") {\n\t\t\t\t\t\tif (!(d[key] <= filters[key])) return;\n\t\t\t\t\t} else if (filters[key][0]==\">\") {\n\t\t\t\t\t\tif (!(d[key] > filters[key])) return;\n\t\t\t\t\t} else if (filters[key][0]==\">=\") {\n\t\t\t\t\t\tif (!(d[key] >= filters[key])) return;\n\t\t\t\t\t}\n\t\t\t\t} else {\n\t\t\t\t\tif (d[key]!=filters[key]) return;\n\t\t\t\t}\n\t\t\t}\n\t\t\tret.push(d);\n\t\t});\n\t\treturn ret;\n\t},\n\tcomma_or: function(list) {\n\t\treturn frappe.utils.comma_sep(list, \" \" + __(\"or\") + \" \");\n\t},\n\tcomma_and: function(list) {\n\t\treturn frappe.utils.comma_sep(list, \" \" + __(\"and\") + \" \");\n\t},\n\tcomma_sep: function(list, sep) {\n\t\tif (list instanceof Array) {\n\t\t\tif (list.length==0) {\n\t\t\t\treturn \"\";\n\t\t\t} else if (list.length==1) {\n\t\t\t\treturn list[0];\n\t\t\t} else {\n\t\t\t\treturn list.slice(0, list.length-1).join(\", \") + sep + list.slice(-1)[0];\n\t\t\t}\n\t\t} else {\n\t\t\treturn list;\n\t\t}\n\t},\n\tset_footnote: function(footnote_area, wrapper, txt) {\n\t\tif (!footnote_area) {\n\t\t\tfootnote_area = $('
')\n\t\t\t\t.appendTo(wrapper);\n\t\t}\n\n\t\tif (txt) {\n\t\t\tfootnote_area.html(txt);\n\t\t} else {\n\t\t\tfootnote_area.remove();\n\t\t\tfootnote_area = null;\n\t\t}\n\t\treturn footnote_area;\n\t},\n\tget_args_dict_from_url: function(txt) {\n\t\tvar args = {};\n\t\t$.each(decodeURIComponent(txt).split(\"&\"), function(i, arg) {\n\t\t\targ = arg.split(\"=\");\n\t\t\targs[arg[0]] = arg[1];\n\t\t});\n\t\treturn args;\n\t},\n\tget_url_from_dict: function(args) {\n\t\treturn $.map(args, function(val, key) {\n\t\t\tif (val!==null)\n\t\t\t\treturn encodeURIComponent(key)+\"=\"+encodeURIComponent(val);\n\t\t\telse\n\t\t\t\treturn null;\n\t\t}).join(\"&\") || \"\";\n\t},\n\tvalidate_type: function ( val, type ) {\n\t\t// from https://github.com/guillaumepotier/Parsley.js/blob/master/parsley.js#L81\n\t\tvar regExp;\n\n\t\tswitch ( type ) {\n\t\t\tcase \"phone\":\n\t\t\t\tregExp = /^([0-9 +_\\-,.*#()]){1,20}$/;\n\t\t\t\tbreak;\n\t\t\tcase \"name\":\n\t\t\t\tregExp = /^[\\w][\\w'-]*([ \\w][\\w'-]+)*$/;\n\t\t\t\tbreak;\n\t\t\tcase \"number\":\n\t\t\t\tregExp = /^-?(?:\\d+|\\d{1,3}(?:,\\d{3})+)?(?:\\.\\d+)?$/;\n\t\t\t\tbreak;\n\t\t\tcase \"digits\":\n\t\t\t\tregExp = /^\\d+$/;\n\t\t\t\tbreak;\n\t\t\tcase \"alphanum\":\n\t\t\t\tregExp = /^\\w+$/;\n\t\t\t\tbreak;\n\t\t\tcase \"email\":\n\t\t\t\t// from https://emailregex.com/\n\t\t\t\tregExp = /^(([^<>()[\\]\\\\.,;:\\s@\"]+(\\.[^<>()[\\]\\\\.,;:\\s@\"]+)*)|(\".+\"))@((\\[[0-9]{1,3}\\.[0-9]{1,3}\\.[0-9]{1,3}\\.[0-9]{1,3}])|(([a-zA-Z\\-0-9]+\\.)+[a-zA-Z]{2,}))$/;\n\t\t\t\tbreak;\n\t\t\tcase \"url\":\n\t\t\t\tregExp = /^((([A-Za-z0-9.+-]+:(?:\\/\\/)?)(?:[-;:&=\\+\\,\\w]@)?[A-Za-z0-9.-]+(:[0-9]+)?|(?:www.|[-;:&=\\+\\$,\\w]+@)[A-Za-z0-9.-]+)((?:\\/[\\+~%\\/.\\w-_]*)?\\??(?:[-\\+=&;%@.\\w_]*)#?(?:[\\w]*))?)$/i;\n\t\t\t\tbreak;\n\t\t\tcase \"dateIso\":\n\t\t\t\tregExp = /^(\\d{4})\\D?(0[1-9]|1[0-2])\\D?([12]\\d|0[1-9]|3[01])$/;\n\t\t\t\tbreak;\n\t\t\tdefault:\n\t\t\t\treturn false;\n\t\t}\n\n\t\t// test regExp if not null\n\t\treturn '' !== val ? regExp.test( val ) : false;\n\t},\n\tguess_style: function(text, default_style, _colour) {\n\t\tvar style = default_style || \"default\";\n\t\tvar colour = \"gray\";\n\t\tif (text) {\n\t\t\tif (has_words([\"Pending\", \"Review\", \"Medium\", \"Not Approved\"], text)) {\n\t\t\t\tstyle = \"warning\";\n\t\t\t\tcolour = \"orange\";\n\t\t\t} else if (has_words([\"Open\", \"Urgent\", \"High\", \"Failed\", \"Rejected\", \"Error\"], text)) {\n\t\t\t\tstyle = \"danger\";\n\t\t\t\tcolour = \"red\";\n\t\t\t} else if (has_words([\"Closed\", \"Finished\", \"Converted\", \"Completed\", \"Complete\", \"Confirmed\",\n\t\t\t\t\"Approved\", \"Yes\", \"Active\", \"Available\", \"Paid\", \"Success\"], text)) {\n\t\t\t\tstyle = \"success\";\n\t\t\t\tcolour = \"green\";\n\t\t\t} else if (has_words([\"Submitted\"], text)) {\n\t\t\t\tstyle = \"info\";\n\t\t\t\tcolour = \"blue\";\n\t\t\t}\n\t\t}\n\t\treturn _colour ? colour : style;\n\t},\n\n\tguess_colour: function(text) {\n\t\treturn frappe.utils.guess_style(text, null, true);\n\t},\n\n\tget_indicator_color: function(state) {\n\t\treturn frappe.db.get_list('Workflow State', {filters: {name: state}, fields: ['name', 'style']}).then(res => {\n\t\t\tconst state = res[0];\n\t\t\tif (!state.style) {\n\t\t\t\treturn frappe.utils.guess_colour(state.name);\n\t\t\t}\n\t\t\tconst style = state.style;\n\t\t\tconst colour_map = {\n\t\t\t\t\"Success\": \"green\",\n\t\t\t\t\"Warning\": \"orange\",\n\t\t\t\t\"Danger\": \"red\",\n\t\t\t\t\"Primary\": \"blue\",\n\t\t\t};\n\n\t\t\treturn colour_map[style];\n\t\t});\n\n\t},\n\n\tsort: function(list, key, compare_type, reverse) {\n\t\tif (!list || list.length < 2)\n\t\t\treturn list || [];\n\n\t\tvar sort_fn = {\n\t\t\t\"string\": function(a, b) {\n\t\t\t\treturn cstr(a[key]).localeCompare(cstr(b[key]));\n\t\t\t},\n\t\t\t\"number\": function(a, b) {\n\t\t\t\treturn flt(a[key]) - flt(b[key]);\n\t\t\t}\n\t\t};\n\n\t\tif (!compare_type)\n\t\t\tcompare_type = typeof list[0][key]===\"string\" ? \"string\" : \"number\";\n\n\t\tlist.sort(sort_fn[compare_type]);\n\n\t\tif (reverse) {\n\t\t\tlist.reverse();\n\t\t}\n\n\t\treturn list;\n\t},\n\n\tunique: function(list) {\n\t\tvar dict = {},\n\t\t\tarr = [];\n\t\tfor (var i=0, l=list.length; i < l; i++) {\n\t\t\tif (!(list[i] in dict)) {\n\t\t\t\tdict[list[i]] = null;\n\t\t\t\tarr.push(list[i]);\n\t\t\t}\n\t\t}\n\t\treturn arr;\n\t},\n\n\tremove_nulls: function(list) {\n\t\tvar new_list = [];\n\t\tfor (var i=0, l=list.length; i < l; i++) {\n\t\t\tif (!is_null(list[i])) {\n\t\t\t\tnew_list.push(list[i]);\n\t\t\t}\n\t\t}\n\t\treturn new_list;\n\t},\n\n\tall: function(lst) {\n\t\tfor (var i=0, l=lst.length; i b[bi] ) {\n\t\t\t\tbi++;\n\t\t\t} else {\n\t\t\t\t/* they're equal */\n\t\t\t\tresult.push(a[ai]);\n\t\t\t\tai++;\n\t\t\t\tbi++;\n\t\t\t}\n\t\t}\n\n\t\treturn result;\n\t},\n\n\tresize_image: function(reader, callback, max_width, max_height) {\n\t\tvar tempImg = new Image();\n\t\tif (!max_width) max_width = 600;\n\t\tif (!max_height) max_height = 400;\n\t\ttempImg.src = reader.result;\n\n\t\ttempImg.onload = function() {\n\t\t\tvar tempW = tempImg.width;\n\t\t\tvar tempH = tempImg.height;\n\t\t\tif (tempW > tempH) {\n\t\t\t\tif (tempW > max_width) {\n\t\t\t\t\ttempH *= max_width / tempW;\n\t\t\t\t\ttempW = max_width;\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tif (tempH > max_height) {\n\t\t\t\t\ttempW *= max_height / tempH;\n\t\t\t\t\ttempH = max_height;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tvar canvas = document.createElement('canvas');\n\t\t\tcanvas.width = tempW;\n\t\t\tcanvas.height = tempH;\n\t\t\tvar ctx = canvas.getContext(\"2d\");\n\t\t\tctx.drawImage(this, 0, 0, tempW, tempH);\n\t\t\tvar dataURL = canvas.toDataURL(\"image/jpeg\");\n\t\t\tsetTimeout(function() {\n\t\t\t\tcallback(dataURL);\n\t\t\t}, 10 );\n\t\t};\n\t},\n\n\tcsv_to_array: function (strData, strDelimiter) {\n\t\t// Check to see if the delimiter is defined. If not,\n\t\t// then default to comma.\n\t\tstrDelimiter = (strDelimiter || \",\");\n\n\t\t// Create a regular expression to parse the CSV values.\n\t\tvar objPattern = new RegExp(\n\t\t\t(\n\t\t\t\t// Delimiters.\n\t\t\t\t\"(\\\\\" + strDelimiter + \"|\\\\r?\\\\n|\\\\r|^)\" +\n\n\t\t\t\t// Quoted fields.\n\t\t\t\t\"(?:\\\"([^\\\"]*(?:\\\"\\\"[^\\\"]*)*)\\\"|\" +\n\n\t\t\t\t// Standard fields.\n\t\t\t\t\"([^\\\"\\\\\" + strDelimiter + \"\\\\r\\\\n]*))\"\n\t\t\t),\n\t\t\t\"gi\"\n\t\t);\n\n\n\t\t// Create an array to hold our data. Give the array\n\t\t// a default empty first row.\n\t\tvar arrData = [[]];\n\n\t\t// Create an array to hold our individual pattern\n\t\t// matching groups.\n\t\tvar arrMatches = null;\n\n\n\t\t// Keep looping over the regular expression matches\n\t\t// until we can no longer find a match.\n\t\twhile ((arrMatches = objPattern.exec( strData ))) {\n\n\t\t\t// Get the delimiter that was found.\n\t\t\tvar strMatchedDelimiter = arrMatches[ 1 ];\n\n\t\t\t// Check to see if the given delimiter has a length\n\t\t\t// (is not the start of string) and if it matches\n\t\t\t// field delimiter. If id does not, then we know\n\t\t\t// that this delimiter is a row delimiter.\n\t\t\tif (\n\t\t\t\tstrMatchedDelimiter.length &&\n\t\t\t\tstrMatchedDelimiter !== strDelimiter\n\t\t\t) {\n\n\t\t\t\t// Since we have reached a new row of data,\n\t\t\t\t// add an empty row to our data array.\n\t\t\t\tarrData.push( [] );\n\n\t\t\t}\n\n\t\t\tvar strMatchedValue;\n\n\t\t\t// Now that we have our delimiter out of the way,\n\t\t\t// let's check to see which kind of value we\n\t\t\t// captured (quoted or unquoted).\n\t\t\tif (arrMatches[ 2 ]) {\n\n\t\t\t\t// We found a quoted value. When we capture\n\t\t\t\t// this value, unescape any double quotes.\n\t\t\t\tstrMatchedValue = arrMatches[ 2 ].replace(\n\t\t\t\t\tnew RegExp( \"\\\"\\\"\", \"g\" ),\n\t\t\t\t\t\"\\\"\"\n\t\t\t\t);\n\n\t\t\t} else {\n\n\t\t\t\t// We found a non-quoted value.\n\t\t\t\tstrMatchedValue = arrMatches[ 3 ];\n\n\t\t\t}\n\n\n\t\t\t// Now that we have our value string, let's add\n\t\t\t// it to the data array.\n\t\t\tarrData[ arrData.length - 1 ].push( strMatchedValue );\n\t\t}\n\n\t\t// Return the parsed data.\n\t\treturn ( arrData );\n\t},\n\n\twarn_page_name_change: function() {\n\t\tfrappe.msgprint(__(\"Note: Changing the Page Name will break previous URL to this page.\"));\n\t},\n\n\tnotify: function(subject, body, route, onclick) {\n\t\tconsole.log('push notifications are evil and deprecated');\n\t},\n\n\tset_title: function(title) {\n\t\tfrappe._original_title = title;\n\t\tif (frappe._title_prefix) {\n\t\t\ttitle = frappe._title_prefix + \" \" + title.replace(/<[^>]*>/g, \"\");\n\t\t}\n\t\tdocument.title = title;\n\n\t\t// save for re-routing\n\t\tconst sub_path = frappe.router.get_sub_path();\n\t\tfrappe.route_titles[sub_path] = title;\n\t},\n\n\tset_title_prefix: function(prefix) {\n\t\tfrappe._title_prefix = prefix;\n\n\t\t// reset the original title\n\t\tfrappe.utils.set_title(frappe._original_title);\n\t},\n\n\tis_image_file: function(filename) {\n\t\tif (!filename) return false;\n\t\t// url can have query params\n\t\tfilename = filename.split('?')[0];\n\t\treturn (/\\.(gif|jpg|jpeg|tiff|png|svg)$/i).test(filename);\n\t},\n\n\tplay_sound: function(name) {\n\t\ttry {\n\t\t\tif (frappe.boot.user.mute_sounds) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tvar audio = $(\"#sound-\" + name)[0];\n\t\t\taudio.volume = audio.getAttribute(\"volume\");\n\t\t\taudio.play();\n\n\t\t} catch (e) {\n\t\t\tconsole.log(\"Cannot play sound\", name, e);\n\t\t\t// pass\n\t\t}\n\n\t},\n\tsplit_emails: function(txt) {\n\t\tvar email_list = [];\n\n\t\tif (!txt) {\n\t\t\treturn email_list;\n\t\t}\n\n\t\t// emails can be separated by comma or newline\n\t\ttxt.split(/[,\\n](?=(?:[^\"]|\"[^\"]*\")*$)/g).forEach(function(email) {\n\t\t\temail = email.trim();\n\t\t\tif (email) {\n\t\t\t\temail_list.push(email);\n\t\t\t}\n\t\t});\n\n\t\treturn email_list;\n\t},\n\tsupportsES6: function() {\n\t\ttry {\n\t\t\tnew Function(\"(a = 0) => a\");\n\t\t\treturn true;\n\t\t} catch (err) {\n\t\t\treturn false;\n\t\t}\n\t}(),\n\tthrottle: function (func, wait, options) {\n\t\tvar context, args, result;\n\t\tvar timeout = null;\n\t\tvar previous = 0;\n\t\tif (!options) options = {};\n\n\t\tlet later = function () {\n\t\t\tprevious = options.leading === false ? 0 : Date.now();\n\t\t\ttimeout = null;\n\t\t\tresult = func.apply(context, args);\n\t\t\tif (!timeout) context = args = null;\n\t\t};\n\n\t\treturn function () {\n\t\t\tvar now = Date.now();\n\t\t\tif (!previous && options.leading === false) previous = now;\n\t\t\tlet remaining = wait - (now - previous);\n\t\t\tcontext = this;\n\t\t\targs = arguments;\n\t\t\tif (remaining <= 0 || remaining > wait) {\n\t\t\t\tif (timeout) {\n\t\t\t\t\tclearTimeout(timeout);\n\t\t\t\t\ttimeout = null;\n\t\t\t\t}\n\t\t\t\tprevious = now;\n\t\t\t\tresult = func.apply(context, args);\n\t\t\t\tif (!timeout) context = args = null;\n\t\t\t} else if (!timeout && options.trailing !== false) {\n\t\t\t\ttimeout = setTimeout(later, remaining);\n\t\t\t}\n\t\t\treturn result;\n\t\t};\n\t},\n\tdebounce: function(func, wait, immediate) {\n\t\tvar timeout;\n\t\treturn function() {\n\t\t\tvar context = this, args = arguments;\n\t\t\tvar later = function() {\n\t\t\t\ttimeout = null;\n\t\t\t\tif (!immediate) func.apply(context, args);\n\t\t\t};\n\t\t\tvar callNow = immediate && !timeout;\n\t\t\tclearTimeout(timeout);\n\t\t\ttimeout = setTimeout(later, wait);\n\t\t\tif (callNow) func.apply(context, args);\n\t\t};\n\t},\n\tget_form_link: function(doctype, name, html=false, display_text=null, query_params_obj=null) {\n\t\tdisplay_text = display_text || name;\n\t\tname = encodeURIComponent(name);\n\t\tlet route = `/app/${encodeURIComponent(doctype.toLowerCase().replace(/ /g, '-'))}/${name}`;\n\t\tif (query_params_obj) {\n\t\t\troute += frappe.utils.make_query_string(query_params_obj);\n\t\t}\n\t\tif (html) {\n\t\t\treturn `${display_text}`;\n\t\t}\n\t\treturn route;\n\t},\n\tget_route_label(route_str) {\n\t\tlet route = route_str.split('/');\n\n\t\tif (route[2] === 'Report' || route[0] === 'query-report') {\n\t\t\treturn __('{0} Report', [route[3] || route[1]]);\n\t\t}\n\t\tif (route[0] === 'List') {\n\t\t\treturn __('{0} List', [route[1]]);\n\t\t}\n\t\tif (route[0] === 'modules') {\n\t\t\treturn __('{0} Modules', [route[1]]);\n\t\t}\n\t\tif (route[0] === 'dashboard') {\n\t\t\treturn __('{0} Dashboard', [route[1]]);\n\t\t}\n\t\treturn __(frappe.utils.to_title_case(route[0], true));\n\t},\n\treport_column_total: function(values, column, type) {\n\t\tif (column.column.disable_total) {\n\t\t\treturn '';\n\t\t} else if (values.length > 0) {\n\t\t\tif (column.column.fieldtype == \"Percent\" || type === \"mean\") {\n\t\t\t\treturn values.reduce((a, b) => a + flt(b)) / values.length;\n\t\t\t} else if (column.column.fieldtype == \"Int\") {\n\t\t\t\treturn values.reduce((a, b) => a + cint(b));\n\t\t\t} else if (frappe.model.is_numeric_field(column.column.fieldtype)) {\n\t\t\t\treturn values.reduce((a, b) => a + flt(b));\n\t\t\t} else {\n\t\t\t\treturn null;\n\t\t\t}\n\t\t} else {\n\t\t\treturn null;\n\t\t}\n\t},\n\tsetup_search($wrapper, el_class, text_class, data_attr) {\n\t\tconst $search_input = $wrapper.find('[data-element=\"search\"]').show();\n\t\t$search_input.focus().val('');\n\t\tconst $elements = $wrapper.find(el_class).show();\n\n\t\t$search_input.off('keyup').on('keyup', () => {\n\t\t\tlet text_filter = $search_input.val().toLowerCase();\n\t\t\t// Replace trailing and leading spaces\n\t\t\ttext_filter = text_filter.replace(/^\\s+|\\s+$/g, '');\n\t\t\tfor (let i = 0; i < $elements.length; i++) {\n\t\t\t\tconst text_element = $elements.eq(i).find(text_class);\n\t\t\t\tconst text = text_element.text().toLowerCase();\n\n\t\t\t\tlet name = '';\n\t\t\t\tif (data_attr && text_element.attr(data_attr)) {\n\t\t\t\t\tname = text_element.attr(data_attr).toLowerCase();\n\t\t\t\t}\n\n\t\t\t\tif (text.includes(text_filter) || name.includes(text_filter)) {\n\t\t\t\t\t$elements.eq(i).css('display', '');\n\t\t\t\t} else {\n\t\t\t\t\t$elements.eq(i).css('display', 'none');\n\t\t\t\t}\n\t\t\t}\n\t\t});\n\t},\n\tdeep_equal(a, b) {\n\t\treturn deep_equal(a, b);\n\t},\n\n\tfile_name_ellipsis(filename, length) {\n\t\tlet first_part_length = length * 2 / 3;\n\t\tlet last_part_length = length - first_part_length;\n\t\tlet parts = filename.split('.');\n\t\tlet extn = parts.pop();\n\t\tlet name = parts.join('');\n\t\tlet first_part = name.slice(0, first_part_length);\n\t\tlet last_part = name.slice(-last_part_length);\n\t\tif (name.length > length) {\n\t\t\treturn `${first_part}...${last_part}.${extn}`;\n\t\t} else {\n\t\t\treturn filename;\n\t\t}\n\t},\n\tget_decoded_string(dataURI) {\n\t\t// decodes base64 to string\n\t\tlet parts = dataURI.split(',');\n\t\tconst encoded_data = parts[1];\n\t\tlet decoded = atob(encoded_data);\n\t\ttry {\n\t\t\tconst escaped = escape(decoded);\n\t\t\tdecoded = decodeURIComponent(escaped);\n\n\t\t} catch (e) {\n\t\t\t// pass decodeURIComponent failure\n\t\t\t// just return atob response\n\t\t}\n\t\treturn decoded;\n\t},\n\tcopy_to_clipboard(string) {\n\t\tconst show_success_alert = () => {\n\t\t\tfrappe.show_alert({\n\t\t\t\tindicator: 'green',\n\t\t\t\tmessage: __('Copied to clipboard.')\n\t\t\t});\n\t\t};\n\t\tif (navigator.clipboard && window.isSecureContext) {\n\t\t\tnavigator.clipboard.writeText(string).then(show_success_alert);\n\t\t} else {\n\t\t\tlet input = $(\"