rubyfox-server 2.16.0.4 → 2.16.3.1

Sign up to get free protection for your applications and to get access to all the features.
@@ -2017,6 +2017,9 @@ class ConfigInterfaceBuilder
2017
2017
 
2018
2018
  // Remove form inside static tab panes (predefined tab panes in html)
2019
2019
  $(`#${this._mainContainerId} > #tabPanels > div.tab-pane > form`).remove();
2020
+
2021
+ // Remove "active" class from static tab panes (otherwise this class messes with the tab navigator functioning)
2022
+ $(`#${this._mainContainerId} > #tabPanels > div.tab-pane`).removeClass('active');
2020
2023
  }
2021
2024
 
2022
2025
  disableInterface(disable)
@@ -2615,4 +2618,4 @@ class ConfigurationParameter
2615
2618
  /***/ })
2616
2619
 
2617
2620
  }]);
2618
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-12~module-13~module-9.bundle.js","sources":["webpack://application/./src/components/uibuilder/config-check-box.js","webpack://application/./src/components/uibuilder/config-drop-down-list.js","webpack://application/./src/components/uibuilder/config-dual-list.js","webpack://application/./src/components/uibuilder/config-form-item.js","webpack://application/./src/components/uibuilder/config-grid.js","webpack://application/./src/components/uibuilder/config-label.js","webpack://application/./src/components/uibuilder/config-numeric-stepper.js","webpack://application/./src/components/uibuilder/config-text-input.js","webpack://application/./src/components/uibuilder/config-vector-3d.js","webpack://application/./src/components/uibuilder/widgets/list-item-editor.js","webpack://application/./src/components/uibuilder/widgets/vector-3d-input.js","webpack://application/./src/utils/uibuilder/config-form-item-factory.js","webpack://application/./src/utils/uibuilder/config-interface-builder.js","webpack://application/./src/utils/uibuilder/configuration-parameter.js"],"sourcesContent":["import {ConfigFormItem} from './config-form-item';\nimport {ConfigLabel} from './config-label';\n\nexport class ConfigCheckBox extends ConfigFormItem\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super(configParam, editEnabled, inDialog);\n\t}\n\n\t/**\n\t * Create widget to render the ConfigParameter value.\n\t * If parameter is not editable, a simple label is used.\n\t * @override\n\t */\n\t_generateInnerWidget()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Set widget configuration\n\t\t\tlet config = {\n\t\t\t\ttype: 'checkbox',\n\t\t\t\tclass: '',\n\t\t\t\tid: this._data.name,\n\t\t\t\tname: this._data.name,\n\t\t\t\t'data-role': 'switch',\n\t\t\t};\n\n\t\t\t// Set widget attributes (see parent class)\n\t\t\tthis._setWidgetAttributes(config);\n\n\t\t\t// Set additional widget attributes based on validation rules (see parent class)\n\t\t\tthis._setWidgetValidationAttributes(config);\n\n\t\t\t// Create widget's html\n\t\t\tthis._widgetHtml = $('<input>', config);\n\t\t}\n\t\telse\n\t\t\tthis._widgetHtml = new ConfigLabel();\n\n\t\t// Return component\n\t\treturn this._widgetHtml;\n\t}\n\n\t/**\n\t * Initialize widget.\n\t * @override\n\t */\n   _initialize()\n   {\n\t   if (this._data.editable)\n\t   {\n\t\t   // Initialize kendo widget\n\t\t   kendo.init(this._widgetHtml);\n\n\t\t   // Save ref. to widget\n\t\t   this._innerWidget = this._widgetHtml.data('kendoSwitch');\n\n\t\t   // Enable value commit binding\n\t\t   this._innerWidget.bind('change', $.proxy(this._onValueInput, this));\n\t   }\n\n\t   // Proceed with initialization\n\t   super._initialize();\n   }\n\n\t/**\n\t * Set widget's value.\n\t * If parameter is not editable, the label text is set.\n\t * @override\n\t */\n\t_setWidgetValue()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._innerWidget.value(this._data.value);\n\t\telse\n\t\t\tthis._widgetHtml.value = this._data.value;\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n\n\t/**\n\t * Set widget's disabled state.\n\t * @override\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._innerWidget.enable(this._editEnabled);\n\t}\n\n\t/**\n\t * Update Configuration Parameter value.\n\t * @override\n\t */\n\t_onValueInput(e)\n\t{\n\t\t// Update Configuration Parameter to new value\n\t\tthis._data.value = this._innerWidget.value();\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-check-box'))\n\twindow.customElements.define('config-check-box', ConfigCheckBox);\n","import {ConfigFormItem} from './config-form-item';\nimport {ConfigLabel} from './config-label';\n\nexport class ConfigDropDownList extends ConfigFormItem\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super(configParam, editEnabled, inDialog);\n\t}\n\n\t/**\n\t * Create widget to render the ConfigParameter value.\n\t * If parameter is not editable, a simple label is used.\n\t * @override\n\t */\n\t_generateInnerWidget()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Set widget configuration\n\t\t\tlet config = {\n\t\t\t\tclass: 'form-control',\n\t\t\t\tid: this._data.name,\n\t\t\t\tname: this._data.name,\n\t\t\t\t'data-role': 'dropdownlist',\n\t\t\t};\n\n\t\t\t// Set widget attributes (see parent class)\n\t\t\tthis._setWidgetAttributes(config);\n\n\t\t\t// Set additional widget attributes based on validation rules (see parent class)\n\t\t\tthis._setWidgetValidationAttributes(config);\n\n\t\t\t// Create widget's html\n\t\t\tthis._widgetHtml = $('<input>', config);\n\t\t}\n\t\telse\n\t\t\tthis._widgetHtml = new ConfigLabel();\n\n\t\t// Return component\n\t\treturn this._widgetHtml;\n\t}\n\n\t/**\n\t * Initialize widget.\n\t * @override\n\t */\n   _initialize()\n   {\n\t   if (this._data.editable)\n\t   {\n\t\t   // Initialize kendo widget\n\t\t   kendo.init(this._widgetHtml);\n\n\t\t   // Save ref. to widget\n\t\t   this._innerWidget = this._widgetHtml.data('kendoDropDownList');\n\n\t\t   // Set list items\n\t\t   this._innerWidget.setDataSource(this._getDataSource(this._data.dataProvider))\n\n\t\t   // Enable value commit binding\n\t\t   this._widgetHtml.bind('change', $.proxy(this._onValueInput, this));\n\t   }\n\n\t   // Proceed with initialization\n\t   super._initialize();\n   }\n\n\t/**\n\t * Set widget's value.\n\t * If parameter is not editable, the label text is set.\n\t * @override\n\t */\n\t_setWidgetValue()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._innerWidget.value(this._data.value);\n\t\telse\n\t\t\tthis._widgetHtml.value = this._data.value;\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n\n\t/**\n\t * Set widget's disabled state.\n\t * @override\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._innerWidget.wrapper.attr('disabled', !this._editEnabled);\n\t}\n\n\t/**\n\t * Update Configuration Parameter value.\n\t * @override\n\t */\n\t_onValueInput(e)\n\t{\n\t\t// Update Configuration Parameter to new value\n\t\tthis._data.value = this._innerWidget.value();\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n\n\t_getDataSource(dpString)\n\t{\n\t\tif (dpString)\n\t\t\treturn dpString.split(',');\n\n\t\t// In case the dataprovider is empty, add at least the current value\n\t\telse\n\t\t\treturn [this._data.value];\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-drop-down-list'))\n\twindow.customElements.define('config-drop-down-list', ConfigDropDownList);\n","import {ConfigFormItem} from './config-form-item';\nimport {ConfigLabel} from './config-label';\n\nexport class ConfigDualList extends ConfigFormItem\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super(configParam, editEnabled, inDialog);\n\t}\n\n\t/**\n\t * Create widget to render the ConfigParameter.\n\t * @override\n\t */\n\t_generateInnerWidget()\n\t{\n\t\tthis._widgetHtml = $('<div>');\n\n\t\tconst availableId = this._getId(this._data.name, 'available');\n\t\tconst selectedId = this._getId(this._data.name, 'selected');\n\n\t\t// Create header for labels\n\t\tlet header = $('<div>', {class: 'form-label-container dual-list-labels'});\n\n\t\theader.append($('<label>', {\n\t\t\tclass: 'font-italic form-label dual-list-left-col' + (!this._data.editable ? ' no-interact' : ''),\n\t\t\tfor: availableId,\n\t\t}).text('Available'));\n\n\t\theader.append($('<label>', {\n\t\t\tclass: 'font-italic font-weight-bold form-label dual-list-right-col' + (!this._data.editable ? ' no-interact' : ''),\n\t\t\tfor: selectedId,\n\t\t}).text('Selected'));\n\n\t\tthis._widgetHtml.append(header);\n\n\t\t// Add available items list\n\t\tthis._availableListHtml = $('<select>', {\n\t\t\tid: availableId,\n\t\t\tclass: 'dual-list-left-col' + (!this._data.editable ? ' no-interact' : ''),\n\t\t});\n\t\tthis._widgetHtml.append(this._availableListHtml);\n\n\t\t// Add selected items list\n\t\tthis._selectedListHtml = $('<select>', {\n\t\t\tid: selectedId,\n\t\t\tclass: 'dual-list-right-col' + (!this._data.editable ? ' no-interact' : ''),\n\t\t});\n\t\tthis._widgetHtml.append(this._selectedListHtml);\n\n\t\t// Return component\n\t\treturn this._widgetHtml;\n\t}\n\n\t// IDs containing a \".\" cause issues to connected lists\n\t_getId(name, suffix)\n\t{\n\t\treturn name.replace('.', '_') + '-' + suffix;\n\t}\n\n\t/**\n\t * Initialize widget.\n\t * @override\n\t */\n\t_initialize()\n\t{\n\t\t// Initialize \"avalable\" listbox\n\t\tthis._availableList = this._availableListHtml.kendoListBox({\n            connectWith: this._getId(this._data.name, 'selected'),\n            toolbar: {\n                tools: this._data.editable ? ['transferTo', 'transferFrom', 'transferAllTo', 'transferAllFrom'] : []\n            },\n\t\t\ttemplate: \"<div>#:value#</div>\",\n\t\t\tselectable: 'multiple',\n        }).data('kendoListBox');\n\n\t\t// Initialize \"selected\" listbox\n        this._selectedList = this._selectedListHtml.kendoListBox({\n\t\t\ttemplate: \"<div>#:value#</div>\",\n\t\t\tselectable: 'multiple',\n\t\t\t// The following listeners can't be used because events are fired before the datasource is actually updated\n\t\t\t// We have to use a change event listener on the datasource (see below), even if not optimal\n\t\t\t//add: $.proxy(this._onValueInput, this),\n\t\t\t//remove: $.proxy(this._onValueInput, this),\n\t\t}).data('kendoListBox');\n\n\t\t// Proceed with initialization\n\t\tsuper._initialize();\n\t}\n\n\t/**\n\t * Set widget's datasource.\n\t * @override\n\t */\n\t_setWidgetValue()\n\t{\n\t\tlet availableArr = this._data.dataProvider != '' ? this._data.dataProvider.split(',') : [];\n\t\tlet selectedArr = this._data.value != '' ? this._data.value.split(',') : [];\n\n\t\t// Remove selected values from available values\n\t\tif (selectedArr.length > 0)\n\t\t{\n\t\t\tlet temp = [];\n\n\t\t\tfor (let val of availableArr)\n\t\t\t{\n\t\t\t\tif (selectedArr.indexOf(val) == -1)\n\t\t\t\t\ttemp.push(val);\n\t\t\t}\n\n\t\t\tavailableArr = temp;\n\t\t}\n\n\t\t// Convert lists of strings to lists of objects\n\t\tlet availableValues = [];\n\t\tfor (let val of availableArr)\n\t\t\tavailableValues.push({value: val});\n\n\t\tlet selectedValues = [];\n\t\tfor (let val of selectedArr)\n\t\t\tselectedValues.push({value: val});\n\n\t\t// Clear selection\n\t\tthis._availableList.clearSelection();\n\t\tthis._selectedList.clearSelection();\n\n\t\t// Set datasources\n\t\tthis._availableList.setDataSource(new kendo.data.DataSource({\n\t\t\tdata: availableValues\n\t\t}));\n\n\t\tthis._selectedList.setDataSource(new kendo.data.DataSource({\n\t\t\tdata: selectedValues,\n\t\t\t// We listen to the change event instead of the add/remove events on the listbox, because those are fired before the datasource is updated\n\t\t\t// This is not optimal because the event is fired for each item added to or removed from the datasource\n\t\t\tchange: $.proxy(this._onValueInput, this)\n\t\t}));\n\n\t\t// Disable editing\n\t\tif (!this._data.editable)\n\t\t{\n\t\t\tthis._availableList.enable('.k-item', false);\n\t\t\tthis._selectedList.enable('.k-item', false);\n\t\t}\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n\n\t/**\n\t * Set widget's disabled state.\n\t * @override\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Clear selection\n\t\t\tthis._availableList.clearSelection();\n\t\t\tthis._selectedList.clearSelection();\n\n\t\t\t// Enable/disable lists\n\t\t\tthis._availableList.wrapper.attr('disabled', !this._editEnabled);\n\t\t\tthis._selectedList.wrapper.attr('disabled', !this._editEnabled);\n\t\t}\n\t}\n\n\t/**\n\t * Update Configuration Parameter value.\n\t * @override\n\t */\n\t_onValueInput(e)\n\t{\n\t\tlet listData = this._selectedList.dataSource.data();\n\n\t\t// Update Configuration Parameter to new value\n\t\tthis._data.value = listData.map(e => e.value).join(',');\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-dual-list'))\n\twindow.customElements.define('config-dual-list', ConfigDualList);\n","export class ConfigFormItem extends HTMLElement\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super();\n\n\t\tthis.id = 'form-item-' + configParam.name;\n\t\tthis._editEnabled = editEnabled;\n\t\tthis._data = configParam;\n\n\t\t// Create form item view\n\t\tthis._buildView(inDialog);\n\n\t\t// Initialize form item\n\t\tthis._initialize();\n\t}\n\n\tconnectedCallback()\n\t{\n\t\t// Trigger event\n\t\t// NOTE: when a ConfigFormItem is instantiated, the _triggerEvent method is called as soon as its value is set.\n\t\t// When this happenso, due to the fact that the object is not yet in the DOM, the event is not catched by the listener\n\t\t// (which is attached to the outer container). So forcing the event to trigger again as soon as the ConfigFormItem\n\t\t// is appended to the DOM is needed.\n\t\tthis._triggerEvent();\n\t}\n\n\tset data(configParam)\n\t{\n\t\tthis._data = configParam;\n\t\tthis._setWidgetValue();\n\t}\n\n\tget data()\n\t{\n\t\treturn this._data;\n\t}\n\n\tset editEnabled(enable)\n\t{\n\t\tif (enable != this._editEnabled)\n\t\t{\n\t\t\tthis._editEnabled = enable;\n\t\t\tthis._setWidgetEditEnabled();\n\t\t}\n\t}\n\n\tget editEnabled()\n\t{\n\t\treturn this._editEnabled;\n\t}\n\n\t_buildView(isInsideDialog)\n\t{\n\t\tif (!isInsideDialog)\n\t\t{\n\t\t\t// Set additional classes for inner widget\n\t\t\tlet classNames = '';\n\n\t\t\tswitch (this._data.type)\n\t\t\t{\n\t\t\t\tcase 'DualList':\n\t\t\t\t\tclassNames = 'col-sm-7 col-lg-8';\n\t\t\t\t\tbreak;\n\t\t\t\tcase 'DataGrid':\n\t\t\t\t\tclassNames = 'col-sm'; // Use 'col-sm-7 col-lg-8' for DataGrid too?\n\t\t\t\t\tbreak;\n\t\t\t\tdefault:\n\t\t\t\t\tclassNames = 'col-sm-auto';\n\n\t\t\t}\n\n\t\t\t// Generate boilerplate html, surrounding the actual widget (label, numeric stepper, etc)\n\t\t\tthis.innerHTML = `\n\t\t\t\t<div class=\"form-group position-relative row\">\n\t\t\t\t\t<div class=\"col-sm-5 col-lg-4 col-form-label form-label-container\">\n\t\t\t\t\t\t<label for=\"${this._data.name}\" class=\"form-label ${(this._data.clientOnly ? 'client-only' : '')}\">${this._data.label} <i class=\"fas fa-question-circle text-muted help\" title=\"${this._data.tooltip}\"></i></label>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div class=\"inner-widget align-self-center ${classNames}\">\n\t\t\t\t\t\t<span class=\"k-invalid-msg\" data-for=\"${this._data.name}\"></span>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t`;\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis.innerHTML = `\n\t\t\t\t<div class=\"form-group position-relative\">\n\t\t\t\t\t<div class=\"col-form-label form-label-container\">\n\t\t\t\t\t\t<label for=\"${this._data.name}\" class=\"form-label ${(this._data.clientOnly ? 'client-only' : '')}\">${this._data.label} <i class=\"fas fa-question-circle text-muted help\" title=\"${this._data.tooltip}\"></i></label>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div class=\"inner-widget\">\n\t\t\t\t\t\t<span class=\"k-invalid-msg\" data-for=\"${this._data.name}\"></span>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t`;\n\t\t}\n\n\t\t// Create inner widget (must be overridden)\n\t\tlet widget = this._generateInnerWidget();\n\n\t\t// Append inner widget\n\t\t$(this).find('.inner-widget').prepend(widget);\n\t}\n\n\t/**\n\t * TO BE OVERRIDDEN\n\t */\n\t_generateInnerWidget()\n\t{\n\t\t// Show an error, should be overridden\n\t\tconsole.error(`Unable to create ${this._data.type} form item for configuration parameter ${this.id}`);\n\t}\n\n\t/**\n\t * Set attributes on the widget configuration object.\n\t */\n\t_setWidgetAttributes(config)\n\t{\n\t\tconst attribs = this._data.attributes;\n\n\t\tif (attribs)\n\t\t{\n\t\t\tfor (let attr in attribs)\n\t\t\t{\n\t\t\t\tconfig[attr] = attribs[attr];\n\n\t\t\t\tif (attr == 'pattern')\n\t\t\t\t\tconfig['data-pattern-msg'] = 'Contains invalid characters';\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Set additional attributes on the widget configuration object to properly validate input.\n\t */\n\t_setWidgetValidationAttributes(config)\n\t{\n\t\tconst val = this._data.validator;\n\n\t\tif (val != null && val != '')\n\t\t{\n\t\t\tif (val == 'ip')\n\t\t\t{\n\t\t\t\tconfig['pattern'] = '^(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$';\n\t\t\t\tconfig['data-pattern-msg'] = 'Invalid IP address';\n\t\t\t\tconfig['required'] = true;\n\t\t\t\tconfig['data-required-msg'] = 'Required';\n\t\t\t}\n\n\t\t\telse if (val == 'notNull')\n\t\t\t{\n\t\t\t\tconfig['required'] = true;\n\t\t\t\tconfig['data-required-msg'] = 'Required';\n\t\t\t}\n\n\t\t\telse if (val == 'pwd')\n\t\t\t{\n\t\t\t\tconfig['pattern'] = '^.{6,}$';\n\t\t\t\tconfig['data-pattern-msg'] = 'Minimum length: 6 characters';\n\t\t\t}\n\n\t\t\telse if (val == 'posNum')\n\t\t\t{\n\t\t\t\tconfig['pattern'] = '^[0-9]\\d*$';\n\t\t\t\tconfig['data-pattern-msg'] = 'Non-negative number required';\n\t\t\t}\n\n\t\t\telse if (val == 'aoi')\n\t\t\t{\n\t\t\t\t// Nothing to do\n\t\t\t\t// See Kendo validation initialization in config-interface-builder.js\n\t\t\t}\n\n\t\t\telse if (val == 'url')\n\t\t\t{\n\t\t\t\tconfig['type'] = 'url';\n\t\t\t\tconfig['data-url-msg'] = 'Invalid URL';\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Initialize form item.\n\t *\n\t * NOTE: must be overridden if inner widget requires special initialization (for example Kendo widgets)\n\t */\n\t_initialize()\n\t{\n\t\t// Set value\n \t   this._setWidgetValue();\n\n \t   // Set edit enabled\n \t   this._setWidgetEditEnabled();\n\t}\n\n\t/**\n\t * TO BE OVERRIDDEN\n\t */\n\t_setWidgetValue()\n\t{\n\t\t// Nothing to do, must be overridden\n\t}\n\n\t/**\n\t * TO BE OVERRIDDEN\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\t// Nothing to do, must be overridden\n\t}\n\n\t/**\n\t * TO BE OVERRIDDEN\n\t */\n\t_onValueInput(e)\n\t{\n\t\t// Nothing to do, must be overridden\n\t}\n\n\t_triggerEvent()\n\t{\n\t\tif (this._data.trigger)\n\t\t{\n\t\t\tlet event = new CustomEvent('value-set', {detail: null, bubbles: true, cancelable: true});\n\t\t\tthis.dispatchEvent(event);\n\t\t}\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-form-item'))\n\twindow.customElements.define('config-form-item', ConfigFormItem);\n","import {ConfigFormItem} from './config-form-item';\nimport {ConfigLabel} from './config-label';\nimport {ListItemEditor} from './widgets/list-item-editor';\n\nexport class ConfigGrid extends ConfigFormItem\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super(configParam, editEnabled, inDialog);\n\t}\n\n\t/**\n\t * Create widget to render the ConfigParameter.\n\t * @override\n\t */\n\t_generateInnerWidget()\n\t{\n\t\t// Create main widget's html\n\t\tthis._widgetHtml = $('<div>', {class: ''});\n\n\t\t// Set grid widget configuration\n\t\tlet gridConfig = {\n\t\t\tid: this._data.name,\n\t\t\tname: this._data.name,\n\t\t\tclass: 'limited-height' + (!this._data.editable ? ' no-interact' : '')\n\t\t};\n\n\t\t// Append grid to main html; grid will be converted to Kendo widget during initialization\n\t\tthis._widgetHtml.append($('<div>', gridConfig));\n\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// BUTTONS\n\n\t\t\t// Create buttons container\n\t\t\tlet buttons = $('<div>', {class: 'mt-2 text-right'});\n\n\t\t\t// Append buttons to container\n\t\t\tthis._addButton = $('<button>', {type: 'button', class: 'k-button k-secondary', title: 'Add'}).append($('<i class=\"fas fa-plus\"></i>'));\n\t\t\tthis._editButton = $('<button>', {type: 'button', class: 'k-button k-secondary ml-2', title: 'Edit', disabled: true}).append($('<i class=\"fas fa-pen\"></i>'));\n\t\t\tthis._removeButton = $('<button>', {type: 'button', class: 'k-button k-secondary ml-2', title: 'Remove', disabled: true}).append($('<i class=\"fas fa-times\"></i>'));\n\n\t\t\tbuttons.append(this._addButton);\n\t\t\tbuttons.append(this._editButton);\n\t\t\tbuttons.append(this._removeButton);\n\n\t\t\t// Append buttons container to main html\n\t\t\tthis._widgetHtml.append(buttons);\n\n\t\t\t// Create edit dialog\n\t\t\t// NOTE: data-dismiss=\"modal\" on the close/cancel buttons was removed to work around an issue with nested modals;\n\t\t\t// the custom \"data-cancel\" attribute is used to add a custom listener to the buttons\n\t\t\tthis._editDialog = $(`\n\t\t\t\t<div class=\"modal\" tabindex=\"-1\" role=\"dialog\" aria-labelledby=\"modalTitle\" aria-hidden=\"true\" data-keyboard=\"false\" data-backdrop=\"static\">\n\t\t\t\t\t<div class=\"modal-dialog modal-dialog-centered\" role=\"document\">\n\t\t\t\t\t\t<div class=\"modal-content\">\n\t\t\t\t\t\t\t<div class=\"modal-header\">\n\t\t\t\t\t\t\t\t<h5 class=\"modal-title text-primary\" id=\"modalTitle\">${this._data.label}</h5>\n\t\t\t\t\t\t\t\t<button type=\"button\" class=\"close\" aria-label=\"Close\" data-cancel=\"modal\">\n\t\t\t\t\t\t\t\t\t<span aria-hidden=\"true\">&times;</span>\n\t\t\t\t\t\t\t\t</button>\n\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t<div class=\"modal-body in-flow-invalid-msg\">\n\n\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t<div class=\"modal-footer flex-column\">\n\t\t\t\t\t\t\t\t<div class=\"d-flex w-100\">\n\t\t\t\t\t\t\t\t\t<div class=\"flex-grow-1 text-left\">\n\t\t\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary\">...</button>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t<div class=\"flex-grow-1 text-right\">\n\t\t\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-secondary\" data-cancel=\"modal\">Cancel</button>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t`);\n\n\t\t\t// Add listener to dialog hide event\n\t\t\tthis._editDialog.on('hidden.bs.modal', $.proxy(this._onEditPanelHidden, this));\n\n\t\t\t// Add listener to main button click event\n\t\t\t$('button.k-primary', this._editDialog).on('click', $.proxy(this._onSubmitBtClick, this));\n\n\t\t\t// Add listener to close/cancel buttons click event\n\t\t\t$('button[data-cancel=\"modal\"]', this._editDialog).on('click', $.proxy(this._onCancelBtClick, this));\n\n\t\t\t// Append edit dialog to main html\n\t\t\tthis._widgetHtml.append(this._editDialog);\n\t\t}\n\n\t\t// Return component\n\t\treturn this._widgetHtml;\n\t}\n\n\t/**\n\t * Initialize widget.\n\t * @override\n\t */\n\t_initialize()\n\t{\n\t\tlet columns = [];\n\t\tfor (let subConfigParam of this._data.defaultListItem)\n\t\t{\n\t\t\tlet col = {\n\t\t\t\tfield: subConfigParam.name,\n\t\t\t\ttitle: subConfigParam.label,\n\t\t\t\twidth: 120\n\t\t\t}\n\n\t\t\t// Display V or X for booleans\n\t\t\tif (typeof subConfigParam.value === 'boolean')\n\t\t\t\tcol.template = `#= ${subConfigParam.name} ? '<i class=\"fas fa-check\"></i>' : '<i class=\"fas fa-times\"></i>' #`;\n\n\t\t\t// Hide passwords\n\t\t\tif (subConfigParam.type == 'TextInput' && subConfigParam.attributes != null && subConfigParam.attributes.type == 'password')\n\t\t\t\tcol.template = `#= '•'.repeat(data.${subConfigParam.name}.length) #`;\n\n\t\t\tcolumns.push(col);\n\t\t}\n\n\t\t// Initialize grid\n\t\tlet gridHtml = this._widgetHtml.find(`#${$.escapeSelector(this._data.name)}`);\n\n\t\tgridHtml.kendoGrid({\n\t\t\tresizable: true,\n\t\t\tselectable: this._data.editable ? 'row' : false,\n\t\t\tchange: $.proxy(this._onGridSelectionChange, this),\n\t\t\tcolumns: columns,\n\t\t\tnoRecords: {\n\t\t\t\ttemplate: 'No items.'\n\t\t\t}\n\t\t});\n\n\t\t// Save ref. to widget\n\t\tthis._gridWidget = gridHtml.data('kendoGrid');\n\n\t\t// Show tootip if grid's cell content exceeds cell width (ellipsis is displayed by Kendo Grid)\n\t\tgridHtml.kendoTooltip({\n\t\t\tfilter: 'td',\n\t\t\tshow: function(e) {\n\t\t\t\t// Never show tooltip...\n\t\t\t\tthis.content.parent().css('visibility', 'hidden');\n\n\t\t\t\t// ...unless content is returned (see below) due to cell width being exceeded\n\t\t\t\tif (this.content.text() != '')\n\t\t\t\t\tthis.content.parent().css('visibility', 'visible');\n\t\t\t},\n\t\t\thide: function() {\n\t\t\t\tthis.content.parent().css('visibility', 'hidden');\n\t\t\t},\n\t\t\tcontent: function(e) {\n\t\t\t\tlet element = e.target[0];\n\t\t\t\tif (element.offsetWidth < element.scrollWidth)\n\t\t\t\t\treturn e.target.text();\n\t\t\t\telse\n\t\t\t\t\treturn '';\n\t\t\t}\n\t\t});\n\n\t\t/*\n\t\t// Initialize button tooltips\n\t\tthis._widgetHtml.kendoTooltip({\n\t\t\tfilter: 'button',\n\t\t\tcontent: function(e) {\n\t\t\t\treturn `<div class=\"help-tooltip\">${e.target.data('title')}</div>`;\n\t\t\t}\n\t\t});\n\t\t*/\n\n\t\t// Add button listeners\n\t\tif (this._data.editable)\n\t\t{\n\t\t\tthis._addButton.click($.proxy(this._onAddClick, this));\n\t\t\tthis._editButton.click($.proxy(this._onEditClick, this));\n\t\t\tthis._removeButton.click($.proxy(this._onRemoveClick, this));\n\t\t}\n\n\t\t// Proceed with initialization\n\t\tsuper._initialize();\n\t}\n\n\t/**\n\t * Set widget's datasource.\n\t * @override\n\t */\n\t_setWidgetValue()\n\t{\n\t\tlet dataSource = new kendo.data.DataSource({\n\t\t\tdata: this._data.listValues\n\t\t});\n\n\t\t// Read current horizontal scroll value\n\t\tconst scrollLeft = $('.k-grid-content', this._gridWidget.wrapper).scrollLeft();\n\n\t\t// Clear grid selection if any\n\t\tthis._gridWidget.clearSelection();\n\n\t\t// Set updated grid's datasource\n\t\tthis._gridWidget.setDataSource(dataSource);\n\n\t\t// Set horizontal scroll\n\t\t$('.k-grid-content', this._gridWidget.wrapper).scrollLeft(scrollLeft);\n\t}\n\n\t/**\n\t * Set widget's disabled state.\n\t * @override\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Deselect item\n\t\t\tthis._gridWidget.clearSelection();\n\n\t\t\t// Enable/disable grid\n\t\t\tthis._gridWidget.wrapper.attr('disabled', !this._editEnabled);\n\n\t\t\t// Enable \"Add\" button\n\t\t\tif (this._editEnabled)\n\t\t\t\tthis._addButton.attr('disabled', false);\n\n\t\t\t// Disable all buttons\n\t\t\telse\n\t\t\t{\n\t\t\t\tthis._addButton.attr('disabled', true);\n\t\t\t\tthis._editButton.attr('disabled', true);\n\t\t\t\tthis._removeButton.attr('disabled', true);\n\t\t\t}\n\t\t}\n\t}\n\n\t_onGridSelectionChange(e)\n\t{\n\t\tlet selectedRows = this._gridWidget.select();\n\t\tlet selectedDataItems = [];\n\n\t\tfor (let i = 0; i < selectedRows.length; i++)\n\t\t{\n\t\t\tlet dataItem = this._gridWidget.dataItem(selectedRows[i]);\n\t\t\tselectedDataItems.push(dataItem);\n\t\t}\n\n\t\t// Enable/disable edit button\n\t\tif (this._editButton)\n\t\t\tthis._editButton.prop('disabled', selectedDataItems.length == 0);\n\n\t\t// Enable/disable remove button\n\t\tif (this._removeButton)\n\t\t\tthis._removeButton.prop('disabled', selectedDataItems.length == 0);\n    }\n\n\t_onRemoveClick()\n\t{\n\t\tlet selectedIndex = this._gridWidget.select().index();\n\n\t\t// Remove item from list\n\t\tthis._data.removeListItem(selectedIndex);\n\n\t\t// Regenerate datagrid's datasource\n\t\tthis._setWidgetValue();\n\t}\n\n\t_onAddClick()\n\t{\n\t\t// Clone default item and add to list\n\t\tlet newListItem = [];\n\t\tfor (let subCP of this._data.defaultListItem)\n\t\t\tnewListItem.push(subCP.clone(true));\n\n\t\t// Create edit popup\n\t\tthis._openEditPanel(newListItem);\n\t}\n\n\t_onEditClick()\n\t{\n\t\tlet selectedIndex = this._gridWidget.select().index();\n\n\t\t// Clone selected item and add to list\n\t\tlet clonedListItem = [];\n\t\tfor (let subCP of this._data.listItems[selectedIndex])\n\t\t\tclonedListItem.push(subCP.clone(true));\n\n\t\t// Create edit popup\n\t\tthis._openEditPanel(clonedListItem, selectedIndex);\n\t}\n\n\t_openEditPanel(subConfigParamsArray, editIndex = -1)\n\t{\n\t\t// Check if this configuration item is inside a modal window;\n\t\t// if yes, the edit panel (which is a modal as well) must be configured to remove the dark background\n\t\tif ($(this).parents('.modal').length > 0)\n\t\t\t$('.modal', $(this)).attr('data-backdrop', false);\n\n\t\t// Create dialog content\n\t\tthis._itemEditor = new ListItemEditor();\n\t\tthis._itemEditor.data = subConfigParamsArray;\n\t\tthis._itemEditor.index = editIndex;\n\n\t\tlet itemEditor = $(this._itemEditor);\n\n\t\t// Append content to dialog\n\t\t$('.modal-body', this._editDialog).append(itemEditor);\n\n\t\t// Set dialog main button text\n\t\t$('button.k-primary', this._editDialog).html(editIndex > -1 ? '<i class=\"fas fa-pen mr-1\"></i>Update' : '<i class=\"fas fa-plus mr-1\"></i>Add');\n\n\t\t// Display dialog\n\t\tthis._editDialog.modal('show');\n\t}\n\n\t_onSubmitBtClick()\n\t{\n\t\tif (this._itemEditor.validate())\n\t\t{\n\t\t\tlet data = this._itemEditor.data;\n\t\t\tlet index = this._itemEditor.index;\n\n\t\t\t// Hide modal\n\t\t\tthis._editDialog.modal('hide');\n\n\t\t\t// Complete editing\n\t\t\tthis._onEditComplete(data, index);\n\t\t}\n\t}\n\n\t_onCancelBtClick()\n\t{\n\t\t// Hide modal\n\t\tthis._editDialog.modal('hide');\n\t}\n\n\t_onEditPanelHidden(e)\n\t{\n\t\t// Remove content from dialog\n\t\tthis._itemEditor.remove();\n\n\t\t// Set dialog main button text\n\t\t$('button.k-primary', this._editDialog).html('...');\n\n\t\tthis._itemEditor = null;\n\t}\n\n\t_onEditComplete(listItem, editIndex)\n\t{\n\t\t// An existing list item was updated\n\t\tif (editIndex > -1)\n\t\t\tthis._data.updateListItem(listItem, editIndex);\n\n\t\t// A new list item was added; add it to the configuration parameter\n\t\telse\n\t\t\tthis._data.addListItem(listItem);\n\n\t\t// Regenerate datagrid's datasource\n\t\tthis._setWidgetValue();\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-grid'))\n\twindow.customElements.define('config-grid', ConfigGrid);\n","export class ConfigLabel extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\n\t\tthis.setAttribute('class','config-label');\n\t}\n\n\tset value(val)\n\t{\n\t\tif (typeof val === 'boolean')\n\t\t\tthis.innerHTML = (val ? 'true' : 'false');\n\t\telse if (typeof val === 'number')\n\t\t\tthis.innerHTML = (val ? val : 0);\n\t\telse\n\t\t\tthis.innerHTML = (val != '' ? val : '&mdash;');\n\t}\n\n\tget value()\n\t{\n\t\treturn this.textContent;\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-label'))\n\twindow.customElements.define('config-label', ConfigLabel);\n","import {ConfigFormItem} from './config-form-item';\nimport {ConfigLabel} from './config-label';\n\nexport class ConfigNumericStepper extends ConfigFormItem\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super(configParam, editEnabled, inDialog);\n\t}\n\n\t/**\n\t * Create widget to render the ConfigParameter value.\n\t * If parameter is not editable, a simple label is used.\n\t * @override\n\t */\n\t_generateInnerWidget()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Set widget configuration\n\t\t\tlet config = {\n\t\t\t\ttype: 'number',\n\t\t\t\tclass: 'form-control',\n\t\t\t\tid: this._data.name,\n\t\t\t\tname: this._data.name,\n\t\t\t\t'data-role': 'numerictextbox',\n\t\t\t\t'data-required-msg': 'Required',\n\t\t\t\t'data-format': '#',\n\t\t\t\trequired: 'required',\n\t\t\t};\n\n\t\t\t// Set widget attributes (see parent class)\n\t\t\tthis._setWidgetAttributes(config);\n\n\t\t\t// Set additional widget attributes based on validation rules (see parent class)\n\t\t\tthis._setWidgetValidationAttributes(config);\n\n\t\t\t// Create widget's html\n\t\t\tthis._widgetHtml = $('<input>', config);\n\t\t}\n\t\telse\n\t\t\tthis._widgetHtml = new ConfigLabel();\n\n\t\t// Return component\n\t\treturn this._widgetHtml;\n\t}\n\n\t/**\n\t * Initialize widget.\n\t * @override\n\t */\n   _initialize()\n   {\n\t   if (this._data.editable)\n\t   {\n\t\t   // Initialize kendo widget\n\t\t   kendo.init(this._widgetHtml);\n\n\t\t   // Save ref. to widget\n\t\t   this._innerWidget = this._widgetHtml.data('kendoNumericTextBox');\n\n\t\t   // Enable value commit binding\n\t\t   this._innerWidget.bind('change', $.proxy(this._onValueInput, this));\n\t   }\n\n\t   // Proceed with initialization\n\t   super._initialize();\n   }\n\n\t/**\n\t * Set widget's value.\n\t * If parameter is not editable, the label text is set.\n\t * @override\n\t */\n\t_setWidgetValue()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._innerWidget.value(this._data.value);\n\t\telse\n\t\t\tthis._widgetHtml.value = this._data.value;\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n\n\t/**\n\t * Set widget's disabled state.\n\t * @override\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._innerWidget.enable(this._editEnabled);\n\t}\n\n\t/**\n\t * Update Configuration Parameter value.\n\t * @override\n\t */\n\t_onValueInput(e)\n\t{\n\t\t// Update Configuration Parameter to new value\n\t\tthis._data.value = Number(this._innerWidget.value());\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-numeric-stepper'))\n\twindow.customElements.define('config-numeric-stepper', ConfigNumericStepper);\n","import {ConfigFormItem} from './config-form-item';\nimport {ConfigLabel} from './config-label';\n\nexport class ConfigTextInput extends ConfigFormItem\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super(configParam, editEnabled, inDialog);\n\t}\n\n\t/**\n\t * Create widget to render the ConfigParameter value.\n\t * If parameter is not editable, a simple label is used.\n\t * @override\n\t */\n\t_generateInnerWidget()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Set widget configuration\n\t\t\tlet config = {\n\t\t\t\ttype: 'text',\n\t\t\t\tclass: 'form-control k-textbox',\n\t\t\t\tid: this._data.name,\n\t\t\t\tname: this._data.name,\n\t\t\t\tautocomplete: 'off',\n\t\t\t};\n\n\t\t\t// Set widget attributes\n\t\t\tthis._setWidgetAttributes(config);\n\n\t\t\t// Set additional widget attributes based on validation rules\n\t\t\tthis._setWidgetValidationAttributes(config);\n\n\t\t\t// Create widget's html\n\t\t\tthis._widgetHtml = $('<input>', config);\n\n\t\t\t// Enable value commit binding\n\t\t\tthis._widgetHtml.on('change', $.proxy(this._onValueInput, this));\n\t\t}\n\t\telse\n\t\t\tthis._widgetHtml = new ConfigLabel();\n\n\t\t// Return component\n\t\treturn this._widgetHtml;\n\t}\n\n\t/**\n\t * Set widget's value.\n\t * If parameter is not editable, the label text is set.\n\t * @override\n\t */\n\t_setWidgetValue()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._widgetHtml.val(this._data.value);\n\t\telse\n\t\t\tthis._widgetHtml.value = this._data.value;\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n\n\t/**\n\t * Set widget's disabled state.\n\t * @override\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._widgetHtml.attr('disabled', !this._editEnabled);\n\t}\n\n\t/**\n\t * Update Configuration Parameter value.\n\t * @override\n\t */\n\t_onValueInput(e)\n\t{\n\t\t// Update Configuration Parameter to new value\n\t\tthis._data.value = this._widgetHtml.val();\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-text-input'))\n\twindow.customElements.define('config-text-input', ConfigTextInput);\n","import {ConfigFormItem} from './config-form-item';\nimport {ConfigLabel} from './config-label';\nimport {Vector3DInput} from './widgets/vector-3d-input';\n\nexport class ConfigVector3D extends ConfigFormItem\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super(configParam, editEnabled, inDialog);\n\t}\n\n\t/**\n\t * Create widget to render the ConfigParameter value.\n\t * If parameter is not editable, a simple label is used.\n\t * @override\n\t */\n\t_generateInnerWidget()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Create widget's html\n\t\t\tthis._widgetHtml = new Vector3DInput(this._data.name, this._data.validator == 'aoi');\n\n\t\t\t// Set widget attributes\n\t\t\tthis._setWidgetAttributes(this._widgetHtml);\n\n\t\t\t// Enable value commit binding\n\t\t\t$(this._widgetHtml).on('change', $.proxy(this._onValueInput, this));\n\t\t}\n\t\telse\n\t\t\tthis._widgetHtml = new ConfigLabel();\n\n\t\t// Return component\n\t\treturn this._widgetHtml;\n\t}\n\n\t/**\n\t * Set widget's value.\n\t * If parameter is not editable, the label text is set.\n\t * @override\n\t */\n\t_setWidgetValue()\n\t{\n\t\tthis._widgetHtml.value = this._data.value;\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n\n\t/**\n\t * Set widget's disabled state.\n\t * @override\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t$(this._widgetHtml).attr('disabled', !this._editEnabled);\n\t\t}\n\t}\n\n\t/**\n\t * Update Configuration Parameter value.\n\t * @override\n\t */\n\t_onValueInput(e)\n\t{\n\t\t// Update Configuration Parameter to new value\n\t\tthis._data.value = this._widgetHtml.value;\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-vector-3d'))\n\twindow.customElements.define('config-vector-3d', ConfigVector3D);\n","import {ConfigFormItemFactory} from '../../../utils/uibuilder/config-form-item-factory';\n\nexport class ListItemEditor extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\t}\n\n\tset data(subConfigParamsArray)\n\t{\n\t\tthis._data = subConfigParamsArray;\n\n\t\tthis._buildView();\n\t}\n\n\tget data()\n\t{\n\t\treturn this._data;\n\t}\n\n\tset index(index)\n\t{\n\t\tthis._index = index;\n\t}\n\n\tget index()\n\t{\n\t\treturn this._index;\n\t}\n\n\t_buildView()\n\t{\n\t\t// Generate container form\n\t\tthis._form = $('<form>', {});\n\n\t\t// Append form\n\t\t$(this).append(this._form);\n\n\t\t// Generate form fields\n\t\tfor (let configParam of this._data)\n\t\t{\n\t\t\t// Create form item\n\t\t\tlet formItem = ConfigFormItemFactory.create(configParam, true, true);\n\t\t\tformItem.data = configParam;\n\n\t\t\t// Add form item to form\n\t\t\tthis._form.append(formItem);\n\t\t}\n\n\t\t// Initialize kendo validation on form\n\t\tthis._validator = this._form.kendoValidator({\n\t\t\tvalidateOnBlur: true,\n\t\t\trules: {\n\t\t\t\t// Add rule to validate AOI form items?\n\t\t\t\t// (see: https://demos.telerik.com/kendo-ui/validator/custom-validation)\n\t\t\t\taoi: function (input) {\n\t\t\t\t\tif (input.is('[data-aoi-msg]') && input.val() != '')\n\t\t\t\t\t{\n\t\t\t\t\t\tif (input.val() == '0,0,0')\n\t\t\t\t\t\t\treturn false;\n                    }\n\n                    return true;\n                }\n\t\t\t}\n\t\t}).data('kendoValidator');\n\t}\n\n\tvalidate()\n\t{\n\t\treturn this._validator.validate();\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('list-item-editor'))\n\twindow.customElements.define('list-item-editor', ListItemEditor);\n","export class Vector3DInput extends HTMLElement\n{\n\tconstructor(id, isValidable)\n\t{\n\t    super();\n\n\t\tthis.id = id;\n\t\tthis.name = id;\n\n\t\tthis._isValidable = isValidable;\n\n\t\tthis._initialize();\n\t}\n\n\tset enableClear(value)\n\t{\n\t\tif (value)\n\t\t\tthis._clearButton.show();\n\t\telse\n\t\t\tthis._clearButton.hide();\n\t}\n\n\tset allowNegative(value)\n\t{\n\t\tif (value)\n\t\t{\n\t\t\tthis._widgetX.setOptions( {min: null} );\n\t\t\tthis._widgetY.setOptions( {min: null} );\n\t\t\tthis._widgetZ.setOptions( {min: null} );\n\t\t}\n\t}\n\n\tset value(val)\n\t{\n\t\tvar coords = val.split(',');\n\n\t\tif (coords.length >= 1)\n\t\t\tthis._widgetX.value(coords[0]);\n\n\t\tif (coords.length >= 2)\n\t\t\tthis._widgetY.value(coords[1]);\n\n\t\tif (coords.length >= 3)\n\t\t\tthis._widgetZ.value(coords[2]);\n\n\t\tif (this._isValidable)\n\t\t\tthis._inputVal.val(this.value);\n\t}\n\n\tget value()\n\t{\n\t\tif (this._widgetX.value() == null && this._widgetY.value() == null && this._widgetZ.value() == null)\n\t\t\treturn '';\n\t\telse\n\t\t\treturn this._widgetX.value() + ',' + this._widgetY.value() + ',' + this._widgetZ.value();\n\t}\n\n\t_initialize()\n\t{\n\t\t// Generate container form\n\t\tthis._container = $('<div>', {\n\t\t\tclass: 'form-inline'\n\t\t});\n\n\t\t// Append container\n\t\t$(this).append(this._container);\n\n\t\t// Set inputs configuration\n\t\tlet configHtml = {\n\t\t\ttype: 'number',\n\t\t\tclass: 'form-control short-4',\n\t\t};\n\n\t\t// Set widget configuration\n\t\tlet configWidget = {\n\t\t\tmin: 0,\n\t\t\tspinners: false,\n\t\t\tformat: '#.######',\n\t\t\tdecimals: 6,\n\t\t\tround: false,\n\t\t\tspinners: false,\n\t\t\trestrictDecimals: false,\n\t\t\tchange: $.proxy(this._onChange, this)\n\t\t};\n\n\t\t// Create widgets\n\t\tthis._inputX = $('<input>', configHtml);\n\t\tthis._container.append(this._inputX);\n\t\tthis._widgetX = this._inputX.kendoNumericTextBox(configWidget).data('kendoNumericTextBox');\n\n\t\tthis._container.append('<span class=\"px-1\">,</span>');\n\n\t\tthis._inputY = $('<input>', configHtml);\n\t\tthis._container.append(this._inputY);\n\t\tthis._widgetY = this._inputY.kendoNumericTextBox(configWidget).data('kendoNumericTextBox');\n\n\t\tthis._container.append('<span class=\"px-1\">,</span>');\n\n\t\tthis._inputZ = $('<input>', configHtml);\n\t\tthis._container.append(this._inputZ);\n\t\tthis._widgetZ = this._inputZ.kendoNumericTextBox(configWidget).data('kendoNumericTextBox');\n\n\t\tthis._container.append('<span class=\"px-1\"></span>'); // Additional spacer\n\n\t\t// Create invisible field to apply overall validation\n\t\tif (this._isValidable)\n\t\t{\n\t\t\tthis._inputVal = $('<input>', {name: `${this.name}-custom-validate`, 'data-aoi-msg': 'Values can\\'t all be 0'});\n\t\t\tthis._container.append(this._inputVal);\n\t\t\tthis._container.append(`<span class=\"k-invalid-msg\" data-for=\"${this.name}-custom-validate\"></span>`)\n\t\t\tthis._inputVal.hide();\n\t\t}\n\n\t\t// Create and append Clear button\n\t\tthis._clearButton = $('<button>', {type: 'button', class: 'k-button k-secondary my-1', title: 'Clear'}).append($('<i class=\"fas fa-times\"></i>'));\n\t\tthis._clearButton.on('click', $.proxy(this._onClearClick, this));\n\t\tthis._container.append(this._clearButton);\n\n\t\t// Hide button by default\n\t\tthis._clearButton.hide();\n\t}\n\n\t_onChange()\n\t{\n\t\t// Empty strings are not allowed\n\t\tif (this._widgetX.value() == null)\n\t\t\tthis._widgetX.value(0);\n\n\t\tif (this._widgetY.value() == null)\n\t\t\tthis._widgetY.value(0);\n\n\t\tif (this._widgetZ.value() == null)\n\t\t\tthis._widgetZ.value(0);\n\n\t\tthis._dispatchCommit();\n\t}\n\n\t_onClearClick()\n\t{\n\t\tthis._widgetX.value('');\n\t\tthis._widgetY.value('');\n\t\tthis._widgetZ.value('');\n\n\t\tthis._dispatchCommit();\n\t}\n\n\t_dispatchCommit()\n\t{\n\t\tif (this._isValidable)\n\t\t\tthis._inputVal.val(this.value);\n\n\t\tthis.dispatchEvent(new Event('change'));\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('vector-3d-input'))\n\twindow.customElements.define('vector-3d-input', Vector3DInput);\n","import {ConfigFormItem} from '../../components/uibuilder/config-form-item';\n\nimport {ConfigNumericStepper} from '../../components/uibuilder/config-numeric-stepper';\nimport {ConfigTextInput} from '../../components/uibuilder/config-text-input';\nimport {ConfigCheckBox} from '../../components/uibuilder/config-check-box';\nimport {ConfigDropDownList} from '../../components/uibuilder/config-drop-down-list';\nimport {ConfigGrid} from '../../components/uibuilder/config-grid';\nimport {ConfigDualList} from '../../components/uibuilder/config-dual-list';\nimport {ConfigVector3D} from '../../components/uibuilder/config-vector-3d';\n\nexport class ConfigFormItemFactory\n{\n\tstatic create(configParam, editEnabled, inDialog = false)\n\t{\n\t\tswitch (configParam.type)\n\t\t{\n\t\t\tcase 'TextInput':\n\t\t\t\treturn new ConfigTextInput(configParam, editEnabled, inDialog);\n\t\t\t\tbreak;\n\n\t\t\tcase 'CheckBox':\n\t\t\t\treturn new ConfigCheckBox(configParam, editEnabled, inDialog);\n\t\t\t\tbreak;\n\n\t\t\tcase 'NumericStepper':\n\t\t\t\treturn new ConfigNumericStepper(configParam, editEnabled, inDialog);\n\t\t\t\tbreak;\n\n\t\t\tcase 'ComboBox':\n\t\t\t\treturn new ConfigDropDownList(configParam, editEnabled, inDialog);\n\t\t\t\tbreak;\n\n\t\t\tcase 'DataGrid':\n\t\t\t\treturn new ConfigGrid(configParam, editEnabled, inDialog);\n\t\t\t\tbreak;\n\n\t\t\tcase 'DualList':\n\t\t\t\treturn new ConfigDualList(configParam, editEnabled, inDialog);\n\t\t\t\tbreak;\n\n\t\t\tcase 'Vector3D':\n\t\t\t\treturn new ConfigVector3D(configParam, editEnabled, inDialog);\n\t\t\t\tbreak;\n\n\t\t\tdefault:\n\t\t\t\treturn new ConfigFormItem(configParam, editEnabled, inDialog); // Will log an error for missing form item type\n\t\t}\n\t}\n}\n","import {ConfigurationParameter} from './configuration-parameter';\nimport {ConfigFormItemFactory} from './config-form-item-factory';\n\nexport class ConfigInterfaceBuilder\n{\n\tconstructor()\n\t{\n\t\t// Set some constants\n\t\tthis.TAB_PREFIX = 'tab-'\n\t\tthis.TAB_PANE_PREFIX = 'tabpane-';\n\t\tthis.SEPARATOR_BEFORE = 'before';\n\t\tthis.SEPARATOR_AFTER = 'after';\n\t}\n\n\tdump(modifiedOnly = false)\n\t{\n\t\tlet dumpStr = '';\n\n\t\tfor (let cp of this._configParams)\n\t\t{\n\t\t\tif (modifiedOnly)\n\t\t\t{\n\t\t\t\tif (cp.isModified)\n\t\t\t\t\tdumpStr += cp.toString() + '\\n';\n\t\t\t}\n\t\t\telse\n\t\t\t\tdumpStr += cp.toString() + '\\n';\n\t\t}\n\n\t\tconsole.log(dumpStr);\n\t}\n\n\tbuildInterface(data, mainContainerId, disableEdit = false, tabSuffix = '')\n\t{\n\t\tthis._mainContainerId = mainContainerId;\n\t\tthis._configParams = new Array();\n\t\tthis._validator = null;\n\n\t\tlet hasNewFormItem = false;\n\n\t\t//console.log(data.getDump())\n\n\t\tfor (let i = 0; i < data.size(); i++)\n\t\t{\n\t\t\t// PARSE DATA\n\n\t\t\tlet configParam = ConfigurationParameter.fromSfsObject(data.get(i));\n\t\t\tthis._configParams.push(configParam);\n\n\t\t\t// Get tab and tab pane id from group id\n\t\t\tconst tabId = this.TAB_PREFIX + configParam.categoryId + (tabSuffix ? '_' + tabSuffix : '');\n\t\t\tconst tabPaneId = this.TAB_PANE_PREFIX + configParam.categoryId + (tabSuffix ? '_' + tabSuffix : '');\n\n\t\t\t// BUILD INTERFACE :: TABS\n\n\t\t\t// Check if a tab specific for this group already exists inside the mainContainer: if not, create it\n\t\t\t// (a tab already exists if it was created in a previous loop)\n\t\t\tlet tab = $(`#${mainContainerId} > #tabs #${tabId}`);\n\n\t\t\tif (tab.length == 0)\n\t\t\t{\n\t\t\t\t// Create tab for tab pane\n\t\t\t\ttab = $('<li>', {class: 'nav-item'});\n\t\t\t\ttab.append($('<a>', {\n\t\t\t\t\tclass: 'nav-link' + (i == 0 ? ' active' : ''),\n\t\t\t\t\tid: tabId,\n\t\t\t\t\t'data-toggle': 'tab',\n\t\t\t\t\thref: '#' + tabPaneId,\n\t\t\t\t\trole: 'tab',\n\t\t\t\t\t'aria-controls': tabPaneId,\n\t\t\t\t\t'aria-selected': (i == 0 ? 'true' : 'false'),\n\t\t\t\t\thtml: configParam.category,\n\t\t\t\t}));\n\n\t\t\t\t// Add tab to container\n\t\t\t\t$(`#${mainContainerId} > #tabs`).append(tab);\n\t\t\t}\n\n\t\t\t// BUILD INTERFACE :: TAB PANES\n\n\t\t\t// Check if a tab pane specific for this group already exists inside the mainContainer: if not, create it\n\t\t\t// (a tab pane already exists if it was created in a previous loop or if it exists statically in the html - in case it is needed to add some static content)\n\t\t\tlet tabPane = $(`#${mainContainerId} > #tabPanels > #${tabPaneId}`);\n\n\t\t\tif (tabPane.length == 0)\n\t\t\t{\n\t\t\t\t// Create tab pane\n\t\t\t\ttabPane = $('<div>', {\n\t\t\t\t\tclass: 'tab-pane' + (i == 0 ? ' show active' : ''),\n\t\t\t\t\tid: tabPaneId,\n\t\t\t\t\trole: 'tabpanel',\n\t\t\t\t\t'aria-labelledby': tabId,\n\t\t\t\t\t'data-dynamic': 'true',\n\t\t\t\t});\n\n\t\t\t\t// Add tab pane to container\n\t\t\t\t$(`#${mainContainerId} > #tabPanels`).append(tabPane);\n\t\t\t}\n\n\t\t\t// BUILD INTERFACE :: TAB PANES' FORM\n\n\t\t\t// Check if a form already exists inside the tab pane: if not, create it\n\t\t\tlet form = tabPane.find('form');\n\n\t\t\tif (form.length == 0)\n\t\t\t{\n\t\t\t\t// Create form\n\t\t\t\tform = $('<form>', {\n\t\t\t\t\tclass: '',\n\t\t\t\t\tautocomplete: 'off'\n\t\t\t\t});\n\n\t\t\t\t// Create an inner fieldset; this might be useful to easily disable the whole form at once (actually we don't use it because Kendo widgets are not disabled automatically)\n\t\t\t\tform.append(\n\t\t\t\t\t$('<fieldset>', {\n\t\t\t\t\t\tclass: ''\n\t\t\t\t\t})\n\t\t\t\t);\n\n\t\t\t\t// Add form to tab pane\n\t\t\t\ttabPane.prepend(form);\n\t\t\t}\n\n\t\t\t// Get fieldset, which is the actual form items container\n\t\t\tlet fieldset = form.find('fieldset');\n\n\t\t\t// BUILD INTERFACE :: TAB PANES' FORM ITEMS\n\n\t\t\t// Check if form item already exists in fieldset; if yes, just update its data\n\t\t\tlet formItem = fieldset.find(`#form-item-${$.escapeSelector(configParam.name)}`);\n\n\t\t\tif (formItem.length == 0)\n\t\t\t{\n\t\t\t\thasNewFormItem = true;\n\n\t\t\t\tformItem = ConfigFormItemFactory.create(configParam, !disableEdit);\n\n\t\t\t\t// Add separator before\n\t\t\t\tif (configParam.separator != null && configParam.separator.pos == 'before')\n\t\t\t\t\tfieldset.append(this._buildSeparator(configParam.separator));\n\n\t\t\t\t// Add form item to form\n\t\t\t\tfieldset.append(formItem);\n\n\t\t\t\t// Add separator after\n\t\t\t\tif (configParam.separator != null && configParam.separator.pos == 'after')\n\t\t\t\t\tfieldset.append(this._buildSeparator(configParam.separator));\n\t\t\t}\n\t\t\telse\n\t\t\t\tformItem[0].data = configParam;\n\t\t}\n\n\t\t// Add listener to show help tooltips\n\t\tlet allTabPanes = $(`#${mainContainerId} > #tabPanels > div.tab-pane`);\n\t\tallTabPanes.kendoTooltip({\n\t\t\tfilter: 'i[title].help',\n\t\t\tposition: 'right',\n\t\t\twidth: '250px',\n\t\t\tcontent: function(e) {\n\t\t\t\treturn `<div class=\"help-tooltip\">${e.target.data('title')}</div>`;\n\t\t\t}\n\t\t});\n\n\t\t// Initialize kendo validation on forms' main container\n\t\tthis._validator = $(`#${mainContainerId}`).kendoValidator({\n\t\t\tvalidateOnBlur: true,\n\t\t\trules: {\n\t\t\t\t// Add rule to validate AOI form items\n\t\t\t\t// (see: https://demos.telerik.com/kendo-ui/validator/custom-validation)\n\t\t\t\taoi: function (input) {\n\t\t\t\t\tif (input.is('[data-aoi-msg]') && input.val() != '')\n\t\t\t\t\t{\n\t\t\t\t\t\tif (input.val() == '0,0,0')\n\t\t\t\t\t\t\treturn false;\n                    }\n\n                    return true;\n                }\n\t\t\t}\n\t\t}).data('kendoValidator');\n\t}\n\n\tdestroyInterface()\n\t{\n\t\t// Destroy all Kendo widgets in forms\n\t\tkendo.destroy($(`#${this._mainContainerId} > #tabPanels > div.tab-pane > form`));\n\n\t\t// Remove all tabs\n\t\t$(`#${this._mainContainerId} > #tabs`).empty();\n\n\t\t// Remove dynamic tab panes (tab panes created by Interface Builder)\n\t\t$(`#${this._mainContainerId} > #tabPanels > div.tab-pane[data-dynamic=\"true\"]`).remove();\n\n\t\t// Remove form inside static tab panes (predefined tab panes in html)\n\t\t$(`#${this._mainContainerId} > #tabPanels > div.tab-pane > form`).remove();\n\t}\n\n\tdisableInterface(disable)\n\t{\n\t\t// Enable/disable all config form items\n\t\t$(`#${this._mainContainerId} *[id^='form-item-']`).prop('editEnabled', !disable);\n\t}\n\n\t_buildSeparator(separator)\n\t{\n\t\tif (separator.text == null)\n\t\t\treturn $(`<hr class=\"config-form-separator\">`);\n\n\t\telse\n\t\t\treturn $(`<label class=\"config-form-separator-label mb-3\">${separator.text}</label>`);\n\t}\n\n\tgetChangedData()\n\t{\n\t\tlet changes = new SFS2X.SFSArray();\n\n\t\tfor (var cp of this._configParams)\n\t\t{\n\t\t\tif (cp.isModified)\n\t\t\t\tchanges.addSFSObject(cp.toSfsObject());\n\t\t}\n\n\t\treturn changes;\n\t}\n\n\tresetIsModified()\n\t{\n\t\tfor (let cp of this._configParams)\n\t\t{\n\t\t\tif (cp.isModified)\n\t\t\t\tcp.resetIsModified();\n\t\t}\n\t}\n\n\tcheckIsValid()\n\t{\n\t\treturn this._validator.validate();\n\t}\n\n\tresetValidation()\n\t{\n\t\tthis._validator.hideMessages();\n\n\t\t// The method above doesn't remove the k-invalid classes and aria-invalid=\"true\" attributes from inputs\n\t\t// Let's do it manually\n\t\t$(`#${this._mainContainerId} .k-invalid`).removeClass('k-invalid');\n\t\t$(`#${this._mainContainerId} [aria-invalid=\"true\"]`).removeAttr('aria-invalid');\n\t}\n\n\tgetConfigFormItem(configParamName)\n\t{\n\t\tlet formItem = $(`#${this._mainContainerId}`).find(`#form-item-${$.escapeSelector(configParamName)}`);\n\n\t\tif (formItem.length > 0)\n\t\t\treturn formItem[0];\n\t\telse\n\t\t\treturn null;\n\t}\n\n\tactivateFirstTabPanel()\n\t{\n\t\tlet configParam = this._configParams[0];\n\t\tconst tabPaneId = this.TAB_PANE_PREFIX + configParam.categoryId;\n\t\tlet tabPane = $(`#${this._mainContainerId} > #tabPanels > #${tabPaneId}`);\n\t\ttabPane.addClass('show active');\n\t}\n}\n","export class ConfigurationParameter\n{\n\tstatic fromSfsObject(element)\n\t{\n\t\tlet cp = new ConfigurationParameter();\n\n\t\t// Parse common data\n\t\tcp.name = element.getUtfString('name');\n\t\tcp.label = element.getUtfString('label');\n\t\tcp.category = element.getUtfString('category');\n\t\tcp.tooltip = element.getUtfString('tooltip');\n\t\tcp.type = element.getUtfString('type');\n\t\tcp.value = element.get('value');\n\t\tcp.validator = element.getUtfString('validator');\n\t\tcp.editable = (element.containsKey('edit') ? element.getBool('edit') : true);\n\t\tcp.trigger = (element.containsKey('trigger') ? element.getBool('trigger') : false);\n\t\tcp.triggerData = element.getSFSArray('triggerData');\n\t\tcp.clientOnly = (element.containsKey('clientOnly') ? element.getBool('clientOnly') : false);\n\t\tcp.dataProvider = element.getUtfString('dataProvider');\n\n\t\t// Parse component specific attributes\n\t\tlet tmpAttributes = element.getSFSObject('attributes');\n\t\tif (tmpAttributes != null)\n\t\t{\n\t\t\tlet attributes = {};\n\n\t\t\tlet keys = tmpAttributes.getKeysArray();\n\t\t\tfor (let key of keys)\n\t\t\t\tattributes[key] = tmpAttributes.get(key);\n\n\t\t\tcp.attributes = attributes;\n\t\t}\n\n\t\t// Parse separator settings\n\t\tlet tmpSeparator = element.getSFSObject('separator');\n\t\tif (tmpSeparator != null)\n\t\t{\n\t\t\tlet separator = {};\n\n\t\t\tlet keys1 = tmpSeparator.getKeysArray();\n\t\t\tfor (let key1 of keys1)\n\t\t\t\tseparator[key1] = tmpSeparator.get(key1);\n\n\t\t\tcp.separator = separator;\n\t\t}\n\n\t\t// Parse default list item\n\t\tlet tmpDefaultListItem = element.getSFSArray('defaultListItem');\n\t\tif (tmpDefaultListItem != null && tmpDefaultListItem.size() > 0)\n\t\t{\n\t\t\tlet defaultListItem = [];\n\n\t\t\tfor (let i = 0; i < tmpDefaultListItem.size(); i++)\n\t\t\t\tdefaultListItem.push(ConfigurationParameter.fromSfsObject(tmpDefaultListItem.getSFSObject(i)));\n\n\t\t\tcp.defaultListItem = defaultListItem;\n\n\t\t\t// Parse list values\n\t\t\tlet listValues = [];\n\n\t\t\tlet tmpListValues = element.getSFSArray('listValues');\n\t\t\tif (tmpListValues != null && tmpListValues.size() > 0)\n\t\t\t{\n\t\t\t\tfor (let v = 0; v < tmpListValues.size(); v++)\n\t\t\t\t{\n\t\t\t\t\tlet listValueObj = tmpListValues.getSFSObject(v);\n\t\t\t\t\tlet obj = {};\n\n\t\t\t\t\tlet keys2 = listValueObj.getKeysArray();\n\t\t\t\t\tfor (let key2 of keys2)\n\t\t\t\t\t\tobj[key2] = listValueObj.get(key2);\n\n\t\t\t\t\tlistValues.push(obj);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tcp.listValues = listValues;\n\n\t\t\t// If we have a list, on the server-side items could be represented by a class\n\t\t\tcp.clazz = element.getUtfString('clazz');\n\t\t}\n\n\t\treturn cp;\n\t}\n\n\tconstructor()\n\t{\n\t\t/* CONSTANTS */\n\t\tthis.DEFAULT_CATEGORY_NAME = 'General';\n\t\tthis.DEFAULT_CATEGORY_ID = 'general';\n\n\t\t/* PUBLIC VARS */\n\n\t\tthis.name = '';\n\t\tthis.label = '';\n\t\tthis.tooltip = '';\n\t\tthis.type = null;\n\t\tthis.trigger = false;\n\t\tthis.triggerData = null;\n\t\tthis.clientOnly = false;\n\t\tthis.editable = true;\n\t\tthis.attributes = null;\n\t\tthis.dataProvider = null;\n\n\t\tthis.separator = null;\t\t\t// Parameter used to create a separator before or after the config parameter\n\t\tthis.defaultListItem = null;\t\t// List of sub-ConfigurationParameters, each containing the default values\n\t\tthis.clazz = null;\t\t\t\t// Name of the class representing the list item (not used in case of primiteve data types)\n\n\t\t/* PRIVATE VARS */\n\n\t\tthis._category = this.DEFAULT_CATEGORY_NAME;\n\t\tthis._categoryId = this.DEFAULT_CATEGORY_ID;\n\t\tthis._value = null;\n\t\tthis._initialValue = null;\t\t// Save the initial value of the configuration parameter, to check if the value was modified\n\t\tthis._validator = null;\n\n\t\tthis._listItems = [];\t\t\t// Array of arrays of ConfigurationParameters\n\t\tthis._listItemsChanged = false;\t// Flag to be set in case a list item is added or removed\n\t}\n\n\t//---------------------------------------------\n\t// GETTERS / SETTERS\n\t//---------------------------------------------\n\n\tset category(val)\n\t{\n\t\tif (val)\n\t\t{\n\t\t\tthis._category = val;\n\t\t\tthis._setIdFromCategoryName(this._category);\n\t\t}\n\t}\n\n\tget category()\n\t{\n\t\treturn this._category;\n\t}\n\n\tset value(val)\n\t{\n\t\tif (this._value != val)\n\t\t{\n\t\t\t// If value is null, then we are setting this for the first time and\n\t\t\t// we want to save the initial value, to check later if it has been modified\n\t\t\tif (this._value == null)\n\t\t\t\tthis._initialValue = val;\n\n\t\t\tthis._value = val;\n\t\t}\n\t}\n\n\tget value()\n\t{\n\t\treturn this._value;\n\t}\n\n\tset validator(val)\n\t{\n\t\tif (val)\n\t\t\tthis._validator = val;\n\t}\n\n\tget validator()\n\t{\n\t\treturn this._validator;\n\t}\n\n\t/**\n\t * An array of objects; each object contains the name-value pairs used to\n\t * populate the list of sub-configuration parameters arrays, based on defaultListItem.\n\t */\n\tset listValues(arr)\n\t{\n\t\tthis._setSubConfigurationParams(arr);\n\t}\n\n\tget listValues()\n\t{\n\t\treturn this._getSubConfigurationParamsValues();\n\t}\n\n\t//---------------------------------------------\n\t// GETTERS ONLY\n\t//---------------------------------------------\n\n\tget isModified()\n\t{\n\t\tlet _isModified = false;\n\n\t\t// If the parameter is used on the client only (for example in a custom trigger)\n\t\t// then we never have to consider it as modified, to prevent it being sent to the server\n\t\tif (!this.clientOnly)\n\t\t{\n\t\t\tif (this._value != this._initialValue || this._listItemsChanged)\n\t\t\t\t_isModified = true;\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Check sub parameters\n\t\t\t\touterLoop: for (let listItem of this._listItems)\n\t\t\t\t{\n\t\t\t\t\tfor (let subCP of listItem)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (subCP.isModified)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t_isModified = true;\n\t\t\t\t\t\t\tbreak outerLoop;\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}\n\n\t\treturn _isModified;\n\t}\n\n\tget categoryId()\n\t{\n\t\treturn this._categoryId;\n\t}\n\n\tget listItems()\n\t{\n\t\treturn this._listItems;\n\t}\n\n\t//---------------------------------------------\n\t// PUBLIC METHODS\n\t//---------------------------------------------\n\n\t/**\n\t * Return a clone of this ConfigurationParameter.\n\t */\n\tclone(cloneValue = false)\n\t{\n\t\tlet cp = new ConfigurationParameter();\n\t\tcp.name = this.name;\n\t\tcp.label = this.label;\n\t\tcp.category = this.category;\n\t\tcp.tooltip = this.tooltip;\n\t\tcp.type = this.type;\n\t\tcp.validator = this.validator;\n\t\tcp.trigger = this.trigger;\n\t\tcp.triggerData = (this.triggerData != null ? SFS2X.SFSArray.newFromBinaryData(this.triggerData.toBinary()) : null);\n\t\tcp.clientOnly = this.clientOnly;\n\t\tcp.dataProvider = this.dataProvider;\n\n\t\tif (cloneValue)\n\t\t\tcp.value = this.value;\n\n\t\tif (this.attributes != null)\n\t\t{\n\t\t\tcp.attributes = new Object();\n\t\t\tfor (let s1 in this.attributes)\n\t\t\t\tcp.attributes[s1] = this.attributes[s1];\n\t\t}\n\n\t\tif (this.separator != null)\n\t\t{\n\t\t\tcp.separator = new Object()\n\t\t\tfor (let s2 in this.separator)\n\t\t\t\tcp.separator[s2] = this.separator[s2];\n\t\t}\n\n\t\tif (this.defaultListItem != null)\n\t\t{\n\t\t\tlet clonedDefaultListItems = [];\n\n\t\t\tfor (let subCP of this.defaultListItem)\n\t\t\t\tclonedDefaultListItems.push(subCP.clone(cloneValue));\n\n\t\t\tcp.defaultListItem = clonedDefaultListItems;\n\t\t}\n\n\t\tcp.listValues = this.listValues; // No need to clone this, as the listValues setter already does it\n\t\tcp.clazz = this.clazz;\n\n\t\treturn cp;\n\t}\n\n\t/**\n\t * Reset initial value by copying the current value.\n\t */\n\tresetIsModified()\n\t{\n\t\tthis._initialValue = this._value;\n\n\t\t// Reset sub-parameters\n\t\tif (this._listItems != null)\n\t\t{\n\t\t\tfor (let listItem of this._listItems)\n\t\t\t{\n\t\t\t\tfor (let subCP of listItem)\n\t\t\t\t\tsubCP.resetIsModified();\n\t\t\t}\n\t\t}\n\n\t\tthis._listItemsChanged = false;\n\t}\n\n\taddListItem(newListItem)\n\t{\n\t\tthis._listItems.push(newListItem);\n\t\tthis._listItemsChanged = true;\n\t}\n\n\tupdateListItem(listItem, itemIndex)\n\t{\n\t\tthis._listItems[itemIndex] = listItem;\n\t\tthis._listItemsChanged = true;\n\t}\n\n\tremoveListItem(itemIndex)\n\t{\n\t\tthis._listItems.splice(itemIndex, 1);\n\t\tthis._listItemsChanged = true;\n\t}\n\n\ttoSfsObject()\n\t{\n\t\tlet obj = new SFS2X.SFSObject();\n\n\t\t// Set changed setting name\n\t\tobj.putUtfString('name', this.name);\n\n\t\t// Set changed setting class, if any\n\t\tif (this.clazz != null)\n\t\t\tobj.putUtfString('clazz', this.clazz);\n\n\t\tif (this.value != null)\n\t\t{\n\t\t\t// Set changed setting value\n\t\t\tif (typeof this.value === 'boolean')\n\t\t\t\tobj.putBool('value', this.value);\n\t\t\telse if (typeof this.value === 'number')\n\t\t\t\tobj.putInt('value', this.value);\n\t\t\telse\n\t\t\t\tobj.putText('value', this.value);\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// Set changed setting list of values\n\n\t\t\tlet listItems = new SFS2X.SFSArray();\n\n\t\t\tfor (let a of this._listItems)\n\t\t\t{\n\t\t\t\tif (a.length == 1) // We have just one sub config param; no need to parse it complitely\n\t\t\t\t{\n\t\t\t\t\t// Simple list\n\t\t\t\t\tlet tempObj = a[0].toSfsObject();\n\t\t\t\t\tlet wa = tempObj.getWrappedItem('value');\n\t\t\t\t\tlistItems.add(wa.value, wa.type);\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t// Complex list\n\n\t\t\t\t\tlet values = new SFS2X.SFSArray();\n\n\t\t\t\t\tfor (let subCp of a)\n\t\t\t\t\t\tvalues.addSFSObject(subCp.toSfsObject());\n\n\t\t\t\t\tlistItems.addSFSArray(values);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tobj.putSFSArray('value', listItems);\n\t\t}\n\n\t\treturn obj;\n\t}\n\n\t/**\n\t * Return a description of the ConfigurationParameter instance.\n\t */\n\ttoString()\n\t{\n\t\tlet s = ``;\n\t\ts += `Configuration parameter: ${this.name}\\n`;\n\t\ts += `\\ttype: ${this.type}\\n`;\n\t\ts += `\\tlabel: ${this.label}\\n`;\n\t\ts += `\\tcategory name: ${this.category}\\n`;\n\t\ts += `\\tcategory id: ${this.categoryId}\\n`;\n\t\ts += `\\ttooltip: ${this.tooltip}\\n`;\n\t\ts += `\\tvalue: ${this.value}\\n`;\n\t\ts += `\\ttrigger: ${this.trigger}\\n`;\n\t\ts += `\\ttrigger data: ${this.triggerData}\\n`;\n\t\ts += `\\tclient only: ${this.clientOnly}\\n`;\n\t\ts += `\\tvalidator: ${this.validator}\\n`;\n\t\ts += `\\tis modified: ${this.isModified}\\n`;\n\n\t\tif (this.attributes != null)\n\t\t{\n\t\t\ts += `\\tcomponent attributes:\\n`;\n\n\t\t\tfor (let s1 in this.attributes)\n\t\t\t\ts += `\\t\\t${s1} --> ${this.attributes[s1]}\\n`;\n\t\t}\n\n\t\tif (this.dataProvider != null)\n\t\t\ts += `\\tdata provider: ${this.dataProvider}\\n`;\n\n\t\tif (this.separator != null)\n\t\t{\n\t\t\ts += `\\tseparator:\\n`;\n\n\t\t\tfor (let s2 in this.separator)\n\t\t\t\ts += `\\t\\t${s2} --> ${this.separator[s2]}\\n`;\n\t\t}\n\n\t\tif (this._listItems != null && this._listItems.length > 0)\n\t\t{\n\t\t\ts += `\\t# list items: ${this._listItems.length}\\n`;\n\n\t\t\tfor (let i = 0; i < this._listItems.length; i++)\n\t\t\t{\n\t\t\t\ts += `\\tlist item ${i} sub-parameters:\\n`;\n\t\t\t\tfor (let e = 0; e < this._listItems[i].length; e++)\n\t\t\t\t\ts += `\\t\\t${this._listItems[i][e].toCompactString()}\\n`;\n\t\t\t}\n\n\t\t\ts += `\\tclass name: ${this.clazz}\\n`;\n\t\t}\n\n\t\treturn s;\n\t}\n\n\t/**\n\t * Return a compact description of the ConfigurationParameter instance.\n\t */\n\ttoCompactString()\n\t{\n\t\treturn `Configuration parameter '${this.name}': ${this.value} ${this.isModified ? '[X]' : '[ ]'}`;\n\t}\n\n\t//---------------------------------------------\n\t// PRIVATE METHODS\n\t//---------------------------------------------\n\n\t/**\n\t * Retrieve the category id form the category name.\n\t * Spaces and invalid characters are removed; words are separated using capitals.\n\t */\n\t_setIdFromCategoryName(categoryName)\n\t{\n\t\tthis._categoryId = categoryName;\n\n\t\t// Strip invalid characters\n\t\tvar pattern = /[^0-9a-zA-Z]/g;\n\t\tthis._categoryId = this._categoryId.replace(pattern, ' ');\n\n\t\t// Capitalize words\n\t\tvar words = this._categoryId.split(' ');\n\t\tthis._categoryId = '';\n\n\t\tfor (let i = 0; i < words.length; i++)\n\t\t{\n\t\t\tlet word = words[i];\n\t\t\tif (word.length > 0)\n\t\t\t\tthis._categoryId += (i > 0 ? word.substr(0,1).toUpperCase() : word.substr(0,1).toLowerCase()) + (word.length > 1 ? word.substr(1) : \"\");\n\t\t}\n\n\t\tif (this._categoryId.length == 0)\n\t\t\tthis._categoryId = this.DEFAULT_CATEGORY_ID;\n\t}\n\n\t_setSubConfigurationParams(_listValues)\n\t{\n\t\tthis._listItems = [];\n\n\t\tfor (let obj of _listValues)\n\t\t{\n\t\t\tlet listItem = [];\n\n\t\t\tfor (let defaultCP of this.defaultListItem)\n\t\t\t{\n\t\t\t\tlet subCP = defaultCP.clone(false);\n\t\t\t\tsubCP.value = obj[subCP.name];\n\n\t\t\t\tlistItem.push(subCP);\n\t\t\t}\n\n\t\t\tthis._listItems.push(listItem);\n\t\t}\n\t}\n\n\t_getSubConfigurationParamsValues()\n\t{\n\t\tlet _listValues = [];\n\n\t\tfor (let listItem of this._listItems)\n\t\t{\n\t\t\tlet obj = {};\n\n\t\t\tfor (let subCP of listItem)\n\t\t\t{\n\t\t\t\tif (subCP.value != null)\n\t\t\t\t\tobj[subCP.name] = subCP.value;\n\t\t\t}\n\n\t\t\t_listValues.push(obj);\n\t\t}\n\n\t\treturn _listValues;\n\t}\n}\n"],"mappings":";;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC5GA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACxHA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACzLA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACxOA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC3WA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AC3BA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC/GA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACzFA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC7EA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC7EA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC7JA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AChDA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC1QA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;A","sourceRoot":""}
2621
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-12~module-13~module-9.bundle.js","sources":["webpack://application/./src/components/uibuilder/config-check-box.js","webpack://application/./src/components/uibuilder/config-drop-down-list.js","webpack://application/./src/components/uibuilder/config-dual-list.js","webpack://application/./src/components/uibuilder/config-form-item.js","webpack://application/./src/components/uibuilder/config-grid.js","webpack://application/./src/components/uibuilder/config-label.js","webpack://application/./src/components/uibuilder/config-numeric-stepper.js","webpack://application/./src/components/uibuilder/config-text-input.js","webpack://application/./src/components/uibuilder/config-vector-3d.js","webpack://application/./src/components/uibuilder/widgets/list-item-editor.js","webpack://application/./src/components/uibuilder/widgets/vector-3d-input.js","webpack://application/./src/utils/uibuilder/config-form-item-factory.js","webpack://application/./src/utils/uibuilder/config-interface-builder.js","webpack://application/./src/utils/uibuilder/configuration-parameter.js"],"sourcesContent":["import {ConfigFormItem} from './config-form-item';\nimport {ConfigLabel} from './config-label';\n\nexport class ConfigCheckBox extends ConfigFormItem\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super(configParam, editEnabled, inDialog);\n\t}\n\n\t/**\n\t * Create widget to render the ConfigParameter value.\n\t * If parameter is not editable, a simple label is used.\n\t * @override\n\t */\n\t_generateInnerWidget()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Set widget configuration\n\t\t\tlet config = {\n\t\t\t\ttype: 'checkbox',\n\t\t\t\tclass: '',\n\t\t\t\tid: this._data.name,\n\t\t\t\tname: this._data.name,\n\t\t\t\t'data-role': 'switch',\n\t\t\t};\n\n\t\t\t// Set widget attributes (see parent class)\n\t\t\tthis._setWidgetAttributes(config);\n\n\t\t\t// Set additional widget attributes based on validation rules (see parent class)\n\t\t\tthis._setWidgetValidationAttributes(config);\n\n\t\t\t// Create widget's html\n\t\t\tthis._widgetHtml = $('<input>', config);\n\t\t}\n\t\telse\n\t\t\tthis._widgetHtml = new ConfigLabel();\n\n\t\t// Return component\n\t\treturn this._widgetHtml;\n\t}\n\n\t/**\n\t * Initialize widget.\n\t * @override\n\t */\n   _initialize()\n   {\n\t   if (this._data.editable)\n\t   {\n\t\t   // Initialize kendo widget\n\t\t   kendo.init(this._widgetHtml);\n\n\t\t   // Save ref. to widget\n\t\t   this._innerWidget = this._widgetHtml.data('kendoSwitch');\n\n\t\t   // Enable value commit binding\n\t\t   this._innerWidget.bind('change', $.proxy(this._onValueInput, this));\n\t   }\n\n\t   // Proceed with initialization\n\t   super._initialize();\n   }\n\n\t/**\n\t * Set widget's value.\n\t * If parameter is not editable, the label text is set.\n\t * @override\n\t */\n\t_setWidgetValue()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._innerWidget.value(this._data.value);\n\t\telse\n\t\t\tthis._widgetHtml.value = this._data.value;\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n\n\t/**\n\t * Set widget's disabled state.\n\t * @override\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._innerWidget.enable(this._editEnabled);\n\t}\n\n\t/**\n\t * Update Configuration Parameter value.\n\t * @override\n\t */\n\t_onValueInput(e)\n\t{\n\t\t// Update Configuration Parameter to new value\n\t\tthis._data.value = this._innerWidget.value();\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-check-box'))\n\twindow.customElements.define('config-check-box', ConfigCheckBox);\n","import {ConfigFormItem} from './config-form-item';\nimport {ConfigLabel} from './config-label';\n\nexport class ConfigDropDownList extends ConfigFormItem\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super(configParam, editEnabled, inDialog);\n\t}\n\n\t/**\n\t * Create widget to render the ConfigParameter value.\n\t * If parameter is not editable, a simple label is used.\n\t * @override\n\t */\n\t_generateInnerWidget()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Set widget configuration\n\t\t\tlet config = {\n\t\t\t\tclass: 'form-control',\n\t\t\t\tid: this._data.name,\n\t\t\t\tname: this._data.name,\n\t\t\t\t'data-role': 'dropdownlist',\n\t\t\t};\n\n\t\t\t// Set widget attributes (see parent class)\n\t\t\tthis._setWidgetAttributes(config);\n\n\t\t\t// Set additional widget attributes based on validation rules (see parent class)\n\t\t\tthis._setWidgetValidationAttributes(config);\n\n\t\t\t// Create widget's html\n\t\t\tthis._widgetHtml = $('<input>', config);\n\t\t}\n\t\telse\n\t\t\tthis._widgetHtml = new ConfigLabel();\n\n\t\t// Return component\n\t\treturn this._widgetHtml;\n\t}\n\n\t/**\n\t * Initialize widget.\n\t * @override\n\t */\n   _initialize()\n   {\n\t   if (this._data.editable)\n\t   {\n\t\t   // Initialize kendo widget\n\t\t   kendo.init(this._widgetHtml);\n\n\t\t   // Save ref. to widget\n\t\t   this._innerWidget = this._widgetHtml.data('kendoDropDownList');\n\n\t\t   // Set list items\n\t\t   this._innerWidget.setDataSource(this._getDataSource(this._data.dataProvider))\n\n\t\t   // Enable value commit binding\n\t\t   this._widgetHtml.bind('change', $.proxy(this._onValueInput, this));\n\t   }\n\n\t   // Proceed with initialization\n\t   super._initialize();\n   }\n\n\t/**\n\t * Set widget's value.\n\t * If parameter is not editable, the label text is set.\n\t * @override\n\t */\n\t_setWidgetValue()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._innerWidget.value(this._data.value);\n\t\telse\n\t\t\tthis._widgetHtml.value = this._data.value;\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n\n\t/**\n\t * Set widget's disabled state.\n\t * @override\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._innerWidget.wrapper.attr('disabled', !this._editEnabled);\n\t}\n\n\t/**\n\t * Update Configuration Parameter value.\n\t * @override\n\t */\n\t_onValueInput(e)\n\t{\n\t\t// Update Configuration Parameter to new value\n\t\tthis._data.value = this._innerWidget.value();\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n\n\t_getDataSource(dpString)\n\t{\n\t\tif (dpString)\n\t\t\treturn dpString.split(',');\n\n\t\t// In case the dataprovider is empty, add at least the current value\n\t\telse\n\t\t\treturn [this._data.value];\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-drop-down-list'))\n\twindow.customElements.define('config-drop-down-list', ConfigDropDownList);\n","import {ConfigFormItem} from './config-form-item';\nimport {ConfigLabel} from './config-label';\n\nexport class ConfigDualList extends ConfigFormItem\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super(configParam, editEnabled, inDialog);\n\t}\n\n\t/**\n\t * Create widget to render the ConfigParameter.\n\t * @override\n\t */\n\t_generateInnerWidget()\n\t{\n\t\tthis._widgetHtml = $('<div>');\n\n\t\tconst availableId = this._getId(this._data.name, 'available');\n\t\tconst selectedId = this._getId(this._data.name, 'selected');\n\n\t\t// Create header for labels\n\t\tlet header = $('<div>', {class: 'form-label-container dual-list-labels'});\n\n\t\theader.append($('<label>', {\n\t\t\tclass: 'font-italic form-label dual-list-left-col' + (!this._data.editable ? ' no-interact' : ''),\n\t\t\tfor: availableId,\n\t\t}).text('Available'));\n\n\t\theader.append($('<label>', {\n\t\t\tclass: 'font-italic font-weight-bold form-label dual-list-right-col' + (!this._data.editable ? ' no-interact' : ''),\n\t\t\tfor: selectedId,\n\t\t}).text('Selected'));\n\n\t\tthis._widgetHtml.append(header);\n\n\t\t// Add available items list\n\t\tthis._availableListHtml = $('<select>', {\n\t\t\tid: availableId,\n\t\t\tclass: 'dual-list-left-col' + (!this._data.editable ? ' no-interact' : ''),\n\t\t});\n\t\tthis._widgetHtml.append(this._availableListHtml);\n\n\t\t// Add selected items list\n\t\tthis._selectedListHtml = $('<select>', {\n\t\t\tid: selectedId,\n\t\t\tclass: 'dual-list-right-col' + (!this._data.editable ? ' no-interact' : ''),\n\t\t});\n\t\tthis._widgetHtml.append(this._selectedListHtml);\n\n\t\t// Return component\n\t\treturn this._widgetHtml;\n\t}\n\n\t// IDs containing a \".\" cause issues to connected lists\n\t_getId(name, suffix)\n\t{\n\t\treturn name.replace('.', '_') + '-' + suffix;\n\t}\n\n\t/**\n\t * Initialize widget.\n\t * @override\n\t */\n\t_initialize()\n\t{\n\t\t// Initialize \"avalable\" listbox\n\t\tthis._availableList = this._availableListHtml.kendoListBox({\n            connectWith: this._getId(this._data.name, 'selected'),\n            toolbar: {\n                tools: this._data.editable ? ['transferTo', 'transferFrom', 'transferAllTo', 'transferAllFrom'] : []\n            },\n\t\t\ttemplate: \"<div>#:value#</div>\",\n\t\t\tselectable: 'multiple',\n        }).data('kendoListBox');\n\n\t\t// Initialize \"selected\" listbox\n        this._selectedList = this._selectedListHtml.kendoListBox({\n\t\t\ttemplate: \"<div>#:value#</div>\",\n\t\t\tselectable: 'multiple',\n\t\t\t// The following listeners can't be used because events are fired before the datasource is actually updated\n\t\t\t// We have to use a change event listener on the datasource (see below), even if not optimal\n\t\t\t//add: $.proxy(this._onValueInput, this),\n\t\t\t//remove: $.proxy(this._onValueInput, this),\n\t\t}).data('kendoListBox');\n\n\t\t// Proceed with initialization\n\t\tsuper._initialize();\n\t}\n\n\t/**\n\t * Set widget's datasource.\n\t * @override\n\t */\n\t_setWidgetValue()\n\t{\n\t\tlet availableArr = this._data.dataProvider != '' ? this._data.dataProvider.split(',') : [];\n\t\tlet selectedArr = this._data.value != '' ? this._data.value.split(',') : [];\n\n\t\t// Remove selected values from available values\n\t\tif (selectedArr.length > 0)\n\t\t{\n\t\t\tlet temp = [];\n\n\t\t\tfor (let val of availableArr)\n\t\t\t{\n\t\t\t\tif (selectedArr.indexOf(val) == -1)\n\t\t\t\t\ttemp.push(val);\n\t\t\t}\n\n\t\t\tavailableArr = temp;\n\t\t}\n\n\t\t// Convert lists of strings to lists of objects\n\t\tlet availableValues = [];\n\t\tfor (let val of availableArr)\n\t\t\tavailableValues.push({value: val});\n\n\t\tlet selectedValues = [];\n\t\tfor (let val of selectedArr)\n\t\t\tselectedValues.push({value: val});\n\n\t\t// Clear selection\n\t\tthis._availableList.clearSelection();\n\t\tthis._selectedList.clearSelection();\n\n\t\t// Set datasources\n\t\tthis._availableList.setDataSource(new kendo.data.DataSource({\n\t\t\tdata: availableValues\n\t\t}));\n\n\t\tthis._selectedList.setDataSource(new kendo.data.DataSource({\n\t\t\tdata: selectedValues,\n\t\t\t// We listen to the change event instead of the add/remove events on the listbox, because those are fired before the datasource is updated\n\t\t\t// This is not optimal because the event is fired for each item added to or removed from the datasource\n\t\t\tchange: $.proxy(this._onValueInput, this)\n\t\t}));\n\n\t\t// Disable editing\n\t\tif (!this._data.editable)\n\t\t{\n\t\t\tthis._availableList.enable('.k-item', false);\n\t\t\tthis._selectedList.enable('.k-item', false);\n\t\t}\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n\n\t/**\n\t * Set widget's disabled state.\n\t * @override\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Clear selection\n\t\t\tthis._availableList.clearSelection();\n\t\t\tthis._selectedList.clearSelection();\n\n\t\t\t// Enable/disable lists\n\t\t\tthis._availableList.wrapper.attr('disabled', !this._editEnabled);\n\t\t\tthis._selectedList.wrapper.attr('disabled', !this._editEnabled);\n\t\t}\n\t}\n\n\t/**\n\t * Update Configuration Parameter value.\n\t * @override\n\t */\n\t_onValueInput(e)\n\t{\n\t\tlet listData = this._selectedList.dataSource.data();\n\n\t\t// Update Configuration Parameter to new value\n\t\tthis._data.value = listData.map(e => e.value).join(',');\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-dual-list'))\n\twindow.customElements.define('config-dual-list', ConfigDualList);\n","export class ConfigFormItem extends HTMLElement\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super();\n\n\t\tthis.id = 'form-item-' + configParam.name;\n\t\tthis._editEnabled = editEnabled;\n\t\tthis._data = configParam;\n\n\t\t// Create form item view\n\t\tthis._buildView(inDialog);\n\n\t\t// Initialize form item\n\t\tthis._initialize();\n\t}\n\n\tconnectedCallback()\n\t{\n\t\t// Trigger event\n\t\t// NOTE: when a ConfigFormItem is instantiated, the _triggerEvent method is called as soon as its value is set.\n\t\t// When this happenso, due to the fact that the object is not yet in the DOM, the event is not catched by the listener\n\t\t// (which is attached to the outer container). So forcing the event to trigger again as soon as the ConfigFormItem\n\t\t// is appended to the DOM is needed.\n\t\tthis._triggerEvent();\n\t}\n\n\tset data(configParam)\n\t{\n\t\tthis._data = configParam;\n\t\tthis._setWidgetValue();\n\t}\n\n\tget data()\n\t{\n\t\treturn this._data;\n\t}\n\n\tset editEnabled(enable)\n\t{\n\t\tif (enable != this._editEnabled)\n\t\t{\n\t\t\tthis._editEnabled = enable;\n\t\t\tthis._setWidgetEditEnabled();\n\t\t}\n\t}\n\n\tget editEnabled()\n\t{\n\t\treturn this._editEnabled;\n\t}\n\n\t_buildView(isInsideDialog)\n\t{\n\t\tif (!isInsideDialog)\n\t\t{\n\t\t\t// Set additional classes for inner widget\n\t\t\tlet classNames = '';\n\n\t\t\tswitch (this._data.type)\n\t\t\t{\n\t\t\t\tcase 'DualList':\n\t\t\t\t\tclassNames = 'col-sm-7 col-lg-8';\n\t\t\t\t\tbreak;\n\t\t\t\tcase 'DataGrid':\n\t\t\t\t\tclassNames = 'col-sm'; // Use 'col-sm-7 col-lg-8' for DataGrid too?\n\t\t\t\t\tbreak;\n\t\t\t\tdefault:\n\t\t\t\t\tclassNames = 'col-sm-auto';\n\n\t\t\t}\n\n\t\t\t// Generate boilerplate html, surrounding the actual widget (label, numeric stepper, etc)\n\t\t\tthis.innerHTML = `\n\t\t\t\t<div class=\"form-group position-relative row\">\n\t\t\t\t\t<div class=\"col-sm-5 col-lg-4 col-form-label form-label-container\">\n\t\t\t\t\t\t<label for=\"${this._data.name}\" class=\"form-label ${(this._data.clientOnly ? 'client-only' : '')}\">${this._data.label} <i class=\"fas fa-question-circle text-muted help\" title=\"${this._data.tooltip}\"></i></label>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div class=\"inner-widget align-self-center ${classNames}\">\n\t\t\t\t\t\t<span class=\"k-invalid-msg\" data-for=\"${this._data.name}\"></span>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t`;\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis.innerHTML = `\n\t\t\t\t<div class=\"form-group position-relative\">\n\t\t\t\t\t<div class=\"col-form-label form-label-container\">\n\t\t\t\t\t\t<label for=\"${this._data.name}\" class=\"form-label ${(this._data.clientOnly ? 'client-only' : '')}\">${this._data.label} <i class=\"fas fa-question-circle text-muted help\" title=\"${this._data.tooltip}\"></i></label>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div class=\"inner-widget\">\n\t\t\t\t\t\t<span class=\"k-invalid-msg\" data-for=\"${this._data.name}\"></span>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t`;\n\t\t}\n\n\t\t// Create inner widget (must be overridden)\n\t\tlet widget = this._generateInnerWidget();\n\n\t\t// Append inner widget\n\t\t$(this).find('.inner-widget').prepend(widget);\n\t}\n\n\t/**\n\t * TO BE OVERRIDDEN\n\t */\n\t_generateInnerWidget()\n\t{\n\t\t// Show an error, should be overridden\n\t\tconsole.error(`Unable to create ${this._data.type} form item for configuration parameter ${this.id}`);\n\t}\n\n\t/**\n\t * Set attributes on the widget configuration object.\n\t */\n\t_setWidgetAttributes(config)\n\t{\n\t\tconst attribs = this._data.attributes;\n\n\t\tif (attribs)\n\t\t{\n\t\t\tfor (let attr in attribs)\n\t\t\t{\n\t\t\t\tconfig[attr] = attribs[attr];\n\n\t\t\t\tif (attr == 'pattern')\n\t\t\t\t\tconfig['data-pattern-msg'] = 'Contains invalid characters';\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Set additional attributes on the widget configuration object to properly validate input.\n\t */\n\t_setWidgetValidationAttributes(config)\n\t{\n\t\tconst val = this._data.validator;\n\n\t\tif (val != null && val != '')\n\t\t{\n\t\t\tif (val == 'ip')\n\t\t\t{\n\t\t\t\tconfig['pattern'] = '^(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$';\n\t\t\t\tconfig['data-pattern-msg'] = 'Invalid IP address';\n\t\t\t\tconfig['required'] = true;\n\t\t\t\tconfig['data-required-msg'] = 'Required';\n\t\t\t}\n\n\t\t\telse if (val == 'notNull')\n\t\t\t{\n\t\t\t\tconfig['required'] = true;\n\t\t\t\tconfig['data-required-msg'] = 'Required';\n\t\t\t}\n\n\t\t\telse if (val == 'pwd')\n\t\t\t{\n\t\t\t\tconfig['pattern'] = '^.{6,}$';\n\t\t\t\tconfig['data-pattern-msg'] = 'Minimum length: 6 characters';\n\t\t\t}\n\n\t\t\telse if (val == 'posNum')\n\t\t\t{\n\t\t\t\tconfig['pattern'] = '^[0-9]\\d*$';\n\t\t\t\tconfig['data-pattern-msg'] = 'Non-negative number required';\n\t\t\t}\n\n\t\t\telse if (val == 'aoi')\n\t\t\t{\n\t\t\t\t// Nothing to do\n\t\t\t\t// See Kendo validation initialization in config-interface-builder.js\n\t\t\t}\n\n\t\t\telse if (val == 'url')\n\t\t\t{\n\t\t\t\tconfig['type'] = 'url';\n\t\t\t\tconfig['data-url-msg'] = 'Invalid URL';\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Initialize form item.\n\t *\n\t * NOTE: must be overridden if inner widget requires special initialization (for example Kendo widgets)\n\t */\n\t_initialize()\n\t{\n\t\t// Set value\n \t   this._setWidgetValue();\n\n \t   // Set edit enabled\n \t   this._setWidgetEditEnabled();\n\t}\n\n\t/**\n\t * TO BE OVERRIDDEN\n\t */\n\t_setWidgetValue()\n\t{\n\t\t// Nothing to do, must be overridden\n\t}\n\n\t/**\n\t * TO BE OVERRIDDEN\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\t// Nothing to do, must be overridden\n\t}\n\n\t/**\n\t * TO BE OVERRIDDEN\n\t */\n\t_onValueInput(e)\n\t{\n\t\t// Nothing to do, must be overridden\n\t}\n\n\t_triggerEvent()\n\t{\n\t\tif (this._data.trigger)\n\t\t{\n\t\t\tlet event = new CustomEvent('value-set', {detail: null, bubbles: true, cancelable: true});\n\t\t\tthis.dispatchEvent(event);\n\t\t}\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-form-item'))\n\twindow.customElements.define('config-form-item', ConfigFormItem);\n","import {ConfigFormItem} from './config-form-item';\nimport {ConfigLabel} from './config-label';\nimport {ListItemEditor} from './widgets/list-item-editor';\n\nexport class ConfigGrid extends ConfigFormItem\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super(configParam, editEnabled, inDialog);\n\t}\n\n\t/**\n\t * Create widget to render the ConfigParameter.\n\t * @override\n\t */\n\t_generateInnerWidget()\n\t{\n\t\t// Create main widget's html\n\t\tthis._widgetHtml = $('<div>', {class: ''});\n\n\t\t// Set grid widget configuration\n\t\tlet gridConfig = {\n\t\t\tid: this._data.name,\n\t\t\tname: this._data.name,\n\t\t\tclass: 'limited-height' + (!this._data.editable ? ' no-interact' : '')\n\t\t};\n\n\t\t// Append grid to main html; grid will be converted to Kendo widget during initialization\n\t\tthis._widgetHtml.append($('<div>', gridConfig));\n\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// BUTTONS\n\n\t\t\t// Create buttons container\n\t\t\tlet buttons = $('<div>', {class: 'mt-2 text-right'});\n\n\t\t\t// Append buttons to container\n\t\t\tthis._addButton = $('<button>', {type: 'button', class: 'k-button k-secondary', title: 'Add'}).append($('<i class=\"fas fa-plus\"></i>'));\n\t\t\tthis._editButton = $('<button>', {type: 'button', class: 'k-button k-secondary ml-2', title: 'Edit', disabled: true}).append($('<i class=\"fas fa-pen\"></i>'));\n\t\t\tthis._removeButton = $('<button>', {type: 'button', class: 'k-button k-secondary ml-2', title: 'Remove', disabled: true}).append($('<i class=\"fas fa-times\"></i>'));\n\n\t\t\tbuttons.append(this._addButton);\n\t\t\tbuttons.append(this._editButton);\n\t\t\tbuttons.append(this._removeButton);\n\n\t\t\t// Append buttons container to main html\n\t\t\tthis._widgetHtml.append(buttons);\n\n\t\t\t// Create edit dialog\n\t\t\t// NOTE: data-dismiss=\"modal\" on the close/cancel buttons was removed to work around an issue with nested modals;\n\t\t\t// the custom \"data-cancel\" attribute is used to add a custom listener to the buttons\n\t\t\tthis._editDialog = $(`\n\t\t\t\t<div class=\"modal\" tabindex=\"-1\" role=\"dialog\" aria-labelledby=\"modalTitle\" aria-hidden=\"true\" data-keyboard=\"false\" data-backdrop=\"static\">\n\t\t\t\t\t<div class=\"modal-dialog modal-dialog-centered\" role=\"document\">\n\t\t\t\t\t\t<div class=\"modal-content\">\n\t\t\t\t\t\t\t<div class=\"modal-header\">\n\t\t\t\t\t\t\t\t<h5 class=\"modal-title text-primary\" id=\"modalTitle\">${this._data.label}</h5>\n\t\t\t\t\t\t\t\t<button type=\"button\" class=\"close\" aria-label=\"Close\" data-cancel=\"modal\">\n\t\t\t\t\t\t\t\t\t<span aria-hidden=\"true\">&times;</span>\n\t\t\t\t\t\t\t\t</button>\n\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t<div class=\"modal-body in-flow-invalid-msg\">\n\n\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t<div class=\"modal-footer flex-column\">\n\t\t\t\t\t\t\t\t<div class=\"d-flex w-100\">\n\t\t\t\t\t\t\t\t\t<div class=\"flex-grow-1 text-left\">\n\t\t\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary\">...</button>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t<div class=\"flex-grow-1 text-right\">\n\t\t\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-secondary\" data-cancel=\"modal\">Cancel</button>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t`);\n\n\t\t\t// Add listener to dialog hide event\n\t\t\tthis._editDialog.on('hidden.bs.modal', $.proxy(this._onEditPanelHidden, this));\n\n\t\t\t// Add listener to main button click event\n\t\t\t$('button.k-primary', this._editDialog).on('click', $.proxy(this._onSubmitBtClick, this));\n\n\t\t\t// Add listener to close/cancel buttons click event\n\t\t\t$('button[data-cancel=\"modal\"]', this._editDialog).on('click', $.proxy(this._onCancelBtClick, this));\n\n\t\t\t// Append edit dialog to main html\n\t\t\tthis._widgetHtml.append(this._editDialog);\n\t\t}\n\n\t\t// Return component\n\t\treturn this._widgetHtml;\n\t}\n\n\t/**\n\t * Initialize widget.\n\t * @override\n\t */\n\t_initialize()\n\t{\n\t\tlet columns = [];\n\t\tfor (let subConfigParam of this._data.defaultListItem)\n\t\t{\n\t\t\tlet col = {\n\t\t\t\tfield: subConfigParam.name,\n\t\t\t\ttitle: subConfigParam.label,\n\t\t\t\twidth: 120\n\t\t\t}\n\n\t\t\t// Display V or X for booleans\n\t\t\tif (typeof subConfigParam.value === 'boolean')\n\t\t\t\tcol.template = `#= ${subConfigParam.name} ? '<i class=\"fas fa-check\"></i>' : '<i class=\"fas fa-times\"></i>' #`;\n\n\t\t\t// Hide passwords\n\t\t\tif (subConfigParam.type == 'TextInput' && subConfigParam.attributes != null && subConfigParam.attributes.type == 'password')\n\t\t\t\tcol.template = `#= '•'.repeat(data.${subConfigParam.name}.length) #`;\n\n\t\t\tcolumns.push(col);\n\t\t}\n\n\t\t// Initialize grid\n\t\tlet gridHtml = this._widgetHtml.find(`#${$.escapeSelector(this._data.name)}`);\n\n\t\tgridHtml.kendoGrid({\n\t\t\tresizable: true,\n\t\t\tselectable: this._data.editable ? 'row' : false,\n\t\t\tchange: $.proxy(this._onGridSelectionChange, this),\n\t\t\tcolumns: columns,\n\t\t\tnoRecords: {\n\t\t\t\ttemplate: 'No items.'\n\t\t\t}\n\t\t});\n\n\t\t// Save ref. to widget\n\t\tthis._gridWidget = gridHtml.data('kendoGrid');\n\n\t\t// Show tootip if grid's cell content exceeds cell width (ellipsis is displayed by Kendo Grid)\n\t\tgridHtml.kendoTooltip({\n\t\t\tfilter: 'td',\n\t\t\tshow: function(e) {\n\t\t\t\t// Never show tooltip...\n\t\t\t\tthis.content.parent().css('visibility', 'hidden');\n\n\t\t\t\t// ...unless content is returned (see below) due to cell width being exceeded\n\t\t\t\tif (this.content.text() != '')\n\t\t\t\t\tthis.content.parent().css('visibility', 'visible');\n\t\t\t},\n\t\t\thide: function() {\n\t\t\t\tthis.content.parent().css('visibility', 'hidden');\n\t\t\t},\n\t\t\tcontent: function(e) {\n\t\t\t\tlet element = e.target[0];\n\t\t\t\tif (element.offsetWidth < element.scrollWidth)\n\t\t\t\t\treturn e.target.text();\n\t\t\t\telse\n\t\t\t\t\treturn '';\n\t\t\t}\n\t\t});\n\n\t\t/*\n\t\t// Initialize button tooltips\n\t\tthis._widgetHtml.kendoTooltip({\n\t\t\tfilter: 'button',\n\t\t\tcontent: function(e) {\n\t\t\t\treturn `<div class=\"help-tooltip\">${e.target.data('title')}</div>`;\n\t\t\t}\n\t\t});\n\t\t*/\n\n\t\t// Add button listeners\n\t\tif (this._data.editable)\n\t\t{\n\t\t\tthis._addButton.click($.proxy(this._onAddClick, this));\n\t\t\tthis._editButton.click($.proxy(this._onEditClick, this));\n\t\t\tthis._removeButton.click($.proxy(this._onRemoveClick, this));\n\t\t}\n\n\t\t// Proceed with initialization\n\t\tsuper._initialize();\n\t}\n\n\t/**\n\t * Set widget's datasource.\n\t * @override\n\t */\n\t_setWidgetValue()\n\t{\n\t\tlet dataSource = new kendo.data.DataSource({\n\t\t\tdata: this._data.listValues\n\t\t});\n\n\t\t// Read current horizontal scroll value\n\t\tconst scrollLeft = $('.k-grid-content', this._gridWidget.wrapper).scrollLeft();\n\n\t\t// Clear grid selection if any\n\t\tthis._gridWidget.clearSelection();\n\n\t\t// Set updated grid's datasource\n\t\tthis._gridWidget.setDataSource(dataSource);\n\n\t\t// Set horizontal scroll\n\t\t$('.k-grid-content', this._gridWidget.wrapper).scrollLeft(scrollLeft);\n\t}\n\n\t/**\n\t * Set widget's disabled state.\n\t * @override\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Deselect item\n\t\t\tthis._gridWidget.clearSelection();\n\n\t\t\t// Enable/disable grid\n\t\t\tthis._gridWidget.wrapper.attr('disabled', !this._editEnabled);\n\n\t\t\t// Enable \"Add\" button\n\t\t\tif (this._editEnabled)\n\t\t\t\tthis._addButton.attr('disabled', false);\n\n\t\t\t// Disable all buttons\n\t\t\telse\n\t\t\t{\n\t\t\t\tthis._addButton.attr('disabled', true);\n\t\t\t\tthis._editButton.attr('disabled', true);\n\t\t\t\tthis._removeButton.attr('disabled', true);\n\t\t\t}\n\t\t}\n\t}\n\n\t_onGridSelectionChange(e)\n\t{\n\t\tlet selectedRows = this._gridWidget.select();\n\t\tlet selectedDataItems = [];\n\n\t\tfor (let i = 0; i < selectedRows.length; i++)\n\t\t{\n\t\t\tlet dataItem = this._gridWidget.dataItem(selectedRows[i]);\n\t\t\tselectedDataItems.push(dataItem);\n\t\t}\n\n\t\t// Enable/disable edit button\n\t\tif (this._editButton)\n\t\t\tthis._editButton.prop('disabled', selectedDataItems.length == 0);\n\n\t\t// Enable/disable remove button\n\t\tif (this._removeButton)\n\t\t\tthis._removeButton.prop('disabled', selectedDataItems.length == 0);\n    }\n\n\t_onRemoveClick()\n\t{\n\t\tlet selectedIndex = this._gridWidget.select().index();\n\n\t\t// Remove item from list\n\t\tthis._data.removeListItem(selectedIndex);\n\n\t\t// Regenerate datagrid's datasource\n\t\tthis._setWidgetValue();\n\t}\n\n\t_onAddClick()\n\t{\n\t\t// Clone default item and add to list\n\t\tlet newListItem = [];\n\t\tfor (let subCP of this._data.defaultListItem)\n\t\t\tnewListItem.push(subCP.clone(true));\n\n\t\t// Create edit popup\n\t\tthis._openEditPanel(newListItem);\n\t}\n\n\t_onEditClick()\n\t{\n\t\tlet selectedIndex = this._gridWidget.select().index();\n\n\t\t// Clone selected item and add to list\n\t\tlet clonedListItem = [];\n\t\tfor (let subCP of this._data.listItems[selectedIndex])\n\t\t\tclonedListItem.push(subCP.clone(true));\n\n\t\t// Create edit popup\n\t\tthis._openEditPanel(clonedListItem, selectedIndex);\n\t}\n\n\t_openEditPanel(subConfigParamsArray, editIndex = -1)\n\t{\n\t\t// Check if this configuration item is inside a modal window;\n\t\t// if yes, the edit panel (which is a modal as well) must be configured to remove the dark background\n\t\tif ($(this).parents('.modal').length > 0)\n\t\t\t$('.modal', $(this)).attr('data-backdrop', false);\n\n\t\t// Create dialog content\n\t\tthis._itemEditor = new ListItemEditor();\n\t\tthis._itemEditor.data = subConfigParamsArray;\n\t\tthis._itemEditor.index = editIndex;\n\n\t\tlet itemEditor = $(this._itemEditor);\n\n\t\t// Append content to dialog\n\t\t$('.modal-body', this._editDialog).append(itemEditor);\n\n\t\t// Set dialog main button text\n\t\t$('button.k-primary', this._editDialog).html(editIndex > -1 ? '<i class=\"fas fa-pen mr-1\"></i>Update' : '<i class=\"fas fa-plus mr-1\"></i>Add');\n\n\t\t// Display dialog\n\t\tthis._editDialog.modal('show');\n\t}\n\n\t_onSubmitBtClick()\n\t{\n\t\tif (this._itemEditor.validate())\n\t\t{\n\t\t\tlet data = this._itemEditor.data;\n\t\t\tlet index = this._itemEditor.index;\n\n\t\t\t// Hide modal\n\t\t\tthis._editDialog.modal('hide');\n\n\t\t\t// Complete editing\n\t\t\tthis._onEditComplete(data, index);\n\t\t}\n\t}\n\n\t_onCancelBtClick()\n\t{\n\t\t// Hide modal\n\t\tthis._editDialog.modal('hide');\n\t}\n\n\t_onEditPanelHidden(e)\n\t{\n\t\t// Remove content from dialog\n\t\tthis._itemEditor.remove();\n\n\t\t// Set dialog main button text\n\t\t$('button.k-primary', this._editDialog).html('...');\n\n\t\tthis._itemEditor = null;\n\t}\n\n\t_onEditComplete(listItem, editIndex)\n\t{\n\t\t// An existing list item was updated\n\t\tif (editIndex > -1)\n\t\t\tthis._data.updateListItem(listItem, editIndex);\n\n\t\t// A new list item was added; add it to the configuration parameter\n\t\telse\n\t\t\tthis._data.addListItem(listItem);\n\n\t\t// Regenerate datagrid's datasource\n\t\tthis._setWidgetValue();\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-grid'))\n\twindow.customElements.define('config-grid', ConfigGrid);\n","export class ConfigLabel extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\n\t\tthis.setAttribute('class','config-label');\n\t}\n\n\tset value(val)\n\t{\n\t\tif (typeof val === 'boolean')\n\t\t\tthis.innerHTML = (val ? 'true' : 'false');\n\t\telse if (typeof val === 'number')\n\t\t\tthis.innerHTML = (val ? val : 0);\n\t\telse\n\t\t\tthis.innerHTML = (val != '' ? val : '&mdash;');\n\t}\n\n\tget value()\n\t{\n\t\treturn this.textContent;\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-label'))\n\twindow.customElements.define('config-label', ConfigLabel);\n","import {ConfigFormItem} from './config-form-item';\nimport {ConfigLabel} from './config-label';\n\nexport class ConfigNumericStepper extends ConfigFormItem\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super(configParam, editEnabled, inDialog);\n\t}\n\n\t/**\n\t * Create widget to render the ConfigParameter value.\n\t * If parameter is not editable, a simple label is used.\n\t * @override\n\t */\n\t_generateInnerWidget()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Set widget configuration\n\t\t\tlet config = {\n\t\t\t\ttype: 'number',\n\t\t\t\tclass: 'form-control',\n\t\t\t\tid: this._data.name,\n\t\t\t\tname: this._data.name,\n\t\t\t\t'data-role': 'numerictextbox',\n\t\t\t\t'data-required-msg': 'Required',\n\t\t\t\t'data-format': '#',\n\t\t\t\trequired: 'required',\n\t\t\t};\n\n\t\t\t// Set widget attributes (see parent class)\n\t\t\tthis._setWidgetAttributes(config);\n\n\t\t\t// Set additional widget attributes based on validation rules (see parent class)\n\t\t\tthis._setWidgetValidationAttributes(config);\n\n\t\t\t// Create widget's html\n\t\t\tthis._widgetHtml = $('<input>', config);\n\t\t}\n\t\telse\n\t\t\tthis._widgetHtml = new ConfigLabel();\n\n\t\t// Return component\n\t\treturn this._widgetHtml;\n\t}\n\n\t/**\n\t * Initialize widget.\n\t * @override\n\t */\n   _initialize()\n   {\n\t   if (this._data.editable)\n\t   {\n\t\t   // Initialize kendo widget\n\t\t   kendo.init(this._widgetHtml);\n\n\t\t   // Save ref. to widget\n\t\t   this._innerWidget = this._widgetHtml.data('kendoNumericTextBox');\n\n\t\t   // Enable value commit binding\n\t\t   this._innerWidget.bind('change', $.proxy(this._onValueInput, this));\n\t   }\n\n\t   // Proceed with initialization\n\t   super._initialize();\n   }\n\n\t/**\n\t * Set widget's value.\n\t * If parameter is not editable, the label text is set.\n\t * @override\n\t */\n\t_setWidgetValue()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._innerWidget.value(this._data.value);\n\t\telse\n\t\t\tthis._widgetHtml.value = this._data.value;\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n\n\t/**\n\t * Set widget's disabled state.\n\t * @override\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._innerWidget.enable(this._editEnabled);\n\t}\n\n\t/**\n\t * Update Configuration Parameter value.\n\t * @override\n\t */\n\t_onValueInput(e)\n\t{\n\t\t// Update Configuration Parameter to new value\n\t\tthis._data.value = Number(this._innerWidget.value());\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-numeric-stepper'))\n\twindow.customElements.define('config-numeric-stepper', ConfigNumericStepper);\n","import {ConfigFormItem} from './config-form-item';\nimport {ConfigLabel} from './config-label';\n\nexport class ConfigTextInput extends ConfigFormItem\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super(configParam, editEnabled, inDialog);\n\t}\n\n\t/**\n\t * Create widget to render the ConfigParameter value.\n\t * If parameter is not editable, a simple label is used.\n\t * @override\n\t */\n\t_generateInnerWidget()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Set widget configuration\n\t\t\tlet config = {\n\t\t\t\ttype: 'text',\n\t\t\t\tclass: 'form-control k-textbox',\n\t\t\t\tid: this._data.name,\n\t\t\t\tname: this._data.name,\n\t\t\t\tautocomplete: 'off',\n\t\t\t};\n\n\t\t\t// Set widget attributes\n\t\t\tthis._setWidgetAttributes(config);\n\n\t\t\t// Set additional widget attributes based on validation rules\n\t\t\tthis._setWidgetValidationAttributes(config);\n\n\t\t\t// Create widget's html\n\t\t\tthis._widgetHtml = $('<input>', config);\n\n\t\t\t// Enable value commit binding\n\t\t\tthis._widgetHtml.on('change', $.proxy(this._onValueInput, this));\n\t\t}\n\t\telse\n\t\t\tthis._widgetHtml = new ConfigLabel();\n\n\t\t// Return component\n\t\treturn this._widgetHtml;\n\t}\n\n\t/**\n\t * Set widget's value.\n\t * If parameter is not editable, the label text is set.\n\t * @override\n\t */\n\t_setWidgetValue()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._widgetHtml.val(this._data.value);\n\t\telse\n\t\t\tthis._widgetHtml.value = this._data.value;\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n\n\t/**\n\t * Set widget's disabled state.\n\t * @override\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\tif (this._data.editable)\n\t\t\tthis._widgetHtml.attr('disabled', !this._editEnabled);\n\t}\n\n\t/**\n\t * Update Configuration Parameter value.\n\t * @override\n\t */\n\t_onValueInput(e)\n\t{\n\t\t// Update Configuration Parameter to new value\n\t\tthis._data.value = this._widgetHtml.val();\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-text-input'))\n\twindow.customElements.define('config-text-input', ConfigTextInput);\n","import {ConfigFormItem} from './config-form-item';\nimport {ConfigLabel} from './config-label';\nimport {Vector3DInput} from './widgets/vector-3d-input';\n\nexport class ConfigVector3D extends ConfigFormItem\n{\n\tconstructor(configParam, editEnabled, inDialog)\n\t{\n\t    super(configParam, editEnabled, inDialog);\n\t}\n\n\t/**\n\t * Create widget to render the ConfigParameter value.\n\t * If parameter is not editable, a simple label is used.\n\t * @override\n\t */\n\t_generateInnerWidget()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Create widget's html\n\t\t\tthis._widgetHtml = new Vector3DInput(this._data.name, this._data.validator == 'aoi');\n\n\t\t\t// Set widget attributes\n\t\t\tthis._setWidgetAttributes(this._widgetHtml);\n\n\t\t\t// Enable value commit binding\n\t\t\t$(this._widgetHtml).on('change', $.proxy(this._onValueInput, this));\n\t\t}\n\t\telse\n\t\t\tthis._widgetHtml = new ConfigLabel();\n\n\t\t// Return component\n\t\treturn this._widgetHtml;\n\t}\n\n\t/**\n\t * Set widget's value.\n\t * If parameter is not editable, the label text is set.\n\t * @override\n\t */\n\t_setWidgetValue()\n\t{\n\t\tthis._widgetHtml.value = this._data.value;\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n\n\t/**\n\t * Set widget's disabled state.\n\t * @override\n\t */\n\t_setWidgetEditEnabled()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t$(this._widgetHtml).attr('disabled', !this._editEnabled);\n\t\t}\n\t}\n\n\t/**\n\t * Update Configuration Parameter value.\n\t * @override\n\t */\n\t_onValueInput(e)\n\t{\n\t\t// Update Configuration Parameter to new value\n\t\tthis._data.value = this._widgetHtml.value;\n\n\t\t// Trigger event\n\t\tthis._triggerEvent();\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('config-vector-3d'))\n\twindow.customElements.define('config-vector-3d', ConfigVector3D);\n","import {ConfigFormItemFactory} from '../../../utils/uibuilder/config-form-item-factory';\n\nexport class ListItemEditor extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\t}\n\n\tset data(subConfigParamsArray)\n\t{\n\t\tthis._data = subConfigParamsArray;\n\n\t\tthis._buildView();\n\t}\n\n\tget data()\n\t{\n\t\treturn this._data;\n\t}\n\n\tset index(index)\n\t{\n\t\tthis._index = index;\n\t}\n\n\tget index()\n\t{\n\t\treturn this._index;\n\t}\n\n\t_buildView()\n\t{\n\t\t// Generate container form\n\t\tthis._form = $('<form>', {});\n\n\t\t// Append form\n\t\t$(this).append(this._form);\n\n\t\t// Generate form fields\n\t\tfor (let configParam of this._data)\n\t\t{\n\t\t\t// Create form item\n\t\t\tlet formItem = ConfigFormItemFactory.create(configParam, true, true);\n\t\t\tformItem.data = configParam;\n\n\t\t\t// Add form item to form\n\t\t\tthis._form.append(formItem);\n\t\t}\n\n\t\t// Initialize kendo validation on form\n\t\tthis._validator = this._form.kendoValidator({\n\t\t\tvalidateOnBlur: true,\n\t\t\trules: {\n\t\t\t\t// Add rule to validate AOI form items?\n\t\t\t\t// (see: https://demos.telerik.com/kendo-ui/validator/custom-validation)\n\t\t\t\taoi: function (input) {\n\t\t\t\t\tif (input.is('[data-aoi-msg]') && input.val() != '')\n\t\t\t\t\t{\n\t\t\t\t\t\tif (input.val() == '0,0,0')\n\t\t\t\t\t\t\treturn false;\n                    }\n\n                    return true;\n                }\n\t\t\t}\n\t\t}).data('kendoValidator');\n\t}\n\n\tvalidate()\n\t{\n\t\treturn this._validator.validate();\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('list-item-editor'))\n\twindow.customElements.define('list-item-editor', ListItemEditor);\n","export class Vector3DInput extends HTMLElement\n{\n\tconstructor(id, isValidable)\n\t{\n\t    super();\n\n\t\tthis.id = id;\n\t\tthis.name = id;\n\n\t\tthis._isValidable = isValidable;\n\n\t\tthis._initialize();\n\t}\n\n\tset enableClear(value)\n\t{\n\t\tif (value)\n\t\t\tthis._clearButton.show();\n\t\telse\n\t\t\tthis._clearButton.hide();\n\t}\n\n\tset allowNegative(value)\n\t{\n\t\tif (value)\n\t\t{\n\t\t\tthis._widgetX.setOptions( {min: null} );\n\t\t\tthis._widgetY.setOptions( {min: null} );\n\t\t\tthis._widgetZ.setOptions( {min: null} );\n\t\t}\n\t}\n\n\tset value(val)\n\t{\n\t\tvar coords = val.split(',');\n\n\t\tif (coords.length >= 1)\n\t\t\tthis._widgetX.value(coords[0]);\n\n\t\tif (coords.length >= 2)\n\t\t\tthis._widgetY.value(coords[1]);\n\n\t\tif (coords.length >= 3)\n\t\t\tthis._widgetZ.value(coords[2]);\n\n\t\tif (this._isValidable)\n\t\t\tthis._inputVal.val(this.value);\n\t}\n\n\tget value()\n\t{\n\t\tif (this._widgetX.value() == null && this._widgetY.value() == null && this._widgetZ.value() == null)\n\t\t\treturn '';\n\t\telse\n\t\t\treturn this._widgetX.value() + ',' + this._widgetY.value() + ',' + this._widgetZ.value();\n\t}\n\n\t_initialize()\n\t{\n\t\t// Generate container form\n\t\tthis._container = $('<div>', {\n\t\t\tclass: 'form-inline'\n\t\t});\n\n\t\t// Append container\n\t\t$(this).append(this._container);\n\n\t\t// Set inputs configuration\n\t\tlet configHtml = {\n\t\t\ttype: 'number',\n\t\t\tclass: 'form-control short-4',\n\t\t};\n\n\t\t// Set widget configuration\n\t\tlet configWidget = {\n\t\t\tmin: 0,\n\t\t\tspinners: false,\n\t\t\tformat: '#.######',\n\t\t\tdecimals: 6,\n\t\t\tround: false,\n\t\t\tspinners: false,\n\t\t\trestrictDecimals: false,\n\t\t\tchange: $.proxy(this._onChange, this)\n\t\t};\n\n\t\t// Create widgets\n\t\tthis._inputX = $('<input>', configHtml);\n\t\tthis._container.append(this._inputX);\n\t\tthis._widgetX = this._inputX.kendoNumericTextBox(configWidget).data('kendoNumericTextBox');\n\n\t\tthis._container.append('<span class=\"px-1\">,</span>');\n\n\t\tthis._inputY = $('<input>', configHtml);\n\t\tthis._container.append(this._inputY);\n\t\tthis._widgetY = this._inputY.kendoNumericTextBox(configWidget).data('kendoNumericTextBox');\n\n\t\tthis._container.append('<span class=\"px-1\">,</span>');\n\n\t\tthis._inputZ = $('<input>', configHtml);\n\t\tthis._container.append(this._inputZ);\n\t\tthis._widgetZ = this._inputZ.kendoNumericTextBox(configWidget).data('kendoNumericTextBox');\n\n\t\tthis._container.append('<span class=\"px-1\"></span>'); // Additional spacer\n\n\t\t// Create invisible field to apply overall validation\n\t\tif (this._isValidable)\n\t\t{\n\t\t\tthis._inputVal = $('<input>', {name: `${this.name}-custom-validate`, 'data-aoi-msg': 'Values can\\'t all be 0'});\n\t\t\tthis._container.append(this._inputVal);\n\t\t\tthis._container.append(`<span class=\"k-invalid-msg\" data-for=\"${this.name}-custom-validate\"></span>`)\n\t\t\tthis._inputVal.hide();\n\t\t}\n\n\t\t// Create and append Clear button\n\t\tthis._clearButton = $('<button>', {type: 'button', class: 'k-button k-secondary my-1', title: 'Clear'}).append($('<i class=\"fas fa-times\"></i>'));\n\t\tthis._clearButton.on('click', $.proxy(this._onClearClick, this));\n\t\tthis._container.append(this._clearButton);\n\n\t\t// Hide button by default\n\t\tthis._clearButton.hide();\n\t}\n\n\t_onChange()\n\t{\n\t\t// Empty strings are not allowed\n\t\tif (this._widgetX.value() == null)\n\t\t\tthis._widgetX.value(0);\n\n\t\tif (this._widgetY.value() == null)\n\t\t\tthis._widgetY.value(0);\n\n\t\tif (this._widgetZ.value() == null)\n\t\t\tthis._widgetZ.value(0);\n\n\t\tthis._dispatchCommit();\n\t}\n\n\t_onClearClick()\n\t{\n\t\tthis._widgetX.value('');\n\t\tthis._widgetY.value('');\n\t\tthis._widgetZ.value('');\n\n\t\tthis._dispatchCommit();\n\t}\n\n\t_dispatchCommit()\n\t{\n\t\tif (this._isValidable)\n\t\t\tthis._inputVal.val(this.value);\n\n\t\tthis.dispatchEvent(new Event('change'));\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('vector-3d-input'))\n\twindow.customElements.define('vector-3d-input', Vector3DInput);\n","import {ConfigFormItem} from '../../components/uibuilder/config-form-item';\n\nimport {ConfigNumericStepper} from '../../components/uibuilder/config-numeric-stepper';\nimport {ConfigTextInput} from '../../components/uibuilder/config-text-input';\nimport {ConfigCheckBox} from '../../components/uibuilder/config-check-box';\nimport {ConfigDropDownList} from '../../components/uibuilder/config-drop-down-list';\nimport {ConfigGrid} from '../../components/uibuilder/config-grid';\nimport {ConfigDualList} from '../../components/uibuilder/config-dual-list';\nimport {ConfigVector3D} from '../../components/uibuilder/config-vector-3d';\n\nexport class ConfigFormItemFactory\n{\n\tstatic create(configParam, editEnabled, inDialog = false)\n\t{\n\t\tswitch (configParam.type)\n\t\t{\n\t\t\tcase 'TextInput':\n\t\t\t\treturn new ConfigTextInput(configParam, editEnabled, inDialog);\n\t\t\t\tbreak;\n\n\t\t\tcase 'CheckBox':\n\t\t\t\treturn new ConfigCheckBox(configParam, editEnabled, inDialog);\n\t\t\t\tbreak;\n\n\t\t\tcase 'NumericStepper':\n\t\t\t\treturn new ConfigNumericStepper(configParam, editEnabled, inDialog);\n\t\t\t\tbreak;\n\n\t\t\tcase 'ComboBox':\n\t\t\t\treturn new ConfigDropDownList(configParam, editEnabled, inDialog);\n\t\t\t\tbreak;\n\n\t\t\tcase 'DataGrid':\n\t\t\t\treturn new ConfigGrid(configParam, editEnabled, inDialog);\n\t\t\t\tbreak;\n\n\t\t\tcase 'DualList':\n\t\t\t\treturn new ConfigDualList(configParam, editEnabled, inDialog);\n\t\t\t\tbreak;\n\n\t\t\tcase 'Vector3D':\n\t\t\t\treturn new ConfigVector3D(configParam, editEnabled, inDialog);\n\t\t\t\tbreak;\n\n\t\t\tdefault:\n\t\t\t\treturn new ConfigFormItem(configParam, editEnabled, inDialog); // Will log an error for missing form item type\n\t\t}\n\t}\n}\n","import {ConfigurationParameter} from './configuration-parameter';\nimport {ConfigFormItemFactory} from './config-form-item-factory';\n\nexport class ConfigInterfaceBuilder\n{\n\tconstructor()\n\t{\n\t\t// Set some constants\n\t\tthis.TAB_PREFIX = 'tab-'\n\t\tthis.TAB_PANE_PREFIX = 'tabpane-';\n\t\tthis.SEPARATOR_BEFORE = 'before';\n\t\tthis.SEPARATOR_AFTER = 'after';\n\t}\n\n\tdump(modifiedOnly = false)\n\t{\n\t\tlet dumpStr = '';\n\n\t\tfor (let cp of this._configParams)\n\t\t{\n\t\t\tif (modifiedOnly)\n\t\t\t{\n\t\t\t\tif (cp.isModified)\n\t\t\t\t\tdumpStr += cp.toString() + '\\n';\n\t\t\t}\n\t\t\telse\n\t\t\t\tdumpStr += cp.toString() + '\\n';\n\t\t}\n\n\t\tconsole.log(dumpStr);\n\t}\n\n\tbuildInterface(data, mainContainerId, disableEdit = false, tabSuffix = '')\n\t{\n\t\tthis._mainContainerId = mainContainerId;\n\t\tthis._configParams = new Array();\n\t\tthis._validator = null;\n\n\t\tlet hasNewFormItem = false;\n\n\t\t//console.log(data.getDump())\n\n\t\tfor (let i = 0; i < data.size(); i++)\n\t\t{\n\t\t\t// PARSE DATA\n\n\t\t\tlet configParam = ConfigurationParameter.fromSfsObject(data.get(i));\n\t\t\tthis._configParams.push(configParam);\n\n\t\t\t// Get tab and tab pane id from group id\n\t\t\tconst tabId = this.TAB_PREFIX + configParam.categoryId + (tabSuffix ? '_' + tabSuffix : '');\n\t\t\tconst tabPaneId = this.TAB_PANE_PREFIX + configParam.categoryId + (tabSuffix ? '_' + tabSuffix : '');\n\n\t\t\t// BUILD INTERFACE :: TABS\n\n\t\t\t// Check if a tab specific for this group already exists inside the mainContainer: if not, create it\n\t\t\t// (a tab already exists if it was created in a previous loop)\n\t\t\tlet tab = $(`#${mainContainerId} > #tabs #${tabId}`);\n\n\t\t\tif (tab.length == 0)\n\t\t\t{\n\t\t\t\t// Create tab for tab pane\n\t\t\t\ttab = $('<li>', {class: 'nav-item'});\n\t\t\t\ttab.append($('<a>', {\n\t\t\t\t\tclass: 'nav-link' + (i == 0 ? ' active' : ''),\n\t\t\t\t\tid: tabId,\n\t\t\t\t\t'data-toggle': 'tab',\n\t\t\t\t\thref: '#' + tabPaneId,\n\t\t\t\t\trole: 'tab',\n\t\t\t\t\t'aria-controls': tabPaneId,\n\t\t\t\t\t'aria-selected': (i == 0 ? 'true' : 'false'),\n\t\t\t\t\thtml: configParam.category,\n\t\t\t\t}));\n\n\t\t\t\t// Add tab to container\n\t\t\t\t$(`#${mainContainerId} > #tabs`).append(tab);\n\t\t\t}\n\n\t\t\t// BUILD INTERFACE :: TAB PANES\n\n\t\t\t// Check if a tab pane specific for this group already exists inside the mainContainer: if not, create it\n\t\t\t// (a tab pane already exists if it was created in a previous loop or if it exists statically in the html - in case it is needed to add some static content)\n\t\t\tlet tabPane = $(`#${mainContainerId} > #tabPanels > #${tabPaneId}`);\n\n\t\t\tif (tabPane.length == 0)\n\t\t\t{\n\t\t\t\t// Create tab pane\n\t\t\t\ttabPane = $('<div>', {\n\t\t\t\t\tclass: 'tab-pane' + (i == 0 ? ' show active' : ''),\n\t\t\t\t\tid: tabPaneId,\n\t\t\t\t\trole: 'tabpanel',\n\t\t\t\t\t'aria-labelledby': tabId,\n\t\t\t\t\t'data-dynamic': 'true',\n\t\t\t\t});\n\n\t\t\t\t// Add tab pane to container\n\t\t\t\t$(`#${mainContainerId} > #tabPanels`).append(tabPane);\n\t\t\t}\n\n\t\t\t// BUILD INTERFACE :: TAB PANES' FORM\n\n\t\t\t// Check if a form already exists inside the tab pane: if not, create it\n\t\t\tlet form = tabPane.find('form');\n\n\t\t\tif (form.length == 0)\n\t\t\t{\n\t\t\t\t// Create form\n\t\t\t\tform = $('<form>', {\n\t\t\t\t\tclass: '',\n\t\t\t\t\tautocomplete: 'off'\n\t\t\t\t});\n\n\t\t\t\t// Create an inner fieldset; this might be useful to easily disable the whole form at once (actually we don't use it because Kendo widgets are not disabled automatically)\n\t\t\t\tform.append(\n\t\t\t\t\t$('<fieldset>', {\n\t\t\t\t\t\tclass: ''\n\t\t\t\t\t})\n\t\t\t\t);\n\n\t\t\t\t// Add form to tab pane\n\t\t\t\ttabPane.prepend(form);\n\t\t\t}\n\n\t\t\t// Get fieldset, which is the actual form items container\n\t\t\tlet fieldset = form.find('fieldset');\n\n\t\t\t// BUILD INTERFACE :: TAB PANES' FORM ITEMS\n\n\t\t\t// Check if form item already exists in fieldset; if yes, just update its data\n\t\t\tlet formItem = fieldset.find(`#form-item-${$.escapeSelector(configParam.name)}`);\n\n\t\t\tif (formItem.length == 0)\n\t\t\t{\n\t\t\t\thasNewFormItem = true;\n\n\t\t\t\tformItem = ConfigFormItemFactory.create(configParam, !disableEdit);\n\n\t\t\t\t// Add separator before\n\t\t\t\tif (configParam.separator != null && configParam.separator.pos == 'before')\n\t\t\t\t\tfieldset.append(this._buildSeparator(configParam.separator));\n\n\t\t\t\t// Add form item to form\n\t\t\t\tfieldset.append(formItem);\n\n\t\t\t\t// Add separator after\n\t\t\t\tif (configParam.separator != null && configParam.separator.pos == 'after')\n\t\t\t\t\tfieldset.append(this._buildSeparator(configParam.separator));\n\t\t\t}\n\t\t\telse\n\t\t\t\tformItem[0].data = configParam;\n\t\t}\n\n\t\t// Add listener to show help tooltips\n\t\tlet allTabPanes = $(`#${mainContainerId} > #tabPanels > div.tab-pane`);\n\t\tallTabPanes.kendoTooltip({\n\t\t\tfilter: 'i[title].help',\n\t\t\tposition: 'right',\n\t\t\twidth: '250px',\n\t\t\tcontent: function(e) {\n\t\t\t\treturn `<div class=\"help-tooltip\">${e.target.data('title')}</div>`;\n\t\t\t}\n\t\t});\n\n\t\t// Initialize kendo validation on forms' main container\n\t\tthis._validator = $(`#${mainContainerId}`).kendoValidator({\n\t\t\tvalidateOnBlur: true,\n\t\t\trules: {\n\t\t\t\t// Add rule to validate AOI form items\n\t\t\t\t// (see: https://demos.telerik.com/kendo-ui/validator/custom-validation)\n\t\t\t\taoi: function (input) {\n\t\t\t\t\tif (input.is('[data-aoi-msg]') && input.val() != '')\n\t\t\t\t\t{\n\t\t\t\t\t\tif (input.val() == '0,0,0')\n\t\t\t\t\t\t\treturn false;\n                    }\n\n                    return true;\n                }\n\t\t\t}\n\t\t}).data('kendoValidator');\n\t}\n\n\tdestroyInterface()\n\t{\n\t\t// Destroy all Kendo widgets in forms\n\t\tkendo.destroy($(`#${this._mainContainerId} > #tabPanels > div.tab-pane > form`));\n\n\t\t// Remove all tabs\n\t\t$(`#${this._mainContainerId} > #tabs`).empty();\n\n\t\t// Remove dynamic tab panes (tab panes created by Interface Builder)\n\t\t$(`#${this._mainContainerId} > #tabPanels > div.tab-pane[data-dynamic=\"true\"]`).remove();\n\n\t\t// Remove form inside static tab panes (predefined tab panes in html)\n\t\t$(`#${this._mainContainerId} > #tabPanels > div.tab-pane > form`).remove();\n\n\t\t// Remove \"active\" class from static tab panes (otherwise this class messes with the tab navigator functioning)\n\t\t$(`#${this._mainContainerId} > #tabPanels > div.tab-pane`).removeClass('active');\n\t}\n\n\tdisableInterface(disable)\n\t{\n\t\t// Enable/disable all config form items\n\t\t$(`#${this._mainContainerId} *[id^='form-item-']`).prop('editEnabled', !disable);\n\t}\n\n\t_buildSeparator(separator)\n\t{\n\t\tif (separator.text == null)\n\t\t\treturn $(`<hr class=\"config-form-separator\">`);\n\n\t\telse\n\t\t\treturn $(`<label class=\"config-form-separator-label mb-3\">${separator.text}</label>`);\n\t}\n\n\tgetChangedData()\n\t{\n\t\tlet changes = new SFS2X.SFSArray();\n\n\t\tfor (var cp of this._configParams)\n\t\t{\n\t\t\tif (cp.isModified)\n\t\t\t\tchanges.addSFSObject(cp.toSfsObject());\n\t\t}\n\n\t\treturn changes;\n\t}\n\n\tresetIsModified()\n\t{\n\t\tfor (let cp of this._configParams)\n\t\t{\n\t\t\tif (cp.isModified)\n\t\t\t\tcp.resetIsModified();\n\t\t}\n\t}\n\n\tcheckIsValid()\n\t{\n\t\treturn this._validator.validate();\n\t}\n\n\tresetValidation()\n\t{\n\t\tthis._validator.hideMessages();\n\n\t\t// The method above doesn't remove the k-invalid classes and aria-invalid=\"true\" attributes from inputs\n\t\t// Let's do it manually\n\t\t$(`#${this._mainContainerId} .k-invalid`).removeClass('k-invalid');\n\t\t$(`#${this._mainContainerId} [aria-invalid=\"true\"]`).removeAttr('aria-invalid');\n\t}\n\n\tgetConfigFormItem(configParamName)\n\t{\n\t\tlet formItem = $(`#${this._mainContainerId}`).find(`#form-item-${$.escapeSelector(configParamName)}`);\n\n\t\tif (formItem.length > 0)\n\t\t\treturn formItem[0];\n\t\telse\n\t\t\treturn null;\n\t}\n\n\tactivateFirstTabPanel()\n\t{\n\t\tlet configParam = this._configParams[0];\n\t\tconst tabPaneId = this.TAB_PANE_PREFIX + configParam.categoryId;\n\t\tlet tabPane = $(`#${this._mainContainerId} > #tabPanels > #${tabPaneId}`);\n\t\ttabPane.addClass('show active');\n\t}\n}\n","export class ConfigurationParameter\n{\n\tstatic fromSfsObject(element)\n\t{\n\t\tlet cp = new ConfigurationParameter();\n\n\t\t// Parse common data\n\t\tcp.name = element.getUtfString('name');\n\t\tcp.label = element.getUtfString('label');\n\t\tcp.category = element.getUtfString('category');\n\t\tcp.tooltip = element.getUtfString('tooltip');\n\t\tcp.type = element.getUtfString('type');\n\t\tcp.value = element.get('value');\n\t\tcp.validator = element.getUtfString('validator');\n\t\tcp.editable = (element.containsKey('edit') ? element.getBool('edit') : true);\n\t\tcp.trigger = (element.containsKey('trigger') ? element.getBool('trigger') : false);\n\t\tcp.triggerData = element.getSFSArray('triggerData');\n\t\tcp.clientOnly = (element.containsKey('clientOnly') ? element.getBool('clientOnly') : false);\n\t\tcp.dataProvider = element.getUtfString('dataProvider');\n\n\t\t// Parse component specific attributes\n\t\tlet tmpAttributes = element.getSFSObject('attributes');\n\t\tif (tmpAttributes != null)\n\t\t{\n\t\t\tlet attributes = {};\n\n\t\t\tlet keys = tmpAttributes.getKeysArray();\n\t\t\tfor (let key of keys)\n\t\t\t\tattributes[key] = tmpAttributes.get(key);\n\n\t\t\tcp.attributes = attributes;\n\t\t}\n\n\t\t// Parse separator settings\n\t\tlet tmpSeparator = element.getSFSObject('separator');\n\t\tif (tmpSeparator != null)\n\t\t{\n\t\t\tlet separator = {};\n\n\t\t\tlet keys1 = tmpSeparator.getKeysArray();\n\t\t\tfor (let key1 of keys1)\n\t\t\t\tseparator[key1] = tmpSeparator.get(key1);\n\n\t\t\tcp.separator = separator;\n\t\t}\n\n\t\t// Parse default list item\n\t\tlet tmpDefaultListItem = element.getSFSArray('defaultListItem');\n\t\tif (tmpDefaultListItem != null && tmpDefaultListItem.size() > 0)\n\t\t{\n\t\t\tlet defaultListItem = [];\n\n\t\t\tfor (let i = 0; i < tmpDefaultListItem.size(); i++)\n\t\t\t\tdefaultListItem.push(ConfigurationParameter.fromSfsObject(tmpDefaultListItem.getSFSObject(i)));\n\n\t\t\tcp.defaultListItem = defaultListItem;\n\n\t\t\t// Parse list values\n\t\t\tlet listValues = [];\n\n\t\t\tlet tmpListValues = element.getSFSArray('listValues');\n\t\t\tif (tmpListValues != null && tmpListValues.size() > 0)\n\t\t\t{\n\t\t\t\tfor (let v = 0; v < tmpListValues.size(); v++)\n\t\t\t\t{\n\t\t\t\t\tlet listValueObj = tmpListValues.getSFSObject(v);\n\t\t\t\t\tlet obj = {};\n\n\t\t\t\t\tlet keys2 = listValueObj.getKeysArray();\n\t\t\t\t\tfor (let key2 of keys2)\n\t\t\t\t\t\tobj[key2] = listValueObj.get(key2);\n\n\t\t\t\t\tlistValues.push(obj);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tcp.listValues = listValues;\n\n\t\t\t// If we have a list, on the server-side items could be represented by a class\n\t\t\tcp.clazz = element.getUtfString('clazz');\n\t\t}\n\n\t\treturn cp;\n\t}\n\n\tconstructor()\n\t{\n\t\t/* CONSTANTS */\n\t\tthis.DEFAULT_CATEGORY_NAME = 'General';\n\t\tthis.DEFAULT_CATEGORY_ID = 'general';\n\n\t\t/* PUBLIC VARS */\n\n\t\tthis.name = '';\n\t\tthis.label = '';\n\t\tthis.tooltip = '';\n\t\tthis.type = null;\n\t\tthis.trigger = false;\n\t\tthis.triggerData = null;\n\t\tthis.clientOnly = false;\n\t\tthis.editable = true;\n\t\tthis.attributes = null;\n\t\tthis.dataProvider = null;\n\n\t\tthis.separator = null;\t\t\t// Parameter used to create a separator before or after the config parameter\n\t\tthis.defaultListItem = null;\t\t// List of sub-ConfigurationParameters, each containing the default values\n\t\tthis.clazz = null;\t\t\t\t// Name of the class representing the list item (not used in case of primiteve data types)\n\n\t\t/* PRIVATE VARS */\n\n\t\tthis._category = this.DEFAULT_CATEGORY_NAME;\n\t\tthis._categoryId = this.DEFAULT_CATEGORY_ID;\n\t\tthis._value = null;\n\t\tthis._initialValue = null;\t\t// Save the initial value of the configuration parameter, to check if the value was modified\n\t\tthis._validator = null;\n\n\t\tthis._listItems = [];\t\t\t// Array of arrays of ConfigurationParameters\n\t\tthis._listItemsChanged = false;\t// Flag to be set in case a list item is added or removed\n\t}\n\n\t//---------------------------------------------\n\t// GETTERS / SETTERS\n\t//---------------------------------------------\n\n\tset category(val)\n\t{\n\t\tif (val)\n\t\t{\n\t\t\tthis._category = val;\n\t\t\tthis._setIdFromCategoryName(this._category);\n\t\t}\n\t}\n\n\tget category()\n\t{\n\t\treturn this._category;\n\t}\n\n\tset value(val)\n\t{\n\t\tif (this._value != val)\n\t\t{\n\t\t\t// If value is null, then we are setting this for the first time and\n\t\t\t// we want to save the initial value, to check later if it has been modified\n\t\t\tif (this._value == null)\n\t\t\t\tthis._initialValue = val;\n\n\t\t\tthis._value = val;\n\t\t}\n\t}\n\n\tget value()\n\t{\n\t\treturn this._value;\n\t}\n\n\tset validator(val)\n\t{\n\t\tif (val)\n\t\t\tthis._validator = val;\n\t}\n\n\tget validator()\n\t{\n\t\treturn this._validator;\n\t}\n\n\t/**\n\t * An array of objects; each object contains the name-value pairs used to\n\t * populate the list of sub-configuration parameters arrays, based on defaultListItem.\n\t */\n\tset listValues(arr)\n\t{\n\t\tthis._setSubConfigurationParams(arr);\n\t}\n\n\tget listValues()\n\t{\n\t\treturn this._getSubConfigurationParamsValues();\n\t}\n\n\t//---------------------------------------------\n\t// GETTERS ONLY\n\t//---------------------------------------------\n\n\tget isModified()\n\t{\n\t\tlet _isModified = false;\n\n\t\t// If the parameter is used on the client only (for example in a custom trigger)\n\t\t// then we never have to consider it as modified, to prevent it being sent to the server\n\t\tif (!this.clientOnly)\n\t\t{\n\t\t\tif (this._value != this._initialValue || this._listItemsChanged)\n\t\t\t\t_isModified = true;\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Check sub parameters\n\t\t\t\touterLoop: for (let listItem of this._listItems)\n\t\t\t\t{\n\t\t\t\t\tfor (let subCP of listItem)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (subCP.isModified)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t_isModified = true;\n\t\t\t\t\t\t\tbreak outerLoop;\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}\n\n\t\treturn _isModified;\n\t}\n\n\tget categoryId()\n\t{\n\t\treturn this._categoryId;\n\t}\n\n\tget listItems()\n\t{\n\t\treturn this._listItems;\n\t}\n\n\t//---------------------------------------------\n\t// PUBLIC METHODS\n\t//---------------------------------------------\n\n\t/**\n\t * Return a clone of this ConfigurationParameter.\n\t */\n\tclone(cloneValue = false)\n\t{\n\t\tlet cp = new ConfigurationParameter();\n\t\tcp.name = this.name;\n\t\tcp.label = this.label;\n\t\tcp.category = this.category;\n\t\tcp.tooltip = this.tooltip;\n\t\tcp.type = this.type;\n\t\tcp.validator = this.validator;\n\t\tcp.trigger = this.trigger;\n\t\tcp.triggerData = (this.triggerData != null ? SFS2X.SFSArray.newFromBinaryData(this.triggerData.toBinary()) : null);\n\t\tcp.clientOnly = this.clientOnly;\n\t\tcp.dataProvider = this.dataProvider;\n\n\t\tif (cloneValue)\n\t\t\tcp.value = this.value;\n\n\t\tif (this.attributes != null)\n\t\t{\n\t\t\tcp.attributes = new Object();\n\t\t\tfor (let s1 in this.attributes)\n\t\t\t\tcp.attributes[s1] = this.attributes[s1];\n\t\t}\n\n\t\tif (this.separator != null)\n\t\t{\n\t\t\tcp.separator = new Object()\n\t\t\tfor (let s2 in this.separator)\n\t\t\t\tcp.separator[s2] = this.separator[s2];\n\t\t}\n\n\t\tif (this.defaultListItem != null)\n\t\t{\n\t\t\tlet clonedDefaultListItems = [];\n\n\t\t\tfor (let subCP of this.defaultListItem)\n\t\t\t\tclonedDefaultListItems.push(subCP.clone(cloneValue));\n\n\t\t\tcp.defaultListItem = clonedDefaultListItems;\n\t\t}\n\n\t\tcp.listValues = this.listValues; // No need to clone this, as the listValues setter already does it\n\t\tcp.clazz = this.clazz;\n\n\t\treturn cp;\n\t}\n\n\t/**\n\t * Reset initial value by copying the current value.\n\t */\n\tresetIsModified()\n\t{\n\t\tthis._initialValue = this._value;\n\n\t\t// Reset sub-parameters\n\t\tif (this._listItems != null)\n\t\t{\n\t\t\tfor (let listItem of this._listItems)\n\t\t\t{\n\t\t\t\tfor (let subCP of listItem)\n\t\t\t\t\tsubCP.resetIsModified();\n\t\t\t}\n\t\t}\n\n\t\tthis._listItemsChanged = false;\n\t}\n\n\taddListItem(newListItem)\n\t{\n\t\tthis._listItems.push(newListItem);\n\t\tthis._listItemsChanged = true;\n\t}\n\n\tupdateListItem(listItem, itemIndex)\n\t{\n\t\tthis._listItems[itemIndex] = listItem;\n\t\tthis._listItemsChanged = true;\n\t}\n\n\tremoveListItem(itemIndex)\n\t{\n\t\tthis._listItems.splice(itemIndex, 1);\n\t\tthis._listItemsChanged = true;\n\t}\n\n\ttoSfsObject()\n\t{\n\t\tlet obj = new SFS2X.SFSObject();\n\n\t\t// Set changed setting name\n\t\tobj.putUtfString('name', this.name);\n\n\t\t// Set changed setting class, if any\n\t\tif (this.clazz != null)\n\t\t\tobj.putUtfString('clazz', this.clazz);\n\n\t\tif (this.value != null)\n\t\t{\n\t\t\t// Set changed setting value\n\t\t\tif (typeof this.value === 'boolean')\n\t\t\t\tobj.putBool('value', this.value);\n\t\t\telse if (typeof this.value === 'number')\n\t\t\t\tobj.putInt('value', this.value);\n\t\t\telse\n\t\t\t\tobj.putText('value', this.value);\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// Set changed setting list of values\n\n\t\t\tlet listItems = new SFS2X.SFSArray();\n\n\t\t\tfor (let a of this._listItems)\n\t\t\t{\n\t\t\t\tif (a.length == 1) // We have just one sub config param; no need to parse it complitely\n\t\t\t\t{\n\t\t\t\t\t// Simple list\n\t\t\t\t\tlet tempObj = a[0].toSfsObject();\n\t\t\t\t\tlet wa = tempObj.getWrappedItem('value');\n\t\t\t\t\tlistItems.add(wa.value, wa.type);\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t// Complex list\n\n\t\t\t\t\tlet values = new SFS2X.SFSArray();\n\n\t\t\t\t\tfor (let subCp of a)\n\t\t\t\t\t\tvalues.addSFSObject(subCp.toSfsObject());\n\n\t\t\t\t\tlistItems.addSFSArray(values);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tobj.putSFSArray('value', listItems);\n\t\t}\n\n\t\treturn obj;\n\t}\n\n\t/**\n\t * Return a description of the ConfigurationParameter instance.\n\t */\n\ttoString()\n\t{\n\t\tlet s = ``;\n\t\ts += `Configuration parameter: ${this.name}\\n`;\n\t\ts += `\\ttype: ${this.type}\\n`;\n\t\ts += `\\tlabel: ${this.label}\\n`;\n\t\ts += `\\tcategory name: ${this.category}\\n`;\n\t\ts += `\\tcategory id: ${this.categoryId}\\n`;\n\t\ts += `\\ttooltip: ${this.tooltip}\\n`;\n\t\ts += `\\tvalue: ${this.value}\\n`;\n\t\ts += `\\ttrigger: ${this.trigger}\\n`;\n\t\ts += `\\ttrigger data: ${this.triggerData}\\n`;\n\t\ts += `\\tclient only: ${this.clientOnly}\\n`;\n\t\ts += `\\tvalidator: ${this.validator}\\n`;\n\t\ts += `\\tis modified: ${this.isModified}\\n`;\n\n\t\tif (this.attributes != null)\n\t\t{\n\t\t\ts += `\\tcomponent attributes:\\n`;\n\n\t\t\tfor (let s1 in this.attributes)\n\t\t\t\ts += `\\t\\t${s1} --> ${this.attributes[s1]}\\n`;\n\t\t}\n\n\t\tif (this.dataProvider != null)\n\t\t\ts += `\\tdata provider: ${this.dataProvider}\\n`;\n\n\t\tif (this.separator != null)\n\t\t{\n\t\t\ts += `\\tseparator:\\n`;\n\n\t\t\tfor (let s2 in this.separator)\n\t\t\t\ts += `\\t\\t${s2} --> ${this.separator[s2]}\\n`;\n\t\t}\n\n\t\tif (this._listItems != null && this._listItems.length > 0)\n\t\t{\n\t\t\ts += `\\t# list items: ${this._listItems.length}\\n`;\n\n\t\t\tfor (let i = 0; i < this._listItems.length; i++)\n\t\t\t{\n\t\t\t\ts += `\\tlist item ${i} sub-parameters:\\n`;\n\t\t\t\tfor (let e = 0; e < this._listItems[i].length; e++)\n\t\t\t\t\ts += `\\t\\t${this._listItems[i][e].toCompactString()}\\n`;\n\t\t\t}\n\n\t\t\ts += `\\tclass name: ${this.clazz}\\n`;\n\t\t}\n\n\t\treturn s;\n\t}\n\n\t/**\n\t * Return a compact description of the ConfigurationParameter instance.\n\t */\n\ttoCompactString()\n\t{\n\t\treturn `Configuration parameter '${this.name}': ${this.value} ${this.isModified ? '[X]' : '[ ]'}`;\n\t}\n\n\t//---------------------------------------------\n\t// PRIVATE METHODS\n\t//---------------------------------------------\n\n\t/**\n\t * Retrieve the category id form the category name.\n\t * Spaces and invalid characters are removed; words are separated using capitals.\n\t */\n\t_setIdFromCategoryName(categoryName)\n\t{\n\t\tthis._categoryId = categoryName;\n\n\t\t// Strip invalid characters\n\t\tvar pattern = /[^0-9a-zA-Z]/g;\n\t\tthis._categoryId = this._categoryId.replace(pattern, ' ');\n\n\t\t// Capitalize words\n\t\tvar words = this._categoryId.split(' ');\n\t\tthis._categoryId = '';\n\n\t\tfor (let i = 0; i < words.length; i++)\n\t\t{\n\t\t\tlet word = words[i];\n\t\t\tif (word.length > 0)\n\t\t\t\tthis._categoryId += (i > 0 ? word.substr(0,1).toUpperCase() : word.substr(0,1).toLowerCase()) + (word.length > 1 ? word.substr(1) : \"\");\n\t\t}\n\n\t\tif (this._categoryId.length == 0)\n\t\t\tthis._categoryId = this.DEFAULT_CATEGORY_ID;\n\t}\n\n\t_setSubConfigurationParams(_listValues)\n\t{\n\t\tthis._listItems = [];\n\n\t\tfor (let obj of _listValues)\n\t\t{\n\t\t\tlet listItem = [];\n\n\t\t\tfor (let defaultCP of this.defaultListItem)\n\t\t\t{\n\t\t\t\tlet subCP = defaultCP.clone(false);\n\t\t\t\tsubCP.value = obj[subCP.name];\n\n\t\t\t\tlistItem.push(subCP);\n\t\t\t}\n\n\t\t\tthis._listItems.push(listItem);\n\t\t}\n\t}\n\n\t_getSubConfigurationParamsValues()\n\t{\n\t\tlet _listValues = [];\n\n\t\tfor (let listItem of this._listItems)\n\t\t{\n\t\t\tlet obj = {};\n\n\t\t\tfor (let subCP of listItem)\n\t\t\t{\n\t\t\t\tif (subCP.value != null)\n\t\t\t\t\tobj[subCP.name] = subCP.value;\n\t\t\t}\n\n\t\t\t_listValues.push(obj);\n\t\t}\n\n\t\treturn _listValues;\n\t}\n}\n"],"mappings":";;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC5GA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACxHA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACzLA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACxOA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC3WA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AC3BA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC/GA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACzFA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC7EA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC7EA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC7JA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AChDA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC7QA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;A","sourceRoot":""}
@@ -1223,6 +1223,7 @@ class ZoneMonitor extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModule"
1223
1223
  this.REQ_SET_USER_SETTINGS = 'setUserSettings';
1224
1224
  this.REQ_RELOAD_ZONE_EXT = 'reloadZoneExt';
1225
1225
  this.REQ_RELOAD_WORDS = 'reloadWords';
1226
+ this.REQ_DISCONNECT_USER = 'disconnUser';
1226
1227
  this.REQ_KICK_USER = 'kickUser';
1227
1228
  this.REQ_BAN_USER = 'banUser';
1228
1229
  this.REQ_GET_ROOM_CONFIG = 'getRoomConfig';
@@ -1508,6 +1509,7 @@ class ZoneMonitor extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModule"
1508
1509
  }).data('kendoNumericTextBox');
1509
1510
 
1510
1511
  // Add button click listeners
1512
+ $('#znm-disconnectBt').on('click', $.proxy(this._onDisconnectBtClick, this));
1511
1513
  $('#znm-kickBt').on('click', $.proxy(this._onKickBtClick, this));
1512
1514
  $('#znm-banBt').on('click', $.proxy(this._onBanBtClick, this));
1513
1515
 
@@ -1789,6 +1791,7 @@ class ZoneMonitor extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModule"
1789
1791
  $(window).off('resize');
1790
1792
  $('#znm-reloadWordsBt').off('click');
1791
1793
  $('#znm-reloadZoneExtBt').off('click');
1794
+ $('#znm-disconnectBt').off('click');
1792
1795
  $('#znm-kickBt').off('click');
1793
1796
  $('#znm-banBt').off('click');
1794
1797
 
@@ -1955,7 +1958,8 @@ class ZoneMonitor extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModule"
1955
1958
  // NPC
1956
1959
  const isNpc = data.getSFSObject('monitored').getBool('npc');
1957
1960
 
1958
- // Disable kick and ban buttons
1961
+ // Disable disconnect, kick and ban buttons
1962
+ $('#znm-disconnectBt').attr('disabled', isNpc);
1959
1963
  $('#znm-kickBt').attr('disabled', isNpc);
1960
1964
  $('#znm-banBt').attr('disabled', isNpc);
1961
1965
 
@@ -2622,6 +2626,19 @@ class ZoneMonitor extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModule"
2622
2626
  }
2623
2627
  }
2624
2628
 
2629
+ _onDisconnectBtClick()
2630
+ {
2631
+ if (this._monitoredType == this.MONITORED_TYPE_USER)
2632
+ {
2633
+ // Send request to extension
2634
+ let params = new SFS2X.SFSObject();
2635
+ params.putUtfString('zone', this._monitoredZone);
2636
+ params.putUtfString('user', this._monitoredName);
2637
+
2638
+ this.sendExtensionRequest(this.REQ_DISCONNECT_USER, params);
2639
+ }
2640
+ }
2641
+
2625
2642
  _onKickBtClick()
2626
2643
  {
2627
2644
  if (this._monitoredType == this.MONITORED_TYPE_USER)
@@ -3312,4 +3329,4 @@ class UserPropertiesData
3312
3329
  /***/ })
3313
3330
 
3314
3331
  }]);
3315
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-13.bundle.js","sources":["webpack://application/./src/components/charts/chart-utils.js","webpack://application/./src/components/charts/user-count-chart.js","webpack://application/./src/components/module-specific/match-expression-builder.js","webpack://application/./src/components/sidebar-layout.js","webpack://application/./src/modules/zone-monitor.js","webpack://application/./src/utils/match-properties.js"],"sourcesContent":["import {bytesToSize, kBytesToSize} from '../../utils/utilities';\n\nexport function getFormattedDateTime(date, format = 'dd MMM HH:mm:ss')\n{\n\treturn kendo.toString(date, format);\n}\n\nexport function getBasicSharedTemplate(valueFormat = '{0}') {\n\treturn kendo.template(`\n\t\t<div class=\"chart-tooltip-category\"><strong>#: category #</strong></div>\n\t\t# for (var i = 0; i < points.length; i++) { #\n\t\t<div>\n\t\t\t<i class=\"fas fa-square\" style=\"color: #: points[i].color # \"></i>\n\t\t\t#: points[i].series.name# : #: kendo.format('${valueFormat}', points[i].value) #\n\t\t</div>\n\t\t# } #\n\t`);\n}\n\nexport function getMemorySharedTemplate() {\n\treturn function(dataItem) {\n\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t<div class=\"chart-tooltip-category\"><strong>#: category #</strong></div>\n\t\t\t# for (var i = 0; i < points.length; i++) { #\n\t\t\t<div>\n\t\t\t\t<i class=\"fas fa-square\" style=\"color: #: points[i].color # \"></i>\n\t\t\t\t#: points[i].series.name# : #: bytesToSize(points[i].value) #\n\t\t\t</div>\n\t\t\t# } #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getMemoryValueAxisLabelTemplate() {\n\treturn function(dataItem) {\n\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t#: bytesToSize(value) #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getNetworkSharedTemplate(categoryFormat = 'dd MMM HH:mm:ss') {\n\treturn function(dataItem) {\n\t\tdataItem.kBytesToSize = kBytesToSize; // Pass kBytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t<div class=\"chart-tooltip-category\"><strong>#: kendo.toString(category, '${categoryFormat}') #</strong></div>\n\t\t\t# for (var i = 0; i < points.length; i++) { #\n\t\t\t<div>\n\t\t\t\t<i class=\"fas fa-square\" style=\"color: #: points[i].color # \"></i>\n\t\t\t\t#: points[i].series.name# : #: kBytesToSize(points[i].value, 1, '', '/s') #\n\t\t\t</div>\n\t\t\t# } #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getNetworkValueAxisLabelTemplate() {\n\treturn function(dataItem) {\n\t\tdataItem.kBytesToSize = kBytesToSize; // Pass kBytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t#: kBytesToSize(value, 1, '', '/s') #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getUserCountSharedTemplate(categoryFormat = 'dd MMM HH:mm:ss') {\n\treturn function(dataItem) {\n\t\treturn kendo.template(`\n\t\t\t<div class=\"chart-tooltip-category\"><strong>#: kendo.toString(category, '${categoryFormat}') #</strong></div>\n\t\t\t# for (var i = 0; i < points.length; i++) { #\n\t\t\t<div>\n\t\t\t\t<i class=\"fas fa-square\" style=\"color: #: points[i].color # \"></i>\n\t\t\t\t#: points[i].series.name# : #: points[i].value #\n\t\t\t</div>\n\t\t\t# } #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getTimeRangeTooltipTemplate() {\n\treturn function(dataItem) {\n\t\tdataItem.getTooltipString = function(v1, v2) {\n\t\t\tif (v1 == v2)\n\t\t\t{\n\t\t\t\tif (v2 == -24)\n\t\t\t\t\tv2 = -23;\n\n\t\t\t\tv1 = v2 - 1;\n\t\t\t}\n\n\t\t\tlet start = Math.abs(v1);\n\t\t\tlet end = Math.abs(v2);\n\n\t\t\treturn `From ${start} hour${start != 1 ? 's' : ''} ago to ${end > 0 ? ('to ' + end + ' hour' + (end != 1 ? 's' : '') + ' ago') : 'now'}`;\n\t\t}; // Pass utility function to template\n\t\treturn kendo.template(`\n\t\t\t#: getTooltipString(selectionStart, selectionEnd) #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getWRBytesLabelTemplate() {\n\treturn function(dataItem) {\n\t\treturn kendo.template(`\n\t\t\t#: value # #: dataItem.unit #\n\t\t`)(dataItem);\n\t};\n}\n","import {getUserCountSharedTemplate} from './chart-utils';\nimport * as moment from 'moment';\n\nexport class UserCountChart extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\n\t\t// Create chart html\n\t\tlet chartHtml = $('<div>');\n\t\t$(this).append(chartHtml);\n\n\t\t// Initialize chart\n\t\tthis._chartWidget = this._initChartWidget(chartHtml);\n\t}\n\n\t_initChartWidget(chartHtml)\n\t{\n\t\treturn chartHtml.kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 270,\n\t\t\t\tbackground: 'transparent'\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'area',\n\t            labels: {\n\t                visible: false,\n\t                format: '{0}',\n\t                background: 'transparent'\n\t            },\n\t        },\n\t        series: [{\n\t\t\t\tname: 'Users',\n\t            field: 'users',\n\t            categoryField: 'time',\n\t\t\t\tcolor: '#fd7d24',\n\t\t\t\tline: {\n\t\t\t\t\twidth: 2\n\t\t\t\t}\n\t        }],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}',\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0,\n\t        },\n\t        categoryAxis: {\n\t\t\t\tmajorGridLines: {\n\t                visible: true,\n\t\t\t\t\tstep: 60\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: true,\n\t\t\t\t\tstep: 60\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: true,\n\t\t\t\t\tstep: 60\n\t\t\t\t},\n\t\t\t\tbaseUnit: 'minutes',\n\t\t\t\ttype: 'date',\n\t        },\n\t        tooltip: {\n\t\t\t\tshared: true,\n\t            visible: true,\n\t\t\t\tsharedTemplate: getUserCountSharedTemplate('dd MMM HH:mm'),\n\t\t\t\tbackground: '#e4e4e4'\n\t        }\n\t    }).data('kendoChart');\n\t}\n\n\tredraw()\n\t{\n\t\tthis._chartWidget.redraw();\n\t}\n\n\tset range(values)\n\t{\n\t\tthis._range = values;\n\n\t\tthis._updateXAxisLimits();\n\t}\n\n\tget range()\n\t{\n\t\treturn this._range;\n\t}\n\n\taddHistoryEntries(userCountData, samplingRateSeconds)\n\t{\n\t\tlet date = new Date();\n\t\tlet start = moment(date);\n\t\tstart.subtract((samplingRateSeconds * userCountData.length), 's')\n\n\t\tlet dataSource = new kendo.data.DataSource();\n\n\t\tfor (let i = 1; i < userCountData.length; i++)\n\t\t{\n\t\t\tlet date = start.clone();\n\t\t\tdate.add(i * samplingRateSeconds, 's');\n\n\t\t\tthis.addEntry(userCountData[i], dataSource, date.toDate(), false);\n\t\t}\n\n\t\t// Assign datasource to chart\n\t\tthis._chartWidget.setDataSource(dataSource);\n\n\t\t// Update axis\n\t\tthis._updateXAxisLimits();\n\t}\n\n\taddEntry(userCount, dataSource = null, date = null, updateAxis = true)\n\t{\n\t\tif (date == null)\n\t\t\tdate = new Date();\n\n\t\tlet newEntry = {\n\t\t\ttime: date,\n\t\t\tusers: userCount\n\t\t};\n\n\t\t// Add entry to data source\n\t\tif (dataSource == null)\n\t\t\tdataSource = this._dataSource;\n\n\t\tdataSource.add(newEntry);\n\n\t\tif (updateAxis)\n\t\t\tthis._updateXAxisLimits();\n\t}\n\n\t_updateXAxisLimits()\n\t{\n\t\tlet ds = this._dataSource;\n\n\t\tif (ds.total() > 0)\n\t\t{\n\t\t\tconst chartMinimum = moment(this._lastDate).add(this._range[0], 'h');\n\t\t\tconst chartMaximum = moment(this._lastDate).add(this._range[1], 'h');\n\n\t\t\tthis._chartWidget.options.categoryAxis.min = chartMinimum.toDate();\n\t\t\tthis._chartWidget.options.categoryAxis.max = chartMaximum.toDate();\n\n\t\t\tthis._chartWidget.redraw();\n\t\t}\n\t}\n\n\tget _dataSource()\n\t{\n\t\treturn this._chartWidget.dataSource;\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('user-count-chart'))\n\twindow.customElements.define('user-count-chart', UserCountChart);\n","export class MatchExpressionBuilder extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\n\t\tthis.EXPRESSION_UPDATED_EVENT = 'expressionUpdated';\n\n\t\tthis._dialog = $(`\n\t\t\t<div class=\"modal\" id=\"matchExprBuilderModal\" tabindex=\"-1\" role=\"dialog\" aria-labelledby=\"matchExprBuilderModalTitle\" aria-hidden=\"true\">\n\t\t\t\t<div class=\"modal-dialog modal-dialog-centered modal-lg\" role=\"document\">\n\t\t\t\t\t<div class=\"modal-content\">\n\t\t\t\t\t\t<div class=\"modal-header\">\n\t\t\t\t\t\t\t<h5 class=\"modal-title text-primary\" id=\"matchExprBuilderModalTitle\">...</h5>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class=\"modal-body p-0\">\n\t\t\t\t\t\t\t<form class=\"bg-color-default p-3\">\n\t\t\t\t\t\t\t\t<div class=\"d-inline-flex align-items-center radio-button-group inline-form-groups\">\n\t\t\t\t\t\t\t\t\t<div class=\"custom-control custom-radio custom-control-inline mr-2\">\n\t\t\t\t\t\t\t\t\t\t<input type=\"radio\" id=\"andOperatorRB\" name=\"operatorRBG\" class=\"custom-control-input\" value=\"AND\" checked disabled>\n\t\t\t\t\t\t\t\t\t\t<label class=\"custom-control-label\" for=\"andOperatorRB\">AND</label>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t<div class=\"custom-control custom-radio custom-control-inline mr-2\">\n\t\t\t\t\t\t\t\t\t\t<input type=\"radio\" id=\"orOperatorRB\" name=\"operatorRBG\" class=\"custom-control-input\" value=\"OR\" disabled>\n\t\t\t\t\t\t\t\t\t\t<label class=\"custom-control-label\" for=\"orOperatorRB\">OR</label>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t<div class=\"d-inline-block\">\n\t\t\t\t\t\t\t\t\t<div class=\"inline-form-groups my-1\">\n\t\t\t\t\t\t\t\t\t\t<input id=\"varNameCB\"/>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t<div class=\"inline-form-groups my-1\">\n\t\t\t\t\t\t\t\t\t\t<input id=\"typeDD\" class=\"matchTypeDropdown\"/>\n\t\t\t\t\t\t\t\t\t\t<input id=\"conditionDD\" class=\"matchConditionDropdown\"/>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t<div class=\"d-inline inline-form-groups\">\n\t\t\t\t\t\t\t\t\t<input id=\"valueIn\" class=\"k-textbox mt-1\" placeholder=\"Value\"/>\n\t\t\t\t\t\t\t\t\t<button id=\"matchExprAddBt\" type=\"button\" class=\"k-button k-secondary mt-1\"><i class=\"fas fa-plus mr-1\"></i>Add</button>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t</form>\n\n\t\t\t\t\t\t\t<div id=\"matchExprGrid\" class=\"m-3\"></div>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class=\"modal-footer d-flex\">\n\t\t\t\t\t\t\t<div class=\"flex-grow-1 text-left\">\n\t\t\t\t\t\t\t\t<button id=\"matchExprApplyBt\" type=\"button\" class=\"k-button k-primary mr-2\"><i class=\"fas fa-check-circle mr-1\"></i>Apply</button>\n\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t<div class=\"flex-grow-1 text-right\">\n\t\t\t\t\t\t\t\t<button id=\"matchExprRemoveBt\" type=\"button\" class=\"k-button k-secondary\" disabled><i class=\"fas fa-minus-circle mr-1\"></i>Remove selected</button>\n\t\t\t\t\t\t\t\t<button id=\"matchExprClearBt\" type=\"button\" class=\"k-button k-secondary\"><i class=\"fas fa-times-circle mr-1\"></i>Clear all</button>\n\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t`);\n\n\t\t$('#matchExprAddBt', this._dialog).on('click', $.proxy(this._onAddBtClick, this));\n\t\t$('#matchExprApplyBt', this._dialog).on('click', $.proxy(this._onApplyBtClick, this));\n\t\t$('#matchExprRemoveBt', this._dialog).on('click', $.proxy(this._onRemoveBtClick, this));\n\t\t$('#matchExprClearBt', this._dialog).on('click', $.proxy(this._onClearBtClick, this));\n\n\t\t// INITIALIZE FORM ITEMS\n\n\t\tthis._varNameCb = $('#varNameCB', this._dialog).kendoComboBox({\n\t\t\tdataTextField: 'label',\n\t\t\tdataValueField: 'value',\n\t\t\tplaceholder: 'Variable name',\n\t\t}).data('kendoComboBox');\n\n\t\t// Set types dropdown data source\n\t\t// NOTE: in order to avoid XxxMatch objects to be wrapped by Kendo data source, we have to pass them using _ property names\n\t\tlet typeDS = [];\n\n\t\t// Boolean ----------------\n\t\ttypeDS.push({\n\t\t\tlabel: 'Boolean',\n\t\t\t_value: SFS2X.BoolMatch.EQUALS,\n\t\t\tconditions: [\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.BoolMatch.EQUALS.symbol,\n\t\t\t\t\t_value: SFS2X.BoolMatch.EQUALS,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.BoolMatch.NOT_EQUALS.symbol,\n\t\t\t\t\t_value: SFS2X.BoolMatch.NOT_EQUALS,\n\t\t\t\t}\n\t\t\t]\n\t\t});\n\n\t\t// Number ----------------\n\t\ttypeDS.push({\n\t\t\tlabel: 'Number',\n\t\t\t_value: SFS2X.NumberMatch.EQUALS,\n\t\t\tconditions: [\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.NumberMatch.EQUALS.symbol,\n\t\t\t\t\t_value: SFS2X.NumberMatch.EQUALS,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.NumberMatch.NOT_EQUALS.symbol,\n\t\t\t\t\t_value: SFS2X.NumberMatch.NOT_EQUALS,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.NumberMatch.GREATER_THAN.symbol,\n\t\t\t\t\t_value: SFS2X.NumberMatch.GREATER_THAN,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.NumberMatch.LESS_THAN.symbol,\n\t\t\t\t\t_value: SFS2X.NumberMatch.LESS_THAN,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.NumberMatch.GREATER_THAN_OR_EQUAL_TO.symbol,\n\t\t\t\t\t_value: SFS2X.NumberMatch.GREATER_THAN_OR_EQUAL_TO,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.NumberMatch.LESS_THAN_OR_EQUAL_TO.symbol,\n\t\t\t\t\t_value: SFS2X.NumberMatch.LESS_THAN_OR_EQUAL_TO,\n\t\t\t\t}\n\t\t\t]\n\t\t});\n\n\t\t// String ----------------\n\t\ttypeDS.push({\n\t\t\tlabel: 'String',\n\t\t\t_value: SFS2X.StringMatch.EQUALS,\n\t\t\tconditions: [\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.StringMatch.EQUALS.symbol,\n\t\t\t\t\t_value: SFS2X.StringMatch.EQUALS,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.StringMatch.NOT_EQUALS.symbol,\n\t\t\t\t\t_value: SFS2X.StringMatch.NOT_EQUALS,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.StringMatch.CONTAINS.symbol,\n\t\t\t\t\t_value: SFS2X.StringMatch.CONTAINS,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.StringMatch.STARTS_WITH.symbol,\n\t\t\t\t\t_value: SFS2X.StringMatch.STARTS_WITH,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.StringMatch.ENDS_WITH.symbol,\n\t\t\t\t\t_value: SFS2X.StringMatch.ENDS_WITH,\n\t\t\t\t}\n\t\t\t]\n\t\t});\n\n\t\tthis._typeDd = $('#typeDD', this._dialog).kendoDropDownList({\n\t\t\tdataSource: typeDS,\n\t\t\tdataTextField: 'label',\n\t\t\tdataValueField: '_value',\n\t\t\toptionLabel: {\n\t\t\t\tlabel: 'Type',\n\t\t\t\t_value: ''\n\t\t\t},\n\t\t\toptionLabelTemplate: '<span class=\"text-muted\">#:label#</span>',\n\t\t\tchange: $.proxy(function() {\n\t\t\t\tif (this._typeDd && this._typeDd.select())\n\t\t\t\t{\n\t\t\t\t\tlet selectedType = this._typeDd.dataItem(this._typeDd.select());\n\t\t\t\t\tthis._conditionDd.setDataSource(selectedType.conditions);\n\t\t\t\t\tthis._conditionDd.select(1);\n\t\t\t\t}\n\t\t\t}, this)\n\t\t}).data('kendoDropDownList');\n\n\t\tthis._conditionDd = $('#conditionDD', this._dialog).kendoDropDownList({\n\t\t\tdataTextField: 'label',\n\t\t\tdataValueField: '_value',\n\t\t\toptionLabel: {\n\t\t\t\tlabel: 'Condition',\n\t\t\t\t_value: ''\n\t\t\t},\n\t\t\toptionLabelTemplate: '<span class=\"text-muted\">#:label#</span>',\n\t\t}).data('kendoDropDownList');\n\n\t\tthis._valueIn = $('#valueIn', this._dialog);\n\n\t\t// INITIALIZE GRID\n\n\t\tthis._grid = $('#matchExprGrid', this._dialog).kendoGrid({\n\t\t\tdataSource: [],\n\t\t\tresizable: true,\n\t\t\tselectable: 'row',\n\t\t\tchange: $.proxy(this._onGridSelectionChange, this),\n\t\t\tnoRecords: {\n\t\t\t\ttemplate: 'No entries.'\n\t\t\t},\n\t\t\tcolumns: [\n\t\t\t\t{\n\t\t\t\t\tfield: 'operator',\n\t\t\t\t\ttitle: 'Operator',\n\t\t\t\t\twidth: 90\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'label',\n\t\t\t\t\ttitle: 'Name',\n\t\t\t\t\twidth: 120\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'typeLabel',\n\t\t\t\t\ttitle: 'Type',\n\t\t\t\t\twidth: 85\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'symbol',\n\t\t\t\t\ttitle: 'Condition',\n\t\t\t\t\twidth: 95\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'value',\n\t\t\t\t\ttitle: 'Value',\n\t\t\t\t\twidth: 200\n\t\t\t\t},\n\t\t\t]\n\t\t}).data('kendoGrid');\n\t}\n\n\tget title()\n\t{\n\t\treturn $('#matchExprBuilderModalTitle', this._dialog).text();\n\t}\n\n\tset title(value)\n\t{\n\t\t$('#matchExprBuilderModalTitle', this._dialog).text(value);\n\t}\n\n\tget predefinedVarNames()\n\t{\n\t\treturn this._predefinedVarNames;\n\t}\n\n\tset predefinedVarNames(value)\n\t{\n\t\tthis._predefinedVarNames = value;\n\n\t\tthis._varNameCb.setDataSource(value);\n\t}\n\n\tget filterExpression()\n\t{\n\t\treturn this._filterExpression;\n\t}\n\n\tshow()\n\t{\n\t\tthis._dialog.modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t\tshow: true\n\t\t});\n\t}\n\n\t_onGridSelectionChange()\n\t{\n\t\tlet selectedRows = this._grid.select();\n\t\tlet selectedDataItems = [];\n\n\t\tfor (let i = 0; i < selectedRows.length; i++)\n\t\t{\n\t\t\tlet dataItem = this._grid.dataItem(selectedRows[i]);\n\t\t\tselectedDataItems.push(dataItem);\n\t\t}\n\n\t\t// Enable/disable remove button\n\t\t$('#matchExprRemoveBt', this._dialog).attr('disabled', selectedDataItems.length == 0);\n\t}\n\n\t_onAddBtClick()\n\t{\n\t\tif (this._varNameCb.value() != '' && this._typeDd.select())\n\t\t{\n\t\t\t// Variable name and label\n\t\t\tlet varName, varLabel;\n\n\t\t\tif (this._varNameCb.select() > -1)\n\t\t\t{\n\t\t\t\tlet dataItem = this._varNameCb.dataItem(this._varNameCb.select());\n\t\t\t\tvarName = dataItem.value;\n\t\t\t\tvarLabel = dataItem.label;\n\t\t\t}\n\t\t\telse\n\t\t\t\tvarName = varLabel = this._varNameCb.value();\n\n\t\t\t// Variable type's label\n\t\t\tlet typeLabel = this._typeDd.dataItem(this._typeDd.select()).label;\n\n\t\t\t// Match condition\n\t\t\tlet condition = this._conditionDd.dataItem(this._conditionDd.select())._value;\n\n\t\t\t// Expression operator\n\t\t\tlet operator = this._grid.dataSource.total() > 0 ? $('input[name=operatorRBG]:checked', this._dialog).val() : null;\n\n\t\t\t// Variable value\n\t\t\tlet val = this._valueIn.val();\n\t\t\tlet varValue;\n\n\t\t\tif (condition instanceof SFS2X.BoolMatch)\n\t\t\t\tvarValue = (val == 'true' ? true : false);\n\t\t\telse if (condition instanceof SFS2X.NumberMatch)\n\t\t\t\tvarValue = (isNaN(Number(val)) ? 0 : Number(val));\n\t\t\telse\n\t\t\t\tvarValue = val;\n\n\t\t\t// Add item to grid\n\t\t\tlet exprPart = this._getExpressionPart(varLabel, varName, varValue, typeLabel, condition, operator);\n\t\t\tthis._grid.dataSource.add(exprPart);\n\n\t\t\t// Reset inputs\n\t\t\tthis._varNameCb.value('');\n\t\t\tthis._valueIn.val('');\n\n\t\t\t// Enable operator radio buttons\n\t\t\tthis._enableOperatorRb(true);\n\t\t}\n\t}\n\n\t_onApplyBtClick()\n\t{\n\t\t// Hide dialog\n\t\tthis._dialog.modal('hide');\n\n\t\t// Clear grid selection\n\t\tthis._grid.clearSelection();\n\n\t\t// Build final match expression\n\t\tthis._filterExpression = null;\n\n\t\tif (this._grid.dataSource.total() > 0)\n\t\t{\n\t\t\tlet expPart = this._grid.dataSource.at(0);\n\t\t\tthis._filterExpression = new SFS2X.MatchExpression(expPart.name, expPart._condition, expPart.value);\n\n\t\t\tfor (let i = 1; i < this._grid.dataSource.total(); i++)\n\t\t\t{\n\t\t\t\texpPart = this._grid.dataSource.at(i);\n\n\t\t\t\tif (expPart.operator == SFS2X.LogicOperator.AND.id)\n\t\t\t\t\tthis._filterExpression = this._filterExpression.and(expPart.name, expPart._condition, expPart.value);\n\t\t\t\telse if (expPart.operator == SFS2X.LogicOperator.OR.id)\n\t\t\t\t\tthis._filterExpression = this._filterExpression.or(expPart.name, expPart._condition, expPart.value);\n\t\t\t}\n\t\t}\n\n\t\t// Fire event\n\t\tlet evt = new CustomEvent(this.EXPRESSION_UPDATED_EVENT, {\n\t\t    \tdetail: null,\n\t\t\t\tbubbles: false,\n\t\t\t\tcancelable: false\n\t\t\t});\n\n\t\tthis.dispatchEvent(evt);\n\t}\n\n\t_onRemoveBtClick()\n\t{\n\t\tlet dataItem = this._grid.dataItem(this._grid.select());\n\t\tthis._grid.dataSource.remove(dataItem);\n\n\t\t// Clear grid selection\n\t\tthis._grid.clearSelection();\n\n\t\tif (this._grid.dataSource.total() == 0)\n\t\t{\n\t\t\t// Disable operator radio buttons\n\t\t\tthis._enableOperatorRb(false);\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// Reset operator on first entry\n\t\t\tthis._grid.dataSource.at(0).operator = null;\n\t\t\tthis._grid.refresh();\n\t\t}\n\t}\n\n\t_onClearBtClick()\n\t{\n\t\t// Reset form\n\t\t$('#andOperatorRB', this._dialog).attr('checked', true);\n\t\t$('#orOperatorRB', this._dialog).attr('checked', false);\n\t\tthis._enableOperatorRb(false);\n\n\t\tthis._varNameCb.value('');\n\t\tthis._typeDd.select(0);\n\t\tthis._conditionDd.select(0);\n\n\t\tthis._valueIn.val('');\n\n\t\t// Clear grid selection\n\t\tthis._grid.clearSelection();\n\n\t\t// Clear table\n\t\tthis._grid.setDataSource([]);\n\t}\n\n\t_getExpressionPart(varLabel, varName, varValue, typeLabel, condition, operator)\n\t{\n\t\treturn {\n\t\t\t'label': varLabel,\n\t\t\t'name': varName,\n\t\t\t'value': varValue,\n\t\t\t'typeLabel': typeLabel,\n\t\t\t'symbol': condition.symbol,\n\t\t\t'operator': operator,\n\t\t\t'_condition': condition,\n\t\t};\n\t}\n\n\t_enableOperatorRb(enable)\n\t{\n\t\t$('#andOperatorRB', this._dialog).attr('disabled', !enable);\n\t\t$('#orOperatorRB', this._dialog).attr('disabled', !enable);\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('match-expression-builder'))\n\twindow.customElements.define('match-expression-builder', MatchExpressionBuilder);\n","export class SidebarLayout extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\n\t\t// Attach a shadow root\n\t\tconst shadowRoot = this.attachShadow({mode: 'open'});\n\t\tshadowRoot.innerHTML = `\n\t\t\t<style>\n\t\t\t\t:host {\n\t\t\t\t\tdisplay: flex;\n\t\t\t\t\tflex-direction: row;\n\t\t\t\t}\n\n\t\t\t\t@media (max-width: 575.98px) {\n\t\t\t\t\t:host(.split-xs) ::slotted(:not([aria-selected=\"true\"])) {\n\t\t\t\t\t\tdisplay: none !important;\n\t\t\t\t    }\n\n\t\t\t\t\t:host(.split-xs) ::slotted([aria-selected=\"true\"]) {\n\t\t\t\t\t\tflex-grow: 1;\n\t\t\t\t    }\n\t\t\t\t}\n\n\t\t\t\t@media (max-width: 767.98px) {\n\t\t\t\t\t:host(.split-sm) ::slotted(:not([aria-selected=\"true\"])) {\n\t\t\t\t\t\tdisplay: none !important;\n\t\t\t\t    }\n\n\t\t\t\t\t:host(.split-sm) ::slotted([aria-selected=\"true\"]) {\n\t\t\t\t\t\tflex-grow: 1;\n\t\t\t\t    }\n\t\t\t\t}\n\n\t\t\t\t@media (max-width: 991.98px) {\n\t\t\t\t\t:host(.split-md) ::slotted(:not([aria-selected=\"true\"])) {\n\t\t\t\t\t\tdisplay: none !important;\n\t\t\t\t    }\n\n\t\t\t\t\t:host(.split-md) ::slotted([aria-selected=\"true\"]) {\n\t\t\t\t\t\tflex-grow: 1;\n\t\t\t\t    }\n\t\t\t\t}\n\n\t\t\t\t@media (max-width: 1199.98px) {\n\t\t\t\t\t:host(.split-lg) ::slotted(:not([aria-selected=\"true\"])) {\n\t\t\t\t\t\tdisplay: none !important;\n\t\t\t\t    }\n\n\t\t\t\t\t:host(.split-lg) ::slotted([aria-selected=\"true\"]) {\n\t\t\t\t\t\tflex-grow: 1;\n\t\t\t\t    }\n\t\t\t\t}\n\n\t\t\t\t.side-col::slotted(*) {\n\t\t\t\t}\n\n\t\t\t\t.main-col::slotted(*) {\n\t\t\t\t\tflex-grow: 1;\n\t\t\t\t}\n\t\t\t</style>\n\n\t\t\t<slot class=\"side-col\" name=\"side-column\"></slot>\n\t\t\t<slot class=\"main-col\" name=\"main-column\"></slot>\n\t\t`;\n\n\t\t// Set initial selection\n\t\tthis.selectedIndex = 0;\n\t}\n\n\tget selectedPanel()\n\t{\n\t\treturn this._selectedPanel;\n\t}\n\n\tset selectedPanel(element) // 'side' or 'main'\n\t{\n\t\tif (element != null && element.parentNode == this)\n\t\t{\n\t\t\tthis._selectedPanel = element;\n\n\t\t\tfor (let element of this.children)\n\t\t\t{\n\t\t\t\tif (element == this._selectedPanel)\n\t\t\t\t\telement.setAttribute('aria-selected', 'true');\n\t\t\t\telse\n\t\t\t\t\telement.removeAttribute('aria-selected');\n\t\t\t}\n\t\t}\n\t\telse\n\t\t{\n\t\t\tconsole.error('Element is not a child of SidebarLayout');\n\t\t}\n\t}\n\n\tget selectedIndex()\n\t{\n\t\treturn Array.from(this.children).indexOf(this._selectedPanel);\n\t}\n\n\tset selectedIndex(index)\n\t{\n\t\tif (this.children.length > 0)\n\t\t{\n\t\t\tif (this.children[index] == null)\n\t\t\t{\n\t\t\t\tconsole.error('Invalid SidebarLayout index');\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tlet element = this.children[index];\n\t\t\tthis.selectedPanel = element;\n\t\t}\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('sidebar-layout'))\n\twindow.customElements.define('sidebar-layout', SidebarLayout);\n","import {BaseModule} from './base-module';\nimport {SidebarLayout} from '../components/sidebar-layout';\nimport {MatchExpressionBuilder} from '../components/module-specific/match-expression-builder';\nimport {ConfigInterfaceBuilder} from '../utils/uibuilder/config-interface-builder';\nimport {capitalizeFirst, filterClassName, roundToDecimals, scaleBytes} from '../utils/utilities';\nimport {RoomPropertiesData,UserPropertiesData} from '../utils/match-properties';\nimport {UserCountChart} from '../components/charts/user-count-chart';\nimport {getTimeRangeTooltipTemplate, getWRBytesLabelTemplate} from '../components/charts/chart-utils';\n\nexport default class ZoneMonitor extends BaseModule\n{\n\tconstructor()\n\t{\n\t    super('zoneMonitor');\n\n\t\tthis.MONITORED_TYPE_ZONE = 'zone';\n\t\tthis.MONITORED_TYPE_ROOM = 'room';\n\t\tthis.MONITORED_TYPE_USER = 'user';\n\n\t\tthis.ANY_LABEL = '[any]';\n\t\tthis.DEFAULT_GROUP_NAME = 'default';\n\t\tthis.MAX_EXTENSION_LOG_SIZE = 300;\n\t\tthis.KICK_BAN_DEFAULT_DELAY = 5;\n\n\t\t// Outgoing requests\n\t\tthis.REQ_GET_DATA = 'getData';\n\t\tthis.REQ_SET_ZONE_SETTINGS = 'setZoneSettings';\n\t\tthis.REQ_SET_ROOM_SETTINGS = 'setRoomSettings';\n\t\tthis.REQ_SET_USER_SETTINGS = 'setUserSettings';\n\t\tthis.REQ_RELOAD_ZONE_EXT = 'reloadZoneExt';\n\t\tthis.REQ_RELOAD_WORDS = 'reloadWords';\n\t\tthis.REQ_KICK_USER = 'kickUser';\n\t\tthis.REQ_BAN_USER = 'banUser';\n\t\tthis.REQ_GET_ROOM_CONFIG = 'getRoomConfig';\n\t\tthis.REQ_CREATE_ROOM = 'createRoom';\n\t\tthis.REQ_REMOVE_ROOM = 'removeRoom';\n\t\tthis.REQ_ADMIN_MSG = 'adminMsg';\n\n\t\t// Incoming responses\n\t\tthis.RESP_DATA = 'data';\n\t\tthis.RESP_UNEXPECTED_ERROR = 'unexpError';\n\t\tthis.RESP_UPDATE_ERROR = 'updateError';\n\t\tthis.RESP_ROOM_CONFIG = 'roomConfig';\n\t\tthis.RESP_ROOM_CONFIG_ERROR = 'roomConfigErr';\n\t\tthis.RESP_ROOM_CREATED = 'roomCreated';\n\t\tthis.RESP_ROOM_CREATION_ERROR = 'roomError';\n\t\tthis.RESP_LOG_MESSAGES = 'log'; // This response doesn't have a corresponding request because it is managed by a server event listener\n\n\t\tthis._currentRequestId = -1;\n\t}\n\n\t//------------------------------------\n\t// COMMON MODULE INTERFACE METHODS\n\t// This members are used by the main controller\n\t// to communicate with the module's controller.\n\t//------------------------------------\n\n\tinitialize(idData, shellController)\n\t{\n\t\t// Call super method\n\t\tsuper.initialize(idData, shellController);\n\n\t\t// Set monitoring interface\n\t\tthis._setMonitoringInterface();\n\n\t\t// Initialize progress bar\n\t\t$('#znm-progressBar').kendoProgressBar({\n\t\t\tmin: 0,\n            max: 100,\n\t\t\tvalue: false,\n            type: 'value',\n            animation: {\n                duration: 400\n            }\n        });\n\n\t\t// LISTS\n\n\t\t// Initialize interval dropdown\n\t\tthis._intervalDropDown = $('#znm-intervalDD').kendoDropDownList({\n\t\t\tvalueTemplate: '<span class=\"text-muted pr-1\">Interval:</span><span>#:data.text#</span>',\n\t\t\tchange: $.proxy(this._onUpdateIntervalChange, this)\n\t\t}).data('kendoDropDownList');\n\n\t\t// Initialize pane bar\n\t\tthis._accordion = $('#znm-panelbar').kendoPanelBar({\n            expandMode: 'single',\n\t\t\tchange: $.proxy(this._onEntitiesPanelChange, this)\n        }).data('kendoPanelBar');\n\n\t\t// Initialize lists\n\t\tthis._zoneListBox = $(\"#znm-zoneList\").kendoListBox({\n\t\t\ttemplate: '<div>#:data.name# (#:data.users#)</div>',\n\t\t\tchange: $.proxy(this._onZoneSelected, this)\n\t\t}).data('kendoListBox');\n\n\t\tthis._zoneListBox.wrapper.find('.k-list').on('dblclick', '.k-item', $.proxy(this._onMonitorSelectionBtClick, this));\n\n\t\tthis._roomListBox = $(\"#znm-roomList\").kendoListBox({\n\t\t\ttemplate: '<div>#:data.name# (#:data.users#)</div>',\n\t\t\tchange: $.proxy(this._onRoomSelected, this)\n\t\t}).data('kendoListBox');\n\n\t\tthis._roomListBox.wrapper.find('.k-list').on('dblclick', '.k-item', $.proxy(this._onMonitorSelectionBtClick, this));\n\n\t\tthis._userListBox = $(\"#znm-userList\").kendoListBox({\n\t\t\ttemplate: '<div>#:data.name#</div>',\n\t\t\tchange: $.proxy(this._onUserSelected, this)\n\t\t}).data('kendoListBox');\n\n\t\tthis._userListBox.wrapper.find('.k-list').on('dblclick', '.k-item', $.proxy(this._onMonitorSelectionBtClick, this));\n\n\t\t// Initialize Room Groups dropdown\n\t\tthis._groupsDropDown = $('#znm-roomGroupsDD').kendoDropDownList({\n\t\t\tdataTextField: 'label',\n  \t\t\tdataValueField: 'name',\n\t\t\tchange: $.proxy(this._onGroupChange, this)\n\t\t}).data('kendoDropDownList');\n\n\t\t// Build initial groups list\n\t\tthis._setGroupsDataProvider(null);\n\n\t\t// // Create interface builder instances\n\t\tthis._interfaceBuilder = new ConfigInterfaceBuilder();\n\t\tthis._roomCreationIBuilder = new ConfigInterfaceBuilder();\n\n\t\t// Add listener to show limit warning tooltips\n\t\t$('#znm-panelbar').kendoTooltip({\n\t\t\tfilter: 'i[title].limit-warning',\n\t\t\tposition: 'right',\n\t\t\twidth: '250px',\n\t\t\tcontent: function(e) {\n\t\t\t\treturn `<div class=\"help-tooltip\">${e.target.data('title')}</div>`;\n\t\t\t}\n\t\t});\n\n\t\t// ROOM CREATION PANEL\n\n\t\t// Add click listener to Create & Remove Room buttons\n\t\t$('#znm-createRoomBt').on('click', $.proxy(this._onCreateRoomBtClick, this));\n\t\t$('#znm-removeRoomBt').on('click', $.proxy(this._onRemoveRoomBtClick, this));\n\n\t\t// Configure room creation panel\n\t\t$('#znm-createRoomModal').modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t\tshow: false,\n\t\t});\n\n\t\t// Add listener for room creation panel's create button click\n\t\t$('#znm-roomCreatorCreateBt').on('click', $.proxy(this._onRoomCreatorCreateBtClick, this));\n\n\t\t// Add listener for room creation panel hide\n\t\t$('#znm-createRoomModal').on('hidden.bs.modal', $.proxy(this._onCreateRoomModalHidden, this));\n\n\t\t// ROOM & USER FILTERING PANELS\n\n\t\t// Setup match expression builder panel for rooms\n\t\tthis._roomFilter = document.getElementById('znm-roomFilter');\n\t\tthis._roomFilter.title = 'Room Filter Expression Builder';\n\t\tthis._roomFilter.predefinedVarNames = RoomPropertiesData.propertiesArray;\n\t\t$(this._roomFilter).on(this._roomFilter.EXPRESSION_UPDATED_EVENT, $.proxy(this._onRoomFilterUpdated, this));\n\n\t\t$('#znm-filterRoomBt').on('click', $.proxy(this._onShowRoomFilterBtClick, this));\n\t\t$('#znm-applyRoomFilterCB').on('change', $.proxy(this._onApplyBtChange, this));\n\n\t\t// Setup match expression builder panel for users\n\t\tthis._userFilter = document.getElementById('znm-userFilter');\n\t\tthis._userFilter.title = 'User Filter Expression Builder';\n\t\tthis._userFilter.predefinedVarNames = UserPropertiesData.propertiesArray;\n\t\t$(this._userFilter).on(this._userFilter.EXPRESSION_UPDATED_EVENT, $.proxy(this._onUserFilterUpdated, this));\n\n\t\t$('#znm-filterUserBt').on('click', $.proxy(this._onShowUserFilterBtClick, this));\n\t\t$('#znm-applyUserFilterCB').on('change', $.proxy(this._onApplyBtChange, this));\n\n\t\t// MAIN CONTROLS\n\n\t\t$('#znm-monitorBt').on('click', $.proxy(this._onMonitorSelectionBtClick, this));\n\t\t$('#znm-sendMessageBt').on('click', $.proxy(this._onSendAdminMsgBtClick, this));\n\t\t$('#znm-messageIn').on('keyup', $.proxy(this._onSendAdminMsgInKeyUp, this));\n\n\t\t// MONITORING CONTROLS\n\n\t\t$('#znm-closeMonitorBt').on('click', $.proxy(this._onCloseMonitorBtClick, this));\n\t\t$('#znm-editBt').on('click', $.proxy(this._onEditBtClick, this));\n\t\t$('#znm-cancelBt').on('click', $.proxy(this._onCancelBtClick, this));\n\t\t$('#znm-submitBt').on('click', $.proxy(this._onSubmitBtClick, this));\n\n\t\t// Hide edit controls\n\t\t$('#znm-editControls').hide();\n\n\t\t// ZONE MONITOR ---------------------\n\n\t\t// Set reference to user count chart\n\t\tthis._userCountChart = document.getElementById('znm-userCountChart');\n\n\t\t// Initialize time range slider\n\t\tthis._timeSlider = $('#znm-timeSlider').kendoRangeSlider({\n\t\t\tmin: -24,\n            max: 0,\n            smallStep: 1,\n            largeStep: 0,\n\t\t\tchange: $.proxy(this._onTimeRangeChange, this),\n\t\t\ttooltip: {\n\t\t\t\ttemplate: getTimeRangeTooltipTemplate()\n\t\t\t},\n\t\t\tleftDragHandleTitle: 'Drag',\n\t\t\trightDragHandleTitle: 'Drag',\n\t\t}).data('kendoRangeSlider');\n\n\t\t// Reset time range on user count chart\n\t\tthis._onTimeRangeChange();\n\n\t\t// Add button click listeners\n\t\t$('#znm-reloadWordsBt').on('click', $.proxy(this._onWordsReloadBtClick, this));\n\t\t$('#znm-reloadZoneExtBt').on('click', $.proxy(this._onZoneExtReloadBtClick, this));\n\n\t\t// Initialize extension log grid\n\t\tthis._zoneExtLogGrid = $('#znm-zoneExtLogGrid').kendoGrid({\n\t\t\tdataSource: [],\n\t\t\tresizable: true,\n\t\t\tselectable: false,\n\t\t\tnoRecords: {\n\t\t\t\ttemplate: 'No entries.'\n\t\t\t},\n\t\t\tcolumns: [\n\t\t\t\t{\n\t\t\t\t\tfield: 'timestamp',\n\t\t\t\t\ttitle: 'Timestamp',\n\t\t\t\t\twidth: 180\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'level',\n\t\t\t\t\ttitle: 'Level',\n\t\t\t\t\twidth: 100\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'message',\n\t\t\t\t\ttitle: 'Message',\n\t\t\t\t\twidth: 400\n\t\t\t\t},\n\t\t\t],\n\t\t\tdataBound: function() {\n\t\t\t    this.content.scrollTop(this.tbody.height()); // Scroll to bottom\n\t\t\t}\n\t\t}).data('kendoGrid');\n\n\t\t// ROOM MONITOR ---------------------\n\n\t\t// Initialize extension log grid\n\t\tthis._roomExtLogGrid = $('#znm-roomExtLogGrid').kendoGrid({\n\t\t\tdataSource: [],\n\t\t\tresizable: true,\n\t\t\tselectable: false,\n\t\t\tnoRecords: {\n\t\t\t\ttemplate: 'No entries.'\n\t\t\t},\n\t\t\tcolumns: [\n\t\t\t\t{\n\t\t\t\t\tfield: 'timestamp',\n\t\t\t\t\ttitle: 'Timestamp',\n\t\t\t\t\twidth: 180\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'level',\n\t\t\t\t\ttitle: 'Level',\n\t\t\t\t\twidth: 100\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'message',\n\t\t\t\t\ttitle: 'Message',\n\t\t\t\t\twidth: 400\n\t\t\t\t},\n\t\t\t],\n\t\t\tdataBound: function() {\n\t\t\t    this.content.scrollTop(this.tbody.height()); // Scroll to bottom\n\t\t\t}\n\t\t}).data('kendoGrid');\n\n\t\t// USER MONITOR ---------------------\n\n\t\tthis._kickDelayIn = $('#znm-kickDelayNS').kendoNumericTextBox({\n\t\t    min: 0,\n\t\t    step: 1,\n\t\t    format: '#',\n\t\t\tplaceholder: 'Delay (s)'\n\t\t}).data('kendoNumericTextBox');\n\n\t\tthis._banModeDd = $('#znm-banModeDD').kendoDropDownList({\n\t\t\tdataSource: ['IP', 'NAME'],\n\t\t\tautoWidth: true,\n\t\t\toptionLabel: {\n\t\t\t\ttext: 'Mode'\n\t\t\t},\n\t\t\toptionLabelTemplate: '<span class=\"text-muted\">#:text#</span>',\n\t\t}).data('kendoDropDownList');\n\n\t\tthis._banDurUnitDd = $('#znm-banDurationUnitDD').kendoDropDownList({\n\t\t\tautoWidth: true,\n\t\t}).data('kendoDropDownList');\n\n\t\tthis._banDurationIn = $('#znm-banDurationNS').kendoNumericTextBox({\n\t\t    min: 1,\n\t\t    step: 1,\n\t\t    format: '#',\n\t\t\tplaceholder: 'Duration'\n\t\t}).data('kendoNumericTextBox');\n\n\t\tthis._banDelayIn = $('#znm-banDelayNS').kendoNumericTextBox({\n\t\t    min: 0,\n\t\t    step: 1,\n\t\t    format: '#',\n\t\t\tplaceholder: 'Delay (s)'\n\t\t}).data('kendoNumericTextBox');\n\n\t\t// Add button click listeners\n\t\t$('#znm-kickBt').on('click', $.proxy(this._onKickBtClick, this));\n\t\t$('#znm-banBt').on('click', $.proxy(this._onBanBtClick, this));\n\n\t\t// Charts\n\n\t\tthis._packetQueueChart = $('#znm-packetQueueChart').kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 100\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'bar',\n\t            labels: {\n\t                visible: true,\n\t                format: '{0}%',\n\t                background: 'transparent',\n\t\t\t\t\tposition: 'insideBase'\n\t            },\n\t\t\t\tgap: .5,\n\t\t\t\tspacing: 0,\n\t\t\t\toverlay: {\n\t\t\t\t\tgradient: 'none'\n\t\t\t\t},\n\t\t\t\tborder: {\n\t\t\t\t\twidth: 0\n\t\t\t\t}\n\t        },\n\t        series: [\n\t\t\t\t{\n\t\t\t\t\tname: 'Packet queue',\n\t\t            field: 'value',\n\t\t            categoryField: 'category',\n\t\t\t\t\tcolor: '#FB7D33'\n\t\t        }\n\t\t\t],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}%'\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0,\n\t    \t\tmax: 100\n\t        },\n\t        categoryAxis: {\n\t            majorGridLines: {\n\t                visible: false\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: false\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: false\n\t\t\t\t}\n\t        },\n\t\t\ttooltip: {\n\t\t\t\tvisible: false,\n\t        }\n\t    }).data('kendoChart');\n\n\t\tthis._droppedMsgChart = $('#znm-droppedMsgChart').kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 100\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'bar',\n\t            labels: {\n\t                visible: true,\n\t                format: '{0}',\n\t                background: 'transparent',\n\t\t\t\t\tposition: 'insideBase',\n\t            },\n\t\t\t\tgap: .5,\n\t\t\t\tspacing: 0,\n\t\t\t\toverlay: {\n\t\t\t\t\tgradient: 'none'\n\t\t\t\t},\n\t\t\t\tborder: {\n\t\t\t\t\twidth: 0\n\t\t\t\t}\n\t        },\n\t        series: [\n\t\t\t\t{\n\t\t\t\t\tname: 'Dropped messages',\n\t\t            field: 'value',\n\t\t            categoryField: 'category',\n\t\t\t\t\tcolor: '#FB7D33'\n\t\t        }\n\t\t\t],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}'\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0,\n\t        },\n\t        categoryAxis: {\n\t            majorGridLines: {\n\t                visible: false\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: false\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: false\n\t\t\t\t}\n\t        },\n\t\t\ttooltip: {\n\t\t\t\tvisible: false,\n\t        }\n\t    }).data('kendoChart');\n\n\t\tthis._writtenDataChart = $('#znm-writtenDataChart').kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 100\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'bar',\n\t            labels: {\n\t                visible: true,\n\t                format: '{0}',\n\t                background: 'transparent',\n\t\t\t\t\tposition: 'insideBase',\n\t\t\t\t\ttemplate: getWRBytesLabelTemplate(),\n\t            },\n\t\t\t\tgap: .5,\n\t\t\t\tspacing: 0,\n\t\t\t\toverlay: {\n\t\t\t\t\tgradient: 'none'\n\t\t\t\t},\n\t\t\t\tborder: {\n\t\t\t\t\twidth: 0\n\t\t\t\t}\n\t        },\n\t        series: [\n\t\t\t\t{\n\t\t\t\t\tname: 'Written data',\n\t\t            field: 'value',\n\t\t            categoryField: 'category',\n\t\t\t\t\tcolor: '#FB7D33'\n\t\t        }\n\t\t\t],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}'\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0,\n\t        },\n\t        categoryAxis: {\n\t            majorGridLines: {\n\t                visible: false\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: false\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: false\n\t\t\t\t}\n\t        },\n\t\t\ttooltip: {\n\t\t\t\tvisible: false,\n\t        }\n\t    }).data('kendoChart');\n\n\t\tthis._readDataChart = $('#znm-readDataChart').kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 100\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'bar',\n\t            labels: {\n\t                visible: true,\n\t                format: '{0}',\n\t                background: 'transparent',\n\t\t\t\t\tposition: 'insideBase',\n\t\t\t\t\ttemplate: getWRBytesLabelTemplate(),\n\t            },\n\t\t\t\tgap: .5,\n\t\t\t\tspacing: 0,\n\t\t\t\toverlay: {\n\t\t\t\t\tgradient: 'none'\n\t\t\t\t},\n\t\t\t\tborder: {\n\t\t\t\t\twidth: 0\n\t\t\t\t}\n\t        },\n\t        series: [\n\t\t\t\t{\n\t\t\t\t\tname: 'Read data',\n\t\t            field: 'value',\n\t\t            categoryField: 'category',\n\t\t\t\t\tcolor: '#FB7D33'\n\t\t        }\n\t\t\t],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}'\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0,\n\t        },\n\t        categoryAxis: {\n\t            majorGridLines: {\n\t                visible: false\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: false\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: false\n\t\t\t\t}\n\t        },\n\t\t\ttooltip: {\n\t\t\t\tvisible: false,\n\t        }\n\t    }).data('kendoChart');\n\n\t\t//-------------------------------\n\n\t\t// Add listener to redraw all charts in case of window resize\n\t\t$(window).on('resize', $.proxy(this._onWindowResize, this));\n\t\tthis._onWindowResize(); // Also do it immediately\n\n\t\t// Request data to server\n\t\tthis._requestData(true);\n\t}\n\n\tdestroy()\n\t{\n\t\t// Call super method\n\t\tsuper.destroy();\n\n\t\t// Close monitoring\n\t\tthis._onCloseMonitorBtClick();\n\n\t\t// Remove doubleclick event listeners\n\t\tthis._zoneListBox.wrapper.find('.k-list').off('dblclick');\n\t\tthis._roomListBox.wrapper.find('.k-list').off('dblclick');\n\t\tthis._userListBox.wrapper.find('.k-list').off('dblclick');\n\n\t\t// Remove other event listeners\n\t\t$('#znm-createRoomBt').off('click');\n\t\t$('#znm-removeRoomBt').off('click');\n\t\t$('#znm-roomCreatorCreateBt').off('click');\n\t\t$('#znm-filterRoomBt').off('click');\n\t\t$('#znm-applyRoomFilterCB').off('change');\n\t\t$('#znm-filterUserBt').off('click');\n\t\t$('#znm-applyUserFilterCB').off('change');\n\t\t$('#znm-monitorBt').off('click');\n\t\t$('#znm-sendMessageBt').off('click');\n\t\t$('#znm-messageIn').off('keyup');\n\t\t$('#znm-closeMonitorBt').off('click');\n\t\t$('#znm-editBt').off('click');\n\t\t$('#znm-cancelBt').off('click');\n\t\t$('#znm-submitBt').off('click');\n\t\t$(window).off('resize');\n\t\t$('#znm-reloadWordsBt').off('click');\n\t\t$('#znm-reloadZoneExtBt').off('click');\n\t\t$('#znm-kickBt').off('click');\n\t\t$('#znm-banBt').off('click');\n\n\t\t// Clear request scheduling\n\t\tclearTimeout(this._requestTimer);\n\n\t\t// Hide room creation panel\n\t\t$('#znm-createRoomModal').modal('hide');\n\n\t\t// Remove room creation panel listener\n\t\t$('#znm-createRoomModal').off('hidden.bs.modal');\n\n\t\t// Remove filter panels listeners\n\t\t$(this._roomFilter).off(this._roomFilter.EXPRESSION_UPDATED_EVENT);\n\t\t$(this._userFilter).off(this._userFilter.EXPRESSION_UPDATED_EVENT);\n\t}\n\n\tonExtensionCommand(command, data)\n\t{\n\t\t// Data received\n\t\tif (command == this.RESP_DATA)\n\t\t{\n\t\t\t// We have to check if the response id corresponds to the current request id;\n\t\t\t// if not, the response is discarded as it refers to an outdatet request\n\t\t\tconst responseId = data.getInt('id');\n\n\t\t\tif (responseId == this._currentRequestId)\n\t\t\t{\n\t\t\t\t// --- ZONES LIST ---\n\t\t\t\tif (data.containsKey('zones'))\n\t\t\t\t{\n\t\t\t\t\tthis._setZonesDataProvider(data.getSFSArray('zones'));\n\n\t\t\t\t\tif (this._selectedZone)\n\t\t\t\t\t\tthis._setScopeLabel();\n\t\t\t\t}\n\n\t\t\t\t// --- ROOMS LIST ---\n\t\t\t\tif (data.containsKey('rooms'))\n\t\t\t\t{\n\t\t\t\t\tthis._setGroupsDataProvider(data.getUtfStringArray('groups'));\n\n\t\t\t\t\t// -------------------------------\n\n\t\t\t\t\t// If a null rooms list is returned, the zone must have been deleted!\n\t\t\t\t\tif (!data.isNull('rooms'))\n\t\t\t\t\t{\n\t\t\t\t\t\tthis._setRoomsDataProvider(data.getSFSArray('rooms'));\n\n\t\t\t\t\t\tif (!this._selectedRoom)\n\t\t\t\t\t\t\tthis._setScopeLabel();\n\n\t\t\t\t\t\t// -------------------------------\n\n\t\t\t\t\t\t// Show warning if the rooms list exceeds the server-side limit\n\t\t\t\t\t\tif (data.containsKey('rSize'))\n\t\t\t\t\t\t\tthis._showLimitExceededWarning($('#znm-roomListWarning'), 'The received list is incomplete, because its size (' + data.getInt('rSize') + ' Rooms) exceeded this client\\'s limit of ' + data.getInt('rLimit') + '; you should refine your search');\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tthis._showLimitExceededWarning($('#znm-roomListWarning'), '');\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tif (this._activePanelType == 'room')\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tthis._accordion.select('[data-item-type=\"zone\"]');\n\t\t\t\t\t\t\tthis._accordion.expand('[data-item-type=\"zone\"]');\n\t\t\t\t\t\t\tthis._onZoneSelected();\n\t\t\t\t\t\t\tthis._onEntitiesPanelChange();\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// --- USERS LIST ---\n\t\t\t\tif (data.containsKey('users'))\n\t\t\t\t{\n\t\t\t\t\t// If a null users list is returned, the room must have been deleted!\n\t\t\t\t\tif (!data.isNull('users'))\n\t\t\t\t\t{\n\t\t\t\t\t\tthis._setUsersDataProvider(data.getSFSArray('users'));\n\n\t\t\t\t\t\t// Show warning if the users list exceeds the server-side limit\n\t\t\t\t\t\tif (data.containsKey('uSize'))\n\t\t\t\t\t\t\tthis._showLimitExceededWarning($('#znm-userListWarning'), 'The received list is incomplete, because its size (' + data.getInt('uSize') + ' users) exceeds this client\\'s limit of ' + data.getInt('uLimit') + '; you should refine your search');\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tthis._showLimitExceededWarning($('#znm-userListWarning'), '');\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tif (this._activePanelType == 'user')\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tthis._accordion.select('[data-item-type=\"room\"]');\n\t\t\t\t\t\t\tthis._accordion.expand('[data-item-type=\"room\"]');\n\t\t\t\t\t\t\tthis._onRoomSelected();\n\t\t\t\t\t\t\tthis._onEntitiesPanelChange();\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// --- MONITORED DATA ---\n\t\t\t\tif (data.containsKey('monitored'))\n\t\t\t\t{\n\t\t\t\t\tconst zone = data.getSFSObject('monitored').getUtfString('zone');\n\t\t\t\t\tconst type = data.getSFSObject('monitored').getUtfString('type');\n\t\t\t\t\tconst name = data.getSFSObject('monitored').getUtfString('name');\n\n\t\t\t\t\tif (zone == this._monitoredZone && type == this._monitoredType && name == this._monitoredName)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (!data.getSFSObject('monitored').isNull('params'))\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tconst params = data.getSFSObject('monitored').getSFSArray('params');\n\n\t\t\t\t\t\t\tthis._setMonitoringInterface();\n\n\t\t\t\t\t\t\t// Build interface\n\t\t\t\t\t\t\tif (!this._isEditing)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tthis._interfaceBuilder.buildInterface(params, `znm-${this._monitoredType}TabNavigator`, true);\n\n\t\t\t\t\t\t\t\tif (!this._skipInitTabs)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t$(`#znm-${this._monitoredType}TabNavigator #tabs`).scrollingTabs({\n\t\t\t\t\t\t\t\t\t\tbootstrapVersion: 4,\n\t\t\t\t\t\t\t\t\t\tscrollToTabEdge: true,\n\t\t\t\t\t\t\t\t\t\tenableSwiping: true,\n\t\t\t\t\t\t\t\t\t\tdisableScrollArrowsOnFullyScrolled: true,\n\t\t\t\t\t\t\t\t\t\tcssClassLeftArrow: 'fa fa-chevron-left',\n\t\t\t\t\t\t\t\t\t\tcssClassRightArrow: 'fa fa-chevron-right'\n\t\t\t\t\t\t\t\t\t});\n\n\t\t\t\t\t\t\t\t\tthis._skipInitTabs = true;\n\n\t\t\t\t\t\t\t\t\t// Set first tab panel as active\n\t\t\t\t\t\t\t\t\tthis._interfaceBuilder.activateFirstTabPanel();\n\n\t\t\t\t\t\t\t\t\t// Redraw time range slider to set its width\n\t\t\t\t\t\t\t\t\tthis._timeSlider.resize();\n\n\t\t\t\t\t\t\t\t\t// Add listener to redraw all charts on tab change\n\t\t\t\t\t\t\t\t\t// (to work around an issue with the chart resizing to default width when pane is hidden)\n\t\t\t\t\t\t\t\t\t$('a[data-toggle=\"tab\"]').on('shown.bs.tab', $.proxy(this._onWindowResize, this));\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t// Disable categories\n\t\t\t\t\t\t\tlet disabledCat = (data.getSFSObject('monitored').containsKey('exclude') ? data.getSFSObject('monitored').getUtfStringArray('exclude') : []);\n\n\t\t\t\t\t\t\tfor (let cat of disabledCat)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tlet navLink = $(`#znm-${this._monitoredType}TabNavigator #tabs #tab-${cat}`);\n\n\t\t\t\t\t\t\t\tnavLink.addClass('disabled');\n\t\t\t\t\t\t\t\tnavLink.attr('tabindex', -1);\n\t\t\t\t\t\t\t\tnavLink.attr('aria-disabled', true);\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tif (this._monitoredType == this.MONITORED_TYPE_ZONE)\n\t\t\t\t\t\t\t\tthis._showZoneTrafficData(data.getSFSObject('monitored').getSFSObject('traffic'));\n\n\t\t\t\t\t\t\tif (this._monitoredType == this.MONITORED_TYPE_USER)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t// Stats\n\t\t\t\t\t\t\t\tthis._showUserStatsData(data.getSFSObject('monitored').getSFSObject('stats'));\n\n\t\t\t\t\t\t\t\t// NPC\n\t\t\t\t\t\t\t\tconst isNpc = data.getSFSObject('monitored').getBool('npc');\n\n\t\t\t\t\t\t\t\t// Disable kick and ban buttons\n\t\t\t\t\t\t\t\t$('#znm-kickBt').attr('disabled', isNpc);\n\t\t\t\t\t\t\t\t$('#znm-banBt').attr('disabled', isNpc);\n\n\t\t\t\t\t\t\t\t// Geo-location\n\t\t\t\t\t\t\t\tconst geoLocData = data.getSFSObject('monitored').getSFSObject('geoLoc');\n\t\t\t\t\t\t\t\tthis._setGeoLocationUI(geoLocData);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t// Show alert\n\t\t\t\t\t\t\tthis.shellCtrl.showSimpleAlert(`The ${capitalizeFirst(this._monitoredType)} being monitored is no more available on the server, please select another one.`, false);\n\n\t\t\t\t\t\t\t// Cancel monitoring\n\t\t\t\t\t\t\tthis._onCloseMonitorBtClick();\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}\n\n\t\telse if (command == this.RESP_UPDATE_ERROR)\n\t\t{\n\t\t\tif (data.getUtfString('zone') == this._monitoredZone\n\t\t\t\t&& data.getUtfString('type') ==  this._monitoredType\n\t\t\t\t&& data.getUtfString('name') ==  this._monitoredName)\n\t\t\t{\n\t\t\t\t// Show alert\n\t\t\t\tthis.shellCtrl.showSimpleAlert(`Unable to update ${capitalizeFirst(this._monitoredType)} settings; the following error was reported: ${data.getUtfString('message')}.`);\n\t\t\t}\n\t\t}\n\n\t\telse if (command == this.RESP_ROOM_CONFIG)\n\t\t{\n\t\t\t// Re-enable Create button\n\t\t\t$('#znm-createRoomBt').attr('disabled', false);\n\n\t\t\t// Show Room creation panel\n\t\t\tthis._showRoomCreationPanel(data.getSFSArray('settings'));\n\t\t}\n\n\t\telse if (command == this.RESP_ROOM_CONFIG_ERROR)\n\t\t{\n\t\t\t// Re-enable Create button\n\t\t\t$('#znm-createRoomBt').attr('disabled', false);\n\n\t\t\t// Show an alert\n\t\t\tthis.shellCtrl.showSimpleAlert(data.getUtfString('error'));\n\t\t}\n\n\t\telse if (command == this.RESP_ROOM_CREATED)\n\t\t{\n\t\t\t// Re-enable room creation panel\n\t\t\tthis._enableCreateRoomPanel(true);\n\n\t\t\t// Hide room creation panel\n\t\t\t$('#znm-createRoomModal').modal('hide');\n\n\t\t\t// Update rooms list\n\t\t\tthis._requestData(true);\n\t\t}\n\n\t\telse if (command == this.RESP_ROOM_CREATION_ERROR)\n\t\t{\n\t\t\t// Display error message\n\t\t\t$('#znm-createRoomError').text(data.getUtfString('error'));\n\t\t\t$('#znm-createRoomError').show();\n\n\t\t\t// Re-enable room creation panel\n\t\t\tthis._enableCreateRoomPanel(true);\n\t\t}\n\n\t\t// Unexpected error received\n\t\t// This error is returned in case something went wrong when retrieving the monitored entity parameters,\n\t\t// so we have to stop requesting the same entity\n\t\telse if (command == this.RESP_UNEXPECTED_ERROR)\n\t\t{\n\t\t\t// Cancel monitoring\n\t\t\tthis._onCloseMonitorBtClick();\n\n\t\t\t// Show an alert\n\t\t\tthis.shellCtrl.showSimpleAlert(data.getUtfString('error'));\n\t\t}\n\n\t\t// Extension (Zone or Room) log messages received\n\t\telse if (command == this.RESP_LOG_MESSAGES)\n\t\t{\n\t\t\tlet logEntries = data.getSFSArray('entries');\n\n\t\t\tfor (let e = 0; e < logEntries.size(); e++)\n\t\t\t{\n\t\t\t\tlet logEntry = logEntries.getSFSObject(e);\n\n\t\t\t\tconst logZone = logEntry.getUtfString('zone');\n\t\t\t\tconst logRoom = logEntry.getUtfString('room');\n\n\t\t\t\t// Check if the zone is currently monitored\n\t\t\t\tif (this._monitoredZone == logZone)\n\t\t\t\t{\n\t\t\t\t\t// console.log(logEntry.getLong('time'))\n\t\t\t\t\t// console.log(new Date(logEntry.getLong('time'))\n\n\t\t\t\t\tlet ele = {\n\t\t\t\t\t\ttimestamp: kendo.toString(new Date(logEntry.getLong('time')), 'dd/MM/yyyy HH:mm:ss'),\n\t\t\t\t\t\tlevel: logEntry.getUtfString('level'),\n\t\t\t\t\t\tmessage: logEntry.getUtfString('msg'),\n\t\t\t\t\t};\n\n\t\t\t\t\tif (this._monitoredType == this.MONITORED_TYPE_ZONE && logRoom == '')\n\t\t\t\t\t{\n\t\t\t\t\t\t// Add to zone extension log\n\t\t\t\t\t\tthis._addLogEntryToGrid(this._zoneExtLogGrid, ele);\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tif (this._monitoredType == this.MONITORED_TYPE_ROOM && this._monitoredName == logRoom)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t// Add to room extension log\n\t\t\t\t\t\t\tthis._addLogEntryToGrid(this._roomExtLogGrid, ele);\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}\n\t}\n\n\t//---------------------------------\n\t// UI EVENT LISTENERS\n\t//---------------------------------\n\n\t_onUpdateIntervalChange()\n\t{\n\t\t// Request data to server\n\t\tthis._requestData(false);\n\t}\n\n\t_onEntitiesPanelChange()\n\t{\n\t\t// Set main controls state\n\t\tthis._setMainControlsEnabled();\n\n\t\t// Request data to server\n\t\tthis._requestData(true);\n\t}\n\n\t/**\n\t * Update the Scope label in the Rooms panel.\n\t */\n\t_onZoneSelected()\n\t{\n\t\t// Clear Room and User lists\n\t\tthis._resetRoomList();\n\t\tthis._resetUserList();\n\n\t\t// Set scope label\n\t\tthis._setScopeLabel();\n\n\t\t// Enable/disable Room and User tools\n\t\tconst disabled = this._selectedZone == undefined;\n\n\t\t// (NOTE: this enables/disables whole fieldset; this doesn't affect Kendo DropDown, so we have to do it manually)\n\t\t$('#roomTools').attr('disabled', disabled);\n\t\t$('#userTools').attr('disabled', disabled);\n\n\t\tthis._groupsDropDown.enable(!disabled);\n\n\t\t// Set main controls state\n\t\tthis._setMainControlsEnabled();\n\t}\n\n\t/**\n\t * Update the Scope label in the Users panel.\n\t */\n\t_onRoomSelected()\n\t{\n\t\t// Clear User list\n\t\tthis._resetUserList();\n\n\t\t// Set scope label\n\t\tthis._setScopeLabel();\n\n\t\t// Enable Remove button\n\t\t$('#znm-removeRoomBt').attr('disabled', this._selectedRoom == undefined);\n\n\t\t// Set main controls state\n\t\tthis._setMainControlsEnabled();\n\t}\n\n\t/**\n\t * Update the scope label in the Users panel.\n\t */\n\t_onGroupChange()\n\t{\n\t\t// Clear Room and User lists\n\t\tthis._resetRoomList();\n\t\tthis._resetUserList();\n\n\t\t// Set scope label\n\t\tthis._setScopeLabel();\n\n\t\t// Request data to server\n\t\tthis._requestData(true);\n\n\t\t// Set main controls state\n\t\tthis._setMainControlsEnabled();\n\t}\n\n\t_onUserSelected()\n\t{\n\t\t// Set main controls state\n\t\tthis._setMainControlsEnabled();\n\t}\n\n\t/**\n\t * Show room creation panel.\n\t */\n\t_onCreateRoomBtClick()\n\t{\n\t\t// Disable Create button\n\t\t$('#znm-createRoomBt').attr('disabled', true);\n\n\t\t// Request default room settings to extension\n\t\tthis.sendExtensionRequest(this.REQ_GET_ROOM_CONFIG);\n\t}\n\n\t/**\n\t * Remove existing room.\n\t */\n\t_onRemoveRoomBtClick()\n\t{\n\t\tthis.shellCtrl.showConfirmWarning('Are you sure you want to remove the selected Room?', $.proxy(this._onRemoveRemoveConfirm, this));\n\t}\n\n\t_onRemoveRemoveConfirm()\n\t{\n\t\tconst selectedZone = this._zoneListBox.dataItem(this._zoneListBox.select());\n\t\tlet selectedRoom = this._roomListBox.dataItem(this._roomListBox.select());\n\n\t\tif (selectedRoom)\n\t\t{\n\t\t\t// Send request to server\n\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\tparams.putUtfString('zone', selectedZone.name);\n\t\t\tparams.putUtfString('room', selectedRoom.name);\n\n\t\t\tthis.sendExtensionRequest(this.REQ_REMOVE_ROOM, params);\n\n\t\t\t// Request data to server to update the rooms list\n\t\t\tthis._requestData(true);\n\t\t}\n\t}\n\n\t/**\n\t * Remove content of room creation panel.\n\t */\n\t_onCreateRoomModalHidden(e)\n\t{\n\t\t// If a nested modal is opened, its closing causes the parent modal to be closed too (Bootstrap doesn't support nested modals)\n\t\t// As a workaround, here we reset the css class 'modal-open' when the nested modal is closed and skip everything else\n\t\t// In addition we had to use a custom listener on the close/cancel buttons of the nested modal (see config-grid.js for example)\n\t\tif (e.target !== document.getElementById('znm-createRoomModal'))\n\t\t{\n\t\t\t$('body').addClass('modal-open')\n\t\t\treturn;\n\t\t}\n\n\t\t// Destroy scrolling tabs in room creation panel\n\t\t$('#znm-roomCreatorTabNav #tabs').scrollingTabs('destroy');\n\n\t\t// Remove all tab navigator content\n\t\tthis._roomCreationIBuilder.destroyInterface();\n\n\t\t// Remove listener for custom actions triggered by configuration interface\n\t\t$('#znm-roomCreatorTabNav').off('value-set');\n\n\t\t// Reset and hide error message\n\t\tthis._resetRoomCreationError();\n\t}\n\n\t_onConfigValueSet(e) // SAME METHOD DUPLICATED IN zone-configurator.js\n\t{\n\t\tconst configParam = e.target.data;\n\n\t\t// Handle extension name/type dropdowns update and update the main class dropdown datasource accordingly\n\t\tif (configParam.name == 'extension.name' || configParam.name == 'extension.type' || configParam.name == 'extension.filterClass')\n\t\t{\n\t\t\t// All involved ConfigFormItems must be available and initialized to proceed\n\t\t\tconst nameFormItem = this._roomCreationIBuilder.getConfigFormItem('extension.name');\n\t\t\tconst typeFormItem = this._roomCreationIBuilder.getConfigFormItem('extension.type');\n\t\t\tconst classFormItem = this._roomCreationIBuilder.getConfigFormItem('extension.file');\n\t\t\tconst filterFormItem = this._roomCreationIBuilder.getConfigFormItem('extension.filterClass');\n\n\t\t\tif (nameFormItem != null && typeFormItem != null && classFormItem != null && filterFormItem != null)\n\t\t\t{\n\t\t\t\tconst source = nameFormItem.data;\n\t\t\t\tlet classesList = [];\n\n\t\t\t\tlet data = source.triggerData;\n\t\t\t\tfor (let i = 0; i < data.size(); i++)\n\t\t\t\t{\n\t\t\t\t\tlet ext = data.getSFSObject(i);\n\n\t\t\t\t\tif (ext.getUtfString('name') == nameFormItem.data.value && ext.getUtfString('type') == typeFormItem.data.value)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet classes = ext.getUtfString('classesString').split(',');\n\n\t\t\t\t\t\tif (filterFormItem.data.value == true)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet filteredClasses = classes.filter(filterClassName);\n\t\t\t\t\t\t\tclasses = filteredClasses;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tclassesList = classesList.concat(classes);\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tlet currentClass = classFormItem.data.value;\n\n\t\t\t\t// If the classes list doesn't contain the current value, create an empty entry and reset the value\n\t\t\t\tif (classesList.indexOf(currentClass) < 0)\n\t\t\t\t{\n\t\t\t\t\tif (classesList.length == 0)\n\t\t\t\t\t{\n\t\t\t\t\t\tclassesList.push('');\n\t\t\t\t\t\tcurrentClass = '';\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t\tcurrentClass = classesList[0];\n\t\t\t\t}\n\n\t\t\t\tlet mainClassDropDown = classFormItem._innerWidget;\n\t\t\t\tmainClassDropDown.setDataSource(classesList);\n\n\t\t\t\tclassFormItem.data.value = currentClass;\n\t\t\t\tclassFormItem._setWidgetValue();\n\t\t\t}\n\t\t}\n\t}\n\n\t_onRoomCreatorCreateBtClick()\n\t{\n\t\t// Reset and hide error message\n\t\tthis._resetRoomCreationError();\n\n\t\t// Check validity\n\t\tif (this._roomCreationIBuilder.checkIsValid())\n\t\t{\n\t\t\tlet changes = this._roomCreationIBuilder.getChangedData();\n\n\t\t\tif (changes.size() > 0)\n\t\t\t{\n\t\t\t\t// Check the room name against the rooms list (duplicate names not allowed!)\n\t\t\t\tif (true)//this._validateNewRoomName(changes))\n\t\t\t\t{\n\t\t\t\t\t// Disable room creation panel\n\t\t\t\t\tthis._enableCreateRoomPanel(false);\n\n\t\t\t\t\t// Send settings to server instance\n\t\t\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\t\t\tparams.putSFSArray('settings', changes);\n\t\t\t\t\tparams.putUtfString('zName', this._zoneListBox.dataItem(this._zoneListBox.select()).name);\n\t\t\t\t\tparams.putBool('notify', $('#znm-notifyClientsCB').prop('checked'));\n\n\t\t\t\t\t// Send settings to extension\n\t\t\t\t\tthis.sendExtensionRequest(this.REQ_CREATE_ROOM, params);\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t// Show alert\n\t\t\t\t\tthis.shellCtrl.showSimpleAlert('Unable to submit configuration because the Room name already exists; duplicate names are not allowed.', true);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// Show alert\n\t\t\tthis.shellCtrl.showSimpleAlert('Unable to submit Room configuration due to an invalid value; please verify the highlighted form fields in all tabs.', true);\n\t\t}\n\t}\n\n\t_onShowRoomFilterBtClick()\n\t{\n\t\tthis._roomFilter.show();\n\t}\n\n\t_onShowUserFilterBtClick()\n\t{\n\t\tthis._userFilter.show();\n\t}\n\n\t_onApplyBtChange()\n\t{\n\t\t// Request data to server\n\t\tthis._requestData(true);\n\t}\n\n\t_onRoomFilterUpdated()\n\t{\n\t\t// Enable/disable \"Apply\" checkbox\n\t\t$('#znm-applyRoomFilterCB').attr('disabled', this._roomFilter.filterExpression == null);\n\n\t\t// Request data to server\n\t\tthis._requestData(true);\n\t}\n\n\t_onUserFilterUpdated()\n\t{\n\t\t// Enable/disable \"Apply\" checkbox\n\t\t$('#znm-applyUserFilterCB').attr('disabled', this._userFilter.filterExpression == null);\n\n\t\t// Request data to server\n\t\tthis._requestData(true);\n\t}\n\n\t_onMonitorSelectionBtClick()\n\t{\n\t\tif (this._selectedZone)\n\t\t{\n\t\t\t// Cancel previous monitoring\n\t\t\tthis._onCloseMonitorBtClick();\n\n\t\t\tlet getZoneUserCountHistory = false;\n\n\t\t\tthis._monitoredZone = this._selectedZone.name;\n\n\t\t\t// Zone must be monitored\n\t\t\tif (this._activePanelType == 'zone')\n\t\t\t{\n\t\t\t\tthis._monitoredType = this.MONITORED_TYPE_ZONE;\n\t\t\t\tthis._monitoredName = this._monitoredZone;\n\n\t\t\t\tgetZoneUserCountHistory = true;\n\t\t\t}\n\n\t\t\t// Room must be monitored\n\t\t\tif (this._activePanelType == 'room' && this._selectedRoom != undefined)\n\t\t\t{\n\t\t\t\tthis._monitoredType = this.MONITORED_TYPE_ROOM;\n\t\t\t\tthis._monitoredName = this._selectedRoom.name;\n\t\t\t}\n\n\t\t\t// User must be monitored\n\t\t\tif (this._activePanelType == 'user' && this._selectedUser != undefined)\n\t\t\t{\n\t\t\t\tthis._monitoredType = this.MONITORED_TYPE_USER;\n\t\t\t\tthis._monitoredName = this._selectedUser.name;\n\t\t\t}\n\n\t\t\t// Show loading bar\n\t\t\tthis._switchView('znm-loading');\n\n\t\t\t// Request data to server\n\t\t\tthis._requestData(true, getZoneUserCountHistory);\n\t\t}\n\t}\n\n\t_onSendAdminMsgInKeyUp(event)\n\t{\n\t\tif (event.key !== 'Enter')\n\t\t\treturn;\n\n\t\tevent.preventDefault();\n\t\tthis._onSendAdminMsgBtClick();\n\t}\n\n\t_onSendAdminMsgBtClick()\n\t{\n\t\tif (this._selectedZone && $('#znm-messageIn').val() != '')\n\t\t{\n\t\t\t// Build request parameters\n\t\t\tlet params = new SFS2X.SFSObject();\n\n\t\t\t// Always include zone\n\t\t\tparams.putUtfString('zone', this._selectedZone.name);\n\n\t\t\t// Send message to room?\n\t\t\tif (this._activePanelType == 'room' && this._selectedRoom != undefined)\n\t\t\t\tparams.putUtfString('room', this._selectedRoom.name);\n\n\t\t\t// Send message to user?\n\t\t\tif (this._activePanelType == 'user' && this._selectedUser != undefined)\n\t\t\t\tparams.putUtfString('user', this._selectedUser.name);\n\n\t\t\t// Add message\n\t\t\tparams.putUtfString('msg', $('#znm-messageIn').val());\n\n\t\t\t// Send request to extension\n\t\t\tthis.sendExtensionRequest(this.REQ_ADMIN_MSG, params);\n\n\t\t\t// Clear text input\n\t\t\t$('#znm-messageIn').val('');\n\t\t}\n\t}\n\n\t_onCloseMonitorBtClick()\n\t{\n\t\t// Save ref. for later usage\n\t\tconst monitoredType = this._monitoredType;\n\n\t\tthis._monitoredZone = null;\n\t\tthis._monitoredType = null;\n\t\tthis._monitoredName = null;\n\n\t\t// Clear geolocation UI\n\t\tthis._clearGeoLocationUI();\n\n\t\t// Clear log datagrids dataproviders\n\t\t// NOTE: a try-catch is used here because, for an unknown reason, Kendo is throwing an error when resetting the data sources when the module is destroyed\n\t\ttry\n\t\t{\n\t\t\tthis._zoneExtLogGrid.setDataSource([]);\n\t\t\tthis._roomExtLogGrid.setDataSource([]);\n\t\t}\n\t\tcatch (e) { /* Ignore */ }\n\n\t\t// Leave edit mode\n\t\tthis._onCancelBtClick();\n\n\t\t// Rearrange monitoring interface\n\t\tthis._setMonitoringInterface();\n\n\t\t// Clear tab navigator\n\t\tthis._clearTabs(monitoredType);\n\t}\n\n\t_onEditBtClick()\n\t{\n\t\tif (!this._isEditing)\n\t\t{\n\t\t\t// Show edit controls\n\t\t\t$('#znm-editControls').show();\n\n\t\t\t// Hide edit button\n\t\t\t$('#znm-editBt').hide();\n\n\t\t\t// Enable editable fields\n\t\t\tthis._interfaceBuilder.disableInterface(false);\n\n\t\t\tthis._isEditMode = true;\n\t\t}\n\t}\n\n\t_onCancelBtClick()\n\t{\n\t\tif (this._isEditing)\n\t\t{\n\t\t\t// Hide edit controls\n\t\t\t$('#znm-editControls').hide();\n\n\t\t\t// Show edit button\n\t\t\t$('#znm-editBt').show();\n\n\t\t\t// Disable editable fields\n\t\t\tthis._interfaceBuilder.disableInterface(true);\n\n\t\t\tthis._isEditMode = false;\n\n\t\t\t// Hide validation messages\n\t\t\tthis._interfaceBuilder.resetValidation();\n\n\t\t\t// Request data to update interface (updating suspended in edit mode)\n\t\t\tthis._requestData(true);\n\t\t}\n\t}\n\n\t_onSubmitBtClick()\n\t{\n\t\tif (this._isEditing)\n\t\t{\n\t\t\t// Check validity\n\t\t\tif (this._interfaceBuilder.checkIsValid())\n\t\t\t{\n\t\t\t\tlet changes = this._interfaceBuilder.getChangedData();\n\n\t\t\t\tif (changes.size() > 0)\n\t\t\t\t{\n\t\t\t\t\t//console.log(changes.getDump())\n\n\t\t\t\t\t// Send settings to server instance\n\t\t\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\t\t\tparams.putSFSArray('settings', changes);\n\t\t\t\t\tparams.putUtfString('zone', this._monitoredZone);\n\n\t\t\t\t\tif (this._monitoredType == this.MONITORED_TYPE_ROOM)\n\t\t\t\t\t{\n\t\t\t\t\t\t// Submit room settings\n\t\t\t\t\t\tparams.putUtfString('room', this._monitoredName);\n\t\t\t\t\t\tthis.sendExtensionRequest(this.REQ_SET_ROOM_SETTINGS, params);\n\t\t\t\t\t}\n\t\t\t\t\telse if (this._monitoredType == this.MONITORED_TYPE_USER)\n\t\t\t\t\t{\n\t\t\t\t\t\t// Submit user settings\n\t\t\t\t\t\tparams.putUtfString('user', this._monitoredName);\n\t\t\t\t\t\tthis.sendExtensionRequest(this.REQ_SET_USER_SETTINGS, params);\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\t// Submit zone settings\n\t\t\t\t\t\tthis.sendExtensionRequest(this.REQ_SET_ZONE_SETTINGS, params);\n\t\t\t\t\t}\n\n\t\t\t\t\t// Leave edit mode\n\t\t\t\t\tthis._onCancelBtClick();\n\t\t\t\t}\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Show alert\n\t\t\t\tthis.shellCtrl.showSimpleAlert('Unable to submit changes due to an invalid value; please verify the highlighted form fields in all tabs.', true);\n\t\t\t}\n\t\t}\n\t}\n\n\t_onTimeRangeChange()\n\t{\n\t\tlet values = this._timeSlider.value();\n\n\t\tif (values[0] == values[1])\n\t\t{\n\t\t\tif (values[1] == -24)\n\t\t\t\tvalues[1] = -23;\n\n\t\t\tvalues[0] = values[1] - 1;\n\n\t\t\t// Reset the time range slider value (we need to use setTimeout\n\t\t\t// bacause doing it in the change event listener doesn't redraw the slider)\n\t\t\tsetTimeout($.proxy( function(values) { this._timeSlider.value(values); }, this), 10, values);\n\t\t}\n\n\t\tthis._userCountChart.range = values;\n\t}\n\n\t_onWindowResize()\n\t{\n\t\t// Redraw charts\n\t\tthis._userCountChart.redraw();\n\t\tthis._packetQueueChart.redraw();\n\t\tthis._droppedMsgChart.redraw();\n\t\tthis._writtenDataChart.redraw();\n\t\tthis._readDataChart.redraw();\n\n\t\t// Redraw time range slider\n\t\tthis._timeSlider.resize();\n\t}\n\n\t_onWordsReloadBtClick()\n\t{\n\t\tif (this._monitoredType == this.MONITORED_TYPE_ZONE)\n\t\t{\n\t\t\t// Send request to extension\n\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\tparams.putUtfString('zone', this._monitoredName);\n\n\t\t\tthis.sendExtensionRequest(this.REQ_RELOAD_WORDS, params);\n\t\t}\n\t}\n\n\t_onZoneExtReloadBtClick()\n\t{\n\t\tif (this._monitoredType == this.MONITORED_TYPE_ZONE)\n\t\t{\n\t\t\t// Send request to extension\n\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\tparams.putUtfString('zone', this._monitoredName);\n\n\t\t\tthis.sendExtensionRequest(this.REQ_RELOAD_ZONE_EXT, params);\n\t\t}\n\t}\n\n\t_onKickBtClick()\n\t{\n\t\tif (this._monitoredType == this.MONITORED_TYPE_USER)\n\t\t{\n\t\t\t// Send request to extension\n\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\tparams.putUtfString('zone', this._monitoredZone);\n\t\t\tparams.putUtfString('user', this._monitoredName);\n\t\t\tparams.putUtfString('msg', $('#znm-kickMsgIn').val());\n\t\t\tparams.putInt('delay', (this._kickDelayIn.value() != null ? Number(this._kickDelayIn.value()) : this.KICK_BAN_DEFAULT_DELAY));\n\n\t\t\tthis.sendExtensionRequest(this.REQ_KICK_USER, params);\n\n\t\t\t// Reset message\n\t\t\t$('#znm-kickMsgIn').val('');\n\t\t}\n\t}\n\n\t_onBanBtClick()\n\t{\n\t\tif (this._monitoredType == this.MONITORED_TYPE_USER)\n\t\t{\n\t\t\t// Send request to extension\n\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\tparams.putUtfString('zone', this._monitoredZone);\n\t\t\tparams.putUtfString('user', this._monitoredName);\n\t\t\tparams.putUtfString('msg', $('#znm-banMsgIn').val());\n\t\t\tparams.putUtfString('mode', this._banModeDd.select() > 0 ? this._banModeDd.value() : 'NAME');\n\t\t\tparams.putInt('delay', (this._banDelayIn.value() != null ? Number(this._banDelayIn.value()) : this.KICK_BAN_DEFAULT_DELAY));\n\n\t\t\tlet duration = this._banDurationIn.value() != null ? Number(this._banDurationIn.value()) : 1;\n\n\t\t\tif (this._banDurUnitDd.value() == 'hours')\n\t\t\t\tduration = duration * 60;\n\t\t\telse if (this._banDurUnitDd.value() == 'days')\n\t\t\t\tduration = duration * 60 * 24;\n\n\t\t\tparams.putInt('duration', duration);\n\n\t\t\tthis.sendExtensionRequest(this.REQ_BAN_USER, params);\n\n\t\t\t// Reset message\n\t\t\t$('#znm-banMsgIn').val('');\n\t\t}\n\t}\n\n\t//------------------------------------\n\t// PRIVATE METHODS\n\t//------------------------------------\n\n\t_setMonitoringInterface()\n\t{\n\t\tconst enabled = (this._monitoredType != null && this._monitoredName != null);\n\n\t\t// Show/hide header and footer\n\t\tif (enabled)\n\t\t{\n\t\t\tthis._switchView(`znm-${this._monitoredType}Monitor`);\n\n\t\t\t// Header title\n\t\t\tlet title = 'Now monitoring ';\n\n\t\t\tif (this._monitoredType == this.MONITORED_TYPE_USER)\n\t\t\t\ttitle += `User: <strong>${this._monitoredName}</strong> (Zone: ${this._monitoredZone})`;\n\t\t\telse if (this._monitoredType == this.MONITORED_TYPE_ROOM)\n\t\t\t\ttitle += `Room: <strong>${this._monitoredName}</strong> (Zone: ${this._monitoredZone})`;\n\t\t\telse\n\t\t\t\ttitle += `Zone: <strong>${this._monitoredZone}</strong>`;\n\n\t\t\t$('#znm-monitoredHeader').html(title);\n\t\t\t$('#znm-monitoredFooter').show();\n\n\t\t\t// Show panel\n\t\t\tthis._switchPanel('znm-mainPanel');\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis._switchView('znm-select');\n\n\t\t\t$('#znm-monitoredHeader').html('');\n\t\t\t$('#znm-monitoredFooter').hide();\n\n\t\t\t// Show panel\n\t\t\tthis._switchPanel('znm-sidebarPanel');\n\t\t}\n\t}\n\n\t_switchView(viewId)\n\t{\n\t\tdocument.getElementById('znm-viewstack').selectedElement = document.getElementById(viewId);\n\t}\n\n\t_switchPanel(panelId)\n\t{\n\t\tdocument.getElementById('znm-view').selectedPanel = document.getElementById(panelId);\n\t}\n\n\t/**\n\t * Build the polling request to be sent to the server, based on the selections and other settings in the accordion.\n\t */\n\t_requestData(newId = false, getZoneUserCountHistory = false)\n\t{\n\t\t// Clear previous request scheduling\n\t\tclearTimeout(this._requestTimer);\n\n\t\t// Check if connection is still available\n\t\tif (this.smartFox.isConnected)\n\t\t{\n\t\t\tif (newId)\n\t\t\t\tthis._currentRequestId++;\n\n\t\t\t// Build request parameters\n\t\t\tlet params = new SFS2X.SFSObject();\n\n\t\t\t// The updated zones/rooms/users lists are requested if the corresponding panel is displayed only\n\t\t\tconst getZones = (this._activePanelType == 'zone');\n\t\t\tconst getRooms = (this._activePanelType == 'room' && this._selectedZone != undefined);\n\t\t\tconst getUsers = (this._activePanelType == 'user' && this._selectedZone != undefined);\n\n\t\t\t// --- ZONES ---\n\t\t\tparams.putBool('zones', getZones);\n\n\t\t\t// ROOMS & USERS\n\t\t\tif (getRooms || getUsers)\n\t\t\t{\n\t\t\t\t// Common filtering params:\n\n\t\t\t\t// a) zone name\n\t\t\t\tparams.putUtfString('zone', this._selectedZone.name);\n\n\t\t\t\t// b) group name\n\t\t\t\tparams.putUtfString('group', this._selectedGroup.name);\n\t\t\t}\n\n\t\t\t// ROOMS\n\t\t\tparams.putBool('rooms', getRooms);\n\t\t\tif (getRooms)\n\t\t\t{\n\t\t\t\t// Filtering params:\n\n\t\t\t\t// c) advanced filtering params\n\t\t\t\tif ($('#znm-applyRoomFilterCB').prop('checked') && this._roomFilter.filterExpression != null)\n\t\t\t\t\tparams.putSFSArray('rFilter', this._roomFilter.filterExpression.toSFSArray());\n\t\t\t}\n\n\t\t\t// USERS\n\t\t\tparams.putBool('users', getUsers);\n\t\t\tif (getUsers)\n\t\t\t{\n\t\t\t\t// Filtering params:\n\n\t\t\t\t// c) room name\n\t\t\t\tif (this._selectedRoom)\n\t\t\t\t\tparams.putUtfString('room', this._selectedRoom.name);\n\n\t\t\t\t// d) advanced filtering params\n\t\t\t\tif ($('#znm-applyUserFilterCB').prop('checked') && this._userFilter.filterExpression != null)\n\t\t\t\t\tparams.putSFSArray('uFilter', this._userFilter.filterExpression.toSFSArray());\n\t\t\t}\n\n\t\t\t// --- CURRENTLY MONITORED DATA ---\n\t\t\tif (this._monitoredType != null && this._monitoredName != null)\n\t\t\t{\n\t\t\t\tlet monitoredObj = this._getBasicMonitoredObject();\n\n\t\t\t\tif (this._monitoredType == this.MONITORED_TYPE_ZONE)\n\t\t\t\t\tmonitoredObj.putBool('uc', getZoneUserCountHistory);\n\n\t\t\t\tparams.putSFSObject('monitored', monitoredObj);\n\t\t\t}\n\n\t\t\t// Set request id; when a response is received, we check its id matches the current request id: if not, the response is discarded\n\t\t\tparams.putInt('id', this._currentRequestId);\n\n\t\t\t// Send request to extension\n\t\t\tthis.sendExtensionRequest(this.REQ_GET_DATA, params);\n\n\t\t\t// Schedule next request\n\t\t\tthis._requestTimer = setTimeout($.proxy(this._requestData, this), Number(this._intervalDropDown.value()) * 1000);\n\t\t}\n\t}\n\n\t_getBasicMonitoredObject()\n\t{\n\t\tlet monitoredObj = new SFS2X.SFSObject();\n\n\t\tmonitoredObj.putUtfString('zone', this._monitoredZone);\n\t\tmonitoredObj.putUtfString('type', this._monitoredType);\n\t\tmonitoredObj.putUtfString('name', this._monitoredName);\n\n\t\treturn monitoredObj;\n\t}\n\n\t_setZonesDataProvider(zonesData)\n\t{\n\t\tthis._setListDataProvider(this._zoneListBox, zonesData, true);\n\t}\n\n\t_setRoomsDataProvider(roomsData)\n\t{\n\t\tthis._setListDataProvider(this._roomListBox, roomsData, true);\n\n\t\t// Set Remove button state\n\t\t$('#znm-removeRoomBt').attr('disabled', this._selectedRoom == undefined);\n\t}\n\n\t_setUsersDataProvider(usersData)\n\t{\n\t\tthis._setListDataProvider(this._userListBox, usersData);\n\t}\n\n\t_setListDataProvider(listWidget, listData, countUsers = false)\n\t{\n\t\t// Save reference to selected item name\n\t\t// NOTE: as we are substituting the whole data source, we need it to re-select the same item after update\n\t\tlet selectedDataItem = listWidget.dataItem(listWidget.select());\n\t\tlet selectedDataItemName = selectedDataItem ? selectedDataItem.name : null;\n\n\t\tlet dataArray = [];\n\n\t\t// Convert data coming from server\n\t\tfor (let i = 0; i < listData.size(); i++)\n\t\t{\n\t\t\tlet itemData = listData.getSFSObject(i);\n\n\t\t\tlet item = {};\n\t\t\titem.name = itemData.getUtfString('name');\n\n\t\t\tif (countUsers)\n\t\t\t\titem.users = itemData.getInt('users');\n\n\t\t\tdataArray.push(item);\n\t\t}\n\n\t\t// Create new data source\n\t\tlet listDS = new kendo.data.DataSource({\n\t\t\tdata: dataArray,\n\t\t\tschema: {\n\t\t\t\tmodel: {\n\t\t\t\t\tid: 'name'\n\t\t\t\t}\n\t\t\t},\n\t\t\tsort: {\n\t\t\t\tfield: 'name',\n\t\t\t\tdir: 'asc'\n\t\t\t}\n\t\t});\n\n\t\t// Assign data source to list\n\t\tlistWidget.setDataSource(listDS);\n\n\t\t// Select again previously selected item\n\t\tif (selectedDataItemName != null)\n\t\t{\n\t\t\tselectedDataItem = listDS.get(selectedDataItemName);\n\n\t\t\tif (selectedDataItem)\n\t\t\t{\n\t\t\t\tlet selectedElement = listWidget.wrapper.find('[data-uid=\"' + selectedDataItem.uid + '\"]');\n\t\t\t\tlistWidget.select(selectedElement);\n\t\t\t}\n\t\t}\n\n\t\t// Reset main control state\n\t\tthis._setMainControlsEnabled();\n\t}\n\n\t_setGroupsDataProvider(groupsData)\n\t{\n\t\t// Create Groups array, adding common entries\n\t\tlet groupsArray = [\n\t\t\tthis._getGroupObj('', this.ANY_LABEL),\n\t\t\tthis._getGroupObj(this.DEFAULT_GROUP_NAME)\n\t\t];\n\n\t\t// Save reference to selected group name\n\t\t// NOTE: as we are substituting the whole data source, we need it to re-select the same item after update\n\t\tlet selectedDataItem = this._selectedGroup;\n\t\tlet selectedDataItemName = selectedDataItem ? selectedDataItem.name : null;\n\n\t\tif (groupsData != null)\n\t\t{\n\t\t\tfor (let i = 0; i < groupsData.length; i++)\n\t\t\t{\n\t\t\t\tif (groupsData[i] != this.DEFAULT_GROUP_NAME) // Default group was added before\n\t\t\t\t\tgroupsArray.push(this._getGroupObj(groupsData[i]));\n\t\t\t}\n\t\t}\n\n\t\t// Create new data source\n\t\tlet groupDS = new kendo.data.DataSource({\n\t\t\tdata: groupsArray,\n\t\t\tschema: {\n\t\t\t\tmodel: {\n\t\t\t\t\tid: 'name'\n\t\t\t\t}\n\t\t\t}\n\t\t});\n\n\t\t// Assign data source to dropdown\n\t\tthis._groupsDropDown.setDataSource(groupDS);\n\n\t\t//--------------------------------\n\n\t\t// Select previously selected item\n\t\tselectedDataItem = groupDS.get(selectedDataItemName);\n\n\t\t// If group is not found, select the default one\n\t\tif (selectedDataItem == undefined)\n\t\t\tselectedDataItem = groupDS.get(this.DEFAULT_GROUP_NAME);\n\n\t\tthis._groupsDropDown.select(function(dataItem) {\n\t\t\treturn dataItem.name === selectedDataItem.name;\n\t\t});\n\t}\n\n\t_getGroupObj(name, label = null)\n\t{\n\t\treturn {\n\t\t\tname: name,\n\t\t\tlabel: (label != null ? label : name)\n\t\t}\n\t}\n\n\t/**\n\t * Update the Scope label in the Rooms panel and in the Users panel.\n\t */\n\t_setScopeLabel()\n\t{\n\t\tlet roomScope = '';\n\t\tlet userScope = '';\n\n\t\tif (this._selectedZone)\n\t\t{\n\t\t\tlet zoneName = this._selectedZone.name;\n\t\t\troomScope = this._getScopeLabelPart('Zone:', zoneName);\n\n\t\t\tlet selectedGroup = this._selectedGroup;\n\t\t\tlet groupName = (selectedGroup && selectedGroup.name != '') ? selectedGroup.name : this.ANY_LABEL;\n\t\t\tlet roomName = (this._selectedRoom) ? this._selectedRoom.name : this.ANY_LABEL;\n\n\t\t\tuserScope = this._getScopeLabelPart('Zone:', zoneName) + '<br>' + this._getScopeLabelPart('Group:', groupName) + '<br>' + this._getScopeLabelPart('Room:', roomName);\n\t\t}\n\n\t\t$('#znm-roomScopeLb').html(roomScope);\n\t\t$('#znm-userScopeLb').html(userScope);\n\t}\n\n\t_getScopeLabelPart(label, text)\n\t{\n\t\treturn `<span class=\"text-muted\">${label}</span> <span>${text}</span>`;\n\t}\n\n\t_resetRoomList()\n\t{\n\t\tthis._roomListBox.setDataSource([]);\n\n\t\t// Also disable Remove button\n\t\t$('#znm-removeRoomBt').attr('disabled', true);\n\t}\n\n\t_resetUserList()\n\t{\n\t\tthis._userListBox.setDataSource([]);\n\t}\n\n\t_setMainControlsEnabled()\n\t{\n\t\tlet enabled = (this._activePanelType == 'zone' && this._selectedZone != undefined)\n\t\t\t\t\t|| (this._activePanelType == 'room' && this._selectedRoom != undefined)\n\t\t\t\t\t|| (this._activePanelType == 'user' && this._selectedUser != undefined);\n\n\t\t$('#znm-mainControls').attr('disabled', !enabled);\n\n\t\t$('#znm-monitorTargetLb').text(enabled ? capitalizeFirst(this._activePanelType) : 'selection');\n\t\t$('#znm-messageIn').attr('placeholder', 'Message to ' + (enabled ? capitalizeFirst(this._activePanelType) : 'selection'));\n\t}\n\n\t_showLimitExceededWarning(icon, title)\n\t{\n\t\ticon.attr('title', title);\n\n\t\tif (title != '')\n\t\t\ticon.removeClass('hidden');\n\t\telse\n\t\t\ticon.addClass('hidden');\n\t}\n\n\t_showRoomCreationPanel(roomSettings)\n\t{\n\t\t// Build user interface based on passed data\n\t\tthis._roomCreationIBuilder.buildInterface(roomSettings, 'znm-roomCreatorTabNav', false, 'rc');\n\n\t\t// Enable scrolling tabs\n\t\t$('#znm-roomCreatorTabNav > #tabs').scrollingTabs({\n\t\t\tbootstrapVersion: 4,\n\t\t\tscrollToTabEdge: true,\n\t\t\tenableSwiping: true,\n\t\t\tdisableScrollArrowsOnFullyScrolled: true,\n\t\t\tcssClassLeftArrow: 'fa fa-chevron-left',\n\t\t\tcssClassRightArrow: 'fa fa-chevron-right'\n\t\t});\n\n\t\t// Reset Notify clients checkbox\n\t\t$('#znm-notifyClientsCB').prop('checked', true);\n\n\t\t// Set listener for custom actions triggered by configuration interface\n\t\t$('#znm-roomCreatorTabNav').on('value-set', $.proxy(this._onConfigValueSet, this));\n\n\t\t// Display panel\n\t\t$('#znm-createRoomModal').modal('show');\n\n\t\t// Reset and hide error message\n\t\tthis._resetRoomCreationError();\n\t}\n\n\t_resetRoomCreationError()\n\t{\n\t\t// Reset and hide error message\n\t\t$('#znm-createRoomError').text('');\n\t\t$('#znm-createRoomError').hide();\n\t}\n\n\t_validateNewRoomName(changes)\n\t{\n\t\tfor (let i = 0; i < changes.size(); i++)\n\t\t{\n\t\t\tconst setting = changes.getSFSObject(i);\n\n\t\t\tif (setting.containsKey('name') && setting.getUtfString('name') == 'name')\n\t\t\t{\n\t\t\t\t// Get name value\n\t\t\t\tconst name = setting.getText('value');\n\n\t\t\t\t// Get data source\n\t\t\t\tconst ds = this._roomListBox.dataSource.data();\n\n\t\t\t\t// Check if name exists in data source\n\t\t\t\tfor (let j = 0; j < ds.length; j++)\n\t\t\t\t{\n\t\t\t\t\tif (ds[j].name == name)\n\t\t\t\t\t\treturn false;\n\t\t\t\t}\n\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t_enableCreateRoomPanel(enable)\n\t{\n\t\tlet modalElement = $('#znm-createRoomModal');\n\n\t\t// Enable modal close buttons\n\t\t$('button[data-dismiss=\"modal\"]', modalElement).attr('disabled', !enable);\n\n\t\t// Enable create button\n\t\t$('#znm-roomCreatorCreateBt', modalElement).attr('disabled', !enable);\n\n\t\t// Enable checkbox\n\t\t$('#znm-notifyClientsCB', modalElement).attr('disabled', !enable);\n\n\t\t// Enable configuration interface\n\t\tthis._roomCreationIBuilder.disableInterface(!enable);\n\t}\n\n\t_clearTabs(monitoredType)\n\t{\n\t\t// Destroy scrolling tabs\n\t\t$(`#znm-${monitoredType}TabNavigator #tabs`).scrollingTabs('destroy');\n\n\t\t// Remove all tab navigator content\n\t\tthis._interfaceBuilder.destroyInterface();\n\n\t\t// Set all tab panels as inactive\n\t\t$(`#znm-${monitoredType}TabNavigator #tabPanels .tab-pane`).removeClass('show active');\n\n\t\t// Remove tab change event listener\n\t\t$('a[data-toggle=\"tab\"]').off('shown.bs.tab');\n\n\t\tthis._skipInitTabs = false;\n\t}\n\n\t_showZoneTrafficData(data)\n\t{\n\t\t// Build user count monitor history\n\t\tif (data.containsKey('history'))\n\t\t{\n\t\t\tlet userCountData = data.getIntArray('history');\n\t\t\tlet samplingRateMins = data.getInt('rate');\n\n\t\t\tthis._userCountChart.addHistoryEntries(userCountData, samplingRateMins * 60);\n\t\t}\n\n\t\t// Add last user count value\n\t\tthis._userCountChart.addEntry(data.getInt('current'));\n\t}\n\n\t_addLogEntryToGrid(grid, item)\n\t{\n\t\tlet ds = grid.dataSource;\n\n\t\t// Check items limit\n\t\tif (ds.total() == this.MAX_EXTENSION_LOG_SIZE)\n\t\t\tds.remove(ds.at(0));\n\n\t\tds.add(item);\n\t}\n\n\t_showUserStatsData(data)\n\t{\n\t\t// Packet queue filling %\n\t\tlet packetQueue = roundToDecimals(data.getFloat('packet'), 2);\n\t\tthis._packetQueueChart.setDataSource([{category: 'Queue', value: packetQueue}]);\n\n\t\t// Dropped messages\n\t\tlet droppedMsg = data.getInt('dropped');\n\t\tthis._droppedMsgChart.setDataSource([{category: 'Dropped', value: droppedMsg}]);\n\n\t\t// Written data amount\n\t\tlet writtenDataObj = scaleBytes(data.getLong('wBytes'), 2);\n\t\tthis._writtenDataChart.setDataSource([{category: 'Written', value: writtenDataObj.value, unit: writtenDataObj.unit}]);\n\n\t\t// Read data amount\n\t\tlet readDataObj = scaleBytes(data.getLong('rBytes'), 2);\n\t\tthis._readDataChart.setDataSource([{category: 'Read', value: readDataObj.value, unit: readDataObj.unit}]);\n\t}\n\n\t_setGeoLocationUI(geoLocData)\n\t{\n\t\tconst location = geoLocData.getUtfString('location');\n\t\tconst error = geoLocData.getUtfString('error');\n\t\tconst latitude = geoLocData.getUtfString('latitude');\n\t\tconst longitude = geoLocData.getUtfString('longitude');\n\n\t\t// Show location\n\t\t$('#znm-geoLocationLb').text(location);\n\n\t\t// Log error\n\t\tif (error != '')\n\t\t\tthis.shellCtrl.logMessage(error, 'warn');\n\n\t\t// Enable/disable Show map button\n\t\tif (latitude != '' && longitude != '')\n\t\t{\n\t\t\t$('#znm-showMapBt').attr('disabled', false);\n\n\t\t\tconst url = `https://www.google.com/maps/search/?api=1&query=${Number(latitude) + ',' + Number(longitude)}`;\n\t\t\t$('#znm-showMapBt').attr('href', url);\n\t\t}\n\t}\n\n\t_clearGeoLocationUI()\n\t{\n\t\t$('#znm-geoLocationLb').text('');\n\t\t$('#znm-showMapBt').attr('disabled', true);\n\t\t$('#znm-showMapBt').attr('href', '#');\n\t}\n\n\t//---------------------------------\n\t// PRIVATE GETTERS\n\t//---------------------------------\n\n\tget _activePanelType()\n\t{\n\t\treturn this._accordion.select().data('itemType');\n\t}\n\n\tget _selectedZone()\n\t{\n\t\treturn this._zoneListBox.dataItem(this._zoneListBox.select());\n\t}\n\n\tget _selectedGroup()\n\t{\n\t\treturn this._groupsDropDown.dataItem(this._groupsDropDown.select());\n\t}\n\n\tget _selectedRoom()\n\t{\n\t\treturn this._roomListBox.dataItem(this._roomListBox.select());\n\t}\n\n\tget _selectedUser()\n\t{\n\t\treturn this._userListBox.dataItem(this._userListBox.select());\n\t}\n\n\tget _isEditing()\n\t{\n\t\treturn (this._monitoredType != null && this._monitoredName != null && this._isEditMode);\n\t}\n}\n","export class RoomPropertiesData\n{\n\tstatic get LABELS() {\n\t\tlet map = new Map();\n\n\t\tmap.set(SFS2X.RoomProperties.NAME, 'name');\n\t\tmap.set(SFS2X.RoomProperties.GROUP_ID, 'groupId');\n\t\tmap.set(SFS2X.RoomProperties.MAX_USERS, 'maxUsers');\n\t\tmap.set(SFS2X.RoomProperties.MAX_SPECTATORS, 'maxSpectators');\n\t\tmap.set(SFS2X.RoomProperties.USER_COUNT, 'userCount');\n\t\tmap.set(SFS2X.RoomProperties.SPECTATOR_COUNT, 'spectatorCount');\n\t\tmap.set(SFS2X.RoomProperties.IS_GAME, 'isGame');\n\t\tmap.set(SFS2X.RoomProperties.IS_PRIVATE, 'isPrivate');\n\t\tmap.set(SFS2X.RoomProperties.HAS_FREE_PLAYER_SLOTS, 'hasFreePlayerSlots');\n\t\tmap.set(SFS2X.RoomProperties.IS_TYPE_SFSGAME, 'isSFSGameType');\n\n\t\treturn map;\n\t}\n\n\tstatic getLabel(propName)\n\t{\n\t\treturn 'Room.' + RoomPropertiesData.LABELS.get(propName);\n\t}\n\n\tstatic get propertiesArray()\n\t{\n\t\tconst arr = [];\n\n\t\tfor (const [key, value] of RoomPropertiesData.LABELS.entries())\n\t\t{\n\t\t\tarr.push({\n\t\t\t\tlabel: RoomPropertiesData.getLabel(key),\n\t\t\t\tvalue: key\n\t\t\t});\n\t\t}\n\n\t\treturn arr;\n\t}\n}\n\nexport class UserPropertiesData\n{\n\tstatic get LABELS() {\n\t\tlet map = new Map();\n\n\t\tmap.set(SFS2X.UserProperties.NAME, 'name');\n\t\tmap.set(SFS2X.UserProperties.IS_PLAYER, 'isPlayer');\n\t\tmap.set(SFS2X.UserProperties.IS_SPECTATOR, 'isSpectator');\n\t\tmap.set(SFS2X.UserProperties.IS_NPC, 'isNPC');\n\t\tmap.set(SFS2X.UserProperties.PRIVILEGE_ID, 'privilegeId');\n\n\t\treturn map;\n\t}\n\n\tstatic getLabel(propName)\n\t{\n\t\treturn 'User.' + UserPropertiesData.LABELS.get(propName);\n\t}\n\n\tstatic get propertiesArray()\n\t{\n\t\tconst arr = [];\n\n\t\tfor (const [key, value] of UserPropertiesData.LABELS.entries())\n\t\t{\n\t\t\tarr.push({\n\t\t\t\tlabel: UserPropertiesData.getLabel(key),\n\t\t\t\tvalue: key\n\t\t\t});\n\t\t}\n\n\t\treturn arr;\n\t}\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AC7GA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACnKA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACtaA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACvHA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC3+DA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;A","sourceRoot":""}
3332
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-13.bundle.js","sources":["webpack://application/./src/components/charts/chart-utils.js","webpack://application/./src/components/charts/user-count-chart.js","webpack://application/./src/components/module-specific/match-expression-builder.js","webpack://application/./src/components/sidebar-layout.js","webpack://application/./src/modules/zone-monitor.js","webpack://application/./src/utils/match-properties.js"],"sourcesContent":["import {bytesToSize, kBytesToSize} from '../../utils/utilities';\n\nexport function getFormattedDateTime(date, format = 'dd MMM HH:mm:ss')\n{\n\treturn kendo.toString(date, format);\n}\n\nexport function getBasicSharedTemplate(valueFormat = '{0}') {\n\treturn kendo.template(`\n\t\t<div class=\"chart-tooltip-category\"><strong>#: category #</strong></div>\n\t\t# for (var i = 0; i < points.length; i++) { #\n\t\t<div>\n\t\t\t<i class=\"fas fa-square\" style=\"color: #: points[i].color # \"></i>\n\t\t\t#: points[i].series.name# : #: kendo.format('${valueFormat}', points[i].value) #\n\t\t</div>\n\t\t# } #\n\t`);\n}\n\nexport function getMemorySharedTemplate() {\n\treturn function(dataItem) {\n\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t<div class=\"chart-tooltip-category\"><strong>#: category #</strong></div>\n\t\t\t# for (var i = 0; i < points.length; i++) { #\n\t\t\t<div>\n\t\t\t\t<i class=\"fas fa-square\" style=\"color: #: points[i].color # \"></i>\n\t\t\t\t#: points[i].series.name# : #: bytesToSize(points[i].value) #\n\t\t\t</div>\n\t\t\t# } #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getMemoryValueAxisLabelTemplate() {\n\treturn function(dataItem) {\n\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t#: bytesToSize(value) #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getNetworkSharedTemplate(categoryFormat = 'dd MMM HH:mm:ss') {\n\treturn function(dataItem) {\n\t\tdataItem.kBytesToSize = kBytesToSize; // Pass kBytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t<div class=\"chart-tooltip-category\"><strong>#: kendo.toString(category, '${categoryFormat}') #</strong></div>\n\t\t\t# for (var i = 0; i < points.length; i++) { #\n\t\t\t<div>\n\t\t\t\t<i class=\"fas fa-square\" style=\"color: #: points[i].color # \"></i>\n\t\t\t\t#: points[i].series.name# : #: kBytesToSize(points[i].value, 1, '', '/s') #\n\t\t\t</div>\n\t\t\t# } #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getNetworkValueAxisLabelTemplate() {\n\treturn function(dataItem) {\n\t\tdataItem.kBytesToSize = kBytesToSize; // Pass kBytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t#: kBytesToSize(value, 1, '', '/s') #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getUserCountSharedTemplate(categoryFormat = 'dd MMM HH:mm:ss') {\n\treturn function(dataItem) {\n\t\treturn kendo.template(`\n\t\t\t<div class=\"chart-tooltip-category\"><strong>#: kendo.toString(category, '${categoryFormat}') #</strong></div>\n\t\t\t# for (var i = 0; i < points.length; i++) { #\n\t\t\t<div>\n\t\t\t\t<i class=\"fas fa-square\" style=\"color: #: points[i].color # \"></i>\n\t\t\t\t#: points[i].series.name# : #: points[i].value #\n\t\t\t</div>\n\t\t\t# } #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getTimeRangeTooltipTemplate() {\n\treturn function(dataItem) {\n\t\tdataItem.getTooltipString = function(v1, v2) {\n\t\t\tif (v1 == v2)\n\t\t\t{\n\t\t\t\tif (v2 == -24)\n\t\t\t\t\tv2 = -23;\n\n\t\t\t\tv1 = v2 - 1;\n\t\t\t}\n\n\t\t\tlet start = Math.abs(v1);\n\t\t\tlet end = Math.abs(v2);\n\n\t\t\treturn `From ${start} hour${start != 1 ? 's' : ''} ago to ${end > 0 ? ('to ' + end + ' hour' + (end != 1 ? 's' : '') + ' ago') : 'now'}`;\n\t\t}; // Pass utility function to template\n\t\treturn kendo.template(`\n\t\t\t#: getTooltipString(selectionStart, selectionEnd) #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getWRBytesLabelTemplate() {\n\treturn function(dataItem) {\n\t\treturn kendo.template(`\n\t\t\t#: value # #: dataItem.unit #\n\t\t`)(dataItem);\n\t};\n}\n","import {getUserCountSharedTemplate} from './chart-utils';\nimport * as moment from 'moment';\n\nexport class UserCountChart extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\n\t\t// Create chart html\n\t\tlet chartHtml = $('<div>');\n\t\t$(this).append(chartHtml);\n\n\t\t// Initialize chart\n\t\tthis._chartWidget = this._initChartWidget(chartHtml);\n\t}\n\n\t_initChartWidget(chartHtml)\n\t{\n\t\treturn chartHtml.kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 270,\n\t\t\t\tbackground: 'transparent'\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'area',\n\t            labels: {\n\t                visible: false,\n\t                format: '{0}',\n\t                background: 'transparent'\n\t            },\n\t        },\n\t        series: [{\n\t\t\t\tname: 'Users',\n\t            field: 'users',\n\t            categoryField: 'time',\n\t\t\t\tcolor: '#fd7d24',\n\t\t\t\tline: {\n\t\t\t\t\twidth: 2\n\t\t\t\t}\n\t        }],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}',\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0,\n\t        },\n\t        categoryAxis: {\n\t\t\t\tmajorGridLines: {\n\t                visible: true,\n\t\t\t\t\tstep: 60\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: true,\n\t\t\t\t\tstep: 60\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: true,\n\t\t\t\t\tstep: 60\n\t\t\t\t},\n\t\t\t\tbaseUnit: 'minutes',\n\t\t\t\ttype: 'date',\n\t        },\n\t        tooltip: {\n\t\t\t\tshared: true,\n\t            visible: true,\n\t\t\t\tsharedTemplate: getUserCountSharedTemplate('dd MMM HH:mm'),\n\t\t\t\tbackground: '#e4e4e4'\n\t        }\n\t    }).data('kendoChart');\n\t}\n\n\tredraw()\n\t{\n\t\tthis._chartWidget.redraw();\n\t}\n\n\tset range(values)\n\t{\n\t\tthis._range = values;\n\n\t\tthis._updateXAxisLimits();\n\t}\n\n\tget range()\n\t{\n\t\treturn this._range;\n\t}\n\n\taddHistoryEntries(userCountData, samplingRateSeconds)\n\t{\n\t\tlet date = new Date();\n\t\tlet start = moment(date);\n\t\tstart.subtract((samplingRateSeconds * userCountData.length), 's')\n\n\t\tlet dataSource = new kendo.data.DataSource();\n\n\t\tfor (let i = 1; i < userCountData.length; i++)\n\t\t{\n\t\t\tlet date = start.clone();\n\t\t\tdate.add(i * samplingRateSeconds, 's');\n\n\t\t\tthis.addEntry(userCountData[i], dataSource, date.toDate(), false);\n\t\t}\n\n\t\t// Assign datasource to chart\n\t\tthis._chartWidget.setDataSource(dataSource);\n\n\t\t// Update axis\n\t\tthis._updateXAxisLimits();\n\t}\n\n\taddEntry(userCount, dataSource = null, date = null, updateAxis = true)\n\t{\n\t\tif (date == null)\n\t\t\tdate = new Date();\n\n\t\tlet newEntry = {\n\t\t\ttime: date,\n\t\t\tusers: userCount\n\t\t};\n\n\t\t// Add entry to data source\n\t\tif (dataSource == null)\n\t\t\tdataSource = this._dataSource;\n\n\t\tdataSource.add(newEntry);\n\n\t\tif (updateAxis)\n\t\t\tthis._updateXAxisLimits();\n\t}\n\n\t_updateXAxisLimits()\n\t{\n\t\tlet ds = this._dataSource;\n\n\t\tif (ds.total() > 0)\n\t\t{\n\t\t\tconst chartMinimum = moment(this._lastDate).add(this._range[0], 'h');\n\t\t\tconst chartMaximum = moment(this._lastDate).add(this._range[1], 'h');\n\n\t\t\tthis._chartWidget.options.categoryAxis.min = chartMinimum.toDate();\n\t\t\tthis._chartWidget.options.categoryAxis.max = chartMaximum.toDate();\n\n\t\t\tthis._chartWidget.redraw();\n\t\t}\n\t}\n\n\tget _dataSource()\n\t{\n\t\treturn this._chartWidget.dataSource;\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('user-count-chart'))\n\twindow.customElements.define('user-count-chart', UserCountChart);\n","export class MatchExpressionBuilder extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\n\t\tthis.EXPRESSION_UPDATED_EVENT = 'expressionUpdated';\n\n\t\tthis._dialog = $(`\n\t\t\t<div class=\"modal\" id=\"matchExprBuilderModal\" tabindex=\"-1\" role=\"dialog\" aria-labelledby=\"matchExprBuilderModalTitle\" aria-hidden=\"true\">\n\t\t\t\t<div class=\"modal-dialog modal-dialog-centered modal-lg\" role=\"document\">\n\t\t\t\t\t<div class=\"modal-content\">\n\t\t\t\t\t\t<div class=\"modal-header\">\n\t\t\t\t\t\t\t<h5 class=\"modal-title text-primary\" id=\"matchExprBuilderModalTitle\">...</h5>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class=\"modal-body p-0\">\n\t\t\t\t\t\t\t<form class=\"bg-color-default p-3\">\n\t\t\t\t\t\t\t\t<div class=\"d-inline-flex align-items-center radio-button-group inline-form-groups\">\n\t\t\t\t\t\t\t\t\t<div class=\"custom-control custom-radio custom-control-inline mr-2\">\n\t\t\t\t\t\t\t\t\t\t<input type=\"radio\" id=\"andOperatorRB\" name=\"operatorRBG\" class=\"custom-control-input\" value=\"AND\" checked disabled>\n\t\t\t\t\t\t\t\t\t\t<label class=\"custom-control-label\" for=\"andOperatorRB\">AND</label>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t<div class=\"custom-control custom-radio custom-control-inline mr-2\">\n\t\t\t\t\t\t\t\t\t\t<input type=\"radio\" id=\"orOperatorRB\" name=\"operatorRBG\" class=\"custom-control-input\" value=\"OR\" disabled>\n\t\t\t\t\t\t\t\t\t\t<label class=\"custom-control-label\" for=\"orOperatorRB\">OR</label>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t<div class=\"d-inline-block\">\n\t\t\t\t\t\t\t\t\t<div class=\"inline-form-groups my-1\">\n\t\t\t\t\t\t\t\t\t\t<input id=\"varNameCB\"/>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t<div class=\"inline-form-groups my-1\">\n\t\t\t\t\t\t\t\t\t\t<input id=\"typeDD\" class=\"matchTypeDropdown\"/>\n\t\t\t\t\t\t\t\t\t\t<input id=\"conditionDD\" class=\"matchConditionDropdown\"/>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t<div class=\"d-inline inline-form-groups\">\n\t\t\t\t\t\t\t\t\t<input id=\"valueIn\" class=\"k-textbox mt-1\" placeholder=\"Value\"/>\n\t\t\t\t\t\t\t\t\t<button id=\"matchExprAddBt\" type=\"button\" class=\"k-button k-secondary mt-1\"><i class=\"fas fa-plus mr-1\"></i>Add</button>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t</form>\n\n\t\t\t\t\t\t\t<div id=\"matchExprGrid\" class=\"m-3\"></div>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class=\"modal-footer d-flex\">\n\t\t\t\t\t\t\t<div class=\"flex-grow-1 text-left\">\n\t\t\t\t\t\t\t\t<button id=\"matchExprApplyBt\" type=\"button\" class=\"k-button k-primary mr-2\"><i class=\"fas fa-check-circle mr-1\"></i>Apply</button>\n\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t<div class=\"flex-grow-1 text-right\">\n\t\t\t\t\t\t\t\t<button id=\"matchExprRemoveBt\" type=\"button\" class=\"k-button k-secondary\" disabled><i class=\"fas fa-minus-circle mr-1\"></i>Remove selected</button>\n\t\t\t\t\t\t\t\t<button id=\"matchExprClearBt\" type=\"button\" class=\"k-button k-secondary\"><i class=\"fas fa-times-circle mr-1\"></i>Clear all</button>\n\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t`);\n\n\t\t$('#matchExprAddBt', this._dialog).on('click', $.proxy(this._onAddBtClick, this));\n\t\t$('#matchExprApplyBt', this._dialog).on('click', $.proxy(this._onApplyBtClick, this));\n\t\t$('#matchExprRemoveBt', this._dialog).on('click', $.proxy(this._onRemoveBtClick, this));\n\t\t$('#matchExprClearBt', this._dialog).on('click', $.proxy(this._onClearBtClick, this));\n\n\t\t// INITIALIZE FORM ITEMS\n\n\t\tthis._varNameCb = $('#varNameCB', this._dialog).kendoComboBox({\n\t\t\tdataTextField: 'label',\n\t\t\tdataValueField: 'value',\n\t\t\tplaceholder: 'Variable name',\n\t\t}).data('kendoComboBox');\n\n\t\t// Set types dropdown data source\n\t\t// NOTE: in order to avoid XxxMatch objects to be wrapped by Kendo data source, we have to pass them using _ property names\n\t\tlet typeDS = [];\n\n\t\t// Boolean ----------------\n\t\ttypeDS.push({\n\t\t\tlabel: 'Boolean',\n\t\t\t_value: SFS2X.BoolMatch.EQUALS,\n\t\t\tconditions: [\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.BoolMatch.EQUALS.symbol,\n\t\t\t\t\t_value: SFS2X.BoolMatch.EQUALS,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.BoolMatch.NOT_EQUALS.symbol,\n\t\t\t\t\t_value: SFS2X.BoolMatch.NOT_EQUALS,\n\t\t\t\t}\n\t\t\t]\n\t\t});\n\n\t\t// Number ----------------\n\t\ttypeDS.push({\n\t\t\tlabel: 'Number',\n\t\t\t_value: SFS2X.NumberMatch.EQUALS,\n\t\t\tconditions: [\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.NumberMatch.EQUALS.symbol,\n\t\t\t\t\t_value: SFS2X.NumberMatch.EQUALS,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.NumberMatch.NOT_EQUALS.symbol,\n\t\t\t\t\t_value: SFS2X.NumberMatch.NOT_EQUALS,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.NumberMatch.GREATER_THAN.symbol,\n\t\t\t\t\t_value: SFS2X.NumberMatch.GREATER_THAN,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.NumberMatch.LESS_THAN.symbol,\n\t\t\t\t\t_value: SFS2X.NumberMatch.LESS_THAN,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.NumberMatch.GREATER_THAN_OR_EQUAL_TO.symbol,\n\t\t\t\t\t_value: SFS2X.NumberMatch.GREATER_THAN_OR_EQUAL_TO,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.NumberMatch.LESS_THAN_OR_EQUAL_TO.symbol,\n\t\t\t\t\t_value: SFS2X.NumberMatch.LESS_THAN_OR_EQUAL_TO,\n\t\t\t\t}\n\t\t\t]\n\t\t});\n\n\t\t// String ----------------\n\t\ttypeDS.push({\n\t\t\tlabel: 'String',\n\t\t\t_value: SFS2X.StringMatch.EQUALS,\n\t\t\tconditions: [\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.StringMatch.EQUALS.symbol,\n\t\t\t\t\t_value: SFS2X.StringMatch.EQUALS,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.StringMatch.NOT_EQUALS.symbol,\n\t\t\t\t\t_value: SFS2X.StringMatch.NOT_EQUALS,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.StringMatch.CONTAINS.symbol,\n\t\t\t\t\t_value: SFS2X.StringMatch.CONTAINS,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.StringMatch.STARTS_WITH.symbol,\n\t\t\t\t\t_value: SFS2X.StringMatch.STARTS_WITH,\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tlabel: SFS2X.StringMatch.ENDS_WITH.symbol,\n\t\t\t\t\t_value: SFS2X.StringMatch.ENDS_WITH,\n\t\t\t\t}\n\t\t\t]\n\t\t});\n\n\t\tthis._typeDd = $('#typeDD', this._dialog).kendoDropDownList({\n\t\t\tdataSource: typeDS,\n\t\t\tdataTextField: 'label',\n\t\t\tdataValueField: '_value',\n\t\t\toptionLabel: {\n\t\t\t\tlabel: 'Type',\n\t\t\t\t_value: ''\n\t\t\t},\n\t\t\toptionLabelTemplate: '<span class=\"text-muted\">#:label#</span>',\n\t\t\tchange: $.proxy(function() {\n\t\t\t\tif (this._typeDd && this._typeDd.select())\n\t\t\t\t{\n\t\t\t\t\tlet selectedType = this._typeDd.dataItem(this._typeDd.select());\n\t\t\t\t\tthis._conditionDd.setDataSource(selectedType.conditions);\n\t\t\t\t\tthis._conditionDd.select(1);\n\t\t\t\t}\n\t\t\t}, this)\n\t\t}).data('kendoDropDownList');\n\n\t\tthis._conditionDd = $('#conditionDD', this._dialog).kendoDropDownList({\n\t\t\tdataTextField: 'label',\n\t\t\tdataValueField: '_value',\n\t\t\toptionLabel: {\n\t\t\t\tlabel: 'Condition',\n\t\t\t\t_value: ''\n\t\t\t},\n\t\t\toptionLabelTemplate: '<span class=\"text-muted\">#:label#</span>',\n\t\t}).data('kendoDropDownList');\n\n\t\tthis._valueIn = $('#valueIn', this._dialog);\n\n\t\t// INITIALIZE GRID\n\n\t\tthis._grid = $('#matchExprGrid', this._dialog).kendoGrid({\n\t\t\tdataSource: [],\n\t\t\tresizable: true,\n\t\t\tselectable: 'row',\n\t\t\tchange: $.proxy(this._onGridSelectionChange, this),\n\t\t\tnoRecords: {\n\t\t\t\ttemplate: 'No entries.'\n\t\t\t},\n\t\t\tcolumns: [\n\t\t\t\t{\n\t\t\t\t\tfield: 'operator',\n\t\t\t\t\ttitle: 'Operator',\n\t\t\t\t\twidth: 90\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'label',\n\t\t\t\t\ttitle: 'Name',\n\t\t\t\t\twidth: 120\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'typeLabel',\n\t\t\t\t\ttitle: 'Type',\n\t\t\t\t\twidth: 85\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'symbol',\n\t\t\t\t\ttitle: 'Condition',\n\t\t\t\t\twidth: 95\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'value',\n\t\t\t\t\ttitle: 'Value',\n\t\t\t\t\twidth: 200\n\t\t\t\t},\n\t\t\t]\n\t\t}).data('kendoGrid');\n\t}\n\n\tget title()\n\t{\n\t\treturn $('#matchExprBuilderModalTitle', this._dialog).text();\n\t}\n\n\tset title(value)\n\t{\n\t\t$('#matchExprBuilderModalTitle', this._dialog).text(value);\n\t}\n\n\tget predefinedVarNames()\n\t{\n\t\treturn this._predefinedVarNames;\n\t}\n\n\tset predefinedVarNames(value)\n\t{\n\t\tthis._predefinedVarNames = value;\n\n\t\tthis._varNameCb.setDataSource(value);\n\t}\n\n\tget filterExpression()\n\t{\n\t\treturn this._filterExpression;\n\t}\n\n\tshow()\n\t{\n\t\tthis._dialog.modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t\tshow: true\n\t\t});\n\t}\n\n\t_onGridSelectionChange()\n\t{\n\t\tlet selectedRows = this._grid.select();\n\t\tlet selectedDataItems = [];\n\n\t\tfor (let i = 0; i < selectedRows.length; i++)\n\t\t{\n\t\t\tlet dataItem = this._grid.dataItem(selectedRows[i]);\n\t\t\tselectedDataItems.push(dataItem);\n\t\t}\n\n\t\t// Enable/disable remove button\n\t\t$('#matchExprRemoveBt', this._dialog).attr('disabled', selectedDataItems.length == 0);\n\t}\n\n\t_onAddBtClick()\n\t{\n\t\tif (this._varNameCb.value() != '' && this._typeDd.select())\n\t\t{\n\t\t\t// Variable name and label\n\t\t\tlet varName, varLabel;\n\n\t\t\tif (this._varNameCb.select() > -1)\n\t\t\t{\n\t\t\t\tlet dataItem = this._varNameCb.dataItem(this._varNameCb.select());\n\t\t\t\tvarName = dataItem.value;\n\t\t\t\tvarLabel = dataItem.label;\n\t\t\t}\n\t\t\telse\n\t\t\t\tvarName = varLabel = this._varNameCb.value();\n\n\t\t\t// Variable type's label\n\t\t\tlet typeLabel = this._typeDd.dataItem(this._typeDd.select()).label;\n\n\t\t\t// Match condition\n\t\t\tlet condition = this._conditionDd.dataItem(this._conditionDd.select())._value;\n\n\t\t\t// Expression operator\n\t\t\tlet operator = this._grid.dataSource.total() > 0 ? $('input[name=operatorRBG]:checked', this._dialog).val() : null;\n\n\t\t\t// Variable value\n\t\t\tlet val = this._valueIn.val();\n\t\t\tlet varValue;\n\n\t\t\tif (condition instanceof SFS2X.BoolMatch)\n\t\t\t\tvarValue = (val == 'true' ? true : false);\n\t\t\telse if (condition instanceof SFS2X.NumberMatch)\n\t\t\t\tvarValue = (isNaN(Number(val)) ? 0 : Number(val));\n\t\t\telse\n\t\t\t\tvarValue = val;\n\n\t\t\t// Add item to grid\n\t\t\tlet exprPart = this._getExpressionPart(varLabel, varName, varValue, typeLabel, condition, operator);\n\t\t\tthis._grid.dataSource.add(exprPart);\n\n\t\t\t// Reset inputs\n\t\t\tthis._varNameCb.value('');\n\t\t\tthis._valueIn.val('');\n\n\t\t\t// Enable operator radio buttons\n\t\t\tthis._enableOperatorRb(true);\n\t\t}\n\t}\n\n\t_onApplyBtClick()\n\t{\n\t\t// Hide dialog\n\t\tthis._dialog.modal('hide');\n\n\t\t// Clear grid selection\n\t\tthis._grid.clearSelection();\n\n\t\t// Build final match expression\n\t\tthis._filterExpression = null;\n\n\t\tif (this._grid.dataSource.total() > 0)\n\t\t{\n\t\t\tlet expPart = this._grid.dataSource.at(0);\n\t\t\tthis._filterExpression = new SFS2X.MatchExpression(expPart.name, expPart._condition, expPart.value);\n\n\t\t\tfor (let i = 1; i < this._grid.dataSource.total(); i++)\n\t\t\t{\n\t\t\t\texpPart = this._grid.dataSource.at(i);\n\n\t\t\t\tif (expPart.operator == SFS2X.LogicOperator.AND.id)\n\t\t\t\t\tthis._filterExpression = this._filterExpression.and(expPart.name, expPart._condition, expPart.value);\n\t\t\t\telse if (expPart.operator == SFS2X.LogicOperator.OR.id)\n\t\t\t\t\tthis._filterExpression = this._filterExpression.or(expPart.name, expPart._condition, expPart.value);\n\t\t\t}\n\t\t}\n\n\t\t// Fire event\n\t\tlet evt = new CustomEvent(this.EXPRESSION_UPDATED_EVENT, {\n\t\t    \tdetail: null,\n\t\t\t\tbubbles: false,\n\t\t\t\tcancelable: false\n\t\t\t});\n\n\t\tthis.dispatchEvent(evt);\n\t}\n\n\t_onRemoveBtClick()\n\t{\n\t\tlet dataItem = this._grid.dataItem(this._grid.select());\n\t\tthis._grid.dataSource.remove(dataItem);\n\n\t\t// Clear grid selection\n\t\tthis._grid.clearSelection();\n\n\t\tif (this._grid.dataSource.total() == 0)\n\t\t{\n\t\t\t// Disable operator radio buttons\n\t\t\tthis._enableOperatorRb(false);\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// Reset operator on first entry\n\t\t\tthis._grid.dataSource.at(0).operator = null;\n\t\t\tthis._grid.refresh();\n\t\t}\n\t}\n\n\t_onClearBtClick()\n\t{\n\t\t// Reset form\n\t\t$('#andOperatorRB', this._dialog).attr('checked', true);\n\t\t$('#orOperatorRB', this._dialog).attr('checked', false);\n\t\tthis._enableOperatorRb(false);\n\n\t\tthis._varNameCb.value('');\n\t\tthis._typeDd.select(0);\n\t\tthis._conditionDd.select(0);\n\n\t\tthis._valueIn.val('');\n\n\t\t// Clear grid selection\n\t\tthis._grid.clearSelection();\n\n\t\t// Clear table\n\t\tthis._grid.setDataSource([]);\n\t}\n\n\t_getExpressionPart(varLabel, varName, varValue, typeLabel, condition, operator)\n\t{\n\t\treturn {\n\t\t\t'label': varLabel,\n\t\t\t'name': varName,\n\t\t\t'value': varValue,\n\t\t\t'typeLabel': typeLabel,\n\t\t\t'symbol': condition.symbol,\n\t\t\t'operator': operator,\n\t\t\t'_condition': condition,\n\t\t};\n\t}\n\n\t_enableOperatorRb(enable)\n\t{\n\t\t$('#andOperatorRB', this._dialog).attr('disabled', !enable);\n\t\t$('#orOperatorRB', this._dialog).attr('disabled', !enable);\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('match-expression-builder'))\n\twindow.customElements.define('match-expression-builder', MatchExpressionBuilder);\n","export class SidebarLayout extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\n\t\t// Attach a shadow root\n\t\tconst shadowRoot = this.attachShadow({mode: 'open'});\n\t\tshadowRoot.innerHTML = `\n\t\t\t<style>\n\t\t\t\t:host {\n\t\t\t\t\tdisplay: flex;\n\t\t\t\t\tflex-direction: row;\n\t\t\t\t}\n\n\t\t\t\t@media (max-width: 575.98px) {\n\t\t\t\t\t:host(.split-xs) ::slotted(:not([aria-selected=\"true\"])) {\n\t\t\t\t\t\tdisplay: none !important;\n\t\t\t\t    }\n\n\t\t\t\t\t:host(.split-xs) ::slotted([aria-selected=\"true\"]) {\n\t\t\t\t\t\tflex-grow: 1;\n\t\t\t\t    }\n\t\t\t\t}\n\n\t\t\t\t@media (max-width: 767.98px) {\n\t\t\t\t\t:host(.split-sm) ::slotted(:not([aria-selected=\"true\"])) {\n\t\t\t\t\t\tdisplay: none !important;\n\t\t\t\t    }\n\n\t\t\t\t\t:host(.split-sm) ::slotted([aria-selected=\"true\"]) {\n\t\t\t\t\t\tflex-grow: 1;\n\t\t\t\t    }\n\t\t\t\t}\n\n\t\t\t\t@media (max-width: 991.98px) {\n\t\t\t\t\t:host(.split-md) ::slotted(:not([aria-selected=\"true\"])) {\n\t\t\t\t\t\tdisplay: none !important;\n\t\t\t\t    }\n\n\t\t\t\t\t:host(.split-md) ::slotted([aria-selected=\"true\"]) {\n\t\t\t\t\t\tflex-grow: 1;\n\t\t\t\t    }\n\t\t\t\t}\n\n\t\t\t\t@media (max-width: 1199.98px) {\n\t\t\t\t\t:host(.split-lg) ::slotted(:not([aria-selected=\"true\"])) {\n\t\t\t\t\t\tdisplay: none !important;\n\t\t\t\t    }\n\n\t\t\t\t\t:host(.split-lg) ::slotted([aria-selected=\"true\"]) {\n\t\t\t\t\t\tflex-grow: 1;\n\t\t\t\t    }\n\t\t\t\t}\n\n\t\t\t\t.side-col::slotted(*) {\n\t\t\t\t}\n\n\t\t\t\t.main-col::slotted(*) {\n\t\t\t\t\tflex-grow: 1;\n\t\t\t\t}\n\t\t\t</style>\n\n\t\t\t<slot class=\"side-col\" name=\"side-column\"></slot>\n\t\t\t<slot class=\"main-col\" name=\"main-column\"></slot>\n\t\t`;\n\n\t\t// Set initial selection\n\t\tthis.selectedIndex = 0;\n\t}\n\n\tget selectedPanel()\n\t{\n\t\treturn this._selectedPanel;\n\t}\n\n\tset selectedPanel(element) // 'side' or 'main'\n\t{\n\t\tif (element != null && element.parentNode == this)\n\t\t{\n\t\t\tthis._selectedPanel = element;\n\n\t\t\tfor (let element of this.children)\n\t\t\t{\n\t\t\t\tif (element == this._selectedPanel)\n\t\t\t\t\telement.setAttribute('aria-selected', 'true');\n\t\t\t\telse\n\t\t\t\t\telement.removeAttribute('aria-selected');\n\t\t\t}\n\t\t}\n\t\telse\n\t\t{\n\t\t\tconsole.error('Element is not a child of SidebarLayout');\n\t\t}\n\t}\n\n\tget selectedIndex()\n\t{\n\t\treturn Array.from(this.children).indexOf(this._selectedPanel);\n\t}\n\n\tset selectedIndex(index)\n\t{\n\t\tif (this.children.length > 0)\n\t\t{\n\t\t\tif (this.children[index] == null)\n\t\t\t{\n\t\t\t\tconsole.error('Invalid SidebarLayout index');\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tlet element = this.children[index];\n\t\t\tthis.selectedPanel = element;\n\t\t}\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('sidebar-layout'))\n\twindow.customElements.define('sidebar-layout', SidebarLayout);\n","import {BaseModule} from './base-module';\nimport {SidebarLayout} from '../components/sidebar-layout';\nimport {MatchExpressionBuilder} from '../components/module-specific/match-expression-builder';\nimport {ConfigInterfaceBuilder} from '../utils/uibuilder/config-interface-builder';\nimport {capitalizeFirst, filterClassName, roundToDecimals, scaleBytes} from '../utils/utilities';\nimport {RoomPropertiesData,UserPropertiesData} from '../utils/match-properties';\nimport {UserCountChart} from '../components/charts/user-count-chart';\nimport {getTimeRangeTooltipTemplate, getWRBytesLabelTemplate} from '../components/charts/chart-utils';\n\nexport default class ZoneMonitor extends BaseModule\n{\n\tconstructor()\n\t{\n\t    super('zoneMonitor');\n\n\t\tthis.MONITORED_TYPE_ZONE = 'zone';\n\t\tthis.MONITORED_TYPE_ROOM = 'room';\n\t\tthis.MONITORED_TYPE_USER = 'user';\n\n\t\tthis.ANY_LABEL = '[any]';\n\t\tthis.DEFAULT_GROUP_NAME = 'default';\n\t\tthis.MAX_EXTENSION_LOG_SIZE = 300;\n\t\tthis.KICK_BAN_DEFAULT_DELAY = 5;\n\n\t\t// Outgoing requests\n\t\tthis.REQ_GET_DATA = 'getData';\n\t\tthis.REQ_SET_ZONE_SETTINGS = 'setZoneSettings';\n\t\tthis.REQ_SET_ROOM_SETTINGS = 'setRoomSettings';\n\t\tthis.REQ_SET_USER_SETTINGS = 'setUserSettings';\n\t\tthis.REQ_RELOAD_ZONE_EXT = 'reloadZoneExt';\n\t\tthis.REQ_RELOAD_WORDS = 'reloadWords';\n\t\tthis.REQ_DISCONNECT_USER = 'disconnUser';\n\t\tthis.REQ_KICK_USER = 'kickUser';\n\t\tthis.REQ_BAN_USER = 'banUser';\n\t\tthis.REQ_GET_ROOM_CONFIG = 'getRoomConfig';\n\t\tthis.REQ_CREATE_ROOM = 'createRoom';\n\t\tthis.REQ_REMOVE_ROOM = 'removeRoom';\n\t\tthis.REQ_ADMIN_MSG = 'adminMsg';\n\n\t\t// Incoming responses\n\t\tthis.RESP_DATA = 'data';\n\t\tthis.RESP_UNEXPECTED_ERROR = 'unexpError';\n\t\tthis.RESP_UPDATE_ERROR = 'updateError';\n\t\tthis.RESP_ROOM_CONFIG = 'roomConfig';\n\t\tthis.RESP_ROOM_CONFIG_ERROR = 'roomConfigErr';\n\t\tthis.RESP_ROOM_CREATED = 'roomCreated';\n\t\tthis.RESP_ROOM_CREATION_ERROR = 'roomError';\n\t\tthis.RESP_LOG_MESSAGES = 'log'; // This response doesn't have a corresponding request because it is managed by a server event listener\n\n\t\tthis._currentRequestId = -1;\n\t}\n\n\t//------------------------------------\n\t// COMMON MODULE INTERFACE METHODS\n\t// This members are used by the main controller\n\t// to communicate with the module's controller.\n\t//------------------------------------\n\n\tinitialize(idData, shellController)\n\t{\n\t\t// Call super method\n\t\tsuper.initialize(idData, shellController);\n\n\t\t// Set monitoring interface\n\t\tthis._setMonitoringInterface();\n\n\t\t// Initialize progress bar\n\t\t$('#znm-progressBar').kendoProgressBar({\n\t\t\tmin: 0,\n            max: 100,\n\t\t\tvalue: false,\n            type: 'value',\n            animation: {\n                duration: 400\n            }\n        });\n\n\t\t// LISTS\n\n\t\t// Initialize interval dropdown\n\t\tthis._intervalDropDown = $('#znm-intervalDD').kendoDropDownList({\n\t\t\tvalueTemplate: '<span class=\"text-muted pr-1\">Interval:</span><span>#:data.text#</span>',\n\t\t\tchange: $.proxy(this._onUpdateIntervalChange, this)\n\t\t}).data('kendoDropDownList');\n\n\t\t// Initialize pane bar\n\t\tthis._accordion = $('#znm-panelbar').kendoPanelBar({\n            expandMode: 'single',\n\t\t\tchange: $.proxy(this._onEntitiesPanelChange, this)\n        }).data('kendoPanelBar');\n\n\t\t// Initialize lists\n\t\tthis._zoneListBox = $(\"#znm-zoneList\").kendoListBox({\n\t\t\ttemplate: '<div>#:data.name# (#:data.users#)</div>',\n\t\t\tchange: $.proxy(this._onZoneSelected, this)\n\t\t}).data('kendoListBox');\n\n\t\tthis._zoneListBox.wrapper.find('.k-list').on('dblclick', '.k-item', $.proxy(this._onMonitorSelectionBtClick, this));\n\n\t\tthis._roomListBox = $(\"#znm-roomList\").kendoListBox({\n\t\t\ttemplate: '<div>#:data.name# (#:data.users#)</div>',\n\t\t\tchange: $.proxy(this._onRoomSelected, this)\n\t\t}).data('kendoListBox');\n\n\t\tthis._roomListBox.wrapper.find('.k-list').on('dblclick', '.k-item', $.proxy(this._onMonitorSelectionBtClick, this));\n\n\t\tthis._userListBox = $(\"#znm-userList\").kendoListBox({\n\t\t\ttemplate: '<div>#:data.name#</div>',\n\t\t\tchange: $.proxy(this._onUserSelected, this)\n\t\t}).data('kendoListBox');\n\n\t\tthis._userListBox.wrapper.find('.k-list').on('dblclick', '.k-item', $.proxy(this._onMonitorSelectionBtClick, this));\n\n\t\t// Initialize Room Groups dropdown\n\t\tthis._groupsDropDown = $('#znm-roomGroupsDD').kendoDropDownList({\n\t\t\tdataTextField: 'label',\n  \t\t\tdataValueField: 'name',\n\t\t\tchange: $.proxy(this._onGroupChange, this)\n\t\t}).data('kendoDropDownList');\n\n\t\t// Build initial groups list\n\t\tthis._setGroupsDataProvider(null);\n\n\t\t// // Create interface builder instances\n\t\tthis._interfaceBuilder = new ConfigInterfaceBuilder();\n\t\tthis._roomCreationIBuilder = new ConfigInterfaceBuilder();\n\n\t\t// Add listener to show limit warning tooltips\n\t\t$('#znm-panelbar').kendoTooltip({\n\t\t\tfilter: 'i[title].limit-warning',\n\t\t\tposition: 'right',\n\t\t\twidth: '250px',\n\t\t\tcontent: function(e) {\n\t\t\t\treturn `<div class=\"help-tooltip\">${e.target.data('title')}</div>`;\n\t\t\t}\n\t\t});\n\n\t\t// ROOM CREATION PANEL\n\n\t\t// Add click listener to Create & Remove Room buttons\n\t\t$('#znm-createRoomBt').on('click', $.proxy(this._onCreateRoomBtClick, this));\n\t\t$('#znm-removeRoomBt').on('click', $.proxy(this._onRemoveRoomBtClick, this));\n\n\t\t// Configure room creation panel\n\t\t$('#znm-createRoomModal').modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t\tshow: false,\n\t\t});\n\n\t\t// Add listener for room creation panel's create button click\n\t\t$('#znm-roomCreatorCreateBt').on('click', $.proxy(this._onRoomCreatorCreateBtClick, this));\n\n\t\t// Add listener for room creation panel hide\n\t\t$('#znm-createRoomModal').on('hidden.bs.modal', $.proxy(this._onCreateRoomModalHidden, this));\n\n\t\t// ROOM & USER FILTERING PANELS\n\n\t\t// Setup match expression builder panel for rooms\n\t\tthis._roomFilter = document.getElementById('znm-roomFilter');\n\t\tthis._roomFilter.title = 'Room Filter Expression Builder';\n\t\tthis._roomFilter.predefinedVarNames = RoomPropertiesData.propertiesArray;\n\t\t$(this._roomFilter).on(this._roomFilter.EXPRESSION_UPDATED_EVENT, $.proxy(this._onRoomFilterUpdated, this));\n\n\t\t$('#znm-filterRoomBt').on('click', $.proxy(this._onShowRoomFilterBtClick, this));\n\t\t$('#znm-applyRoomFilterCB').on('change', $.proxy(this._onApplyBtChange, this));\n\n\t\t// Setup match expression builder panel for users\n\t\tthis._userFilter = document.getElementById('znm-userFilter');\n\t\tthis._userFilter.title = 'User Filter Expression Builder';\n\t\tthis._userFilter.predefinedVarNames = UserPropertiesData.propertiesArray;\n\t\t$(this._userFilter).on(this._userFilter.EXPRESSION_UPDATED_EVENT, $.proxy(this._onUserFilterUpdated, this));\n\n\t\t$('#znm-filterUserBt').on('click', $.proxy(this._onShowUserFilterBtClick, this));\n\t\t$('#znm-applyUserFilterCB').on('change', $.proxy(this._onApplyBtChange, this));\n\n\t\t// MAIN CONTROLS\n\n\t\t$('#znm-monitorBt').on('click', $.proxy(this._onMonitorSelectionBtClick, this));\n\t\t$('#znm-sendMessageBt').on('click', $.proxy(this._onSendAdminMsgBtClick, this));\n\t\t$('#znm-messageIn').on('keyup', $.proxy(this._onSendAdminMsgInKeyUp, this));\n\n\t\t// MONITORING CONTROLS\n\n\t\t$('#znm-closeMonitorBt').on('click', $.proxy(this._onCloseMonitorBtClick, this));\n\t\t$('#znm-editBt').on('click', $.proxy(this._onEditBtClick, this));\n\t\t$('#znm-cancelBt').on('click', $.proxy(this._onCancelBtClick, this));\n\t\t$('#znm-submitBt').on('click', $.proxy(this._onSubmitBtClick, this));\n\n\t\t// Hide edit controls\n\t\t$('#znm-editControls').hide();\n\n\t\t// ZONE MONITOR ---------------------\n\n\t\t// Set reference to user count chart\n\t\tthis._userCountChart = document.getElementById('znm-userCountChart');\n\n\t\t// Initialize time range slider\n\t\tthis._timeSlider = $('#znm-timeSlider').kendoRangeSlider({\n\t\t\tmin: -24,\n            max: 0,\n            smallStep: 1,\n            largeStep: 0,\n\t\t\tchange: $.proxy(this._onTimeRangeChange, this),\n\t\t\ttooltip: {\n\t\t\t\ttemplate: getTimeRangeTooltipTemplate()\n\t\t\t},\n\t\t\tleftDragHandleTitle: 'Drag',\n\t\t\trightDragHandleTitle: 'Drag',\n\t\t}).data('kendoRangeSlider');\n\n\t\t// Reset time range on user count chart\n\t\tthis._onTimeRangeChange();\n\n\t\t// Add button click listeners\n\t\t$('#znm-reloadWordsBt').on('click', $.proxy(this._onWordsReloadBtClick, this));\n\t\t$('#znm-reloadZoneExtBt').on('click', $.proxy(this._onZoneExtReloadBtClick, this));\n\n\t\t// Initialize extension log grid\n\t\tthis._zoneExtLogGrid = $('#znm-zoneExtLogGrid').kendoGrid({\n\t\t\tdataSource: [],\n\t\t\tresizable: true,\n\t\t\tselectable: false,\n\t\t\tnoRecords: {\n\t\t\t\ttemplate: 'No entries.'\n\t\t\t},\n\t\t\tcolumns: [\n\t\t\t\t{\n\t\t\t\t\tfield: 'timestamp',\n\t\t\t\t\ttitle: 'Timestamp',\n\t\t\t\t\twidth: 180\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'level',\n\t\t\t\t\ttitle: 'Level',\n\t\t\t\t\twidth: 100\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'message',\n\t\t\t\t\ttitle: 'Message',\n\t\t\t\t\twidth: 400\n\t\t\t\t},\n\t\t\t],\n\t\t\tdataBound: function() {\n\t\t\t    this.content.scrollTop(this.tbody.height()); // Scroll to bottom\n\t\t\t}\n\t\t}).data('kendoGrid');\n\n\t\t// ROOM MONITOR ---------------------\n\n\t\t// Initialize extension log grid\n\t\tthis._roomExtLogGrid = $('#znm-roomExtLogGrid').kendoGrid({\n\t\t\tdataSource: [],\n\t\t\tresizable: true,\n\t\t\tselectable: false,\n\t\t\tnoRecords: {\n\t\t\t\ttemplate: 'No entries.'\n\t\t\t},\n\t\t\tcolumns: [\n\t\t\t\t{\n\t\t\t\t\tfield: 'timestamp',\n\t\t\t\t\ttitle: 'Timestamp',\n\t\t\t\t\twidth: 180\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'level',\n\t\t\t\t\ttitle: 'Level',\n\t\t\t\t\twidth: 100\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'message',\n\t\t\t\t\ttitle: 'Message',\n\t\t\t\t\twidth: 400\n\t\t\t\t},\n\t\t\t],\n\t\t\tdataBound: function() {\n\t\t\t    this.content.scrollTop(this.tbody.height()); // Scroll to bottom\n\t\t\t}\n\t\t}).data('kendoGrid');\n\n\t\t// USER MONITOR ---------------------\n\n\t\tthis._kickDelayIn = $('#znm-kickDelayNS').kendoNumericTextBox({\n\t\t    min: 0,\n\t\t    step: 1,\n\t\t    format: '#',\n\t\t\tplaceholder: 'Delay (s)'\n\t\t}).data('kendoNumericTextBox');\n\n\t\tthis._banModeDd = $('#znm-banModeDD').kendoDropDownList({\n\t\t\tdataSource: ['IP', 'NAME'],\n\t\t\tautoWidth: true,\n\t\t\toptionLabel: {\n\t\t\t\ttext: 'Mode'\n\t\t\t},\n\t\t\toptionLabelTemplate: '<span class=\"text-muted\">#:text#</span>',\n\t\t}).data('kendoDropDownList');\n\n\t\tthis._banDurUnitDd = $('#znm-banDurationUnitDD').kendoDropDownList({\n\t\t\tautoWidth: true,\n\t\t}).data('kendoDropDownList');\n\n\t\tthis._banDurationIn = $('#znm-banDurationNS').kendoNumericTextBox({\n\t\t    min: 1,\n\t\t    step: 1,\n\t\t    format: '#',\n\t\t\tplaceholder: 'Duration'\n\t\t}).data('kendoNumericTextBox');\n\n\t\tthis._banDelayIn = $('#znm-banDelayNS').kendoNumericTextBox({\n\t\t    min: 0,\n\t\t    step: 1,\n\t\t    format: '#',\n\t\t\tplaceholder: 'Delay (s)'\n\t\t}).data('kendoNumericTextBox');\n\n\t\t// Add button click listeners\n\t\t$('#znm-disconnectBt').on('click', $.proxy(this._onDisconnectBtClick, this));\n\t\t$('#znm-kickBt').on('click', $.proxy(this._onKickBtClick, this));\n\t\t$('#znm-banBt').on('click', $.proxy(this._onBanBtClick, this));\n\n\t\t// Charts\n\n\t\tthis._packetQueueChart = $('#znm-packetQueueChart').kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 100\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'bar',\n\t            labels: {\n\t                visible: true,\n\t                format: '{0}%',\n\t                background: 'transparent',\n\t\t\t\t\tposition: 'insideBase'\n\t            },\n\t\t\t\tgap: .5,\n\t\t\t\tspacing: 0,\n\t\t\t\toverlay: {\n\t\t\t\t\tgradient: 'none'\n\t\t\t\t},\n\t\t\t\tborder: {\n\t\t\t\t\twidth: 0\n\t\t\t\t}\n\t        },\n\t        series: [\n\t\t\t\t{\n\t\t\t\t\tname: 'Packet queue',\n\t\t            field: 'value',\n\t\t            categoryField: 'category',\n\t\t\t\t\tcolor: '#FB7D33'\n\t\t        }\n\t\t\t],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}%'\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0,\n\t    \t\tmax: 100\n\t        },\n\t        categoryAxis: {\n\t            majorGridLines: {\n\t                visible: false\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: false\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: false\n\t\t\t\t}\n\t        },\n\t\t\ttooltip: {\n\t\t\t\tvisible: false,\n\t        }\n\t    }).data('kendoChart');\n\n\t\tthis._droppedMsgChart = $('#znm-droppedMsgChart').kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 100\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'bar',\n\t            labels: {\n\t                visible: true,\n\t                format: '{0}',\n\t                background: 'transparent',\n\t\t\t\t\tposition: 'insideBase',\n\t            },\n\t\t\t\tgap: .5,\n\t\t\t\tspacing: 0,\n\t\t\t\toverlay: {\n\t\t\t\t\tgradient: 'none'\n\t\t\t\t},\n\t\t\t\tborder: {\n\t\t\t\t\twidth: 0\n\t\t\t\t}\n\t        },\n\t        series: [\n\t\t\t\t{\n\t\t\t\t\tname: 'Dropped messages',\n\t\t            field: 'value',\n\t\t            categoryField: 'category',\n\t\t\t\t\tcolor: '#FB7D33'\n\t\t        }\n\t\t\t],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}'\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0,\n\t        },\n\t        categoryAxis: {\n\t            majorGridLines: {\n\t                visible: false\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: false\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: false\n\t\t\t\t}\n\t        },\n\t\t\ttooltip: {\n\t\t\t\tvisible: false,\n\t        }\n\t    }).data('kendoChart');\n\n\t\tthis._writtenDataChart = $('#znm-writtenDataChart').kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 100\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'bar',\n\t            labels: {\n\t                visible: true,\n\t                format: '{0}',\n\t                background: 'transparent',\n\t\t\t\t\tposition: 'insideBase',\n\t\t\t\t\ttemplate: getWRBytesLabelTemplate(),\n\t            },\n\t\t\t\tgap: .5,\n\t\t\t\tspacing: 0,\n\t\t\t\toverlay: {\n\t\t\t\t\tgradient: 'none'\n\t\t\t\t},\n\t\t\t\tborder: {\n\t\t\t\t\twidth: 0\n\t\t\t\t}\n\t        },\n\t        series: [\n\t\t\t\t{\n\t\t\t\t\tname: 'Written data',\n\t\t            field: 'value',\n\t\t            categoryField: 'category',\n\t\t\t\t\tcolor: '#FB7D33'\n\t\t        }\n\t\t\t],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}'\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0,\n\t        },\n\t        categoryAxis: {\n\t            majorGridLines: {\n\t                visible: false\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: false\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: false\n\t\t\t\t}\n\t        },\n\t\t\ttooltip: {\n\t\t\t\tvisible: false,\n\t        }\n\t    }).data('kendoChart');\n\n\t\tthis._readDataChart = $('#znm-readDataChart').kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 100\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'bar',\n\t            labels: {\n\t                visible: true,\n\t                format: '{0}',\n\t                background: 'transparent',\n\t\t\t\t\tposition: 'insideBase',\n\t\t\t\t\ttemplate: getWRBytesLabelTemplate(),\n\t            },\n\t\t\t\tgap: .5,\n\t\t\t\tspacing: 0,\n\t\t\t\toverlay: {\n\t\t\t\t\tgradient: 'none'\n\t\t\t\t},\n\t\t\t\tborder: {\n\t\t\t\t\twidth: 0\n\t\t\t\t}\n\t        },\n\t        series: [\n\t\t\t\t{\n\t\t\t\t\tname: 'Read data',\n\t\t            field: 'value',\n\t\t            categoryField: 'category',\n\t\t\t\t\tcolor: '#FB7D33'\n\t\t        }\n\t\t\t],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}'\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0,\n\t        },\n\t        categoryAxis: {\n\t            majorGridLines: {\n\t                visible: false\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: false\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: false\n\t\t\t\t}\n\t        },\n\t\t\ttooltip: {\n\t\t\t\tvisible: false,\n\t        }\n\t    }).data('kendoChart');\n\n\t\t//-------------------------------\n\n\t\t// Add listener to redraw all charts in case of window resize\n\t\t$(window).on('resize', $.proxy(this._onWindowResize, this));\n\t\tthis._onWindowResize(); // Also do it immediately\n\n\t\t// Request data to server\n\t\tthis._requestData(true);\n\t}\n\n\tdestroy()\n\t{\n\t\t// Call super method\n\t\tsuper.destroy();\n\n\t\t// Close monitoring\n\t\tthis._onCloseMonitorBtClick();\n\n\t\t// Remove doubleclick event listeners\n\t\tthis._zoneListBox.wrapper.find('.k-list').off('dblclick');\n\t\tthis._roomListBox.wrapper.find('.k-list').off('dblclick');\n\t\tthis._userListBox.wrapper.find('.k-list').off('dblclick');\n\n\t\t// Remove other event listeners\n\t\t$('#znm-createRoomBt').off('click');\n\t\t$('#znm-removeRoomBt').off('click');\n\t\t$('#znm-roomCreatorCreateBt').off('click');\n\t\t$('#znm-filterRoomBt').off('click');\n\t\t$('#znm-applyRoomFilterCB').off('change');\n\t\t$('#znm-filterUserBt').off('click');\n\t\t$('#znm-applyUserFilterCB').off('change');\n\t\t$('#znm-monitorBt').off('click');\n\t\t$('#znm-sendMessageBt').off('click');\n\t\t$('#znm-messageIn').off('keyup');\n\t\t$('#znm-closeMonitorBt').off('click');\n\t\t$('#znm-editBt').off('click');\n\t\t$('#znm-cancelBt').off('click');\n\t\t$('#znm-submitBt').off('click');\n\t\t$(window).off('resize');\n\t\t$('#znm-reloadWordsBt').off('click');\n\t\t$('#znm-reloadZoneExtBt').off('click');\n\t\t$('#znm-disconnectBt').off('click');\n\t\t$('#znm-kickBt').off('click');\n\t\t$('#znm-banBt').off('click');\n\n\t\t// Clear request scheduling\n\t\tclearTimeout(this._requestTimer);\n\n\t\t// Hide room creation panel\n\t\t$('#znm-createRoomModal').modal('hide');\n\n\t\t// Remove room creation panel listener\n\t\t$('#znm-createRoomModal').off('hidden.bs.modal');\n\n\t\t// Remove filter panels listeners\n\t\t$(this._roomFilter).off(this._roomFilter.EXPRESSION_UPDATED_EVENT);\n\t\t$(this._userFilter).off(this._userFilter.EXPRESSION_UPDATED_EVENT);\n\t}\n\n\tonExtensionCommand(command, data)\n\t{\n\t\t// Data received\n\t\tif (command == this.RESP_DATA)\n\t\t{\n\t\t\t// We have to check if the response id corresponds to the current request id;\n\t\t\t// if not, the response is discarded as it refers to an outdatet request\n\t\t\tconst responseId = data.getInt('id');\n\n\t\t\tif (responseId == this._currentRequestId)\n\t\t\t{\n\t\t\t\t// --- ZONES LIST ---\n\t\t\t\tif (data.containsKey('zones'))\n\t\t\t\t{\n\t\t\t\t\tthis._setZonesDataProvider(data.getSFSArray('zones'));\n\n\t\t\t\t\tif (this._selectedZone)\n\t\t\t\t\t\tthis._setScopeLabel();\n\t\t\t\t}\n\n\t\t\t\t// --- ROOMS LIST ---\n\t\t\t\tif (data.containsKey('rooms'))\n\t\t\t\t{\n\t\t\t\t\tthis._setGroupsDataProvider(data.getUtfStringArray('groups'));\n\n\t\t\t\t\t// -------------------------------\n\n\t\t\t\t\t// If a null rooms list is returned, the zone must have been deleted!\n\t\t\t\t\tif (!data.isNull('rooms'))\n\t\t\t\t\t{\n\t\t\t\t\t\tthis._setRoomsDataProvider(data.getSFSArray('rooms'));\n\n\t\t\t\t\t\tif (!this._selectedRoom)\n\t\t\t\t\t\t\tthis._setScopeLabel();\n\n\t\t\t\t\t\t// -------------------------------\n\n\t\t\t\t\t\t// Show warning if the rooms list exceeds the server-side limit\n\t\t\t\t\t\tif (data.containsKey('rSize'))\n\t\t\t\t\t\t\tthis._showLimitExceededWarning($('#znm-roomListWarning'), 'The received list is incomplete, because its size (' + data.getInt('rSize') + ' Rooms) exceeded this client\\'s limit of ' + data.getInt('rLimit') + '; you should refine your search');\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tthis._showLimitExceededWarning($('#znm-roomListWarning'), '');\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tif (this._activePanelType == 'room')\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tthis._accordion.select('[data-item-type=\"zone\"]');\n\t\t\t\t\t\t\tthis._accordion.expand('[data-item-type=\"zone\"]');\n\t\t\t\t\t\t\tthis._onZoneSelected();\n\t\t\t\t\t\t\tthis._onEntitiesPanelChange();\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// --- USERS LIST ---\n\t\t\t\tif (data.containsKey('users'))\n\t\t\t\t{\n\t\t\t\t\t// If a null users list is returned, the room must have been deleted!\n\t\t\t\t\tif (!data.isNull('users'))\n\t\t\t\t\t{\n\t\t\t\t\t\tthis._setUsersDataProvider(data.getSFSArray('users'));\n\n\t\t\t\t\t\t// Show warning if the users list exceeds the server-side limit\n\t\t\t\t\t\tif (data.containsKey('uSize'))\n\t\t\t\t\t\t\tthis._showLimitExceededWarning($('#znm-userListWarning'), 'The received list is incomplete, because its size (' + data.getInt('uSize') + ' users) exceeds this client\\'s limit of ' + data.getInt('uLimit') + '; you should refine your search');\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tthis._showLimitExceededWarning($('#znm-userListWarning'), '');\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tif (this._activePanelType == 'user')\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tthis._accordion.select('[data-item-type=\"room\"]');\n\t\t\t\t\t\t\tthis._accordion.expand('[data-item-type=\"room\"]');\n\t\t\t\t\t\t\tthis._onRoomSelected();\n\t\t\t\t\t\t\tthis._onEntitiesPanelChange();\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// --- MONITORED DATA ---\n\t\t\t\tif (data.containsKey('monitored'))\n\t\t\t\t{\n\t\t\t\t\tconst zone = data.getSFSObject('monitored').getUtfString('zone');\n\t\t\t\t\tconst type = data.getSFSObject('monitored').getUtfString('type');\n\t\t\t\t\tconst name = data.getSFSObject('monitored').getUtfString('name');\n\n\t\t\t\t\tif (zone == this._monitoredZone && type == this._monitoredType && name == this._monitoredName)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (!data.getSFSObject('monitored').isNull('params'))\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tconst params = data.getSFSObject('monitored').getSFSArray('params');\n\n\t\t\t\t\t\t\tthis._setMonitoringInterface();\n\n\t\t\t\t\t\t\t// Build interface\n\t\t\t\t\t\t\tif (!this._isEditing)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tthis._interfaceBuilder.buildInterface(params, `znm-${this._monitoredType}TabNavigator`, true);\n\n\t\t\t\t\t\t\t\tif (!this._skipInitTabs)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t$(`#znm-${this._monitoredType}TabNavigator #tabs`).scrollingTabs({\n\t\t\t\t\t\t\t\t\t\tbootstrapVersion: 4,\n\t\t\t\t\t\t\t\t\t\tscrollToTabEdge: true,\n\t\t\t\t\t\t\t\t\t\tenableSwiping: true,\n\t\t\t\t\t\t\t\t\t\tdisableScrollArrowsOnFullyScrolled: true,\n\t\t\t\t\t\t\t\t\t\tcssClassLeftArrow: 'fa fa-chevron-left',\n\t\t\t\t\t\t\t\t\t\tcssClassRightArrow: 'fa fa-chevron-right'\n\t\t\t\t\t\t\t\t\t});\n\n\t\t\t\t\t\t\t\t\tthis._skipInitTabs = true;\n\n\t\t\t\t\t\t\t\t\t// Set first tab panel as active\n\t\t\t\t\t\t\t\t\tthis._interfaceBuilder.activateFirstTabPanel();\n\n\t\t\t\t\t\t\t\t\t// Redraw time range slider to set its width\n\t\t\t\t\t\t\t\t\tthis._timeSlider.resize();\n\n\t\t\t\t\t\t\t\t\t// Add listener to redraw all charts on tab change\n\t\t\t\t\t\t\t\t\t// (to work around an issue with the chart resizing to default width when pane is hidden)\n\t\t\t\t\t\t\t\t\t$('a[data-toggle=\"tab\"]').on('shown.bs.tab', $.proxy(this._onWindowResize, this));\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t// Disable categories\n\t\t\t\t\t\t\tlet disabledCat = (data.getSFSObject('monitored').containsKey('exclude') ? data.getSFSObject('monitored').getUtfStringArray('exclude') : []);\n\n\t\t\t\t\t\t\tfor (let cat of disabledCat)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tlet navLink = $(`#znm-${this._monitoredType}TabNavigator #tabs #tab-${cat}`);\n\n\t\t\t\t\t\t\t\tnavLink.addClass('disabled');\n\t\t\t\t\t\t\t\tnavLink.attr('tabindex', -1);\n\t\t\t\t\t\t\t\tnavLink.attr('aria-disabled', true);\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tif (this._monitoredType == this.MONITORED_TYPE_ZONE)\n\t\t\t\t\t\t\t\tthis._showZoneTrafficData(data.getSFSObject('monitored').getSFSObject('traffic'));\n\n\t\t\t\t\t\t\tif (this._monitoredType == this.MONITORED_TYPE_USER)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t// Stats\n\t\t\t\t\t\t\t\tthis._showUserStatsData(data.getSFSObject('monitored').getSFSObject('stats'));\n\n\t\t\t\t\t\t\t\t// NPC\n\t\t\t\t\t\t\t\tconst isNpc = data.getSFSObject('monitored').getBool('npc');\n\n\t\t\t\t\t\t\t\t// Disable disconnect, kick and ban buttons\n\t\t\t\t\t\t\t\t$('#znm-disconnectBt').attr('disabled', isNpc);\n\t\t\t\t\t\t\t\t$('#znm-kickBt').attr('disabled', isNpc);\n\t\t\t\t\t\t\t\t$('#znm-banBt').attr('disabled', isNpc);\n\n\t\t\t\t\t\t\t\t// Geo-location\n\t\t\t\t\t\t\t\tconst geoLocData = data.getSFSObject('monitored').getSFSObject('geoLoc');\n\t\t\t\t\t\t\t\tthis._setGeoLocationUI(geoLocData);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t// Show alert\n\t\t\t\t\t\t\tthis.shellCtrl.showSimpleAlert(`The ${capitalizeFirst(this._monitoredType)} being monitored is no more available on the server, please select another one.`, false);\n\n\t\t\t\t\t\t\t// Cancel monitoring\n\t\t\t\t\t\t\tthis._onCloseMonitorBtClick();\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}\n\n\t\telse if (command == this.RESP_UPDATE_ERROR)\n\t\t{\n\t\t\tif (data.getUtfString('zone') == this._monitoredZone\n\t\t\t\t&& data.getUtfString('type') ==  this._monitoredType\n\t\t\t\t&& data.getUtfString('name') ==  this._monitoredName)\n\t\t\t{\n\t\t\t\t// Show alert\n\t\t\t\tthis.shellCtrl.showSimpleAlert(`Unable to update ${capitalizeFirst(this._monitoredType)} settings; the following error was reported: ${data.getUtfString('message')}.`);\n\t\t\t}\n\t\t}\n\n\t\telse if (command == this.RESP_ROOM_CONFIG)\n\t\t{\n\t\t\t// Re-enable Create button\n\t\t\t$('#znm-createRoomBt').attr('disabled', false);\n\n\t\t\t// Show Room creation panel\n\t\t\tthis._showRoomCreationPanel(data.getSFSArray('settings'));\n\t\t}\n\n\t\telse if (command == this.RESP_ROOM_CONFIG_ERROR)\n\t\t{\n\t\t\t// Re-enable Create button\n\t\t\t$('#znm-createRoomBt').attr('disabled', false);\n\n\t\t\t// Show an alert\n\t\t\tthis.shellCtrl.showSimpleAlert(data.getUtfString('error'));\n\t\t}\n\n\t\telse if (command == this.RESP_ROOM_CREATED)\n\t\t{\n\t\t\t// Re-enable room creation panel\n\t\t\tthis._enableCreateRoomPanel(true);\n\n\t\t\t// Hide room creation panel\n\t\t\t$('#znm-createRoomModal').modal('hide');\n\n\t\t\t// Update rooms list\n\t\t\tthis._requestData(true);\n\t\t}\n\n\t\telse if (command == this.RESP_ROOM_CREATION_ERROR)\n\t\t{\n\t\t\t// Display error message\n\t\t\t$('#znm-createRoomError').text(data.getUtfString('error'));\n\t\t\t$('#znm-createRoomError').show();\n\n\t\t\t// Re-enable room creation panel\n\t\t\tthis._enableCreateRoomPanel(true);\n\t\t}\n\n\t\t// Unexpected error received\n\t\t// This error is returned in case something went wrong when retrieving the monitored entity parameters,\n\t\t// so we have to stop requesting the same entity\n\t\telse if (command == this.RESP_UNEXPECTED_ERROR)\n\t\t{\n\t\t\t// Cancel monitoring\n\t\t\tthis._onCloseMonitorBtClick();\n\n\t\t\t// Show an alert\n\t\t\tthis.shellCtrl.showSimpleAlert(data.getUtfString('error'));\n\t\t}\n\n\t\t// Extension (Zone or Room) log messages received\n\t\telse if (command == this.RESP_LOG_MESSAGES)\n\t\t{\n\t\t\tlet logEntries = data.getSFSArray('entries');\n\n\t\t\tfor (let e = 0; e < logEntries.size(); e++)\n\t\t\t{\n\t\t\t\tlet logEntry = logEntries.getSFSObject(e);\n\n\t\t\t\tconst logZone = logEntry.getUtfString('zone');\n\t\t\t\tconst logRoom = logEntry.getUtfString('room');\n\n\t\t\t\t// Check if the zone is currently monitored\n\t\t\t\tif (this._monitoredZone == logZone)\n\t\t\t\t{\n\t\t\t\t\t// console.log(logEntry.getLong('time'))\n\t\t\t\t\t// console.log(new Date(logEntry.getLong('time'))\n\n\t\t\t\t\tlet ele = {\n\t\t\t\t\t\ttimestamp: kendo.toString(new Date(logEntry.getLong('time')), 'dd/MM/yyyy HH:mm:ss'),\n\t\t\t\t\t\tlevel: logEntry.getUtfString('level'),\n\t\t\t\t\t\tmessage: logEntry.getUtfString('msg'),\n\t\t\t\t\t};\n\n\t\t\t\t\tif (this._monitoredType == this.MONITORED_TYPE_ZONE && logRoom == '')\n\t\t\t\t\t{\n\t\t\t\t\t\t// Add to zone extension log\n\t\t\t\t\t\tthis._addLogEntryToGrid(this._zoneExtLogGrid, ele);\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tif (this._monitoredType == this.MONITORED_TYPE_ROOM && this._monitoredName == logRoom)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t// Add to room extension log\n\t\t\t\t\t\t\tthis._addLogEntryToGrid(this._roomExtLogGrid, ele);\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}\n\t}\n\n\t//---------------------------------\n\t// UI EVENT LISTENERS\n\t//---------------------------------\n\n\t_onUpdateIntervalChange()\n\t{\n\t\t// Request data to server\n\t\tthis._requestData(false);\n\t}\n\n\t_onEntitiesPanelChange()\n\t{\n\t\t// Set main controls state\n\t\tthis._setMainControlsEnabled();\n\n\t\t// Request data to server\n\t\tthis._requestData(true);\n\t}\n\n\t/**\n\t * Update the Scope label in the Rooms panel.\n\t */\n\t_onZoneSelected()\n\t{\n\t\t// Clear Room and User lists\n\t\tthis._resetRoomList();\n\t\tthis._resetUserList();\n\n\t\t// Set scope label\n\t\tthis._setScopeLabel();\n\n\t\t// Enable/disable Room and User tools\n\t\tconst disabled = this._selectedZone == undefined;\n\n\t\t// (NOTE: this enables/disables whole fieldset; this doesn't affect Kendo DropDown, so we have to do it manually)\n\t\t$('#roomTools').attr('disabled', disabled);\n\t\t$('#userTools').attr('disabled', disabled);\n\n\t\tthis._groupsDropDown.enable(!disabled);\n\n\t\t// Set main controls state\n\t\tthis._setMainControlsEnabled();\n\t}\n\n\t/**\n\t * Update the Scope label in the Users panel.\n\t */\n\t_onRoomSelected()\n\t{\n\t\t// Clear User list\n\t\tthis._resetUserList();\n\n\t\t// Set scope label\n\t\tthis._setScopeLabel();\n\n\t\t// Enable Remove button\n\t\t$('#znm-removeRoomBt').attr('disabled', this._selectedRoom == undefined);\n\n\t\t// Set main controls state\n\t\tthis._setMainControlsEnabled();\n\t}\n\n\t/**\n\t * Update the scope label in the Users panel.\n\t */\n\t_onGroupChange()\n\t{\n\t\t// Clear Room and User lists\n\t\tthis._resetRoomList();\n\t\tthis._resetUserList();\n\n\t\t// Set scope label\n\t\tthis._setScopeLabel();\n\n\t\t// Request data to server\n\t\tthis._requestData(true);\n\n\t\t// Set main controls state\n\t\tthis._setMainControlsEnabled();\n\t}\n\n\t_onUserSelected()\n\t{\n\t\t// Set main controls state\n\t\tthis._setMainControlsEnabled();\n\t}\n\n\t/**\n\t * Show room creation panel.\n\t */\n\t_onCreateRoomBtClick()\n\t{\n\t\t// Disable Create button\n\t\t$('#znm-createRoomBt').attr('disabled', true);\n\n\t\t// Request default room settings to extension\n\t\tthis.sendExtensionRequest(this.REQ_GET_ROOM_CONFIG);\n\t}\n\n\t/**\n\t * Remove existing room.\n\t */\n\t_onRemoveRoomBtClick()\n\t{\n\t\tthis.shellCtrl.showConfirmWarning('Are you sure you want to remove the selected Room?', $.proxy(this._onRemoveRemoveConfirm, this));\n\t}\n\n\t_onRemoveRemoveConfirm()\n\t{\n\t\tconst selectedZone = this._zoneListBox.dataItem(this._zoneListBox.select());\n\t\tlet selectedRoom = this._roomListBox.dataItem(this._roomListBox.select());\n\n\t\tif (selectedRoom)\n\t\t{\n\t\t\t// Send request to server\n\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\tparams.putUtfString('zone', selectedZone.name);\n\t\t\tparams.putUtfString('room', selectedRoom.name);\n\n\t\t\tthis.sendExtensionRequest(this.REQ_REMOVE_ROOM, params);\n\n\t\t\t// Request data to server to update the rooms list\n\t\t\tthis._requestData(true);\n\t\t}\n\t}\n\n\t/**\n\t * Remove content of room creation panel.\n\t */\n\t_onCreateRoomModalHidden(e)\n\t{\n\t\t// If a nested modal is opened, its closing causes the parent modal to be closed too (Bootstrap doesn't support nested modals)\n\t\t// As a workaround, here we reset the css class 'modal-open' when the nested modal is closed and skip everything else\n\t\t// In addition we had to use a custom listener on the close/cancel buttons of the nested modal (see config-grid.js for example)\n\t\tif (e.target !== document.getElementById('znm-createRoomModal'))\n\t\t{\n\t\t\t$('body').addClass('modal-open')\n\t\t\treturn;\n\t\t}\n\n\t\t// Destroy scrolling tabs in room creation panel\n\t\t$('#znm-roomCreatorTabNav #tabs').scrollingTabs('destroy');\n\n\t\t// Remove all tab navigator content\n\t\tthis._roomCreationIBuilder.destroyInterface();\n\n\t\t// Remove listener for custom actions triggered by configuration interface\n\t\t$('#znm-roomCreatorTabNav').off('value-set');\n\n\t\t// Reset and hide error message\n\t\tthis._resetRoomCreationError();\n\t}\n\n\t_onConfigValueSet(e) // SAME METHOD DUPLICATED IN zone-configurator.js\n\t{\n\t\tconst configParam = e.target.data;\n\n\t\t// Handle extension name/type dropdowns update and update the main class dropdown datasource accordingly\n\t\tif (configParam.name == 'extension.name' || configParam.name == 'extension.type' || configParam.name == 'extension.filterClass')\n\t\t{\n\t\t\t// All involved ConfigFormItems must be available and initialized to proceed\n\t\t\tconst nameFormItem = this._roomCreationIBuilder.getConfigFormItem('extension.name');\n\t\t\tconst typeFormItem = this._roomCreationIBuilder.getConfigFormItem('extension.type');\n\t\t\tconst classFormItem = this._roomCreationIBuilder.getConfigFormItem('extension.file');\n\t\t\tconst filterFormItem = this._roomCreationIBuilder.getConfigFormItem('extension.filterClass');\n\n\t\t\tif (nameFormItem != null && typeFormItem != null && classFormItem != null && filterFormItem != null)\n\t\t\t{\n\t\t\t\tconst source = nameFormItem.data;\n\t\t\t\tlet classesList = [];\n\n\t\t\t\tlet data = source.triggerData;\n\t\t\t\tfor (let i = 0; i < data.size(); i++)\n\t\t\t\t{\n\t\t\t\t\tlet ext = data.getSFSObject(i);\n\n\t\t\t\t\tif (ext.getUtfString('name') == nameFormItem.data.value && ext.getUtfString('type') == typeFormItem.data.value)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet classes = ext.getUtfString('classesString').split(',');\n\n\t\t\t\t\t\tif (filterFormItem.data.value == true)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet filteredClasses = classes.filter(filterClassName);\n\t\t\t\t\t\t\tclasses = filteredClasses;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tclassesList = classesList.concat(classes);\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tlet currentClass = classFormItem.data.value;\n\n\t\t\t\t// If the classes list doesn't contain the current value, create an empty entry and reset the value\n\t\t\t\tif (classesList.indexOf(currentClass) < 0)\n\t\t\t\t{\n\t\t\t\t\tif (classesList.length == 0)\n\t\t\t\t\t{\n\t\t\t\t\t\tclassesList.push('');\n\t\t\t\t\t\tcurrentClass = '';\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t\tcurrentClass = classesList[0];\n\t\t\t\t}\n\n\t\t\t\tlet mainClassDropDown = classFormItem._innerWidget;\n\t\t\t\tmainClassDropDown.setDataSource(classesList);\n\n\t\t\t\tclassFormItem.data.value = currentClass;\n\t\t\t\tclassFormItem._setWidgetValue();\n\t\t\t}\n\t\t}\n\t}\n\n\t_onRoomCreatorCreateBtClick()\n\t{\n\t\t// Reset and hide error message\n\t\tthis._resetRoomCreationError();\n\n\t\t// Check validity\n\t\tif (this._roomCreationIBuilder.checkIsValid())\n\t\t{\n\t\t\tlet changes = this._roomCreationIBuilder.getChangedData();\n\n\t\t\tif (changes.size() > 0)\n\t\t\t{\n\t\t\t\t// Check the room name against the rooms list (duplicate names not allowed!)\n\t\t\t\tif (true)//this._validateNewRoomName(changes))\n\t\t\t\t{\n\t\t\t\t\t// Disable room creation panel\n\t\t\t\t\tthis._enableCreateRoomPanel(false);\n\n\t\t\t\t\t// Send settings to server instance\n\t\t\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\t\t\tparams.putSFSArray('settings', changes);\n\t\t\t\t\tparams.putUtfString('zName', this._zoneListBox.dataItem(this._zoneListBox.select()).name);\n\t\t\t\t\tparams.putBool('notify', $('#znm-notifyClientsCB').prop('checked'));\n\n\t\t\t\t\t// Send settings to extension\n\t\t\t\t\tthis.sendExtensionRequest(this.REQ_CREATE_ROOM, params);\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t// Show alert\n\t\t\t\t\tthis.shellCtrl.showSimpleAlert('Unable to submit configuration because the Room name already exists; duplicate names are not allowed.', true);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// Show alert\n\t\t\tthis.shellCtrl.showSimpleAlert('Unable to submit Room configuration due to an invalid value; please verify the highlighted form fields in all tabs.', true);\n\t\t}\n\t}\n\n\t_onShowRoomFilterBtClick()\n\t{\n\t\tthis._roomFilter.show();\n\t}\n\n\t_onShowUserFilterBtClick()\n\t{\n\t\tthis._userFilter.show();\n\t}\n\n\t_onApplyBtChange()\n\t{\n\t\t// Request data to server\n\t\tthis._requestData(true);\n\t}\n\n\t_onRoomFilterUpdated()\n\t{\n\t\t// Enable/disable \"Apply\" checkbox\n\t\t$('#znm-applyRoomFilterCB').attr('disabled', this._roomFilter.filterExpression == null);\n\n\t\t// Request data to server\n\t\tthis._requestData(true);\n\t}\n\n\t_onUserFilterUpdated()\n\t{\n\t\t// Enable/disable \"Apply\" checkbox\n\t\t$('#znm-applyUserFilterCB').attr('disabled', this._userFilter.filterExpression == null);\n\n\t\t// Request data to server\n\t\tthis._requestData(true);\n\t}\n\n\t_onMonitorSelectionBtClick()\n\t{\n\t\tif (this._selectedZone)\n\t\t{\n\t\t\t// Cancel previous monitoring\n\t\t\tthis._onCloseMonitorBtClick();\n\n\t\t\tlet getZoneUserCountHistory = false;\n\n\t\t\tthis._monitoredZone = this._selectedZone.name;\n\n\t\t\t// Zone must be monitored\n\t\t\tif (this._activePanelType == 'zone')\n\t\t\t{\n\t\t\t\tthis._monitoredType = this.MONITORED_TYPE_ZONE;\n\t\t\t\tthis._monitoredName = this._monitoredZone;\n\n\t\t\t\tgetZoneUserCountHistory = true;\n\t\t\t}\n\n\t\t\t// Room must be monitored\n\t\t\tif (this._activePanelType == 'room' && this._selectedRoom != undefined)\n\t\t\t{\n\t\t\t\tthis._monitoredType = this.MONITORED_TYPE_ROOM;\n\t\t\t\tthis._monitoredName = this._selectedRoom.name;\n\t\t\t}\n\n\t\t\t// User must be monitored\n\t\t\tif (this._activePanelType == 'user' && this._selectedUser != undefined)\n\t\t\t{\n\t\t\t\tthis._monitoredType = this.MONITORED_TYPE_USER;\n\t\t\t\tthis._monitoredName = this._selectedUser.name;\n\t\t\t}\n\n\t\t\t// Show loading bar\n\t\t\tthis._switchView('znm-loading');\n\n\t\t\t// Request data to server\n\t\t\tthis._requestData(true, getZoneUserCountHistory);\n\t\t}\n\t}\n\n\t_onSendAdminMsgInKeyUp(event)\n\t{\n\t\tif (event.key !== 'Enter')\n\t\t\treturn;\n\n\t\tevent.preventDefault();\n\t\tthis._onSendAdminMsgBtClick();\n\t}\n\n\t_onSendAdminMsgBtClick()\n\t{\n\t\tif (this._selectedZone && $('#znm-messageIn').val() != '')\n\t\t{\n\t\t\t// Build request parameters\n\t\t\tlet params = new SFS2X.SFSObject();\n\n\t\t\t// Always include zone\n\t\t\tparams.putUtfString('zone', this._selectedZone.name);\n\n\t\t\t// Send message to room?\n\t\t\tif (this._activePanelType == 'room' && this._selectedRoom != undefined)\n\t\t\t\tparams.putUtfString('room', this._selectedRoom.name);\n\n\t\t\t// Send message to user?\n\t\t\tif (this._activePanelType == 'user' && this._selectedUser != undefined)\n\t\t\t\tparams.putUtfString('user', this._selectedUser.name);\n\n\t\t\t// Add message\n\t\t\tparams.putUtfString('msg', $('#znm-messageIn').val());\n\n\t\t\t// Send request to extension\n\t\t\tthis.sendExtensionRequest(this.REQ_ADMIN_MSG, params);\n\n\t\t\t// Clear text input\n\t\t\t$('#znm-messageIn').val('');\n\t\t}\n\t}\n\n\t_onCloseMonitorBtClick()\n\t{\n\t\t// Save ref. for later usage\n\t\tconst monitoredType = this._monitoredType;\n\n\t\tthis._monitoredZone = null;\n\t\tthis._monitoredType = null;\n\t\tthis._monitoredName = null;\n\n\t\t// Clear geolocation UI\n\t\tthis._clearGeoLocationUI();\n\n\t\t// Clear log datagrids dataproviders\n\t\t// NOTE: a try-catch is used here because, for an unknown reason, Kendo is throwing an error when resetting the data sources when the module is destroyed\n\t\ttry\n\t\t{\n\t\t\tthis._zoneExtLogGrid.setDataSource([]);\n\t\t\tthis._roomExtLogGrid.setDataSource([]);\n\t\t}\n\t\tcatch (e) { /* Ignore */ }\n\n\t\t// Leave edit mode\n\t\tthis._onCancelBtClick();\n\n\t\t// Rearrange monitoring interface\n\t\tthis._setMonitoringInterface();\n\n\t\t// Clear tab navigator\n\t\tthis._clearTabs(monitoredType);\n\t}\n\n\t_onEditBtClick()\n\t{\n\t\tif (!this._isEditing)\n\t\t{\n\t\t\t// Show edit controls\n\t\t\t$('#znm-editControls').show();\n\n\t\t\t// Hide edit button\n\t\t\t$('#znm-editBt').hide();\n\n\t\t\t// Enable editable fields\n\t\t\tthis._interfaceBuilder.disableInterface(false);\n\n\t\t\tthis._isEditMode = true;\n\t\t}\n\t}\n\n\t_onCancelBtClick()\n\t{\n\t\tif (this._isEditing)\n\t\t{\n\t\t\t// Hide edit controls\n\t\t\t$('#znm-editControls').hide();\n\n\t\t\t// Show edit button\n\t\t\t$('#znm-editBt').show();\n\n\t\t\t// Disable editable fields\n\t\t\tthis._interfaceBuilder.disableInterface(true);\n\n\t\t\tthis._isEditMode = false;\n\n\t\t\t// Hide validation messages\n\t\t\tthis._interfaceBuilder.resetValidation();\n\n\t\t\t// Request data to update interface (updating suspended in edit mode)\n\t\t\tthis._requestData(true);\n\t\t}\n\t}\n\n\t_onSubmitBtClick()\n\t{\n\t\tif (this._isEditing)\n\t\t{\n\t\t\t// Check validity\n\t\t\tif (this._interfaceBuilder.checkIsValid())\n\t\t\t{\n\t\t\t\tlet changes = this._interfaceBuilder.getChangedData();\n\n\t\t\t\tif (changes.size() > 0)\n\t\t\t\t{\n\t\t\t\t\t//console.log(changes.getDump())\n\n\t\t\t\t\t// Send settings to server instance\n\t\t\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\t\t\tparams.putSFSArray('settings', changes);\n\t\t\t\t\tparams.putUtfString('zone', this._monitoredZone);\n\n\t\t\t\t\tif (this._monitoredType == this.MONITORED_TYPE_ROOM)\n\t\t\t\t\t{\n\t\t\t\t\t\t// Submit room settings\n\t\t\t\t\t\tparams.putUtfString('room', this._monitoredName);\n\t\t\t\t\t\tthis.sendExtensionRequest(this.REQ_SET_ROOM_SETTINGS, params);\n\t\t\t\t\t}\n\t\t\t\t\telse if (this._monitoredType == this.MONITORED_TYPE_USER)\n\t\t\t\t\t{\n\t\t\t\t\t\t// Submit user settings\n\t\t\t\t\t\tparams.putUtfString('user', this._monitoredName);\n\t\t\t\t\t\tthis.sendExtensionRequest(this.REQ_SET_USER_SETTINGS, params);\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\t// Submit zone settings\n\t\t\t\t\t\tthis.sendExtensionRequest(this.REQ_SET_ZONE_SETTINGS, params);\n\t\t\t\t\t}\n\n\t\t\t\t\t// Leave edit mode\n\t\t\t\t\tthis._onCancelBtClick();\n\t\t\t\t}\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Show alert\n\t\t\t\tthis.shellCtrl.showSimpleAlert('Unable to submit changes due to an invalid value; please verify the highlighted form fields in all tabs.', true);\n\t\t\t}\n\t\t}\n\t}\n\n\t_onTimeRangeChange()\n\t{\n\t\tlet values = this._timeSlider.value();\n\n\t\tif (values[0] == values[1])\n\t\t{\n\t\t\tif (values[1] == -24)\n\t\t\t\tvalues[1] = -23;\n\n\t\t\tvalues[0] = values[1] - 1;\n\n\t\t\t// Reset the time range slider value (we need to use setTimeout\n\t\t\t// bacause doing it in the change event listener doesn't redraw the slider)\n\t\t\tsetTimeout($.proxy( function(values) { this._timeSlider.value(values); }, this), 10, values);\n\t\t}\n\n\t\tthis._userCountChart.range = values;\n\t}\n\n\t_onWindowResize()\n\t{\n\t\t// Redraw charts\n\t\tthis._userCountChart.redraw();\n\t\tthis._packetQueueChart.redraw();\n\t\tthis._droppedMsgChart.redraw();\n\t\tthis._writtenDataChart.redraw();\n\t\tthis._readDataChart.redraw();\n\n\t\t// Redraw time range slider\n\t\tthis._timeSlider.resize();\n\t}\n\n\t_onWordsReloadBtClick()\n\t{\n\t\tif (this._monitoredType == this.MONITORED_TYPE_ZONE)\n\t\t{\n\t\t\t// Send request to extension\n\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\tparams.putUtfString('zone', this._monitoredName);\n\n\t\t\tthis.sendExtensionRequest(this.REQ_RELOAD_WORDS, params);\n\t\t}\n\t}\n\n\t_onZoneExtReloadBtClick()\n\t{\n\t\tif (this._monitoredType == this.MONITORED_TYPE_ZONE)\n\t\t{\n\t\t\t// Send request to extension\n\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\tparams.putUtfString('zone', this._monitoredName);\n\n\t\t\tthis.sendExtensionRequest(this.REQ_RELOAD_ZONE_EXT, params);\n\t\t}\n\t}\n\n\t_onDisconnectBtClick()\n\t{\n\t\tif (this._monitoredType == this.MONITORED_TYPE_USER)\n\t\t{\n\t\t\t// Send request to extension\n\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\tparams.putUtfString('zone', this._monitoredZone);\n\t\t\tparams.putUtfString('user', this._monitoredName);\n\n\t\t\tthis.sendExtensionRequest(this.REQ_DISCONNECT_USER, params);\n\t\t}\n\t}\n\n\t_onKickBtClick()\n\t{\n\t\tif (this._monitoredType == this.MONITORED_TYPE_USER)\n\t\t{\n\t\t\t// Send request to extension\n\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\tparams.putUtfString('zone', this._monitoredZone);\n\t\t\tparams.putUtfString('user', this._monitoredName);\n\t\t\tparams.putUtfString('msg', $('#znm-kickMsgIn').val());\n\t\t\tparams.putInt('delay', (this._kickDelayIn.value() != null ? Number(this._kickDelayIn.value()) : this.KICK_BAN_DEFAULT_DELAY));\n\n\t\t\tthis.sendExtensionRequest(this.REQ_KICK_USER, params);\n\n\t\t\t// Reset message\n\t\t\t$('#znm-kickMsgIn').val('');\n\t\t}\n\t}\n\n\t_onBanBtClick()\n\t{\n\t\tif (this._monitoredType == this.MONITORED_TYPE_USER)\n\t\t{\n\t\t\t// Send request to extension\n\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\tparams.putUtfString('zone', this._monitoredZone);\n\t\t\tparams.putUtfString('user', this._monitoredName);\n\t\t\tparams.putUtfString('msg', $('#znm-banMsgIn').val());\n\t\t\tparams.putUtfString('mode', this._banModeDd.select() > 0 ? this._banModeDd.value() : 'NAME');\n\t\t\tparams.putInt('delay', (this._banDelayIn.value() != null ? Number(this._banDelayIn.value()) : this.KICK_BAN_DEFAULT_DELAY));\n\n\t\t\tlet duration = this._banDurationIn.value() != null ? Number(this._banDurationIn.value()) : 1;\n\n\t\t\tif (this._banDurUnitDd.value() == 'hours')\n\t\t\t\tduration = duration * 60;\n\t\t\telse if (this._banDurUnitDd.value() == 'days')\n\t\t\t\tduration = duration * 60 * 24;\n\n\t\t\tparams.putInt('duration', duration);\n\n\t\t\tthis.sendExtensionRequest(this.REQ_BAN_USER, params);\n\n\t\t\t// Reset message\n\t\t\t$('#znm-banMsgIn').val('');\n\t\t}\n\t}\n\n\t//------------------------------------\n\t// PRIVATE METHODS\n\t//------------------------------------\n\n\t_setMonitoringInterface()\n\t{\n\t\tconst enabled = (this._monitoredType != null && this._monitoredName != null);\n\n\t\t// Show/hide header and footer\n\t\tif (enabled)\n\t\t{\n\t\t\tthis._switchView(`znm-${this._monitoredType}Monitor`);\n\n\t\t\t// Header title\n\t\t\tlet title = 'Now monitoring ';\n\n\t\t\tif (this._monitoredType == this.MONITORED_TYPE_USER)\n\t\t\t\ttitle += `User: <strong>${this._monitoredName}</strong> (Zone: ${this._monitoredZone})`;\n\t\t\telse if (this._monitoredType == this.MONITORED_TYPE_ROOM)\n\t\t\t\ttitle += `Room: <strong>${this._monitoredName}</strong> (Zone: ${this._monitoredZone})`;\n\t\t\telse\n\t\t\t\ttitle += `Zone: <strong>${this._monitoredZone}</strong>`;\n\n\t\t\t$('#znm-monitoredHeader').html(title);\n\t\t\t$('#znm-monitoredFooter').show();\n\n\t\t\t// Show panel\n\t\t\tthis._switchPanel('znm-mainPanel');\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis._switchView('znm-select');\n\n\t\t\t$('#znm-monitoredHeader').html('');\n\t\t\t$('#znm-monitoredFooter').hide();\n\n\t\t\t// Show panel\n\t\t\tthis._switchPanel('znm-sidebarPanel');\n\t\t}\n\t}\n\n\t_switchView(viewId)\n\t{\n\t\tdocument.getElementById('znm-viewstack').selectedElement = document.getElementById(viewId);\n\t}\n\n\t_switchPanel(panelId)\n\t{\n\t\tdocument.getElementById('znm-view').selectedPanel = document.getElementById(panelId);\n\t}\n\n\t/**\n\t * Build the polling request to be sent to the server, based on the selections and other settings in the accordion.\n\t */\n\t_requestData(newId = false, getZoneUserCountHistory = false)\n\t{\n\t\t// Clear previous request scheduling\n\t\tclearTimeout(this._requestTimer);\n\n\t\t// Check if connection is still available\n\t\tif (this.smartFox.isConnected)\n\t\t{\n\t\t\tif (newId)\n\t\t\t\tthis._currentRequestId++;\n\n\t\t\t// Build request parameters\n\t\t\tlet params = new SFS2X.SFSObject();\n\n\t\t\t// The updated zones/rooms/users lists are requested if the corresponding panel is displayed only\n\t\t\tconst getZones = (this._activePanelType == 'zone');\n\t\t\tconst getRooms = (this._activePanelType == 'room' && this._selectedZone != undefined);\n\t\t\tconst getUsers = (this._activePanelType == 'user' && this._selectedZone != undefined);\n\n\t\t\t// --- ZONES ---\n\t\t\tparams.putBool('zones', getZones);\n\n\t\t\t// ROOMS & USERS\n\t\t\tif (getRooms || getUsers)\n\t\t\t{\n\t\t\t\t// Common filtering params:\n\n\t\t\t\t// a) zone name\n\t\t\t\tparams.putUtfString('zone', this._selectedZone.name);\n\n\t\t\t\t// b) group name\n\t\t\t\tparams.putUtfString('group', this._selectedGroup.name);\n\t\t\t}\n\n\t\t\t// ROOMS\n\t\t\tparams.putBool('rooms', getRooms);\n\t\t\tif (getRooms)\n\t\t\t{\n\t\t\t\t// Filtering params:\n\n\t\t\t\t// c) advanced filtering params\n\t\t\t\tif ($('#znm-applyRoomFilterCB').prop('checked') && this._roomFilter.filterExpression != null)\n\t\t\t\t\tparams.putSFSArray('rFilter', this._roomFilter.filterExpression.toSFSArray());\n\t\t\t}\n\n\t\t\t// USERS\n\t\t\tparams.putBool('users', getUsers);\n\t\t\tif (getUsers)\n\t\t\t{\n\t\t\t\t// Filtering params:\n\n\t\t\t\t// c) room name\n\t\t\t\tif (this._selectedRoom)\n\t\t\t\t\tparams.putUtfString('room', this._selectedRoom.name);\n\n\t\t\t\t// d) advanced filtering params\n\t\t\t\tif ($('#znm-applyUserFilterCB').prop('checked') && this._userFilter.filterExpression != null)\n\t\t\t\t\tparams.putSFSArray('uFilter', this._userFilter.filterExpression.toSFSArray());\n\t\t\t}\n\n\t\t\t// --- CURRENTLY MONITORED DATA ---\n\t\t\tif (this._monitoredType != null && this._monitoredName != null)\n\t\t\t{\n\t\t\t\tlet monitoredObj = this._getBasicMonitoredObject();\n\n\t\t\t\tif (this._monitoredType == this.MONITORED_TYPE_ZONE)\n\t\t\t\t\tmonitoredObj.putBool('uc', getZoneUserCountHistory);\n\n\t\t\t\tparams.putSFSObject('monitored', monitoredObj);\n\t\t\t}\n\n\t\t\t// Set request id; when a response is received, we check its id matches the current request id: if not, the response is discarded\n\t\t\tparams.putInt('id', this._currentRequestId);\n\n\t\t\t// Send request to extension\n\t\t\tthis.sendExtensionRequest(this.REQ_GET_DATA, params);\n\n\t\t\t// Schedule next request\n\t\t\tthis._requestTimer = setTimeout($.proxy(this._requestData, this), Number(this._intervalDropDown.value()) * 1000);\n\t\t}\n\t}\n\n\t_getBasicMonitoredObject()\n\t{\n\t\tlet monitoredObj = new SFS2X.SFSObject();\n\n\t\tmonitoredObj.putUtfString('zone', this._monitoredZone);\n\t\tmonitoredObj.putUtfString('type', this._monitoredType);\n\t\tmonitoredObj.putUtfString('name', this._monitoredName);\n\n\t\treturn monitoredObj;\n\t}\n\n\t_setZonesDataProvider(zonesData)\n\t{\n\t\tthis._setListDataProvider(this._zoneListBox, zonesData, true);\n\t}\n\n\t_setRoomsDataProvider(roomsData)\n\t{\n\t\tthis._setListDataProvider(this._roomListBox, roomsData, true);\n\n\t\t// Set Remove button state\n\t\t$('#znm-removeRoomBt').attr('disabled', this._selectedRoom == undefined);\n\t}\n\n\t_setUsersDataProvider(usersData)\n\t{\n\t\tthis._setListDataProvider(this._userListBox, usersData);\n\t}\n\n\t_setListDataProvider(listWidget, listData, countUsers = false)\n\t{\n\t\t// Save reference to selected item name\n\t\t// NOTE: as we are substituting the whole data source, we need it to re-select the same item after update\n\t\tlet selectedDataItem = listWidget.dataItem(listWidget.select());\n\t\tlet selectedDataItemName = selectedDataItem ? selectedDataItem.name : null;\n\n\t\tlet dataArray = [];\n\n\t\t// Convert data coming from server\n\t\tfor (let i = 0; i < listData.size(); i++)\n\t\t{\n\t\t\tlet itemData = listData.getSFSObject(i);\n\n\t\t\tlet item = {};\n\t\t\titem.name = itemData.getUtfString('name');\n\n\t\t\tif (countUsers)\n\t\t\t\titem.users = itemData.getInt('users');\n\n\t\t\tdataArray.push(item);\n\t\t}\n\n\t\t// Create new data source\n\t\tlet listDS = new kendo.data.DataSource({\n\t\t\tdata: dataArray,\n\t\t\tschema: {\n\t\t\t\tmodel: {\n\t\t\t\t\tid: 'name'\n\t\t\t\t}\n\t\t\t},\n\t\t\tsort: {\n\t\t\t\tfield: 'name',\n\t\t\t\tdir: 'asc'\n\t\t\t}\n\t\t});\n\n\t\t// Assign data source to list\n\t\tlistWidget.setDataSource(listDS);\n\n\t\t// Select again previously selected item\n\t\tif (selectedDataItemName != null)\n\t\t{\n\t\t\tselectedDataItem = listDS.get(selectedDataItemName);\n\n\t\t\tif (selectedDataItem)\n\t\t\t{\n\t\t\t\tlet selectedElement = listWidget.wrapper.find('[data-uid=\"' + selectedDataItem.uid + '\"]');\n\t\t\t\tlistWidget.select(selectedElement);\n\t\t\t}\n\t\t}\n\n\t\t// Reset main control state\n\t\tthis._setMainControlsEnabled();\n\t}\n\n\t_setGroupsDataProvider(groupsData)\n\t{\n\t\t// Create Groups array, adding common entries\n\t\tlet groupsArray = [\n\t\t\tthis._getGroupObj('', this.ANY_LABEL),\n\t\t\tthis._getGroupObj(this.DEFAULT_GROUP_NAME)\n\t\t];\n\n\t\t// Save reference to selected group name\n\t\t// NOTE: as we are substituting the whole data source, we need it to re-select the same item after update\n\t\tlet selectedDataItem = this._selectedGroup;\n\t\tlet selectedDataItemName = selectedDataItem ? selectedDataItem.name : null;\n\n\t\tif (groupsData != null)\n\t\t{\n\t\t\tfor (let i = 0; i < groupsData.length; i++)\n\t\t\t{\n\t\t\t\tif (groupsData[i] != this.DEFAULT_GROUP_NAME) // Default group was added before\n\t\t\t\t\tgroupsArray.push(this._getGroupObj(groupsData[i]));\n\t\t\t}\n\t\t}\n\n\t\t// Create new data source\n\t\tlet groupDS = new kendo.data.DataSource({\n\t\t\tdata: groupsArray,\n\t\t\tschema: {\n\t\t\t\tmodel: {\n\t\t\t\t\tid: 'name'\n\t\t\t\t}\n\t\t\t}\n\t\t});\n\n\t\t// Assign data source to dropdown\n\t\tthis._groupsDropDown.setDataSource(groupDS);\n\n\t\t//--------------------------------\n\n\t\t// Select previously selected item\n\t\tselectedDataItem = groupDS.get(selectedDataItemName);\n\n\t\t// If group is not found, select the default one\n\t\tif (selectedDataItem == undefined)\n\t\t\tselectedDataItem = groupDS.get(this.DEFAULT_GROUP_NAME);\n\n\t\tthis._groupsDropDown.select(function(dataItem) {\n\t\t\treturn dataItem.name === selectedDataItem.name;\n\t\t});\n\t}\n\n\t_getGroupObj(name, label = null)\n\t{\n\t\treturn {\n\t\t\tname: name,\n\t\t\tlabel: (label != null ? label : name)\n\t\t}\n\t}\n\n\t/**\n\t * Update the Scope label in the Rooms panel and in the Users panel.\n\t */\n\t_setScopeLabel()\n\t{\n\t\tlet roomScope = '';\n\t\tlet userScope = '';\n\n\t\tif (this._selectedZone)\n\t\t{\n\t\t\tlet zoneName = this._selectedZone.name;\n\t\t\troomScope = this._getScopeLabelPart('Zone:', zoneName);\n\n\t\t\tlet selectedGroup = this._selectedGroup;\n\t\t\tlet groupName = (selectedGroup && selectedGroup.name != '') ? selectedGroup.name : this.ANY_LABEL;\n\t\t\tlet roomName = (this._selectedRoom) ? this._selectedRoom.name : this.ANY_LABEL;\n\n\t\t\tuserScope = this._getScopeLabelPart('Zone:', zoneName) + '<br>' + this._getScopeLabelPart('Group:', groupName) + '<br>' + this._getScopeLabelPart('Room:', roomName);\n\t\t}\n\n\t\t$('#znm-roomScopeLb').html(roomScope);\n\t\t$('#znm-userScopeLb').html(userScope);\n\t}\n\n\t_getScopeLabelPart(label, text)\n\t{\n\t\treturn `<span class=\"text-muted\">${label}</span> <span>${text}</span>`;\n\t}\n\n\t_resetRoomList()\n\t{\n\t\tthis._roomListBox.setDataSource([]);\n\n\t\t// Also disable Remove button\n\t\t$('#znm-removeRoomBt').attr('disabled', true);\n\t}\n\n\t_resetUserList()\n\t{\n\t\tthis._userListBox.setDataSource([]);\n\t}\n\n\t_setMainControlsEnabled()\n\t{\n\t\tlet enabled = (this._activePanelType == 'zone' && this._selectedZone != undefined)\n\t\t\t\t\t|| (this._activePanelType == 'room' && this._selectedRoom != undefined)\n\t\t\t\t\t|| (this._activePanelType == 'user' && this._selectedUser != undefined);\n\n\t\t$('#znm-mainControls').attr('disabled', !enabled);\n\n\t\t$('#znm-monitorTargetLb').text(enabled ? capitalizeFirst(this._activePanelType) : 'selection');\n\t\t$('#znm-messageIn').attr('placeholder', 'Message to ' + (enabled ? capitalizeFirst(this._activePanelType) : 'selection'));\n\t}\n\n\t_showLimitExceededWarning(icon, title)\n\t{\n\t\ticon.attr('title', title);\n\n\t\tif (title != '')\n\t\t\ticon.removeClass('hidden');\n\t\telse\n\t\t\ticon.addClass('hidden');\n\t}\n\n\t_showRoomCreationPanel(roomSettings)\n\t{\n\t\t// Build user interface based on passed data\n\t\tthis._roomCreationIBuilder.buildInterface(roomSettings, 'znm-roomCreatorTabNav', false, 'rc');\n\n\t\t// Enable scrolling tabs\n\t\t$('#znm-roomCreatorTabNav > #tabs').scrollingTabs({\n\t\t\tbootstrapVersion: 4,\n\t\t\tscrollToTabEdge: true,\n\t\t\tenableSwiping: true,\n\t\t\tdisableScrollArrowsOnFullyScrolled: true,\n\t\t\tcssClassLeftArrow: 'fa fa-chevron-left',\n\t\t\tcssClassRightArrow: 'fa fa-chevron-right'\n\t\t});\n\n\t\t// Reset Notify clients checkbox\n\t\t$('#znm-notifyClientsCB').prop('checked', true);\n\n\t\t// Set listener for custom actions triggered by configuration interface\n\t\t$('#znm-roomCreatorTabNav').on('value-set', $.proxy(this._onConfigValueSet, this));\n\n\t\t// Display panel\n\t\t$('#znm-createRoomModal').modal('show');\n\n\t\t// Reset and hide error message\n\t\tthis._resetRoomCreationError();\n\t}\n\n\t_resetRoomCreationError()\n\t{\n\t\t// Reset and hide error message\n\t\t$('#znm-createRoomError').text('');\n\t\t$('#znm-createRoomError').hide();\n\t}\n\n\t_validateNewRoomName(changes)\n\t{\n\t\tfor (let i = 0; i < changes.size(); i++)\n\t\t{\n\t\t\tconst setting = changes.getSFSObject(i);\n\n\t\t\tif (setting.containsKey('name') && setting.getUtfString('name') == 'name')\n\t\t\t{\n\t\t\t\t// Get name value\n\t\t\t\tconst name = setting.getText('value');\n\n\t\t\t\t// Get data source\n\t\t\t\tconst ds = this._roomListBox.dataSource.data();\n\n\t\t\t\t// Check if name exists in data source\n\t\t\t\tfor (let j = 0; j < ds.length; j++)\n\t\t\t\t{\n\t\t\t\t\tif (ds[j].name == name)\n\t\t\t\t\t\treturn false;\n\t\t\t\t}\n\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t_enableCreateRoomPanel(enable)\n\t{\n\t\tlet modalElement = $('#znm-createRoomModal');\n\n\t\t// Enable modal close buttons\n\t\t$('button[data-dismiss=\"modal\"]', modalElement).attr('disabled', !enable);\n\n\t\t// Enable create button\n\t\t$('#znm-roomCreatorCreateBt', modalElement).attr('disabled', !enable);\n\n\t\t// Enable checkbox\n\t\t$('#znm-notifyClientsCB', modalElement).attr('disabled', !enable);\n\n\t\t// Enable configuration interface\n\t\tthis._roomCreationIBuilder.disableInterface(!enable);\n\t}\n\n\t_clearTabs(monitoredType)\n\t{\n\t\t// Destroy scrolling tabs\n\t\t$(`#znm-${monitoredType}TabNavigator #tabs`).scrollingTabs('destroy');\n\n\t\t// Remove all tab navigator content\n\t\tthis._interfaceBuilder.destroyInterface();\n\n\t\t// Set all tab panels as inactive\n\t\t$(`#znm-${monitoredType}TabNavigator #tabPanels .tab-pane`).removeClass('show active');\n\n\t\t// Remove tab change event listener\n\t\t$('a[data-toggle=\"tab\"]').off('shown.bs.tab');\n\n\t\tthis._skipInitTabs = false;\n\t}\n\n\t_showZoneTrafficData(data)\n\t{\n\t\t// Build user count monitor history\n\t\tif (data.containsKey('history'))\n\t\t{\n\t\t\tlet userCountData = data.getIntArray('history');\n\t\t\tlet samplingRateMins = data.getInt('rate');\n\n\t\t\tthis._userCountChart.addHistoryEntries(userCountData, samplingRateMins * 60);\n\t\t}\n\n\t\t// Add last user count value\n\t\tthis._userCountChart.addEntry(data.getInt('current'));\n\t}\n\n\t_addLogEntryToGrid(grid, item)\n\t{\n\t\tlet ds = grid.dataSource;\n\n\t\t// Check items limit\n\t\tif (ds.total() == this.MAX_EXTENSION_LOG_SIZE)\n\t\t\tds.remove(ds.at(0));\n\n\t\tds.add(item);\n\t}\n\n\t_showUserStatsData(data)\n\t{\n\t\t// Packet queue filling %\n\t\tlet packetQueue = roundToDecimals(data.getFloat('packet'), 2);\n\t\tthis._packetQueueChart.setDataSource([{category: 'Queue', value: packetQueue}]);\n\n\t\t// Dropped messages\n\t\tlet droppedMsg = data.getInt('dropped');\n\t\tthis._droppedMsgChart.setDataSource([{category: 'Dropped', value: droppedMsg}]);\n\n\t\t// Written data amount\n\t\tlet writtenDataObj = scaleBytes(data.getLong('wBytes'), 2);\n\t\tthis._writtenDataChart.setDataSource([{category: 'Written', value: writtenDataObj.value, unit: writtenDataObj.unit}]);\n\n\t\t// Read data amount\n\t\tlet readDataObj = scaleBytes(data.getLong('rBytes'), 2);\n\t\tthis._readDataChart.setDataSource([{category: 'Read', value: readDataObj.value, unit: readDataObj.unit}]);\n\t}\n\n\t_setGeoLocationUI(geoLocData)\n\t{\n\t\tconst location = geoLocData.getUtfString('location');\n\t\tconst error = geoLocData.getUtfString('error');\n\t\tconst latitude = geoLocData.getUtfString('latitude');\n\t\tconst longitude = geoLocData.getUtfString('longitude');\n\n\t\t// Show location\n\t\t$('#znm-geoLocationLb').text(location);\n\n\t\t// Log error\n\t\tif (error != '')\n\t\t\tthis.shellCtrl.logMessage(error, 'warn');\n\n\t\t// Enable/disable Show map button\n\t\tif (latitude != '' && longitude != '')\n\t\t{\n\t\t\t$('#znm-showMapBt').attr('disabled', false);\n\n\t\t\tconst url = `https://www.google.com/maps/search/?api=1&query=${Number(latitude) + ',' + Number(longitude)}`;\n\t\t\t$('#znm-showMapBt').attr('href', url);\n\t\t}\n\t}\n\n\t_clearGeoLocationUI()\n\t{\n\t\t$('#znm-geoLocationLb').text('');\n\t\t$('#znm-showMapBt').attr('disabled', true);\n\t\t$('#znm-showMapBt').attr('href', '#');\n\t}\n\n\t//---------------------------------\n\t// PRIVATE GETTERS\n\t//---------------------------------\n\n\tget _activePanelType()\n\t{\n\t\treturn this._accordion.select().data('itemType');\n\t}\n\n\tget _selectedZone()\n\t{\n\t\treturn this._zoneListBox.dataItem(this._zoneListBox.select());\n\t}\n\n\tget _selectedGroup()\n\t{\n\t\treturn this._groupsDropDown.dataItem(this._groupsDropDown.select());\n\t}\n\n\tget _selectedRoom()\n\t{\n\t\treturn this._roomListBox.dataItem(this._roomListBox.select());\n\t}\n\n\tget _selectedUser()\n\t{\n\t\treturn this._userListBox.dataItem(this._userListBox.select());\n\t}\n\n\tget _isEditing()\n\t{\n\t\treturn (this._monitoredType != null && this._monitoredName != null && this._isEditMode);\n\t}\n}\n","export class RoomPropertiesData\n{\n\tstatic get LABELS() {\n\t\tlet map = new Map();\n\n\t\tmap.set(SFS2X.RoomProperties.NAME, 'name');\n\t\tmap.set(SFS2X.RoomProperties.GROUP_ID, 'groupId');\n\t\tmap.set(SFS2X.RoomProperties.MAX_USERS, 'maxUsers');\n\t\tmap.set(SFS2X.RoomProperties.MAX_SPECTATORS, 'maxSpectators');\n\t\tmap.set(SFS2X.RoomProperties.USER_COUNT, 'userCount');\n\t\tmap.set(SFS2X.RoomProperties.SPECTATOR_COUNT, 'spectatorCount');\n\t\tmap.set(SFS2X.RoomProperties.IS_GAME, 'isGame');\n\t\tmap.set(SFS2X.RoomProperties.IS_PRIVATE, 'isPrivate');\n\t\tmap.set(SFS2X.RoomProperties.HAS_FREE_PLAYER_SLOTS, 'hasFreePlayerSlots');\n\t\tmap.set(SFS2X.RoomProperties.IS_TYPE_SFSGAME, 'isSFSGameType');\n\n\t\treturn map;\n\t}\n\n\tstatic getLabel(propName)\n\t{\n\t\treturn 'Room.' + RoomPropertiesData.LABELS.get(propName);\n\t}\n\n\tstatic get propertiesArray()\n\t{\n\t\tconst arr = [];\n\n\t\tfor (const [key, value] of RoomPropertiesData.LABELS.entries())\n\t\t{\n\t\t\tarr.push({\n\t\t\t\tlabel: RoomPropertiesData.getLabel(key),\n\t\t\t\tvalue: key\n\t\t\t});\n\t\t}\n\n\t\treturn arr;\n\t}\n}\n\nexport class UserPropertiesData\n{\n\tstatic get LABELS() {\n\t\tlet map = new Map();\n\n\t\tmap.set(SFS2X.UserProperties.NAME, 'name');\n\t\tmap.set(SFS2X.UserProperties.IS_PLAYER, 'isPlayer');\n\t\tmap.set(SFS2X.UserProperties.IS_SPECTATOR, 'isSpectator');\n\t\tmap.set(SFS2X.UserProperties.IS_NPC, 'isNPC');\n\t\tmap.set(SFS2X.UserProperties.PRIVILEGE_ID, 'privilegeId');\n\n\t\treturn map;\n\t}\n\n\tstatic getLabel(propName)\n\t{\n\t\treturn 'User.' + UserPropertiesData.LABELS.get(propName);\n\t}\n\n\tstatic get propertiesArray()\n\t{\n\t\tconst arr = [];\n\n\t\tfor (const [key, value] of UserPropertiesData.LABELS.entries())\n\t\t{\n\t\t\tarr.push({\n\t\t\t\tlabel: UserPropertiesData.getLabel(key),\n\t\t\t\tvalue: key\n\t\t\t});\n\t\t}\n\n\t\treturn arr;\n\t}\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AC7GA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACnKA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACtaA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACvHA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC5/DA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;A","sourceRoot":""}
@@ -385,9 +385,6 @@ class ExtensionManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
385
385
  // Create file data source manager
386
386
  this._fileManager = new _managers_file_datasource_manager__WEBPACK_IMPORTED_MODULE_1__["FileDataSourceManager"](libFolder, [libFolder], this._fileSeparator);
387
387
 
388
- // Retrieve HTTP port to be used for files uploading
389
- const uploadHttpPort = data.getInt('httpPort');
390
-
391
388
  // Retrieve module id sent by the server (required because multiple modules use file uploading service)
392
389
  const uploadModuleId = data.getUtfString('modId');
393
390
 
@@ -395,8 +392,9 @@ class ExtensionManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
395
392
  this._uploadTargetConfig = {
396
393
  sessionToken: this.smartFox.sessionToken,
397
394
  host: this.smartFox.config.host,
398
- httpPort: uploadHttpPort,
395
+ port: this.smartFox.config.port,
399
396
  moduleId: uploadModuleId,
397
+ protocol: this.smartFox.config.useSSL ? 'https' : 'http'
400
398
  };
401
399
 
402
400
  // Request Extension files data to server instance
@@ -479,7 +477,7 @@ class ExtensionManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
479
477
  if (requester == this.smartFox.mySelf.name)
480
478
  {
481
479
  // Expand parent
482
- this._filesList.expand($(`#exm-fileList .file-controls[data-item-id="${parentPath}"]`).closest('tr'));
480
+ this._filesList.expand($(`#exm-fileList .file-controls[data-item-id="${$.escapeSelector(parentPath)}"]`).closest('tr'));
483
481
 
484
482
  if (!isUpload)
485
483
  {
@@ -487,7 +485,7 @@ class ExtensionManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
487
485
  this._addFolderModal.modal('hide');
488
486
 
489
487
  // Select upload file
490
- this._filesList.select($(`#exm-fileList .file-controls[data-item-id="${filePath}"]`).closest('tr'));
488
+ this._filesList.select($(`#exm-fileList .file-controls[data-item-id="${$.escapeSelector(filePath)}"]`).closest('tr'));
491
489
  }
492
490
 
493
491
  // Update selection
@@ -537,7 +535,7 @@ class ExtensionManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
537
535
 
538
536
  // Collapse parent if the last of its children was deleted
539
537
  if (parentItem && !parentItem.hasChildren)
540
- this._filesList.collapse($(`#exm-fileList .file-controls[data-item-id="${parentItem.id}"]`).closest('tr'));
538
+ this._filesList.collapse($(`#exm-fileList .file-controls[data-item-id="${$.escapeSelector(parentItem.id)}"]`).closest('tr'));
541
539
  }
542
540
 
543
541
  if (requester == this.smartFox.mySelf.name)
@@ -584,7 +582,7 @@ class ExtensionManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
584
582
  {
585
583
  // Hide control buttons on currently selected item
586
584
  if (this._selectedItem)
587
- $(`#exm-fileList .file-controls[data-item-id="${this._selectedItem.id}"]`).hide();
585
+ $(`#exm-fileList .file-controls[data-item-id="${$.escapeSelector(this._selectedItem.id)}"]`).hide();
588
586
 
589
587
  // Get selected item
590
588
  let selectedRows = this._filesList.select();
@@ -595,7 +593,7 @@ class ExtensionManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
595
593
  this._selectedItem = this._filesList.dataItem(selectedRows[0]);
596
594
 
597
595
  // Show control buttons on new selected item
598
- $(`#exm-fileList .file-controls[data-item-id="${this._selectedItem.id}"]`).show();
596
+ $(`#exm-fileList .file-controls[data-item-id="${$.escapeSelector(this._selectedItem.id)}"]`).show();
599
597
  }
600
598
  else
601
599
  this._selectedItem = null;
@@ -659,8 +657,8 @@ class ExtensionManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
659
657
  $('#exm-clearFilesBt').attr('disabled', true);
660
658
 
661
659
  // Set destination url
662
- const url = 'http://' + this._uploadTargetConfig.host + ':' + this._uploadTargetConfig.httpPort + '/BlueBox/SFS2XFileUpload?sessHashId=' + this._uploadTargetConfig.sessionToken;
663
-
660
+ const url = this._uploadTargetConfig.protocol + '://' + this._uploadTargetConfig.host + ':' + this._uploadTargetConfig.port + '/BlueBox/SFS2XFileUpload?sessHashId=' + this._uploadTargetConfig.sessionToken;
661
+
664
662
  e.sender.options.async.saveUrl = url;
665
663
 
666
664
  // Set payload
@@ -798,4 +796,4 @@ class ExtensionManager extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseMo
798
796
  /***/ })
799
797
 
800
798
  }]);
801
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-6.bundle.js","sources":["webpack://application/./src/managers/file-datasource-manager.js","webpack://application/./src/modules/extension-manager.js"],"sourcesContent":["export class FileDataSourceManager\n{\n\tconstructor(libFolder, protectedFolders, fileSeparator)\n\t{\n\t\tthis._protectedFolders = protectedFolders; // Folders which can't be deleted (but their content can)\n\t\tthis._libFolder = libFolder;\n\t\tthis._fileSeparator = fileSeparator;\n\t}\n\n\tget dataSource()\n\t{\n\t\treturn this._dataSource;\n\t}\n\n\tinit()\n\t{\n\t\tthis._dataSource = new kendo.data.TreeListDataSource({\n\t\t\tdata: [],\n\t\t\tschema: {\n\t\t\t\tmodel: {\n\t\t\t\t\tid: 'id',\n\t\t\t\t\tparentId: 'parentId',\n\t\t\t\t\texpanded: false\n\t\t\t\t}\n\t\t\t},\n\t\t\tsort: { field: 'name', dir: 'asc' }\n\t\t});\n\t}\n\n\taddFile(fileObj, parentLevel = null)\n\t{\n\t\tlet file = {};\n\n\t\tfile.name = fileObj.getUtfString('name');\n\t\tfile.isDir = fileObj.getBool('isDir');\n\t\tfile.lastMod = fileObj.getLong('lastMod');\n\t\tfile.isLib = (file.isDir && file.name == this._libFolder);\n\t\tfile.isProtected = (file.isDir && this._protectedFolders.indexOf(file.name) > -1);\n\t\tfile.size = 0;\n\n\t\tif (parentLevel == null)\n\t\t\tfile.level = 0;\n\t\telse\n\t\t\tfile.level = parentLevel + 1;\n\n\t\tif (fileObj.containsKey('parent'))\n\t\t{\n\t\t\tfile.parentId = fileObj.getUtfString('parent');\n\t\t\tfile.id = file.parentId + this._fileSeparator + file.name;\n\t\t}\n\t\telse\n\t\t{\n\t\t\tfile.parentId = null;\n\t\t\tfile.id = file.name;\n\t\t}\n\n\t\t// Add child files\n\t\tif (file.isDir)\n\t\t{\n\t\t\tlet filesArr = fileObj.getSFSArray('files');\n\n\t\t\tfor (let i = 0; i < filesArr.size(); i++)\n\t\t\t\tfile.size += this.addFile(filesArr.getSFSObject(i), file.level);\n\t\t}\n\t\telse\n\t\t\tfile.size = fileObj.getLong('size');\n\n\t\t// Add file to data source\n\t\tthis._dataSource.add(file);\n\n\t\t// Return file size\n\t\treturn file.size;\n\t}\n\n\tremoveFile(id)\n\t{\n\t\tlet fileItem = this._dataSource.get(id);\n\n\t\tif (fileItem)\n\t\t{\n\t\t\tif (fileItem.parentId)\n\t\t\t{\n\t\t\t\t// Subtract old size from parent size\n\t\t\t\tlet parentItem = this._dataSource.get(fileItem.parentId);\n\t\t\t\tthis._updateParentSize(parentItem, -fileItem.size);\n\t\t\t}\n\n\t\t\tthis._dataSource.remove(fileItem);\n\n\t\t\t// Return parent item\n\t\t\tif (fileItem.parentId)\n\t\t\t\treturn this._dataSource.get(fileItem.parentId);\n\t\t}\n\t}\n\n\tgetFileById(id)\n\t{\n\t\treturn this._dataSource.get(id);\n\t}\n\n\taddFileToParent(fileObj, parentId)\n\t{\n\t\tlet parentItem = this._dataSource.get(parentId);\n\n\t\tif (parentItem != null && parentItem.isDir)\n\t\t{\n\t\t\tconst fileId = parentId + this._fileSeparator + fileObj.getUtfString('name');\n\t\t\tlet fileItem = this._dataSource.get(fileId);\n\n\t\t\tif (fileItem != null)\n\t\t\t{\n\t\t\t\t// Subtract old size from parent size\n\t\t\t\tthis._updateParentSize(parentItem, -fileItem.size);\n\n\t\t\t\t// Update existing item\n\t\t\t\tfileItem.name = fileObj.getUtfString('name');\n\t\t\t\tfileItem.lastMod = fileObj.getLong('lastMod');\n\t\t\t\tfileItem.size = fileObj.getLong('size');\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Add new item\n\t\t\t\tthis.addFile(fileObj, parentItem.level);\n\t\t\t}\n\n\t\t\t// Update parent item size\n\t\t\tthis._updateParentSize(parentItem, fileObj.getLong('size'));\n\n\t\t\treturn fileId;\n\t\t}\n\t\telse\n\t\t\tthrow new Error(`An unexpected error occurred while adding file '${fileObj.getUtfString('name')}' (target: ${parentId}).`);\n\t}\n\n\t_updateParentSize(parentItem, value)\n\t{\n\t\tparentItem.size += value;\n\n\t\tif (parentItem.parentId)\n\t\t{\n\t\t\tlet grandParent = this._dataSource.get(parentItem.parentId);\n\t\t\tthis._updateParentSize(grandParent, value);\n\t\t}\n\t}\n}\n","import {BaseModule} from './base-module';\nimport {FileDataSourceManager} from '../managers/file-datasource-manager';\nimport {bytesToSize} from '../utils/utilities';\n\nexport default class ExtensionManager extends BaseModule\n{\n\tconstructor()\n\t{\n\t    super('extensionMan');\n\n\t\t// Outgoing requests\n\t\tthis.REQ_INIT = 'init';\n\t\tthis.REQ_GET_EXTENSIONS = 'getExtensions';\n\t\tthis.REQ_CREATE_FOLDER = 'createFolder';\n\t\tthis.REQ_DELETE_FILES = 'deleteExtFiles';\n\t\tthis.REQ_RELOAD_EXTENSIONS = 'reloadExt';\n\n\t\t// Incoming responses\n\t\tthis.RESP_LOCKED = 'lock';\n\t\tthis.RESP_INIT = 'init';\n\t\tthis.RESP_EXTENSIONS = 'extensions';\n\t\tthis.RESP_FILE_ADDED = 'fileAdded';\n\t\tthis.RESP_FILES_DELETED = 'filesDeleted';\n\t\tthis.RESP_ERROR = 'error';\n\t}\n\n\t//------------------------------------\n\t// COMMON MODULE INTERFACE METHODS\n\t// This members are used by the main controller\n\t// to communicate with the module's controller.\n\t//------------------------------------\n\n\tinitialize(idData, shellController)\n\t{\n\t\t// Call super method\n\t\tsuper.initialize(idData, shellController);\n\n\t\t// Initialize progress bar\n\t\t$('#exm-progressBar').kendoProgressBar({\n\t\t\tmin: 0,\n            max: 100,\n\t\t\tvalue: false,\n            type: 'value',\n            animation: {\n                duration: 400\n            }\n        });\n\n\t\t// Add listeners to buttons\n\t\t$('#exm-retryBt').on('click', $.proxy(this._onRetryClick, this));\n\t\t$('#exm-refreshBt').on('click', $.proxy(this._onRefreshClick, this));\n\n\t\t// Initialize files list\n\t\tthis._filesList = $('#exm-fileList').kendoTreeList({\n            dataSource: [],\n\t\t\tresizable: true,\n\t\t\tselectable: true,\n            columns: [\n                {\n\t\t\t\t\tfield: 'name',\n\t\t\t\t\ttitle: 'Name',\n\t\t\t\t\ttemplate: kendo.template(`\n\t\t\t\t\t\t<div >\n\t\t\t\t\t\t\t# if (isDir) { #\n\t\t\t\t\t\t\t\t# if (expanded) { #\n\t\t\t\t\t\t\t\t\t<i class=\"fas fa-folder-open\"></i>\n\t\t\t\t\t\t\t\t# } else { #\n\t\t\t\t\t\t\t\t\t<i class=\"fas fa-folder\"></i>\n\t\t\t\t\t\t\t\t# } #\n\t\t\t\t\t\t\t# } else { #\n\t\t\t\t\t\t\t\t<i class=\"far fa-file\"></i>\n\t\t\t\t\t\t\t# } #\n\n\t\t\t\t\t\t\t#: name #\n\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class=\"file-controls flex-grow-1 text-right\" data-item-id=\"#:id#\">\n\t\t\t\t\t\t\t# if (isDir) { #\n\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary my-1 create-folder-bt\"><i class=\"fas fa-folder-plus\"></i></button>\n\t\t\t\t\t\t\t\t# if (level > 0) { #\n\t\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary my-1 upload-files-bt\"><i class=\"fas fa-file-upload\"></i></button>\n\t\t\t\t\t\t\t\t# } #\n\t\t\t\t\t\t\t# } #\n\n\t\t\t\t\t\t\t# if (level > 0 && !isProtected) { #\n\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary my-1 remove-file-bt\"><i class=\"fas fa-minus-circle\"></i></button>\n\t\t\t\t\t\t\t# } #\n\n\t\t\t\t\t\t\t# if (level == 1 && !isLib) { #\n\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary my-1 reload-ext-bt\"><i class=\"fas fa-redo-alt\"></i></button>\n\t\t\t\t\t\t\t# } #\n\t\t\t\t\t\t</div>\n\t\t\t\t\t`),\n\t\t\t\t},\n                {\n\t\t\t\t\tfield: 'size',\n\t\t\t\t\ttitle: 'Size',\n\t\t\t\t\ttemplate: function(dataItem) {\n\t\t\t\t\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\t\t\t\t\treturn kendo.template(`\n\t\t\t\t\t\t\t#: bytesToSize(size, 2, 'KB') #\n\t\t\t\t\t\t`)(dataItem);\n\t\t\t\t\t},\n\t\t\t\t\twidth: 120,\n\t\t\t\t\tminScreenWidth: 576\n\t\t\t\t},\n                {\n\t\t\t\t\tfield: 'lastMod',\n\t\t\t\t\ttitle: 'Last Modified',\n\t\t\t\t\ttemplate: kendo.template(`\n\t\t\t\t\t\t#: kendo.toString(new Date(lastMod), 'dd MMM yyyy HH:mm:ss') #\n\t\t\t\t\t`),\n\t\t\t\t\twidth: 200,\n\t\t\t\t\tminScreenWidth: 768\n\t\t\t\t},\n            ],\n\t\t\tchange: $.proxy(this._onFileSelectedChange, this)\n        }).data('kendoTreeList');\n\n\t\t//-------------------------------------------\n\n\t\t// Add listeners to catch control button clicks on rows\n\t\t$('#exm-fileList').on('click', '.create-folder-bt', $.proxy(this._showAddFolderModalClick, this));\n\t\t$('#exm-fileList').on('click', '.upload-files-bt', $.proxy(this._showUploadFilesModalClick, this));\n\t\t$('#exm-fileList').on('click', '.remove-file-bt', $.proxy(this._onRemoveFileClick, this));\n\t\t$('#exm-fileList').on('click', '.reload-ext-bt', $.proxy(this._onReloadExtClick, this));\n\n\t\t//-------------------------------------------\n\n\t\t// Initialize \"add folder\" modal\n\t\tthis._addFolderModal = $('#exm-addFolderModal');\n\t\tthis._addFolderModal.modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t\tshow: false\n\t\t});\n\n\t\t// Add listener to modal hide event\n\t\tthis._addFolderModal.on('hidden.bs.modal', $.proxy(this._onAddFolderModalHidden, this));\n\n\t\t// Add listener to Add button click\n\t\t$('#exm-addFolderBt').on('click', $.proxy(this._onAddFolderClick, this));\n\n\t\t// Initialize kendo validation on folder name form\n\t\tthis._addFolderValidator = $('#exm-addFolderForm').kendoValidator({}).data('kendoValidator');\n\n\t\t//-------------------------------------------\n\n\t\t// Initialize \"upload files\" modal\n\t\tthis._uploadFilesModal = $('#exm-uploadModal');\n\t\tthis._uploadFilesModal.modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t\tshow: false\n\t\t});\n\n\t\t// Initialize kendo uploader\n\t\tthis._uploader = $('#exm-uploader').kendoUpload({\n\t\t\tmultiple: true,\n\t\t\tasync: {\n\t\t\t\tsaveUrl: 'http://localhost', // This will be changed later in _onUploadStart method\n\t\t\t\tautoUpload: true,\n\t\t\t},\n\t\t\tdirectoryDrop: true,\n\t\t\tupload: $.proxy(this._onUploadStart, this),\n\t\t\tcomplete: $.proxy(this._onUploadEnd, this),\n\t\t\tlocalization: {\n\t\t\t\tselect: 'Select files...'\n\t\t\t}\n\t\t}).data('kendoUpload');\n\n\t\t// Add listener to Upload button click\n\t\t$('#exm-clearFilesBt').on('click', $.proxy(this._onClearFilesClick, this));\n\n\t\t//-------------------------------------------\n\n\t\t// Send initialization request\n\t\tthis.sendExtensionRequest(this.REQ_INIT);\n\t}\n\n\tdestroy()\n\t{\n\t\t// Call super method\n\t\tsuper.destroy();\n\n\t\t$('#exm-retryBt').off('click');\n\t\t$('#exm-refreshBt').off('click');\n\n\t\t$('#exm-fileList').off('click');\n\n\t\tthis._addFolderModal.off('hidden.bs.modal');\n\t\tthis._addFolderModal.modal('dispose');\n\t\t$('#exm-addFolderBt').off('click');\n\n\t\tthis._uploadFilesModal.modal('dispose');\n\t\t$('#exm-clearFilesBt').off('click');\n\t}\n\n\tonExtensionCommand(command, data)\n\t{\n\t\t// Module can be enabled (no locking file exists)\n\t\tif (command == this.RESP_INIT)\n\t\t{\n\t\t\t// Retrieve file separator\n\t\t\tthis._fileSeparator = data.getUtfString('sep');\n\n\t\t\t// Retrieve Extensions' __lib__ folder name\n\t\t\tconst libFolder = data.getUtfString('lib');\n\n\t\t\t// Create file data source manager\n\t\t\tthis._fileManager = new FileDataSourceManager(libFolder, [libFolder], this._fileSeparator);\n\n\t\t\t// Retrieve HTTP port to be used for files uploading\n\t\t\tconst uploadHttpPort = data.getInt('httpPort');\n\n\t\t\t// Retrieve module id sent by the server (required because multiple modules use file uploading service)\n\t\t\tconst uploadModuleId = data.getUtfString('modId');\n\n\t\t\t// Set file uploading target configuration\n\t\t\tthis._uploadTargetConfig = {\n\t\t\t\tsessionToken: this.smartFox.sessionToken,\n\t\t\t\thost: this.smartFox.config.host,\n\t\t\t\thttpPort: uploadHttpPort,\n\t\t\t\tmoduleId: uploadModuleId,\n\t\t\t};\n\n\t\t\t// Request Extension files data to server instance\n\t\t\tthis._refreshDataList();\n\t\t}\n\n\t\t/*\n\t\t * This response is returned if the file UploadsLock.txt exists in the /config folder of the server.\n\t\t * This is an additional security measure to avoid unwanted files to be uploaded by malicius users accessing the server\n\t\t * with the default credentials, in case they have not been changed by the administrator after the installation.\n\t\t * The file must be removed manually before accessing the Extension Manager module for the first time\n\t\t */\n\t\telse if (command == this.RESP_LOCKED)\n\t\t{\n\t\t\t// Show warning\n\t\t\tthis._switchView('exm-locked');\n\t\t}\n\n\t\t// Extensions folders and files\n\t\telse if (command == this.RESP_EXTENSIONS)\n\t\t{\n\t\t\t// Retrieve Extension file list\n\t\t\tlet extensionsObj = data.getSFSObject('extensions');\n\n\t\t\t// Initialize manager\n\t\t\tthis._fileManager.init();\n\n\t\t\t// Add list to manager\n\t\t\tthis._fileManager.addFile(extensionsObj);\n\n\t\t\t// Set TreeList data source\n\t\t\tthis._filesList.setDataSource(this._fileManager.dataSource);\n\n\t\t\t// Expand first level\n\t\t\tthis._filesList.expand($('#exm-fileList tbody>tr:eq(0)'));\n\n\t\t\t// Enable interface\n\t\t\tthis._enableInterface(true);\n\n\t\t\t// Show module's main view\n\t\t\tthis._switchView('exm-main');\n\t\t}\n\n\t\t// An error occurred while managing extension files\n\t\telse if (command == this.RESP_ERROR)\n\t\t{\n\t\t\t// Hide add folder modal\n\t\t\tthis._addFolderModal.modal('hide');\n\n\t\t\t// Re-enable interface\n\t\t\tthis._enableInterface(true);\n\n\t\t\t// Show an alert\n\t\t\tthis.shellCtrl.showSimpleAlert(data.getUtfString('error'));\n\t\t}\n\n\t\t// Extension folder or file added\n\t\telse if (command == this.RESP_FILE_ADDED)\n\t\t{\n\t\t\t// Get name of the user who added the file/folder\n\t\t\tconst requester = data.getUtfString('user');\n\n\t\t\t// Get the object representing the file/folder being added\n\t\t\tconst fileObj = data.getSFSObject('file');\n\n\t\t\t// Get the target folder where the new file/folder should be added\n\t\t\tconst parentPath = data.getUtfString('parent');\n\n\t\t\t// Get the flag notifying this was a file upload\n\t\t\tconst isUpload = data.getBool('isUpload');\n\n\t\t\ttry\n\t\t\t{\n\t\t\t\t// Add/update item on data source\n\t\t\t\tconst filePath = this._fileManager.addFileToParent(fileObj, parentPath);\n\n\t\t\t\t// Refresh view\n\t\t\t\tthis._filesList.refresh();\n\n\t\t\t\tif (requester == this.smartFox.mySelf.name)\n\t\t\t\t{\n\t\t\t\t\t// Expand parent\n\t\t\t\t\tthis._filesList.expand($(`#exm-fileList .file-controls[data-item-id=\"${parentPath}\"]`).closest('tr'));\n\n\t\t\t\t\tif (!isUpload)\n\t\t\t\t\t{\n\t\t\t\t\t\t// Hide modal\n\t\t\t\t\t\tthis._addFolderModal.modal('hide');\n\n\t\t\t\t\t\t// Select upload file\n\t\t\t\t\t\tthis._filesList.select($(`#exm-fileList .file-controls[data-item-id=\"${filePath}\"]`).closest('tr'));\n\t\t\t\t\t}\n\n\t\t\t\t\t// Update selection\n\t\t\t\t\tthis._onFileSelectedChange();\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t// Display notification\n\t\t\t\t\tif (!isUpload)\n\t\t\t\t\t\tthis.shellCtrl.showNotification(`Folder created`, `Administrator ${requester} created folder: <strong>${filePath}</strong>`);\n\t\t\t\t\telse\n\t\t\t\t\t\tthis.shellCtrl.showNotification(`File uploaded`, `Administrator ${requester} uploaded file: <strong>${filePath}</strong>`);\n\t\t\t\t}\n\t\t\t}\n\t\t\tcatch (e)\n\t\t\t{\n\t\t\t\t// This should not happen... data source is corrupted?\n\t\t\t\tif (requester == this.smartFox.mySelf.name)\n\t\t\t\t\tthis.shellCtrl.showSimpleAlert(e.message, true);\n\t\t\t}\n\n\t\t\t// Enable interface\n\t\t\tthis._enableInterface(true);\n\t\t}\n\n\t\t// Extension files deleted\n\t\telse if (command == this.RESP_FILES_DELETED)\n\t\t{\n\t\t\t// Get name of the user who deleted the file/s\n\t\t\tconst requester = data.getUtfString('user');\n\n\t\t\t// Get the list of deleted files\n\t\t\tlet files = data.getSFSArray('files');\n\n\t\t\tlet filesArr = [];\n\n\t\t\t// Update data source\n\t\t\tfor (let j = 0; j < files.size(); j++)\n\t\t\t{\n\t\t\t\tlet path = files.getUtfString(j);\n\t\t\t\tfilesArr.push(path);\n\n\t\t\t\t//------------------------\n\n\t\t\t\t// Remove item from data source; parent item is returned\n\t\t\t\tlet parentItem = this._fileManager.removeFile(path);\n\n\t\t\t\t// Collapse parent if the last of its children was deleted\n\t\t\t\tif (parentItem && !parentItem.hasChildren)\n\t\t\t\t\tthis._filesList.collapse($(`#exm-fileList .file-controls[data-item-id=\"${parentItem.id}\"]`).closest('tr'));\n\t\t\t}\n\n\t\t\tif (requester == this.smartFox.mySelf.name)\n\t\t\t{\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification(`${this._selectedItem.isDir ? 'Folder' : 'File'} deleted`, `${this._selectedItem.isDir ? 'Folder' : 'File'} '${this._selectedItem.name}' deleted successfully`);\n\n\t\t\t\tthis._selectedItem = null;\n\n\t\t\t\tthis._enableInterface(true);\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification(`File deleted`, `Administrator ${requester} deleted the following file${filesArr.length > 1 ? 's' : ''}: <strong>${filesArr.join('<br> ')}</strong>`);\n\t\t\t}\n\n\t\t\t// Reset selection\n\t\t\tthis._onFileSelectedChange();\n\t\t}\n\n\t\t// else if ()\n\t}\n\n\t//---------------------------------\n\t// UI EVENT LISTENERS\n\t//---------------------------------\n\n\t_onRetryClick()\n\t{\n\t\tthis._switchView('exm-init');\n\n\t\t// Re-send initialization request\n\t\tthis.sendExtensionRequest(this.REQ_INIT);\n\t}\n\n\t_onRefreshClick()\n\t{\n\t\tthis._filesList.clearSelection();\n\t\tthis._refreshDataList();\n\t}\n\n\t_onFileSelectedChange()\n\t{\n\t\t// Hide control buttons on currently selected item\n\t\tif (this._selectedItem)\n\t\t\t$(`#exm-fileList .file-controls[data-item-id=\"${this._selectedItem.id}\"]`).hide();\n\n\t\t// Get selected item\n\t\tlet selectedRows = this._filesList.select();\n\n\t\tif (selectedRows.length > 0)\n\t\t{\n\t\t\t// Save ref. to selected item\n\t\t\tthis._selectedItem = this._filesList.dataItem(selectedRows[0]);\n\n\t\t\t// Show control buttons on new selected item\n\t\t\t$(`#exm-fileList .file-controls[data-item-id=\"${this._selectedItem.id}\"]`).show();\n\t\t}\n\t\telse\n\t\t\tthis._selectedItem = null;\n\t}\n\n\t_showAddFolderModalClick()\n\t{\n\t\tif (this._selectedItem && this._selectedItem.isDir)\n\t\t{\n\t\t\tthis._addFolderModal.modal('show');\n\t\t\t$('#exm-folderNameIn').focus();\n\t\t}\n\t}\n\n\t_onAddFolderClick()\n\t{\n\t\t// The parent folder could have been deleted while user is still typing the name of the new child folder\n\t\tif (!this._selectedItem)\n\t\t{\n\t\t\tthis._addFolderModal.modal('hide');\n\t\t\tthis.shellCtrl.showSimpleAlert('Unable to create folder; the parent folder doesn\\'t exist.');\n\t\t\treturn;\n\t\t}\n\n\t\tif (this._addFolderValidator.validate())\n\t\t{\n\t\t\t// Disable modal interface\n\t\t\tthis._enableAddFolderModal(false);\n\n\t\t\tlet data = new SFS2X.SFSObject();\n\t\t\tdata.putUtfString('folder', this._selectedItem.id + this._fileSeparator + $('#exm-folderNameIn').val());\n\n\t\t\t// Send request to server\n\t\t\tthis.sendExtensionRequest(this.REQ_CREATE_FOLDER, data);\n\t\t}\n\t}\n\n\t_onAddFolderModalHidden()\n\t{\n\t\t$('#exm-folderNameIn').val('');\n\t\tthis._resetAddFolderValidation();\n\n\t\t// Enable modal interface\n\t\tthis._enableAddFolderModal(true);\n\t}\n\n\t_showUploadFilesModalClick()\n\t{\n\t\tif (this._selectedItem)\n\t\t\tthis._uploadFilesModal.modal('show');\n\t}\n\n\t_onClearFilesClick()\n\t{\n\t\tthis._uploader.clearAllFiles();\n\t}\n\n\t_onUploadStart(e)\n\t{\n\t\t// Disable clear button\n\t\t$('#exm-clearFilesBt').attr('disabled', true);\n\n\t\t// Set destination url\n\t\tconst url = 'http://' + this._uploadTargetConfig.host + ':' + this._uploadTargetConfig.httpPort + '/BlueBox/SFS2XFileUpload?sessHashId=' + this._uploadTargetConfig.sessionToken;\n\n\t\te.sender.options.async.saveUrl = url;\n\n\t\t// Set payload\n\t\tconst params = new FormData();\n\t\tparams.append('__module', this._uploadTargetConfig.moduleId);\n\t\tparams.append('__target', this._selectedItem.id);\n\n\t\tfor (let f = 0; f < e.files.length; f++)\n\t\t\tparams.append('files[]', e.files[f].rawFile);\n\n\t\te.formData = params;\n\t}\n\n\t_onUploadEnd(e)\n\t{\n\t\t// Enable clear button\n\t\t$('#exm-clearFilesBt').attr('disabled', false);\n\t}\n\n\t_onFilesUploadEnd(response)\n\t{\n\t\t// Nothing to do: we have to wait the upload process completion to be signaled by the server through the dedicated Extension response\n\n\t\t//=================================================================\n\n\t\t// TODO Should we handle this response in some way? For some unknown reason we always get ok=false and status=0\n\t\t// console.log(response)\n\t\t// console.log(response.ok)\n\t\t// console.log(response.status)\n\t}\n\n\t_onRemoveFileClick()\n\t{\n\t\tif (this._selectedItem)\n\t\t\tthis.shellCtrl.showConfirmWarning(`Are you sure you want to delete the selected ${this._selectedItem.isDir ? 'folder' : 'file'}?<br><br>Path: <strong>${this._selectedItem.id}</strong>`, $.proxy(this._onRemoveFileConfirm, this));\n\t}\n\n\t_onRemoveFileConfirm()\n\t{\n\t\t// Disable interface\n\t\tthis._enableInterface(false);\n\n\t\t// Request Extension files removal\n\t\t// NOTE: for compatibility with older AdminTool, the file to be deleted is sent\n\t\t// in an array of strings, even if we can't delete more than 1 file at once in this AdminTool\n\n\t\tlet files = new SFS2X.SFSArray();\n\t\tfiles.addUtfString(this._selectedItem.id);\n\n\t\tlet params = new SFS2X.SFSObject();\n\t\tparams.putSFSArray('files', files);\n\n\t\tthis.sendExtensionRequest(this.REQ_DELETE_FILES, params);\n\t}\n\n\t_onReloadExtClick()\n\t{\n\t\tif (this._selectedItem)\n\t\t{\n\t\t\tlet pathArr = this._selectedItem.id.split(this._fileSeparator);\n\n\t\t\tif (pathArr.length > 1)\n\t\t\t{\n\t\t\t\t// Request Extension reload\n\t\t\t\t// NOTE: for compatibility with older AdminTool, the Extension to be reloaded is sent\n\t\t\t\t// in an array of strings, even if we can't reload more than 1 Extension at once in this AdminTool\n\n\t\t\t\tlet extToReload = [];\n\t\t\t\textToReload.push(pathArr[1]);\n\n\t\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\t\tparams.putUtfStringArray('extensions', extToReload);\n\n\t\t\t\t// Send request to server\n\t\t\t\tthis.sendExtensionRequest(this.REQ_RELOAD_EXTENSIONS, params);\n\t\t\t}\n\t\t}\n\t}\n\n\t//------------------------------------\n\t// PRIVATE METHODS\n\t//------------------------------------\n\n\t_switchView(viewId)\n\t{\n\t\tdocument.getElementById('exm-viewstack').selectedElement = document.getElementById(viewId);\n\t}\n\n\t_enableInterface(enable)\n\t{\n\t\t$('#exm-fileList').attr('disabled', !enable);\n\t\t$('#exm-refreshBt').attr('disabled', !enable);\n\t}\n\n\t_refreshDataList()\n\t{\n\t\t// Disable interface\n\t\tthis._enableInterface(false);\n\n\t\t// Send request to server\n\t\tthis.sendExtensionRequest(this.REQ_GET_EXTENSIONS)\n\t}\n\n\t_resetAddFolderValidation()\n\t{\n\t\tthis._addFolderValidator.hideMessages();\n\n\t\t// The method above doesn't remove the k-invalid classes and aria-invalid=\"true\" attributes from inputs\n\t\t// Let's do it manually\n\t\t$('#exm-addFolderForm .k-invalid').removeClass('k-invalid');\n\t\t$('#exm-addFolderForm [aria-invalid=\"true\"]').removeAttr('aria-invalid');\n\t}\n\n\t_enableAddFolderModal(enable)\n\t{\n\t\t// Disable modal close buttons\n\t\t$('#exm-addFolderModal button[data-dismiss=\"modal\"]').attr('disabled', !enable);\n\n\t\t// Disable add button\n\t\t$('#exm-addFolderBt').attr('disabled', !enable);\n\n\t\t// Disable fieldset\n\t\t$('#exm-addFolderForm').attr('disabled', !enable);\n\t}\n\n\t//---------------------------------\n\t// PRIVATE GETTERS\n\t//---------------------------------\n\n\n}\n"],"mappings":";;;;;;;;;;;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AChJA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;A","sourceRoot":""}
799
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-6.bundle.js","sources":["webpack://application/./src/managers/file-datasource-manager.js","webpack://application/./src/modules/extension-manager.js"],"sourcesContent":["export class FileDataSourceManager\n{\n\tconstructor(libFolder, protectedFolders, fileSeparator)\n\t{\n\t\tthis._protectedFolders = protectedFolders; // Folders which can't be deleted (but their content can)\n\t\tthis._libFolder = libFolder;\n\t\tthis._fileSeparator = fileSeparator;\n\t}\n\n\tget dataSource()\n\t{\n\t\treturn this._dataSource;\n\t}\n\n\tinit()\n\t{\n\t\tthis._dataSource = new kendo.data.TreeListDataSource({\n\t\t\tdata: [],\n\t\t\tschema: {\n\t\t\t\tmodel: {\n\t\t\t\t\tid: 'id',\n\t\t\t\t\tparentId: 'parentId',\n\t\t\t\t\texpanded: false\n\t\t\t\t}\n\t\t\t},\n\t\t\tsort: { field: 'name', dir: 'asc' }\n\t\t});\n\t}\n\n\taddFile(fileObj, parentLevel = null)\n\t{\n\t\tlet file = {};\n\n\t\tfile.name = fileObj.getUtfString('name');\n\t\tfile.isDir = fileObj.getBool('isDir');\n\t\tfile.lastMod = fileObj.getLong('lastMod');\n\t\tfile.isLib = (file.isDir && file.name == this._libFolder);\n\t\tfile.isProtected = (file.isDir && this._protectedFolders.indexOf(file.name) > -1);\n\t\tfile.size = 0;\n\n\t\tif (parentLevel == null)\n\t\t\tfile.level = 0;\n\t\telse\n\t\t\tfile.level = parentLevel + 1;\n\n\t\tif (fileObj.containsKey('parent'))\n\t\t{\n\t\t\tfile.parentId = fileObj.getUtfString('parent');\n\t\t\tfile.id = file.parentId + this._fileSeparator + file.name;\n\t\t}\n\t\telse\n\t\t{\n\t\t\tfile.parentId = null;\n\t\t\tfile.id = file.name;\n\t\t}\n\n\t\t// Add child files\n\t\tif (file.isDir)\n\t\t{\n\t\t\tlet filesArr = fileObj.getSFSArray('files');\n\n\t\t\tfor (let i = 0; i < filesArr.size(); i++)\n\t\t\t\tfile.size += this.addFile(filesArr.getSFSObject(i), file.level);\n\t\t}\n\t\telse\n\t\t\tfile.size = fileObj.getLong('size');\n\n\t\t// Add file to data source\n\t\tthis._dataSource.add(file);\n\n\t\t// Return file size\n\t\treturn file.size;\n\t}\n\n\tremoveFile(id)\n\t{\n\t\tlet fileItem = this._dataSource.get(id);\n\n\t\tif (fileItem)\n\t\t{\n\t\t\tif (fileItem.parentId)\n\t\t\t{\n\t\t\t\t// Subtract old size from parent size\n\t\t\t\tlet parentItem = this._dataSource.get(fileItem.parentId);\n\t\t\t\tthis._updateParentSize(parentItem, -fileItem.size);\n\t\t\t}\n\n\t\t\tthis._dataSource.remove(fileItem);\n\n\t\t\t// Return parent item\n\t\t\tif (fileItem.parentId)\n\t\t\t\treturn this._dataSource.get(fileItem.parentId);\n\t\t}\n\t}\n\n\tgetFileById(id)\n\t{\n\t\treturn this._dataSource.get(id);\n\t}\n\n\taddFileToParent(fileObj, parentId)\n\t{\n\t\tlet parentItem = this._dataSource.get(parentId);\n\n\t\tif (parentItem != null && parentItem.isDir)\n\t\t{\n\t\t\tconst fileId = parentId + this._fileSeparator + fileObj.getUtfString('name');\n\t\t\tlet fileItem = this._dataSource.get(fileId);\n\n\t\t\tif (fileItem != null)\n\t\t\t{\n\t\t\t\t// Subtract old size from parent size\n\t\t\t\tthis._updateParentSize(parentItem, -fileItem.size);\n\n\t\t\t\t// Update existing item\n\t\t\t\tfileItem.name = fileObj.getUtfString('name');\n\t\t\t\tfileItem.lastMod = fileObj.getLong('lastMod');\n\t\t\t\tfileItem.size = fileObj.getLong('size');\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Add new item\n\t\t\t\tthis.addFile(fileObj, parentItem.level);\n\t\t\t}\n\n\t\t\t// Update parent item size\n\t\t\tthis._updateParentSize(parentItem, fileObj.getLong('size'));\n\n\t\t\treturn fileId;\n\t\t}\n\t\telse\n\t\t\tthrow new Error(`An unexpected error occurred while adding file '${fileObj.getUtfString('name')}' (target: ${parentId}).`);\n\t}\n\n\t_updateParentSize(parentItem, value)\n\t{\n\t\tparentItem.size += value;\n\n\t\tif (parentItem.parentId)\n\t\t{\n\t\t\tlet grandParent = this._dataSource.get(parentItem.parentId);\n\t\t\tthis._updateParentSize(grandParent, value);\n\t\t}\n\t}\n}\n","import {BaseModule} from './base-module';\nimport {FileDataSourceManager} from '../managers/file-datasource-manager';\nimport {bytesToSize} from '../utils/utilities';\n\nexport default class ExtensionManager extends BaseModule\n{\n\tconstructor()\n\t{\n\t    super('extensionMan');\n\n\t\t// Outgoing requests\n\t\tthis.REQ_INIT = 'init';\n\t\tthis.REQ_GET_EXTENSIONS = 'getExtensions';\n\t\tthis.REQ_CREATE_FOLDER = 'createFolder';\n\t\tthis.REQ_DELETE_FILES = 'deleteExtFiles';\n\t\tthis.REQ_RELOAD_EXTENSIONS = 'reloadExt';\n\n\t\t// Incoming responses\n\t\tthis.RESP_LOCKED = 'lock';\n\t\tthis.RESP_INIT = 'init';\n\t\tthis.RESP_EXTENSIONS = 'extensions';\n\t\tthis.RESP_FILE_ADDED = 'fileAdded';\n\t\tthis.RESP_FILES_DELETED = 'filesDeleted';\n\t\tthis.RESP_ERROR = 'error';\n\t}\n\n\t//------------------------------------\n\t// COMMON MODULE INTERFACE METHODS\n\t// This members are used by the main controller\n\t// to communicate with the module's controller.\n\t//------------------------------------\n\n\tinitialize(idData, shellController)\n\t{\n\t\t// Call super method\n\t\tsuper.initialize(idData, shellController);\n\n\t\t// Initialize progress bar\n\t\t$('#exm-progressBar').kendoProgressBar({\n\t\t\tmin: 0,\n            max: 100,\n\t\t\tvalue: false,\n            type: 'value',\n            animation: {\n                duration: 400\n            }\n        });\n\n\t\t// Add listeners to buttons\n\t\t$('#exm-retryBt').on('click', $.proxy(this._onRetryClick, this));\n\t\t$('#exm-refreshBt').on('click', $.proxy(this._onRefreshClick, this));\n\n\t\t// Initialize files list\n\t\tthis._filesList = $('#exm-fileList').kendoTreeList({\n            dataSource: [],\n\t\t\tresizable: true,\n\t\t\tselectable: true,\n            columns: [\n                {\n\t\t\t\t\tfield: 'name',\n\t\t\t\t\ttitle: 'Name',\n\t\t\t\t\ttemplate: kendo.template(`\n\t\t\t\t\t\t<div >\n\t\t\t\t\t\t\t# if (isDir) { #\n\t\t\t\t\t\t\t\t# if (expanded) { #\n\t\t\t\t\t\t\t\t\t<i class=\"fas fa-folder-open\"></i>\n\t\t\t\t\t\t\t\t# } else { #\n\t\t\t\t\t\t\t\t\t<i class=\"fas fa-folder\"></i>\n\t\t\t\t\t\t\t\t# } #\n\t\t\t\t\t\t\t# } else { #\n\t\t\t\t\t\t\t\t<i class=\"far fa-file\"></i>\n\t\t\t\t\t\t\t# } #\n\n\t\t\t\t\t\t\t#: name #\n\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class=\"file-controls flex-grow-1 text-right\" data-item-id=\"#:id#\">\n\t\t\t\t\t\t\t# if (isDir) { #\n\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary my-1 create-folder-bt\"><i class=\"fas fa-folder-plus\"></i></button>\n\t\t\t\t\t\t\t\t# if (level > 0) { #\n\t\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary my-1 upload-files-bt\"><i class=\"fas fa-file-upload\"></i></button>\n\t\t\t\t\t\t\t\t# } #\n\t\t\t\t\t\t\t# } #\n\n\t\t\t\t\t\t\t# if (level > 0 && !isProtected) { #\n\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary my-1 remove-file-bt\"><i class=\"fas fa-minus-circle\"></i></button>\n\t\t\t\t\t\t\t# } #\n\n\t\t\t\t\t\t\t# if (level == 1 && !isLib) { #\n\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-primary my-1 reload-ext-bt\"><i class=\"fas fa-redo-alt\"></i></button>\n\t\t\t\t\t\t\t# } #\n\t\t\t\t\t\t</div>\n\t\t\t\t\t`),\n\t\t\t\t},\n                {\n\t\t\t\t\tfield: 'size',\n\t\t\t\t\ttitle: 'Size',\n\t\t\t\t\ttemplate: function(dataItem) {\n\t\t\t\t\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\t\t\t\t\treturn kendo.template(`\n\t\t\t\t\t\t\t#: bytesToSize(size, 2, 'KB') #\n\t\t\t\t\t\t`)(dataItem);\n\t\t\t\t\t},\n\t\t\t\t\twidth: 120,\n\t\t\t\t\tminScreenWidth: 576\n\t\t\t\t},\n                {\n\t\t\t\t\tfield: 'lastMod',\n\t\t\t\t\ttitle: 'Last Modified',\n\t\t\t\t\ttemplate: kendo.template(`\n\t\t\t\t\t\t#: kendo.toString(new Date(lastMod), 'dd MMM yyyy HH:mm:ss') #\n\t\t\t\t\t`),\n\t\t\t\t\twidth: 200,\n\t\t\t\t\tminScreenWidth: 768\n\t\t\t\t},\n            ],\n\t\t\tchange: $.proxy(this._onFileSelectedChange, this)\n        }).data('kendoTreeList');\n\n\t\t//-------------------------------------------\n\n\t\t// Add listeners to catch control button clicks on rows\n\t\t$('#exm-fileList').on('click', '.create-folder-bt', $.proxy(this._showAddFolderModalClick, this));\n\t\t$('#exm-fileList').on('click', '.upload-files-bt', $.proxy(this._showUploadFilesModalClick, this));\n\t\t$('#exm-fileList').on('click', '.remove-file-bt', $.proxy(this._onRemoveFileClick, this));\n\t\t$('#exm-fileList').on('click', '.reload-ext-bt', $.proxy(this._onReloadExtClick, this));\n\n\t\t//-------------------------------------------\n\n\t\t// Initialize \"add folder\" modal\n\t\tthis._addFolderModal = $('#exm-addFolderModal');\n\t\tthis._addFolderModal.modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t\tshow: false\n\t\t});\n\n\t\t// Add listener to modal hide event\n\t\tthis._addFolderModal.on('hidden.bs.modal', $.proxy(this._onAddFolderModalHidden, this));\n\n\t\t// Add listener to Add button click\n\t\t$('#exm-addFolderBt').on('click', $.proxy(this._onAddFolderClick, this));\n\n\t\t// Initialize kendo validation on folder name form\n\t\tthis._addFolderValidator = $('#exm-addFolderForm').kendoValidator({}).data('kendoValidator');\n\n\t\t//-------------------------------------------\n\n\t\t// Initialize \"upload files\" modal\n\t\tthis._uploadFilesModal = $('#exm-uploadModal');\n\t\tthis._uploadFilesModal.modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t\tshow: false\n\t\t});\n\n\t\t// Initialize kendo uploader\n\t\tthis._uploader = $('#exm-uploader').kendoUpload({\n\t\t\tmultiple: true,\n\t\t\tasync: {\n\t\t\t\tsaveUrl: 'http://localhost', // This will be changed later in _onUploadStart method\n\t\t\t\tautoUpload: true,\n\t\t\t},\n\t\t\tdirectoryDrop: true,\n\t\t\tupload: $.proxy(this._onUploadStart, this),\n\t\t\tcomplete: $.proxy(this._onUploadEnd, this),\n\t\t\tlocalization: {\n\t\t\t\tselect: 'Select files...'\n\t\t\t}\n\t\t}).data('kendoUpload');\n\n\t\t// Add listener to Upload button click\n\t\t$('#exm-clearFilesBt').on('click', $.proxy(this._onClearFilesClick, this));\n\n\t\t//-------------------------------------------\n\n\t\t// Send initialization request\n\t\tthis.sendExtensionRequest(this.REQ_INIT);\n\t}\n\n\tdestroy()\n\t{\n\t\t// Call super method\n\t\tsuper.destroy();\n\n\t\t$('#exm-retryBt').off('click');\n\t\t$('#exm-refreshBt').off('click');\n\n\t\t$('#exm-fileList').off('click');\n\n\t\tthis._addFolderModal.off('hidden.bs.modal');\n\t\tthis._addFolderModal.modal('dispose');\n\t\t$('#exm-addFolderBt').off('click');\n\n\t\tthis._uploadFilesModal.modal('dispose');\n\t\t$('#exm-clearFilesBt').off('click');\n\t}\n\n\tonExtensionCommand(command, data)\n\t{\n\t\t// Module can be enabled (no locking file exists)\n\t\tif (command == this.RESP_INIT)\n\t\t{\n\t\t\t// Retrieve file separator\n\t\t\tthis._fileSeparator = data.getUtfString('sep');\n\n\t\t\t// Retrieve Extensions' __lib__ folder name\n\t\t\tconst libFolder = data.getUtfString('lib');\n\n\t\t\t// Create file data source manager\n\t\t\tthis._fileManager = new FileDataSourceManager(libFolder, [libFolder], this._fileSeparator);\n\n\t\t\t// Retrieve module id sent by the server (required because multiple modules use file uploading service)\n\t\t\tconst uploadModuleId = data.getUtfString('modId');\n\n\t\t\t// Set file uploading target configuration\n\t\t\tthis._uploadTargetConfig = {\n\t\t\t\tsessionToken: this.smartFox.sessionToken,\n\t\t\t\thost: this.smartFox.config.host,\n\t\t\t\tport: this.smartFox.config.port,\n\t\t\t\tmoduleId: uploadModuleId,\n\t\t\t\tprotocol: this.smartFox.config.useSSL ? 'https' : 'http'\n\t\t\t};\n\n\t\t\t// Request Extension files data to server instance\n\t\t\tthis._refreshDataList();\n\t\t}\n\n\t\t/*\n\t\t * This response is returned if the file UploadsLock.txt exists in the /config folder of the server.\n\t\t * This is an additional security measure to avoid unwanted files to be uploaded by malicius users accessing the server\n\t\t * with the default credentials, in case they have not been changed by the administrator after the installation.\n\t\t * The file must be removed manually before accessing the Extension Manager module for the first time\n\t\t */\n\t\telse if (command == this.RESP_LOCKED)\n\t\t{\n\t\t\t// Show warning\n\t\t\tthis._switchView('exm-locked');\n\t\t}\n\n\t\t// Extensions folders and files\n\t\telse if (command == this.RESP_EXTENSIONS)\n\t\t{\n\t\t\t// Retrieve Extension file list\n\t\t\tlet extensionsObj = data.getSFSObject('extensions');\n\n\t\t\t// Initialize manager\n\t\t\tthis._fileManager.init();\n\n\t\t\t// Add list to manager\n\t\t\tthis._fileManager.addFile(extensionsObj);\n\n\t\t\t// Set TreeList data source\n\t\t\tthis._filesList.setDataSource(this._fileManager.dataSource);\n\n\t\t\t// Expand first level\n\t\t\tthis._filesList.expand($('#exm-fileList tbody>tr:eq(0)'));\n\n\t\t\t// Enable interface\n\t\t\tthis._enableInterface(true);\n\n\t\t\t// Show module's main view\n\t\t\tthis._switchView('exm-main');\n\t\t}\n\n\t\t// An error occurred while managing extension files\n\t\telse if (command == this.RESP_ERROR)\n\t\t{\n\t\t\t// Hide add folder modal\n\t\t\tthis._addFolderModal.modal('hide');\n\n\t\t\t// Re-enable interface\n\t\t\tthis._enableInterface(true);\n\n\t\t\t// Show an alert\n\t\t\tthis.shellCtrl.showSimpleAlert(data.getUtfString('error'));\n\t\t}\n\n\t\t// Extension folder or file added\n\t\telse if (command == this.RESP_FILE_ADDED)\n\t\t{\n\t\t\t// Get name of the user who added the file/folder\n\t\t\tconst requester = data.getUtfString('user');\n\n\t\t\t// Get the object representing the file/folder being added\n\t\t\tconst fileObj = data.getSFSObject('file');\n\n\t\t\t// Get the target folder where the new file/folder should be added\n\t\t\tconst parentPath = data.getUtfString('parent');\n\n\t\t\t// Get the flag notifying this was a file upload\n\t\t\tconst isUpload = data.getBool('isUpload');\n\n\t\t\ttry\n\t\t\t{\n\t\t\t\t// Add/update item on data source\n\t\t\t\tconst filePath = this._fileManager.addFileToParent(fileObj, parentPath);\n\n\t\t\t\t// Refresh view\n\t\t\t\tthis._filesList.refresh();\n\n\t\t\t\tif (requester == this.smartFox.mySelf.name)\n\t\t\t\t{\n\t\t\t\t\t// Expand parent\n\t\t\t\t\tthis._filesList.expand($(`#exm-fileList .file-controls[data-item-id=\"${$.escapeSelector(parentPath)}\"]`).closest('tr'));\n\n\t\t\t\t\tif (!isUpload)\n\t\t\t\t\t{\n\t\t\t\t\t\t// Hide modal\n\t\t\t\t\t\tthis._addFolderModal.modal('hide');\n\n\t\t\t\t\t\t// Select upload file\n\t\t\t\t\t\tthis._filesList.select($(`#exm-fileList .file-controls[data-item-id=\"${$.escapeSelector(filePath)}\"]`).closest('tr'));\n\t\t\t\t\t}\n\n\t\t\t\t\t// Update selection\n\t\t\t\t\tthis._onFileSelectedChange();\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t// Display notification\n\t\t\t\t\tif (!isUpload)\n\t\t\t\t\t\tthis.shellCtrl.showNotification(`Folder created`, `Administrator ${requester} created folder: <strong>${filePath}</strong>`);\n\t\t\t\t\telse\n\t\t\t\t\t\tthis.shellCtrl.showNotification(`File uploaded`, `Administrator ${requester} uploaded file: <strong>${filePath}</strong>`);\n\t\t\t\t}\n\t\t\t}\n\t\t\tcatch (e)\n\t\t\t{\n\t\t\t\t// This should not happen... data source is corrupted?\n\t\t\t\tif (requester == this.smartFox.mySelf.name)\n\t\t\t\t\tthis.shellCtrl.showSimpleAlert(e.message, true);\n\t\t\t}\n\n\t\t\t// Enable interface\n\t\t\tthis._enableInterface(true);\n\t\t}\n\n\t\t// Extension files deleted\n\t\telse if (command == this.RESP_FILES_DELETED)\n\t\t{\n\t\t\t// Get name of the user who deleted the file/s\n\t\t\tconst requester = data.getUtfString('user');\n\n\t\t\t// Get the list of deleted files\n\t\t\tlet files = data.getSFSArray('files');\n\n\t\t\tlet filesArr = [];\n\n\t\t\t// Update data source\n\t\t\tfor (let j = 0; j < files.size(); j++)\n\t\t\t{\n\t\t\t\tlet path = files.getUtfString(j);\n\t\t\t\tfilesArr.push(path);\n\n\t\t\t\t//------------------------\n\n\t\t\t\t// Remove item from data source; parent item is returned\n\t\t\t\tlet parentItem = this._fileManager.removeFile(path);\n\n\t\t\t\t// Collapse parent if the last of its children was deleted\n\t\t\t\tif (parentItem && !parentItem.hasChildren)\n\t\t\t\t\tthis._filesList.collapse($(`#exm-fileList .file-controls[data-item-id=\"${$.escapeSelector(parentItem.id)}\"]`).closest('tr'));\n\t\t\t}\n\n\t\t\tif (requester == this.smartFox.mySelf.name)\n\t\t\t{\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification(`${this._selectedItem.isDir ? 'Folder' : 'File'} deleted`, `${this._selectedItem.isDir ? 'Folder' : 'File'} '${this._selectedItem.name}' deleted successfully`);\n\n\t\t\t\tthis._selectedItem = null;\n\n\t\t\t\tthis._enableInterface(true);\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification(`File deleted`, `Administrator ${requester} deleted the following file${filesArr.length > 1 ? 's' : ''}: <strong>${filesArr.join('<br> ')}</strong>`);\n\t\t\t}\n\n\t\t\t// Reset selection\n\t\t\tthis._onFileSelectedChange();\n\t\t}\n\n\t\t// else if ()\n\t}\n\n\t//---------------------------------\n\t// UI EVENT LISTENERS\n\t//---------------------------------\n\n\t_onRetryClick()\n\t{\n\t\tthis._switchView('exm-init');\n\n\t\t// Re-send initialization request\n\t\tthis.sendExtensionRequest(this.REQ_INIT);\n\t}\n\n\t_onRefreshClick()\n\t{\n\t\tthis._filesList.clearSelection();\n\t\tthis._refreshDataList();\n\t}\n\n\t_onFileSelectedChange()\n\t{\n\t\t// Hide control buttons on currently selected item\n\t\tif (this._selectedItem)\n\t\t\t$(`#exm-fileList .file-controls[data-item-id=\"${$.escapeSelector(this._selectedItem.id)}\"]`).hide();\n\n\t\t// Get selected item\n\t\tlet selectedRows = this._filesList.select();\n\n\t\tif (selectedRows.length > 0)\n\t\t{\n\t\t\t// Save ref. to selected item\n\t\t\tthis._selectedItem = this._filesList.dataItem(selectedRows[0]);\n\n\t\t\t// Show control buttons on new selected item\n\t\t\t$(`#exm-fileList .file-controls[data-item-id=\"${$.escapeSelector(this._selectedItem.id)}\"]`).show();\n\t\t}\n\t\telse\n\t\t\tthis._selectedItem = null;\n\t}\n\n\t_showAddFolderModalClick()\n\t{\n\t\tif (this._selectedItem && this._selectedItem.isDir)\n\t\t{\n\t\t\tthis._addFolderModal.modal('show');\n\t\t\t$('#exm-folderNameIn').focus();\n\t\t}\n\t}\n\n\t_onAddFolderClick()\n\t{\n\t\t// The parent folder could have been deleted while user is still typing the name of the new child folder\n\t\tif (!this._selectedItem)\n\t\t{\n\t\t\tthis._addFolderModal.modal('hide');\n\t\t\tthis.shellCtrl.showSimpleAlert('Unable to create folder; the parent folder doesn\\'t exist.');\n\t\t\treturn;\n\t\t}\n\n\t\tif (this._addFolderValidator.validate())\n\t\t{\n\t\t\t// Disable modal interface\n\t\t\tthis._enableAddFolderModal(false);\n\n\t\t\tlet data = new SFS2X.SFSObject();\n\t\t\tdata.putUtfString('folder', this._selectedItem.id + this._fileSeparator + $('#exm-folderNameIn').val());\n\n\t\t\t// Send request to server\n\t\t\tthis.sendExtensionRequest(this.REQ_CREATE_FOLDER, data);\n\t\t}\n\t}\n\n\t_onAddFolderModalHidden()\n\t{\n\t\t$('#exm-folderNameIn').val('');\n\t\tthis._resetAddFolderValidation();\n\n\t\t// Enable modal interface\n\t\tthis._enableAddFolderModal(true);\n\t}\n\n\t_showUploadFilesModalClick()\n\t{\n\t\tif (this._selectedItem)\n\t\t\tthis._uploadFilesModal.modal('show');\n\t}\n\n\t_onClearFilesClick()\n\t{\n\t\tthis._uploader.clearAllFiles();\n\t}\n\n\t_onUploadStart(e)\n\t{\n\t\t// Disable clear button\n\t\t$('#exm-clearFilesBt').attr('disabled', true);\n\n\t\t// Set destination url\n\t\tconst url = this._uploadTargetConfig.protocol + '://' + this._uploadTargetConfig.host + ':' + this._uploadTargetConfig.port + '/BlueBox/SFS2XFileUpload?sessHashId=' + this._uploadTargetConfig.sessionToken;\n\t\t\n\t\te.sender.options.async.saveUrl = url;\n\n\t\t// Set payload\n\t\tconst params = new FormData();\n\t\tparams.append('__module', this._uploadTargetConfig.moduleId);\n\t\tparams.append('__target', this._selectedItem.id);\n\n\t\tfor (let f = 0; f < e.files.length; f++)\n\t\t\tparams.append('files[]', e.files[f].rawFile);\n\n\t\te.formData = params;\n\t}\n\n\t_onUploadEnd(e)\n\t{\n\t\t// Enable clear button\n\t\t$('#exm-clearFilesBt').attr('disabled', false);\n\t}\n\n\t_onFilesUploadEnd(response)\n\t{\n\t\t// Nothing to do: we have to wait the upload process completion to be signaled by the server through the dedicated Extension response\n\n\t\t//=================================================================\n\n\t\t// TODO Should we handle this response in some way? For some unknown reason we always get ok=false and status=0\n\t\t// console.log(response)\n\t\t// console.log(response.ok)\n\t\t// console.log(response.status)\n\t}\n\n\t_onRemoveFileClick()\n\t{\n\t\tif (this._selectedItem)\n\t\t\tthis.shellCtrl.showConfirmWarning(`Are you sure you want to delete the selected ${this._selectedItem.isDir ? 'folder' : 'file'}?<br><br>Path: <strong>${this._selectedItem.id}</strong>`, $.proxy(this._onRemoveFileConfirm, this));\n\t}\n\n\t_onRemoveFileConfirm()\n\t{\n\t\t// Disable interface\n\t\tthis._enableInterface(false);\n\n\t\t// Request Extension files removal\n\t\t// NOTE: for compatibility with older AdminTool, the file to be deleted is sent\n\t\t// in an array of strings, even if we can't delete more than 1 file at once in this AdminTool\n\n\t\tlet files = new SFS2X.SFSArray();\n\t\tfiles.addUtfString(this._selectedItem.id);\n\n\t\tlet params = new SFS2X.SFSObject();\n\t\tparams.putSFSArray('files', files);\n\n\t\tthis.sendExtensionRequest(this.REQ_DELETE_FILES, params);\n\t}\n\n\t_onReloadExtClick()\n\t{\n\t\tif (this._selectedItem)\n\t\t{\n\t\t\tlet pathArr = this._selectedItem.id.split(this._fileSeparator);\n\n\t\t\tif (pathArr.length > 1)\n\t\t\t{\n\t\t\t\t// Request Extension reload\n\t\t\t\t// NOTE: for compatibility with older AdminTool, the Extension to be reloaded is sent\n\t\t\t\t// in an array of strings, even if we can't reload more than 1 Extension at once in this AdminTool\n\n\t\t\t\tlet extToReload = [];\n\t\t\t\textToReload.push(pathArr[1]);\n\n\t\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\t\tparams.putUtfStringArray('extensions', extToReload);\n\n\t\t\t\t// Send request to server\n\t\t\t\tthis.sendExtensionRequest(this.REQ_RELOAD_EXTENSIONS, params);\n\t\t\t}\n\t\t}\n\t}\n\n\t//------------------------------------\n\t// PRIVATE METHODS\n\t//------------------------------------\n\n\t_switchView(viewId)\n\t{\n\t\tdocument.getElementById('exm-viewstack').selectedElement = document.getElementById(viewId);\n\t}\n\n\t_enableInterface(enable)\n\t{\n\t\t$('#exm-fileList').attr('disabled', !enable);\n\t\t$('#exm-refreshBt').attr('disabled', !enable);\n\t}\n\n\t_refreshDataList()\n\t{\n\t\t// Disable interface\n\t\tthis._enableInterface(false);\n\n\t\t// Send request to server\n\t\tthis.sendExtensionRequest(this.REQ_GET_EXTENSIONS)\n\t}\n\n\t_resetAddFolderValidation()\n\t{\n\t\tthis._addFolderValidator.hideMessages();\n\n\t\t// The method above doesn't remove the k-invalid classes and aria-invalid=\"true\" attributes from inputs\n\t\t// Let's do it manually\n\t\t$('#exm-addFolderForm .k-invalid').removeClass('k-invalid');\n\t\t$('#exm-addFolderForm [aria-invalid=\"true\"]').removeAttr('aria-invalid');\n\t}\n\n\t_enableAddFolderModal(enable)\n\t{\n\t\t// Disable modal close buttons\n\t\t$('#exm-addFolderModal button[data-dismiss=\"modal\"]').attr('disabled', !enable);\n\n\t\t// Disable add button\n\t\t$('#exm-addFolderBt').attr('disabled', !enable);\n\n\t\t// Disable fieldset\n\t\t$('#exm-addFolderForm').attr('disabled', !enable);\n\t}\n\n\t//---------------------------------\n\t// PRIVATE GETTERS\n\t//---------------------------------\n\n\n}\n"],"mappings":";;;;;;;;;;;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AChJA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;A","sourceRoot":""}
@@ -355,7 +355,7 @@ class SslCertificateManager extends HTMLElement
355
355
  params.append('__module', this._uploadTargetConfig.moduleId);
356
356
 
357
357
  // Set destination url
358
- const url = 'http://' + this._uploadTargetConfig.host + ':' + this._uploadTargetConfig.httpPort + '/BlueBox/SFS2XFileUpload?sessHashId=' + this._uploadTargetConfig.sessionToken;
358
+ const url = this._uploadTargetConfig.protocol + '://' + this._uploadTargetConfig.host + ':' + this._uploadTargetConfig.port + '/BlueBox/SFS2XFileUpload?sessHashId=' + this._uploadTargetConfig.sessionToken;
359
359
 
360
360
  // Start upload
361
361
  fetch(url, {
@@ -590,9 +590,6 @@ class ServerConfigurator extends _base_module__WEBPACK_IMPORTED_MODULE_0__["Base
590
590
  // Initialization data received
591
591
  if (command == this.RESP_INIT)
592
592
  {
593
- // Retrieve HTTP port to be used for SSL certificate file uploading
594
- const uploadHttpPort = data.getInt('httpPort');
595
-
596
593
  // Retrieve module id sent by the server (required because multiple modules use file uploading service)
597
594
  const uploadModuleId = data.getUtfString('modId');
598
595
 
@@ -600,8 +597,9 @@ class ServerConfigurator extends _base_module__WEBPACK_IMPORTED_MODULE_0__["Base
600
597
  this._sslCertManager.uploadTargetConfig = {
601
598
  sessionToken: this.smartFox.sessionToken,
602
599
  host: this.smartFox.config.host,
603
- httpPort: uploadHttpPort,
600
+ port: this.smartFox.config.port,
604
601
  moduleId: uploadModuleId,
602
+ protocol: this.smartFox.config.useSSL ? 'https' : 'http'
605
603
  };
606
604
 
607
605
  // Server sends a flag indicating if file uploads are locked
@@ -844,4 +842,4 @@ class ServerConfigurator extends _base_module__WEBPACK_IMPORTED_MODULE_0__["Base
844
842
  /***/ })
845
843
 
846
844
  }]);
847
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-9.bundle.js","sources":["webpack://application/./src/components/module-specific/ssl-certificate-manager.js","webpack://application/./src/modules/server-configurator.js"],"sourcesContent":["import {bytesToSize} from '../../utils/utilities';\nimport aesjs from 'aes-js';\n\nexport class SslCertificateManager extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\n\t\tthis._modalHtml = `\n\t\t\t<div class=\"modal\" id=\"uploadModal\" tabindex=\"-1\" role=\"dialog\" aria-labelledby=\"uploadModalTitle\" aria-hidden=\"true\">\n\t\t\t\t<div class=\"modal-dialog modal-dialog-centered\" role=\"document\">\n\t\t\t\t\t<div class=\"modal-content\">\n\t\t\t\t\t\t<div class=\"modal-header\">\n\t\t\t\t\t\t\t<h5 class=\"modal-title text-primary\" id=\"uploadModalTitle\">SSL Certificate Manager</h5>\n\t\t\t\t\t\t\t<button type=\"button\" class=\"close\" data-dismiss=\"modal\" aria-label=\"Close\">\n\t\t\t\t\t\t\t\t<span aria-hidden=\"true\">&times;</span>\n\t\t\t\t\t\t\t</button>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class=\"modal-body in-flow-invalid-msg\">\n\t\t\t\t\t\t\t<fieldset id=\"uploadFieldset\">\n\t\t\t\t\t\t\t\t<div id=\"uploaderSubform\">\n\t\t\t\t\t\t\t\t\t<div class=\"form-group\">\n\t\t\t\t\t\t\t\t\t\t<div class=\"col-form-label form-label-container\">\n\t\t\t\t\t\t\t\t\t\t\t<label for=\"uploader\" class=\"form-label\">Certificate keystore (jks) <i class=\"fas fa-question-circle text-muted help\" title=\"SSL certificate's protected keystore file to be uploaded to the server, in jks format\"></i></label>\n\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t<div class=\"inner-widget\">\n\t\t\t\t\t\t\t\t\t\t\t<input type=\"file\" id=\"uploader\" name=\"uploader\" accept=\".jks\" data-upload-msg=\"Select a file\">\n\t\t\t\t\t\t\t\t\t\t\t<span class=\"k-invalid-msg position-static\" data-for=\"uploader\"></span>\n\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t<div id=\"passwordsSubform\">\n\t\t\t\t\t\t\t\t\t<div class=\"form-row\">\n\t\t\t\t\t\t\t\t\t\t<div class=\"form-group col\">\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"col-form-label form-label-container\">\n\t\t\t\t\t\t\t\t\t\t\t\t<label for=\"ksPassword\" class=\"form-label\">Keystore password <i class=\"fas fa-question-circle text-muted help\" title=\"Password used to protect the certificate keystore\"></i></label>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"inner-widget\">\n\t\t\t\t\t\t\t\t\t\t\t\t<input type=\"password\" id=\"ksPassword\" name=\"ksPassword\" class=\"form-control k-textbox\" autocomplete=\"off\" required data-required-msg=\"Required\" />\n\t\t\t\t\t\t\t\t\t\t\t\t<span class=\"k-invalid-msg position-static\" data-for=\"ksPassword\"></span>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t</div>\n\n\t\t\t\t\t\t\t\t\t\t<div class=\"form-group col\">\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"col-form-label form-label-container\">\n\t\t\t\t\t\t\t\t\t\t\t\t<label for=\"confirmKsPassword\" class=\"form-label\">Confirm password <i class=\"fas fa-question-circle text-muted help\" title=\"Keystore password confirmation\"></i></label>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"inner-widget\">\n\t\t\t\t\t\t\t\t\t\t\t\t<input type=\"password\" id=\"confirmKsPassword\" name=\"confirmKsPassword\" class=\"form-control k-textbox\" autocomplete=\"off\" required data-required-msg=\"Required\" />\n\t\t\t\t\t\t\t\t\t\t\t\t<span class=\"k-invalid-msg position-static\" data-for=\"confirmKsPassword\"></span>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t</div>\n\n\t\t\t\t\t\t\t\t\t<p><em>For additional security, enter again and confirm your SFS2X administration password.</em></p>\n\n\t\t\t\t\t\t\t\t\t<div class=\"form-row\">\n\t\t\t\t\t\t\t\t\t\t<div class=\"form-group col\">\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"col-form-label form-label-container\">\n\t\t\t\t\t\t\t\t\t\t\t\t<label for=\"adminPassword\" class=\"form-label\">Admin password <i class=\"fas fa-question-circle text-muted help\" title=\"SmartFoxServer 2X remote administration password\"></i></label>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"inner-widget\">\n\t\t\t\t\t\t\t\t\t\t\t\t<input type=\"password\" id=\"adminPassword\" name=\"adminPassword\" class=\"form-control k-textbox\" autocomplete=\"off\" required data-required-msg=\"Required\" />\n\t\t\t\t\t\t\t\t\t\t\t\t<span class=\"k-invalid-msg position-static\" data-for=\"adminPassword\"></span>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t</div>\n\n\t\t\t\t\t\t\t\t\t\t<div class=\"form-group col\">\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"col-form-label form-label-container\">\n\t\t\t\t\t\t\t\t\t\t\t\t<label for=\"confirmAdminPassword\" class=\"form-label\">Confirm password <i class=\"fas fa-question-circle text-muted help\" title=\"SmartFoxServer 2X remote administration password confirmation\"></i></label>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"inner-widget\">\n\t\t\t\t\t\t\t\t\t\t\t\t<input type=\"password\" id=\"confirmAdminPassword\" name=\"confirmAdminPassword\" class=\"form-control k-textbox\" autocomplete=\"off\" required data-required-msg=\"Required\" />\n\t\t\t\t\t\t\t\t\t\t\t\t<span class=\"k-invalid-msg position-static\" data-for=\"confirmAdminPassword\"></span>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t</fieldset>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class=\"modal-footer flex-column\">\n\t\t\t\t\t\t\t<div class=\"d-flex w-100\">\n\t\t\t\t\t\t\t\t<div class=\"flex-grow-1 text-left\">\n\t\t\t\t\t\t\t\t\t<button id=\"uploadSslButton\" type=\"button\" class=\"k-button k-primary\"><i class=\"fas fa-upload mr-1\"></i>Upload certificate</button>\n\t\t\t\t\t\t\t\t\t<i id=\"uploadSpinner\" class=\"fas fa-circle-notch fa-spin text-primary align-middle ml-1\"></i>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t<div class=\"flex-grow-1 text-right\">\n\t\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-secondary\" data-dismiss=\"modal\">Cancel</button>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t<div id=\"uploadErrorMsg\" class=\"text-danger mt-3\"></div>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t`;\n\n\t\t//-------------------------------------------\n\n\t\t$(this).append(`\n\t\t\t<div class=\"col-sm-5 col-lg-4 col-form-label form-label-container\">\n\t\t\t\t<label class=\"form-label\">Upload certificate <i class=\"fas fa-question-circle text-muted help\" title=\"Upload an SSL certificate's protected keystore to the server\"></i></label>\n\t\t\t</div>\n\t\t\t<div class=\"inner-widget align-self-center align-self-sm-start col-auto\">\n\t\t\t\t<button id=\"manageSslButton\" type=\"button\" class=\"k-button k-primary\" disabled><i class=\"fas fa-cog mr-1\"></i>Manage</button>\n\t\t\t</div>\n\t\t`);\n\n\t\t// Add listeners to Manage button click\n\t\t$('#manageSslButton', $(this)).on('click', $.proxy(this._onManageSslClick, this));\n\t}\n\n\tdestroy()\n\t{\n\t\t// Remove event listener\n\t\t$('#manageSslButton', $(this)).off('click');\n\n\t\t// Hide modal (which in turn destroys it)\n\t\tlet modalElement = $('#uploadModal', $(this));\n\n\t\tif (modalElement)\n\t\t\tmodalElement.modal('hide');\n\t}\n\n\tget enabled()\n\t{\n\t\treturn this._isEnabled;\n\t}\n\n\tset enabled(value)\n\t{\n\t\tthis._isEnabled = value;\n\n\t\t// Enable/disable Manage button\n\t\t$('#manageSslButton', $(this)).attr('disabled', !value);\n\n\t\t// Enable/disable modal\n\t\tlet modalElement = $('#uploadModal', $(this));\n\n\t\tif (modalElement)\n\t\t{\n\t\t\t// Disable modal close buttons\n\t\t\t$('button[data-dismiss=\"modal\"]', modalElement).attr('disabled', !value);\n\n\t\t\t// Disable upload button\n\t\t\t$('#uploadSslButton', modalElement).attr('disabled', !value);\n\n\t\t\t// Disable fieldset\n\t\t\t$('#uploadFieldset', modalElement).attr('disabled', !value);\n\t\t}\n\t}\n\n\tget uploadTargetConfig()\n\t{\n\t\treturn this._uploadTargetConfig;\n\t}\n\n\tset uploadTargetConfig(data)\n\t{\n\t\tthis._uploadTargetConfig = data;\n\t}\n\n\t_onManageSslClick()\n\t{\n\t\t// Initialize and show modal\n\t\tthis._showModal();\n\t}\n\n\t_onUploadSslClick()\n\t{\n\t\tif (this._validate())\n\t\t\tthis._startSslCertUpload();\n\t}\n\n\t_showModal()\n\t{\n\t\t// Append modal html\n\t\t$(this).append(this._modalHtml);\n\n\t\tlet modalElement = $('#uploadModal', $(this));\n\n\t\t// Hide SSL certificate upload spinner and error message container\n\t\t$('#uploadSpinner', modalElement).hide();\n\t\t$('#uploadErrorMsg', modalElement).hide();\n\t\t$('#uploadErrorMsg', modalElement).text('');\n\n\t\t// Add listener to Upload button click\n\t\t$('#uploadSslButton', modalElement).on('click', $.proxy(this._onUploadSslClick, this));\n\n\t\t// Add listener to modal hide event\n\t\tmodalElement.on('hidden.bs.modal', $.proxy(this._destroyModal, this));\n\n\t\t// Initialize kendo uploader\n\t\tthis._uploadWidget = modalElement.find('#uploader').kendoUpload({\n\t\t\tallowedExtensions: ['.jks'],\n\t\t\tmultiple: false,\n\t\t\ttemplate: function(dataItem) {\n\t\t\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\t\t\treturn kendo.template(`\n\t\t\t\t\t<span class='k-progress w-100'></span>\n\t\t\t\t\t<span class=\"\">\n\t\t\t\t\t\t<span class=\"k-file-name\" title=\"#=name#\">#=name#</span>\n\t\t\t\t\t\t<span class=\"k-file-size\">Size: #=bytesToSize(size, 1, 'Bytes')#</span>\n\t\t\t\t\t</span>\n\t\t\t\t`)(dataItem);\n\t\t\t},\n\t        localization: {\n\t            select: 'Select file...'\n\t        }\n        }).data('kendoUpload');\n\n\t\t// Initialize kendo validation on uploader subform\n\t\t// NOTE: we use separate validators to be able to disable 'validateOnBlur' on the uploader,\n\t\t// because it causes the error message to appear as soon as the \"Select file\" button is clicked\n\t\tthis._validator1 = modalElement.find('#uploaderSubform').kendoValidator({\n\t\t\tvalidateOnBlur: false,\n\t\t\trules: {\n\t\t\t\tupload: function(input) {\n\t\t\t\t\tlet valid = false;\n\t\t\t\t\tif (input.is('[type=file]'))\n\t\t\t\t\t\tvalid = input.closest('.k-upload').find('.k-file').length > 0;\n\n\t\t\t\t\treturn valid;\n\t            }\n\t\t\t}\n\t\t}).data('kendoValidator');\n\n\t\t// Initialize kendo validation on passwords subform\n\t\tthis._validator2 = modalElement.find('#passwordsSubform').kendoValidator({\n\t\t\tvalidateOnBlur: true,\n\t\t\trules: {\n\t\t\t\tverifyKsPassword: $.proxy(function(input) {\n\t\t\t\t\tlet valid = true;\n\t\t\t\t\tif (input.is('[name=confirmKsPassword]'))\n\t\t\t\t\t\tvalid = input.val() === $(this).find('#ksPassword').val();\n\t\t\t\t\treturn valid;\n\t\t\t\t}, this),\n\t\t\t\tverifyAdmPassword: $.proxy(function(input) {\n\t\t\t\t\tlet valid = true;\n\t\t\t\t\tif (input.is('[name=confirmAdminPassword]'))\n\t\t\t\t\t\tvalid = input.val() === $(this).find('#adminPassword').val();\n\t\t\t\t\treturn valid;\n\t\t\t\t}, this)\n\t\t\t},\n\t\t\tmessages: {\n\t\t\t\tverifyKsPassword: 'Password not matching',\n\t\t\t\tverifyAdmPassword: 'Password not matching',\n\t\t\t}\n\t\t}).data('kendoValidator');\n\n\t\t// Initialize bootstrap modal\n\t\tmodalElement.modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t});\n\t}\n\n\t_destroyModal()\n\t{\n\t\tlet modalElement = $('#uploadModal', $(this));\n\n\t\tif (modalElement)\n\t\t{\n\t\t\t// Remove listeners\n\t\t\t$('#uploadSslButton', modalElement).off('click');\n\t\t\tmodalElement.off('hidden.bs.modal');\n\n\t\t\t// Destroy everything Kendo\n\t\t\tkendo.destroy(modalElement);\n\n\t\t\t// Dispose modal\n\t\t\tmodalElement.modal('dispose');\n\n\t\t\t// Remove html\n\t\t\tmodalElement.remove();\n\t\t\tmodalElement = null;\n\t\t}\n\t}\n\n\t_validate()\n\t{\n\t\tlet val1 = this._validator1.validate();\n\t\tlet val2 = this._validator2.validate();\n\n\t\treturn val1 && val2;\n\t}\n\n\t_startSslCertUpload()\n\t{\n\t\tif (!this._uploadTargetConfig)\n\t\t\tthrow new Error('Upload target configuration not set');\n\n\t\tlet modalElement = $('#uploadModal', $(this));\n\n\t\tif (modalElement)\n\t\t{\n\t\t\tlet certData = {};\n\t\t\tcertData.file = this._uploadWidget.getFiles()[0];\n\t\t\tcertData.ksPassword = $('#ksPassword', modalElement).val();\n\t\t\tcertData.adminPassword = $('#adminPassword', modalElement).val();\n\n\t\t\t// Disable modal\n\t\t\tthis.enabled = false;\n\n\t\t\t// Hide previous error and show spinner\n\t\t\t$('#uploadSpinner', modalElement).show();\n\t\t\t$('#uploadErrorMsg', modalElement).hide();\n\t\t\t$('#uploadErrorMsg', modalElement).text('');\n\n\t\t\t//=================================================================\n\n\t\t\t// Generate Init Vector\n\t\t\tlet rngValues = [];\n\t\t\tfor (let i = 0; i < 16; i++)\n\t\t\t\trngValues.push(Math.floor(Math.random() * 256));\n\n\t\t\tlet iv = new Uint8Array(rngValues);\n\n\t\t\t// Generate secret key by MD5-encoding admin password + session token\n\t\t\tlet md5Pass = this._binaryMD5(certData.adminPassword + this._uploadTargetConfig.sessionToken);\n\n\t\t\t// Encrypt keystore password via AES (128bit)\n\t\t\tlet encryptedPwd = this._aesEncrypt(certData.ksPassword, md5Pass, iv);\n\n\t\t\t// Encode IV using Base64\n\t\t\tlet encodedIv = this._b64Encode(iv);\n\n\t\t\t// Encode encrypted password using Base64\n\t\t\tlet encodedPwd = this._b64Encode(encryptedPwd);\n\n\t\t\t//=================================================================\n\n\t\t\t// Set parameters to be sent with the certificate keystore file\n\t\t\tconst params = new FormData();\n\t\t\tparams.append('files[]', certData.file.rawFile);\n\t\t\tparams.append('__iv', encodedIv);\n\t\t\tparams.append('__password', encodedPwd);\n\t\t\tparams.append('__module', this._uploadTargetConfig.moduleId);\n\n\t\t\t// Set destination url\n\t\t\tconst url = 'http://' + this._uploadTargetConfig.host + ':' + this._uploadTargetConfig.httpPort + '/BlueBox/SFS2XFileUpload?sessHashId=' + this._uploadTargetConfig.sessionToken;\n\n\t\t\t// Start upload\n\t\t\tfetch(url, {\n\t\t\t\tmethod: 'POST',\n\t\t\t\tbody: params,\n\t\t\t\tmode: 'no-cors'\n\t\t\t}).then($.proxy(this._onSslCertUploadEnd, this));\n\t\t}\n\t}\n\n\t_onSslCertUploadEnd(response)\n\t{\n\t\t// Nothing to do: we have to wait the upload process completion to be signaled by the server through the dedicated Extension response\n\n\t\t//=================================================================\n\n\t\t// TODO Should we handle this response in some way? For some unknown reason we always get ok=false and status=0\n\t\t//console.log(response)\n\t\t//console.log(response.ok)\n\t\t//console.log(response.status)\n\t}\n\n\t/**\n\t * Method called by parent when upload failed.\n\t */\n\tonSslCertUploadError(error)\n\t{\n\t\tlet modalElement = $('#uploadModal', $(this));\n\n\t\tif (modalElement)\n\t\t{\n\t\t\t// Enable modal\n\t\t\tthis.enabled = true;\n\n\t\t\t// Show upload error\n\t\t\t$('#uploadErrorMsg', modalElement).show();\n\t\t\t$('#uploadErrorMsg', modalElement).text(error + '.');\n\n\t\t\t// Hide spinner\n\t\t\t$('#uploadSpinner', modalElement).hide();\n\t\t}\n\t}\n\n\t/**\n\t * Method called by parent when upload is completed successfully.\n\t */\n\tonSslCertUploadSuccess()\n\t{\n\t\tlet modalElement = $('#uploadModal', $(this));\n\n\t\tif (modalElement)\n\t\t{\n\t\t\t// Enable modal\n\t\t\tthis.enabled = true;\n\n\t\t\t// Hide spinner\n\t\t\t$('#uploadSpinner', modalElement).hide();\n\n\t\t\t// Hide modal\n\t\t\tmodalElement.modal('hide');\n\t\t}\n\t}\n\n\t// *****************************************************************\n\n\t/*\n\t * Takes a string and returns the MD5 as Uint8Array\n\t */\n\t_binaryMD5(str)\n\t{\n\t\tlet MD5 = new SFS2X.MD5();\n\t\tlet hexStr = MD5.hex_md5(str);\n\n\t\treturn this._hexByteStringToByteArray(hexStr);\n\t}\n\n\t/*\n\t * Hex bytes ---> Actual byte[] as Uint8Array\n\t */\n\t_hexByteStringToByteArray(hexByteString)\n\t{\n\t    let bytes = new Uint8Array(16); // MD5 fixed output size\n\n\t    for (let i = 0; i < hexByteString.length;)\n\t    {\n\t        let hexByte = hexByteString[i++] + hexByteString[i++];\n\t        let byte = parseInt(hexByte, 16);\n\n\t        bytes[i / 2 - 1] = byte;\n\t    }\n\n\t    return bytes;\n\t}\n\n\t/*\n\t * Encrypt using AES, mode CBC, PKCS#7 padding\n\t *\n\t * text \t\t-> the text we want to encode\n\t * key \t\t-> the AES key\n\t * iv  \t\t-> the AES/CBC init vector\n\t *\n\t * Returns \t-> Uint8Array\n\t */\n\t_aesEncrypt(text, key, iv)\n\t{\n\t\tlet textBytes = aesjs.utils.utf8.toBytes(text); \t\t// Get UTF-8 bytes\n\t\tlet aesCBC = new aesjs.ModeOfOperation.cbc(key, iv);\t// Init CBC mode\n\t\ttextBytes = aesjs.padding.pkcs7.pad(textBytes); \t\t// PKCS#7 padding\n\n\t\t// Encrypt\n\t\treturn aesCBC.encrypt(textBytes);\n\t}\n\n\t/*\n\t * Encode passed byte array --> Base64 representation\n\t * Returns --> string\n\t */\n\t_b64Encode(barray)\n\t{\n\t\treturn btoa(String.fromCharCode.apply(null, barray));\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('ssl-certificate-manager'))\n\twindow.customElements.define('ssl-certificate-manager', SslCertificateManager);\n","import {BaseModule} from './base-module';\nimport {ConfigInterfaceBuilder} from '../utils/uibuilder/config-interface-builder';\nimport {SslCertificateManager} from '../components/module-specific/ssl-certificate-manager';\n\nexport default class ServerConfigurator extends BaseModule\n{\n\tconstructor()\n\t{\n\t    super('serverConfig');\n\n\t\t// Outgoing requests\n\t\tthis.REQ_INIT = 'init';\n\t\tthis.REQ_GET_CONFIG = 'getConfig';\n\t\tthis.REQ_UPDATE_CONFIG = 'updConfig';\n\t\tthis.REQ_UPDATE_GEO_DB = 'updGeoDb';\n\n\t\t// Incoming responses\n\t\tthis.RESP_INIT = 'init';\n\t\tthis.RESP_CONFIG = 'config';\n\t\tthis.RESP_CONFIG_UPDATE_CONFIRM = 'configUpdate';\n\t\tthis.RESP_CONFIG_CHANGED_ALERT = 'configAlert';\n\t\tthis.RESP_SSL_UPLOAD_ERROR = 'sslUploadError';\n\t\tthis.RESP_SSL_UPLOAD_CONFIRM = 'sslUpload';\n\t\tthis.RESP_UPDATE_GEO_DB = 'geoDbUpdate';\n\t}\n\n\t//------------------------------------\n\t// COMMON MODULE INTERFACE METHODS\n\t// This members are used by the main controller\n\t// to communicate with the module's controller.\n\t// This methods override those in BaseModule class.\n\t//------------------------------------\n\n\tinitialize(idData, shellController)\n\t{\n\t\t// Call super method\n\t\tsuper.initialize(idData, shellController);\n\n\t\t// Initialize progress bar\n\t\t$('#src-progressBar').kendoProgressBar({\n\t\t\tmin: 0,\n            max: 100,\n\t\t\tvalue: false,\n            type: 'value',\n            animation: {\n                duration: 400\n            }\n        });\n\n\t\t// Create interface builder instance\n\t\tthis._interfaceBuilder = new ConfigInterfaceBuilder();\n\n\t\t// Add listener to GeoLite2 database update button\n\t\t$('#src-updateGeolocDbButton').on('click', $.proxy(this._onUpdateGeolocDbClick, this));\n\n\t\t// Add listener to interface buttons\n\t\t$('#src-reloadButton').on('click', $.proxy(this._onReloadClick, this));\n\t\t$('#src-submitButton').on('click', $.proxy(this._onSubmitClick, this));\n\n\t\t// Save ref to SSL Manager\n\t\tthis._sslCertManager = document.getElementById('src-sslCertManager');\n\n\t\t//-----------------------------------*/\n\n\t\t// Send initialization request\n\t\tthis.sendExtensionRequest(this.REQ_INIT);\n\t}\n\n\tdestroy()\n\t{\n\t\t// Call super method\n\t\tsuper.destroy();\n\n\t\t// Destroy SSL Certificate Manager\n\t\tthis._sslCertManager.destroy();\n\n\t\t// Remove interface buttons click listeners\n\t\t$('#src-updateGeolocDbButton').off('click');\n\t\t$('#src-reloadButton').off('click');\n\t\t$('#src-submitButton').off('click');\n\n\t\t// Clear tabs container\n\t\tthis._clearTabs();\n\t}\n\n\tonExtensionCommand(command, data)\n\t{\n\t\t// Initialization data received\n\t\tif (command == this.RESP_INIT)\n\t\t{\n\t\t\t// Retrieve HTTP port to be used for SSL certificate file uploading\n\t\t\tconst uploadHttpPort = data.getInt('httpPort');\n\n\t\t\t// Retrieve module id sent by the server (required because multiple modules use file uploading service)\n\t\t\tconst uploadModuleId = data.getUtfString('modId');\n\n\t\t\t// Set SSL upload manager target configuration\n\t\t\tthis._sslCertManager.uploadTargetConfig = {\n\t\t\t\tsessionToken: this.smartFox.sessionToken,\n\t\t\t\thost: this.smartFox.config.host,\n\t\t\t\thttpPort: uploadHttpPort,\n\t\t\t\tmoduleId: uploadModuleId,\n\t\t\t};\n\n\t\t\t// Server sends a flag indicating if file uploads are locked\n\t\t\t// We use it to enable the \"Manage SSL certificate\" button\n\t\t\tthis._sslLocked = data.getBool('lock');\n\n\t\t\tif (!this._sslLocked)\n\t\t\t\t$('#src-manageSslWarn').hide();\n\n\t\t\t// Request configuration data to server instance\n\t\t\tthis.sendExtensionRequest(this.REQ_GET_CONFIG);\n\t\t}\n\n\t\t// Server configuration data received\n\t\telse if (command == this.RESP_CONFIG)\n\t\t{\n\t\t\t// Build user interface based on received data\n\t\t\tthis._interfaceBuilder.buildInterface(data.getSFSArray('settings'), 'src-tabNavigator', false);\n\n\t\t\t// Enable buttons\n\t\t\tthis._enableButtons(true);\n\n\t\t\t// Initialize TabNavigator-ralated widgets\n\t\t\tif (!this._tabNavInitialized)\n\t\t\t{\n\t\t\t\t// Enable scrolling tabs\n\t\t\t\t$('#src-tabNavigator > #tabs').scrollingTabs({\n\t\t\t\t\tbootstrapVersion: 4,\n\t\t\t\t\tscrollToTabEdge: true,\n\t\t\t\t\tenableSwiping: true,\n\t\t\t\t\tdisableScrollArrowsOnFullyScrolled: true,\n\t\t\t\t\tcssClassLeftArrow: 'fa fa-chevron-left',\n\t\t\t\t\tcssClassRightArrow: 'fa fa-chevron-right'\n\t\t\t\t});\n\n\t\t\t\tthis._tabNavInitialized = true;\n\t\t\t}\n\n\t\t\t// Run validation (to remove validation messages if data was reloaded)\n\t\t\tthis._interfaceBuilder.checkIsValid();\n\n\t\t\tthis._switchView('src-main');\n\t\t}\n\n\t\t// Server configuration update confirmation\n\t\telse if (command == this.RESP_CONFIG_UPDATE_CONFIRM)\n\t\t{\n\t\t\t// Enable buttons\n\t\t\tthis._enableButtons(true);\n\n\t\t\t// Enable form items\n\t\t\tthis._interfaceBuilder.disableInterface(false);\n\n\t\t\t// If the current user is the updater, show a notification\n\t\t\t// Otherwise, show a dialog box suggesting to reload\n\t\t\tlet updater = data.getUtfString('user');\n\n\t\t\tif (updater == this.smartFox.mySelf.name)\n\t\t\t{\n\t\t\t\t// Reset the 'modified' flag\n\t\t\t\tthis._interfaceBuilder.resetIsModified();\n\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification('Server settings updated', 'Changes will be applied on next server restart');\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Show alert\n\t\t\t\tthis.shellCtrl.showSimpleAlert(`Administrator ${updater} has modified the server settings; please reload to update your view.`);\n\n\t\t\t\t// Disable submit button\n\t\t\t\t$('#src-submitButton').attr('disabled', true);\n\t\t\t}\n\t\t}\n\n\t\t// Server configuration xml saved by an external process\n\t\telse if (command == this.RESP_CONFIG_CHANGED_ALERT)\n\t\t{\n\t\t\t// Show alert\n\t\t\tthis.shellCtrl.showSimpleAlert(`The system has modified the server settings automatically; please reload to update your view.`);\n\n\t\t\t// Disable submit button\n\t\t\t$('#src-submitButton').attr('disabled', true);\n\t\t}\n\n\t\t// SSL certificate upload error\n\t\telse if (command == this.RESP_SSL_UPLOAD_ERROR)\n\t\t{\n\t\t\tconst error = data.getUtfString('error');\n\n\t\t\t// Log warning\n\t\t\tthis.shellCtrl.logMessage(error, 'error');\n\n\t\t\t// Show error in manager window\n\t\t\tthis._sslCertManager.onSslCertUploadError(error);\n\t\t}\n\n\t\t// SSL certificate upload confirmed\n\t\telse if (command == this.RESP_SSL_UPLOAD_CONFIRM)\n\t\t{\n\t\t\t// Closw manager window\n\t\t\tthis._sslCertManager.onSslCertUploadSuccess();\n\n\t\t\tlet updater = data.getUtfString('user');\n\n\t\t\t// Display notification\n\t\t\tif (updater == this.smartFox.mySelf.name)\n\t\t\t\tthis.shellCtrl.showNotification('SSL certificate', 'SSL certificate keystore was uploaded successfully');\n\t\t\telse\n\t\t\t\tthis.shellCtrl.showNotification('SSL certificate', `Administrator ${updater} has uploaded a new SSL certificate keystore`);\n\n\t\t\t// When a certificate is uploaded, HTTPS is also enabled automatically:\n\t\t\t// we have to update the interface accordingly\n\t\t\tthis._updateConfigFormItemDisplayedValue('webServer.enableHttps', true);\n\t\t}\n\n\t\t// Geolocation database update confirmation\n\t\telse if (command == this.RESP_UPDATE_GEO_DB)\n\t\t{\n\t\t\t// Enable button\n\t\t\t$('#src-updateGeolocDbButton').attr('disabled', false);\n\n\t\t\t// Check success\n\t\t\tif (data.getBool('success'))\n\t\t\t{\n\t\t\t\t// Update displayed date\n\t\t\t\tthis._updateConfigFormItemDisplayedValue('adminHelper.geoDbReleaseDate', data.getUtfString('newRelDate'));\n\n\t\t\t\t// If the current user is the updater, also show a notification\n\t\t\t\tlet updater = data.getUtfString('user');\n\n\t\t\t\tif (updater == this.smartFox.mySelf.name)\n\t\t\t\t\tthis.shellCtrl.showNotification('Geolocation database updated', 'Latest release of the GeoLite2 Country database has been installed successfully');\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Show alert\n\t\t\t\tthis.shellCtrl.showSimpleAlert(data.getUtfString('error'));\n\t\t\t}\n\t\t}\n\t}\n\n\t//------------------------------------\n\t// PRIVATE METHODS\n\t//------------------------------------\n\n\t_enableButtons(enabled)\n\t{\n\t\t$('#src-reloadButton').attr('disabled', !enabled);\n\t\t$('#src-submitButton').attr('disabled', !enabled);\n\t\t$('#src-backupCheck').attr('disabled', !enabled);\n\n\t\t$('#src-updateGeolocDbButton').attr('disabled', !enabled);\n\n\t\tif (!this._sslLocked)\n\t\t\tthis._sslCertManager.enabled = enabled;\n\t}\n\n\t_switchView(viewId)\n\t{\n\t\tdocument.getElementById('src-viewstack').selectedElement = document.getElementById(viewId);\n\t}\n\n\t_clearTabs()\n\t{\n\t\t// Destroy scrolling tabs\n\t\t$('#src-tabNavigator #tabs').scrollingTabs('destroy');\n\n\t\t// Remove all tab navigator content\n\t\tthis._interfaceBuilder.destroyInterface();\n\t}\n\n\t_onUpdateGeolocDbClick()\n\t{\n\t\t// Disable button\n\t\t$('#src-updateGeolocDbButton').attr('disabled', true);\n\n\t\t// Send request to server\n\t\tthis.sendExtensionRequest(this.REQ_UPDATE_GEO_DB);\n\t}\n\n\t_onReloadClick()\n\t{\n\t\t// Disable buttons\n\t\tthis._enableButtons(false);\n\n\t\t// Switch to loading view\n\t\tthis._switchView('src-loading');\n\n\t\t// Hide validation messages\n\t\tthis._interfaceBuilder.resetValidation();\n\n\t\t// Request configuration data to server instance\n\t\tthis.sendExtensionRequest(this.REQ_GET_CONFIG);\n\t}\n\n\t_onSubmitClick()\n\t{\n\t\t// Check validity\n\t\tif (this._interfaceBuilder.checkIsValid())\n\t\t{\n\t\t\tlet changes = this._interfaceBuilder.getChangedData();\n\n\t\t\tif (changes.size() > 0)\n\t\t\t{\n\t\t\t\t// Disable buttons\n\t\t\t\tthis._enableButtons(false);\n\n\t\t\t\t// Disable form items\n\t\t\t\tthis._interfaceBuilder.disableInterface(true);\n\n\t\t\t\t// Send updated settings to server instance\n\t\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\t\tparams.putSFSArray('settings', changes);\n\t\t\t\tparams.putBool('backup', $('#src-backupCheck').prop('checked'));\n\n\t\t\t\tthis.sendExtensionRequest(this.REQ_UPDATE_CONFIG, params);\n\t\t\t}\n\t\t}\n\t\telse\n\t\t\tthis.shellCtrl.showSimpleAlert('Unable to submit configuration changes due to an invalid value; please verify the highlighted form fields in all tabs.', true);\n\t}\n\n\t_updateConfigFormItemDisplayedValue(configParamName, newValue)\n\t{\n\t\t// Get the relevant Configuration Form Item\n\t\tconst configFormItem = this._interfaceBuilder.getConfigFormItem(configParamName);\n\n\t\t// Update Configuration Parameter associated with the Configuration Form Item\n\t\tconfigFormItem.data.value = newValue;\n\t\tconfigFormItem.data.resetIsModified(); // This is needed to avoid the Configuration Parameter to flagged as 'changed'\n\n\t\t// Display the new value of the Configuration Form Item\n\t\tconfigFormItem._setWidgetValue(); // Display the new value in the config form item\n\t}\n}\n"],"mappings":";;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACndA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;A","sourceRoot":""}
845
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-9.bundle.js","sources":["webpack://application/./src/components/module-specific/ssl-certificate-manager.js","webpack://application/./src/modules/server-configurator.js"],"sourcesContent":["import {bytesToSize} from '../../utils/utilities';\nimport aesjs from 'aes-js';\n\nexport class SslCertificateManager extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\n\t\tthis._modalHtml = `\n\t\t\t<div class=\"modal\" id=\"uploadModal\" tabindex=\"-1\" role=\"dialog\" aria-labelledby=\"uploadModalTitle\" aria-hidden=\"true\">\n\t\t\t\t<div class=\"modal-dialog modal-dialog-centered\" role=\"document\">\n\t\t\t\t\t<div class=\"modal-content\">\n\t\t\t\t\t\t<div class=\"modal-header\">\n\t\t\t\t\t\t\t<h5 class=\"modal-title text-primary\" id=\"uploadModalTitle\">SSL Certificate Manager</h5>\n\t\t\t\t\t\t\t<button type=\"button\" class=\"close\" data-dismiss=\"modal\" aria-label=\"Close\">\n\t\t\t\t\t\t\t\t<span aria-hidden=\"true\">&times;</span>\n\t\t\t\t\t\t\t</button>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class=\"modal-body in-flow-invalid-msg\">\n\t\t\t\t\t\t\t<fieldset id=\"uploadFieldset\">\n\t\t\t\t\t\t\t\t<div id=\"uploaderSubform\">\n\t\t\t\t\t\t\t\t\t<div class=\"form-group\">\n\t\t\t\t\t\t\t\t\t\t<div class=\"col-form-label form-label-container\">\n\t\t\t\t\t\t\t\t\t\t\t<label for=\"uploader\" class=\"form-label\">Certificate keystore (jks) <i class=\"fas fa-question-circle text-muted help\" title=\"SSL certificate's protected keystore file to be uploaded to the server, in jks format\"></i></label>\n\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t<div class=\"inner-widget\">\n\t\t\t\t\t\t\t\t\t\t\t<input type=\"file\" id=\"uploader\" name=\"uploader\" accept=\".jks\" data-upload-msg=\"Select a file\">\n\t\t\t\t\t\t\t\t\t\t\t<span class=\"k-invalid-msg position-static\" data-for=\"uploader\"></span>\n\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t<div id=\"passwordsSubform\">\n\t\t\t\t\t\t\t\t\t<div class=\"form-row\">\n\t\t\t\t\t\t\t\t\t\t<div class=\"form-group col\">\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"col-form-label form-label-container\">\n\t\t\t\t\t\t\t\t\t\t\t\t<label for=\"ksPassword\" class=\"form-label\">Keystore password <i class=\"fas fa-question-circle text-muted help\" title=\"Password used to protect the certificate keystore\"></i></label>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"inner-widget\">\n\t\t\t\t\t\t\t\t\t\t\t\t<input type=\"password\" id=\"ksPassword\" name=\"ksPassword\" class=\"form-control k-textbox\" autocomplete=\"off\" required data-required-msg=\"Required\" />\n\t\t\t\t\t\t\t\t\t\t\t\t<span class=\"k-invalid-msg position-static\" data-for=\"ksPassword\"></span>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t</div>\n\n\t\t\t\t\t\t\t\t\t\t<div class=\"form-group col\">\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"col-form-label form-label-container\">\n\t\t\t\t\t\t\t\t\t\t\t\t<label for=\"confirmKsPassword\" class=\"form-label\">Confirm password <i class=\"fas fa-question-circle text-muted help\" title=\"Keystore password confirmation\"></i></label>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"inner-widget\">\n\t\t\t\t\t\t\t\t\t\t\t\t<input type=\"password\" id=\"confirmKsPassword\" name=\"confirmKsPassword\" class=\"form-control k-textbox\" autocomplete=\"off\" required data-required-msg=\"Required\" />\n\t\t\t\t\t\t\t\t\t\t\t\t<span class=\"k-invalid-msg position-static\" data-for=\"confirmKsPassword\"></span>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t</div>\n\n\t\t\t\t\t\t\t\t\t<p><em>For additional security, enter again and confirm your SFS2X administration password.</em></p>\n\n\t\t\t\t\t\t\t\t\t<div class=\"form-row\">\n\t\t\t\t\t\t\t\t\t\t<div class=\"form-group col\">\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"col-form-label form-label-container\">\n\t\t\t\t\t\t\t\t\t\t\t\t<label for=\"adminPassword\" class=\"form-label\">Admin password <i class=\"fas fa-question-circle text-muted help\" title=\"SmartFoxServer 2X remote administration password\"></i></label>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"inner-widget\">\n\t\t\t\t\t\t\t\t\t\t\t\t<input type=\"password\" id=\"adminPassword\" name=\"adminPassword\" class=\"form-control k-textbox\" autocomplete=\"off\" required data-required-msg=\"Required\" />\n\t\t\t\t\t\t\t\t\t\t\t\t<span class=\"k-invalid-msg position-static\" data-for=\"adminPassword\"></span>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t</div>\n\n\t\t\t\t\t\t\t\t\t\t<div class=\"form-group col\">\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"col-form-label form-label-container\">\n\t\t\t\t\t\t\t\t\t\t\t\t<label for=\"confirmAdminPassword\" class=\"form-label\">Confirm password <i class=\"fas fa-question-circle text-muted help\" title=\"SmartFoxServer 2X remote administration password confirmation\"></i></label>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t\t<div class=\"inner-widget\">\n\t\t\t\t\t\t\t\t\t\t\t\t<input type=\"password\" id=\"confirmAdminPassword\" name=\"confirmAdminPassword\" class=\"form-control k-textbox\" autocomplete=\"off\" required data-required-msg=\"Required\" />\n\t\t\t\t\t\t\t\t\t\t\t\t<span class=\"k-invalid-msg position-static\" data-for=\"confirmAdminPassword\"></span>\n\t\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t</fieldset>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class=\"modal-footer flex-column\">\n\t\t\t\t\t\t\t<div class=\"d-flex w-100\">\n\t\t\t\t\t\t\t\t<div class=\"flex-grow-1 text-left\">\n\t\t\t\t\t\t\t\t\t<button id=\"uploadSslButton\" type=\"button\" class=\"k-button k-primary\"><i class=\"fas fa-upload mr-1\"></i>Upload certificate</button>\n\t\t\t\t\t\t\t\t\t<i id=\"uploadSpinner\" class=\"fas fa-circle-notch fa-spin text-primary align-middle ml-1\"></i>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t\t<div class=\"flex-grow-1 text-right\">\n\t\t\t\t\t\t\t\t\t<button type=\"button\" class=\"k-button k-secondary\" data-dismiss=\"modal\">Cancel</button>\n\t\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t<div id=\"uploadErrorMsg\" class=\"text-danger mt-3\"></div>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t`;\n\n\t\t//-------------------------------------------\n\n\t\t$(this).append(`\n\t\t\t<div class=\"col-sm-5 col-lg-4 col-form-label form-label-container\">\n\t\t\t\t<label class=\"form-label\">Upload certificate <i class=\"fas fa-question-circle text-muted help\" title=\"Upload an SSL certificate's protected keystore to the server\"></i></label>\n\t\t\t</div>\n\t\t\t<div class=\"inner-widget align-self-center align-self-sm-start col-auto\">\n\t\t\t\t<button id=\"manageSslButton\" type=\"button\" class=\"k-button k-primary\" disabled><i class=\"fas fa-cog mr-1\"></i>Manage</button>\n\t\t\t</div>\n\t\t`);\n\n\t\t// Add listeners to Manage button click\n\t\t$('#manageSslButton', $(this)).on('click', $.proxy(this._onManageSslClick, this));\n\t}\n\n\tdestroy()\n\t{\n\t\t// Remove event listener\n\t\t$('#manageSslButton', $(this)).off('click');\n\n\t\t// Hide modal (which in turn destroys it)\n\t\tlet modalElement = $('#uploadModal', $(this));\n\n\t\tif (modalElement)\n\t\t\tmodalElement.modal('hide');\n\t}\n\n\tget enabled()\n\t{\n\t\treturn this._isEnabled;\n\t}\n\n\tset enabled(value)\n\t{\n\t\tthis._isEnabled = value;\n\n\t\t// Enable/disable Manage button\n\t\t$('#manageSslButton', $(this)).attr('disabled', !value);\n\n\t\t// Enable/disable modal\n\t\tlet modalElement = $('#uploadModal', $(this));\n\n\t\tif (modalElement)\n\t\t{\n\t\t\t// Disable modal close buttons\n\t\t\t$('button[data-dismiss=\"modal\"]', modalElement).attr('disabled', !value);\n\n\t\t\t// Disable upload button\n\t\t\t$('#uploadSslButton', modalElement).attr('disabled', !value);\n\n\t\t\t// Disable fieldset\n\t\t\t$('#uploadFieldset', modalElement).attr('disabled', !value);\n\t\t}\n\t}\n\n\tget uploadTargetConfig()\n\t{\n\t\treturn this._uploadTargetConfig;\n\t}\n\n\tset uploadTargetConfig(data)\n\t{\n\t\tthis._uploadTargetConfig = data;\n\t}\n\n\t_onManageSslClick()\n\t{\n\t\t// Initialize and show modal\n\t\tthis._showModal();\n\t}\n\n\t_onUploadSslClick()\n\t{\n\t\tif (this._validate())\n\t\t\tthis._startSslCertUpload();\n\t}\n\n\t_showModal()\n\t{\n\t\t// Append modal html\n\t\t$(this).append(this._modalHtml);\n\n\t\tlet modalElement = $('#uploadModal', $(this));\n\n\t\t// Hide SSL certificate upload spinner and error message container\n\t\t$('#uploadSpinner', modalElement).hide();\n\t\t$('#uploadErrorMsg', modalElement).hide();\n\t\t$('#uploadErrorMsg', modalElement).text('');\n\n\t\t// Add listener to Upload button click\n\t\t$('#uploadSslButton', modalElement).on('click', $.proxy(this._onUploadSslClick, this));\n\n\t\t// Add listener to modal hide event\n\t\tmodalElement.on('hidden.bs.modal', $.proxy(this._destroyModal, this));\n\n\t\t// Initialize kendo uploader\n\t\tthis._uploadWidget = modalElement.find('#uploader').kendoUpload({\n\t\t\tallowedExtensions: ['.jks'],\n\t\t\tmultiple: false,\n\t\t\ttemplate: function(dataItem) {\n\t\t\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\t\t\treturn kendo.template(`\n\t\t\t\t\t<span class='k-progress w-100'></span>\n\t\t\t\t\t<span class=\"\">\n\t\t\t\t\t\t<span class=\"k-file-name\" title=\"#=name#\">#=name#</span>\n\t\t\t\t\t\t<span class=\"k-file-size\">Size: #=bytesToSize(size, 1, 'Bytes')#</span>\n\t\t\t\t\t</span>\n\t\t\t\t`)(dataItem);\n\t\t\t},\n\t        localization: {\n\t            select: 'Select file...'\n\t        }\n        }).data('kendoUpload');\n\n\t\t// Initialize kendo validation on uploader subform\n\t\t// NOTE: we use separate validators to be able to disable 'validateOnBlur' on the uploader,\n\t\t// because it causes the error message to appear as soon as the \"Select file\" button is clicked\n\t\tthis._validator1 = modalElement.find('#uploaderSubform').kendoValidator({\n\t\t\tvalidateOnBlur: false,\n\t\t\trules: {\n\t\t\t\tupload: function(input) {\n\t\t\t\t\tlet valid = false;\n\t\t\t\t\tif (input.is('[type=file]'))\n\t\t\t\t\t\tvalid = input.closest('.k-upload').find('.k-file').length > 0;\n\n\t\t\t\t\treturn valid;\n\t            }\n\t\t\t}\n\t\t}).data('kendoValidator');\n\n\t\t// Initialize kendo validation on passwords subform\n\t\tthis._validator2 = modalElement.find('#passwordsSubform').kendoValidator({\n\t\t\tvalidateOnBlur: true,\n\t\t\trules: {\n\t\t\t\tverifyKsPassword: $.proxy(function(input) {\n\t\t\t\t\tlet valid = true;\n\t\t\t\t\tif (input.is('[name=confirmKsPassword]'))\n\t\t\t\t\t\tvalid = input.val() === $(this).find('#ksPassword').val();\n\t\t\t\t\treturn valid;\n\t\t\t\t}, this),\n\t\t\t\tverifyAdmPassword: $.proxy(function(input) {\n\t\t\t\t\tlet valid = true;\n\t\t\t\t\tif (input.is('[name=confirmAdminPassword]'))\n\t\t\t\t\t\tvalid = input.val() === $(this).find('#adminPassword').val();\n\t\t\t\t\treturn valid;\n\t\t\t\t}, this)\n\t\t\t},\n\t\t\tmessages: {\n\t\t\t\tverifyKsPassword: 'Password not matching',\n\t\t\t\tverifyAdmPassword: 'Password not matching',\n\t\t\t}\n\t\t}).data('kendoValidator');\n\n\t\t// Initialize bootstrap modal\n\t\tmodalElement.modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t});\n\t}\n\n\t_destroyModal()\n\t{\n\t\tlet modalElement = $('#uploadModal', $(this));\n\n\t\tif (modalElement)\n\t\t{\n\t\t\t// Remove listeners\n\t\t\t$('#uploadSslButton', modalElement).off('click');\n\t\t\tmodalElement.off('hidden.bs.modal');\n\n\t\t\t// Destroy everything Kendo\n\t\t\tkendo.destroy(modalElement);\n\n\t\t\t// Dispose modal\n\t\t\tmodalElement.modal('dispose');\n\n\t\t\t// Remove html\n\t\t\tmodalElement.remove();\n\t\t\tmodalElement = null;\n\t\t}\n\t}\n\n\t_validate()\n\t{\n\t\tlet val1 = this._validator1.validate();\n\t\tlet val2 = this._validator2.validate();\n\n\t\treturn val1 && val2;\n\t}\n\n\t_startSslCertUpload()\n\t{\n\t\tif (!this._uploadTargetConfig)\n\t\t\tthrow new Error('Upload target configuration not set');\n\n\t\tlet modalElement = $('#uploadModal', $(this));\n\n\t\tif (modalElement)\n\t\t{\n\t\t\tlet certData = {};\n\t\t\tcertData.file = this._uploadWidget.getFiles()[0];\n\t\t\tcertData.ksPassword = $('#ksPassword', modalElement).val();\n\t\t\tcertData.adminPassword = $('#adminPassword', modalElement).val();\n\n\t\t\t// Disable modal\n\t\t\tthis.enabled = false;\n\n\t\t\t// Hide previous error and show spinner\n\t\t\t$('#uploadSpinner', modalElement).show();\n\t\t\t$('#uploadErrorMsg', modalElement).hide();\n\t\t\t$('#uploadErrorMsg', modalElement).text('');\n\n\t\t\t//=================================================================\n\n\t\t\t// Generate Init Vector\n\t\t\tlet rngValues = [];\n\t\t\tfor (let i = 0; i < 16; i++)\n\t\t\t\trngValues.push(Math.floor(Math.random() * 256));\n\n\t\t\tlet iv = new Uint8Array(rngValues);\n\n\t\t\t// Generate secret key by MD5-encoding admin password + session token\n\t\t\tlet md5Pass = this._binaryMD5(certData.adminPassword + this._uploadTargetConfig.sessionToken);\n\n\t\t\t// Encrypt keystore password via AES (128bit)\n\t\t\tlet encryptedPwd = this._aesEncrypt(certData.ksPassword, md5Pass, iv);\n\n\t\t\t// Encode IV using Base64\n\t\t\tlet encodedIv = this._b64Encode(iv);\n\n\t\t\t// Encode encrypted password using Base64\n\t\t\tlet encodedPwd = this._b64Encode(encryptedPwd);\n\n\t\t\t//=================================================================\n\n\t\t\t// Set parameters to be sent with the certificate keystore file\n\t\t\tconst params = new FormData();\n\t\t\tparams.append('files[]', certData.file.rawFile);\n\t\t\tparams.append('__iv', encodedIv);\n\t\t\tparams.append('__password', encodedPwd);\n\t\t\tparams.append('__module', this._uploadTargetConfig.moduleId);\n\n\t\t\t// Set destination url\n\t\t\tconst url = this._uploadTargetConfig.protocol + '://' + this._uploadTargetConfig.host + ':' + this._uploadTargetConfig.port + '/BlueBox/SFS2XFileUpload?sessHashId=' + this._uploadTargetConfig.sessionToken;\n\n\t\t\t// Start upload\n\t\t\tfetch(url, {\n\t\t\t\tmethod: 'POST',\n\t\t\t\tbody: params,\n\t\t\t\tmode: 'no-cors'\n\t\t\t}).then($.proxy(this._onSslCertUploadEnd, this));\n\t\t}\n\t}\n\n\t_onSslCertUploadEnd(response)\n\t{\n\t\t// Nothing to do: we have to wait the upload process completion to be signaled by the server through the dedicated Extension response\n\n\t\t//=================================================================\n\n\t\t// TODO Should we handle this response in some way? For some unknown reason we always get ok=false and status=0\n\t\t//console.log(response)\n\t\t//console.log(response.ok)\n\t\t//console.log(response.status)\n\t}\n\n\t/**\n\t * Method called by parent when upload failed.\n\t */\n\tonSslCertUploadError(error)\n\t{\n\t\tlet modalElement = $('#uploadModal', $(this));\n\n\t\tif (modalElement)\n\t\t{\n\t\t\t// Enable modal\n\t\t\tthis.enabled = true;\n\n\t\t\t// Show upload error\n\t\t\t$('#uploadErrorMsg', modalElement).show();\n\t\t\t$('#uploadErrorMsg', modalElement).text(error + '.');\n\n\t\t\t// Hide spinner\n\t\t\t$('#uploadSpinner', modalElement).hide();\n\t\t}\n\t}\n\n\t/**\n\t * Method called by parent when upload is completed successfully.\n\t */\n\tonSslCertUploadSuccess()\n\t{\n\t\tlet modalElement = $('#uploadModal', $(this));\n\n\t\tif (modalElement)\n\t\t{\n\t\t\t// Enable modal\n\t\t\tthis.enabled = true;\n\n\t\t\t// Hide spinner\n\t\t\t$('#uploadSpinner', modalElement).hide();\n\n\t\t\t// Hide modal\n\t\t\tmodalElement.modal('hide');\n\t\t}\n\t}\n\n\t// *****************************************************************\n\n\t/*\n\t * Takes a string and returns the MD5 as Uint8Array\n\t */\n\t_binaryMD5(str)\n\t{\n\t\tlet MD5 = new SFS2X.MD5();\n\t\tlet hexStr = MD5.hex_md5(str);\n\n\t\treturn this._hexByteStringToByteArray(hexStr);\n\t}\n\n\t/*\n\t * Hex bytes ---> Actual byte[] as Uint8Array\n\t */\n\t_hexByteStringToByteArray(hexByteString)\n\t{\n\t    let bytes = new Uint8Array(16); // MD5 fixed output size\n\n\t    for (let i = 0; i < hexByteString.length;)\n\t    {\n\t        let hexByte = hexByteString[i++] + hexByteString[i++];\n\t        let byte = parseInt(hexByte, 16);\n\n\t        bytes[i / 2 - 1] = byte;\n\t    }\n\n\t    return bytes;\n\t}\n\n\t/*\n\t * Encrypt using AES, mode CBC, PKCS#7 padding\n\t *\n\t * text \t\t-> the text we want to encode\n\t * key \t\t-> the AES key\n\t * iv  \t\t-> the AES/CBC init vector\n\t *\n\t * Returns \t-> Uint8Array\n\t */\n\t_aesEncrypt(text, key, iv)\n\t{\n\t\tlet textBytes = aesjs.utils.utf8.toBytes(text); \t\t// Get UTF-8 bytes\n\t\tlet aesCBC = new aesjs.ModeOfOperation.cbc(key, iv);\t// Init CBC mode\n\t\ttextBytes = aesjs.padding.pkcs7.pad(textBytes); \t\t// PKCS#7 padding\n\n\t\t// Encrypt\n\t\treturn aesCBC.encrypt(textBytes);\n\t}\n\n\t/*\n\t * Encode passed byte array --> Base64 representation\n\t * Returns --> string\n\t */\n\t_b64Encode(barray)\n\t{\n\t\treturn btoa(String.fromCharCode.apply(null, barray));\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('ssl-certificate-manager'))\n\twindow.customElements.define('ssl-certificate-manager', SslCertificateManager);\n","import {BaseModule} from './base-module';\nimport {ConfigInterfaceBuilder} from '../utils/uibuilder/config-interface-builder';\nimport {SslCertificateManager} from '../components/module-specific/ssl-certificate-manager';\n\nexport default class ServerConfigurator extends BaseModule\n{\n\tconstructor()\n\t{\n\t    super('serverConfig');\n\n\t\t// Outgoing requests\n\t\tthis.REQ_INIT = 'init';\n\t\tthis.REQ_GET_CONFIG = 'getConfig';\n\t\tthis.REQ_UPDATE_CONFIG = 'updConfig';\n\t\tthis.REQ_UPDATE_GEO_DB = 'updGeoDb';\n\n\t\t// Incoming responses\n\t\tthis.RESP_INIT = 'init';\n\t\tthis.RESP_CONFIG = 'config';\n\t\tthis.RESP_CONFIG_UPDATE_CONFIRM = 'configUpdate';\n\t\tthis.RESP_CONFIG_CHANGED_ALERT = 'configAlert';\n\t\tthis.RESP_SSL_UPLOAD_ERROR = 'sslUploadError';\n\t\tthis.RESP_SSL_UPLOAD_CONFIRM = 'sslUpload';\n\t\tthis.RESP_UPDATE_GEO_DB = 'geoDbUpdate';\n\t}\n\n\t//------------------------------------\n\t// COMMON MODULE INTERFACE METHODS\n\t// This members are used by the main controller\n\t// to communicate with the module's controller.\n\t// This methods override those in BaseModule class.\n\t//------------------------------------\n\n\tinitialize(idData, shellController)\n\t{\n\t\t// Call super method\n\t\tsuper.initialize(idData, shellController);\n\n\t\t// Initialize progress bar\n\t\t$('#src-progressBar').kendoProgressBar({\n\t\t\tmin: 0,\n            max: 100,\n\t\t\tvalue: false,\n            type: 'value',\n            animation: {\n                duration: 400\n            }\n        });\n\n\t\t// Create interface builder instance\n\t\tthis._interfaceBuilder = new ConfigInterfaceBuilder();\n\n\t\t// Add listener to GeoLite2 database update button\n\t\t$('#src-updateGeolocDbButton').on('click', $.proxy(this._onUpdateGeolocDbClick, this));\n\n\t\t// Add listener to interface buttons\n\t\t$('#src-reloadButton').on('click', $.proxy(this._onReloadClick, this));\n\t\t$('#src-submitButton').on('click', $.proxy(this._onSubmitClick, this));\n\n\t\t// Save ref to SSL Manager\n\t\tthis._sslCertManager = document.getElementById('src-sslCertManager');\n\n\t\t//-----------------------------------*/\n\n\t\t// Send initialization request\n\t\tthis.sendExtensionRequest(this.REQ_INIT);\n\t}\n\n\tdestroy()\n\t{\n\t\t// Call super method\n\t\tsuper.destroy();\n\n\t\t// Destroy SSL Certificate Manager\n\t\tthis._sslCertManager.destroy();\n\n\t\t// Remove interface buttons click listeners\n\t\t$('#src-updateGeolocDbButton').off('click');\n\t\t$('#src-reloadButton').off('click');\n\t\t$('#src-submitButton').off('click');\n\n\t\t// Clear tabs container\n\t\tthis._clearTabs();\n\t}\n\n\tonExtensionCommand(command, data)\n\t{\n\t\t// Initialization data received\n\t\tif (command == this.RESP_INIT)\n\t\t{\n\t\t\t// Retrieve module id sent by the server (required because multiple modules use file uploading service)\n\t\t\tconst uploadModuleId = data.getUtfString('modId');\n\n\t\t\t// Set SSL upload manager target configuration\n\t\t\tthis._sslCertManager.uploadTargetConfig = {\n\t\t\t\tsessionToken: this.smartFox.sessionToken,\n\t\t\t\thost: this.smartFox.config.host,\n\t\t\t\tport: this.smartFox.config.port,\n\t\t\t\tmoduleId: uploadModuleId,\n\t\t\t\tprotocol: this.smartFox.config.useSSL ? 'https' : 'http'\n\t\t\t};\n\n\t\t\t// Server sends a flag indicating if file uploads are locked\n\t\t\t// We use it to enable the \"Manage SSL certificate\" button\n\t\t\tthis._sslLocked = data.getBool('lock');\n\n\t\t\tif (!this._sslLocked)\n\t\t\t\t$('#src-manageSslWarn').hide();\n\n\t\t\t// Request configuration data to server instance\n\t\t\tthis.sendExtensionRequest(this.REQ_GET_CONFIG);\n\t\t}\n\n\t\t// Server configuration data received\n\t\telse if (command == this.RESP_CONFIG)\n\t\t{\n\t\t\t// Build user interface based on received data\n\t\t\tthis._interfaceBuilder.buildInterface(data.getSFSArray('settings'), 'src-tabNavigator', false);\n\n\t\t\t// Enable buttons\n\t\t\tthis._enableButtons(true);\n\n\t\t\t// Initialize TabNavigator-ralated widgets\n\t\t\tif (!this._tabNavInitialized)\n\t\t\t{\n\t\t\t\t// Enable scrolling tabs\n\t\t\t\t$('#src-tabNavigator > #tabs').scrollingTabs({\n\t\t\t\t\tbootstrapVersion: 4,\n\t\t\t\t\tscrollToTabEdge: true,\n\t\t\t\t\tenableSwiping: true,\n\t\t\t\t\tdisableScrollArrowsOnFullyScrolled: true,\n\t\t\t\t\tcssClassLeftArrow: 'fa fa-chevron-left',\n\t\t\t\t\tcssClassRightArrow: 'fa fa-chevron-right'\n\t\t\t\t});\n\n\t\t\t\tthis._tabNavInitialized = true;\n\t\t\t}\n\n\t\t\t// Run validation (to remove validation messages if data was reloaded)\n\t\t\tthis._interfaceBuilder.checkIsValid();\n\n\t\t\tthis._switchView('src-main');\n\t\t}\n\n\t\t// Server configuration update confirmation\n\t\telse if (command == this.RESP_CONFIG_UPDATE_CONFIRM)\n\t\t{\n\t\t\t// Enable buttons\n\t\t\tthis._enableButtons(true);\n\n\t\t\t// Enable form items\n\t\t\tthis._interfaceBuilder.disableInterface(false);\n\n\t\t\t// If the current user is the updater, show a notification\n\t\t\t// Otherwise, show a dialog box suggesting to reload\n\t\t\tlet updater = data.getUtfString('user');\n\n\t\t\tif (updater == this.smartFox.mySelf.name)\n\t\t\t{\n\t\t\t\t// Reset the 'modified' flag\n\t\t\t\tthis._interfaceBuilder.resetIsModified();\n\n\t\t\t\t// Display notification\n\t\t\t\tthis.shellCtrl.showNotification('Server settings updated', 'Changes will be applied on next server restart');\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Show alert\n\t\t\t\tthis.shellCtrl.showSimpleAlert(`Administrator ${updater} has modified the server settings; please reload to update your view.`);\n\n\t\t\t\t// Disable submit button\n\t\t\t\t$('#src-submitButton').attr('disabled', true);\n\t\t\t}\n\t\t}\n\n\t\t// Server configuration xml saved by an external process\n\t\telse if (command == this.RESP_CONFIG_CHANGED_ALERT)\n\t\t{\n\t\t\t// Show alert\n\t\t\tthis.shellCtrl.showSimpleAlert(`The system has modified the server settings automatically; please reload to update your view.`);\n\n\t\t\t// Disable submit button\n\t\t\t$('#src-submitButton').attr('disabled', true);\n\t\t}\n\n\t\t// SSL certificate upload error\n\t\telse if (command == this.RESP_SSL_UPLOAD_ERROR)\n\t\t{\n\t\t\tconst error = data.getUtfString('error');\n\n\t\t\t// Log warning\n\t\t\tthis.shellCtrl.logMessage(error, 'error');\n\n\t\t\t// Show error in manager window\n\t\t\tthis._sslCertManager.onSslCertUploadError(error);\n\t\t}\n\n\t\t// SSL certificate upload confirmed\n\t\telse if (command == this.RESP_SSL_UPLOAD_CONFIRM)\n\t\t{\n\t\t\t// Closw manager window\n\t\t\tthis._sslCertManager.onSslCertUploadSuccess();\n\n\t\t\tlet updater = data.getUtfString('user');\n\n\t\t\t// Display notification\n\t\t\tif (updater == this.smartFox.mySelf.name)\n\t\t\t\tthis.shellCtrl.showNotification('SSL certificate', 'SSL certificate keystore was uploaded successfully');\n\t\t\telse\n\t\t\t\tthis.shellCtrl.showNotification('SSL certificate', `Administrator ${updater} has uploaded a new SSL certificate keystore`);\n\n\t\t\t// When a certificate is uploaded, HTTPS is also enabled automatically:\n\t\t\t// we have to update the interface accordingly\n\t\t\tthis._updateConfigFormItemDisplayedValue('webServer.enableHttps', true);\n\t\t}\n\n\t\t// Geolocation database update confirmation\n\t\telse if (command == this.RESP_UPDATE_GEO_DB)\n\t\t{\n\t\t\t// Enable button\n\t\t\t$('#src-updateGeolocDbButton').attr('disabled', false);\n\n\t\t\t// Check success\n\t\t\tif (data.getBool('success'))\n\t\t\t{\n\t\t\t\t// Update displayed date\n\t\t\t\tthis._updateConfigFormItemDisplayedValue('adminHelper.geoDbReleaseDate', data.getUtfString('newRelDate'));\n\n\t\t\t\t// If the current user is the updater, also show a notification\n\t\t\t\tlet updater = data.getUtfString('user');\n\n\t\t\t\tif (updater == this.smartFox.mySelf.name)\n\t\t\t\t\tthis.shellCtrl.showNotification('Geolocation database updated', 'Latest release of the GeoLite2 Country database has been installed successfully');\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Show alert\n\t\t\t\tthis.shellCtrl.showSimpleAlert(data.getUtfString('error'));\n\t\t\t}\n\t\t}\n\t}\n\n\t//------------------------------------\n\t// PRIVATE METHODS\n\t//------------------------------------\n\n\t_enableButtons(enabled)\n\t{\n\t\t$('#src-reloadButton').attr('disabled', !enabled);\n\t\t$('#src-submitButton').attr('disabled', !enabled);\n\t\t$('#src-backupCheck').attr('disabled', !enabled);\n\n\t\t$('#src-updateGeolocDbButton').attr('disabled', !enabled);\n\n\t\tif (!this._sslLocked)\n\t\t\tthis._sslCertManager.enabled = enabled;\n\t}\n\n\t_switchView(viewId)\n\t{\n\t\tdocument.getElementById('src-viewstack').selectedElement = document.getElementById(viewId);\n\t}\n\n\t_clearTabs()\n\t{\n\t\t// Destroy scrolling tabs\n\t\t$('#src-tabNavigator #tabs').scrollingTabs('destroy');\n\n\t\t// Remove all tab navigator content\n\t\tthis._interfaceBuilder.destroyInterface();\n\t}\n\n\t_onUpdateGeolocDbClick()\n\t{\n\t\t// Disable button\n\t\t$('#src-updateGeolocDbButton').attr('disabled', true);\n\n\t\t// Send request to server\n\t\tthis.sendExtensionRequest(this.REQ_UPDATE_GEO_DB);\n\t}\n\n\t_onReloadClick()\n\t{\n\t\t// Disable buttons\n\t\tthis._enableButtons(false);\n\n\t\t// Switch to loading view\n\t\tthis._switchView('src-loading');\n\n\t\t// Hide validation messages\n\t\tthis._interfaceBuilder.resetValidation();\n\n\t\t// Request configuration data to server instance\n\t\tthis.sendExtensionRequest(this.REQ_GET_CONFIG);\n\t}\n\n\t_onSubmitClick()\n\t{\n\t\t// Check validity\n\t\tif (this._interfaceBuilder.checkIsValid())\n\t\t{\n\t\t\tlet changes = this._interfaceBuilder.getChangedData();\n\n\t\t\tif (changes.size() > 0)\n\t\t\t{\n\t\t\t\t// Disable buttons\n\t\t\t\tthis._enableButtons(false);\n\n\t\t\t\t// Disable form items\n\t\t\t\tthis._interfaceBuilder.disableInterface(true);\n\n\t\t\t\t// Send updated settings to server instance\n\t\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\t\tparams.putSFSArray('settings', changes);\n\t\t\t\tparams.putBool('backup', $('#src-backupCheck').prop('checked'));\n\n\t\t\t\tthis.sendExtensionRequest(this.REQ_UPDATE_CONFIG, params);\n\t\t\t}\n\t\t}\n\t\telse\n\t\t\tthis.shellCtrl.showSimpleAlert('Unable to submit configuration changes due to an invalid value; please verify the highlighted form fields in all tabs.', true);\n\t}\n\n\t_updateConfigFormItemDisplayedValue(configParamName, newValue)\n\t{\n\t\t// Get the relevant Configuration Form Item\n\t\tconst configFormItem = this._interfaceBuilder.getConfigFormItem(configParamName);\n\n\t\t// Update Configuration Parameter associated with the Configuration Form Item\n\t\tconfigFormItem.data.value = newValue;\n\t\tconfigFormItem.data.resetIsModified(); // This is needed to avoid the Configuration Parameter to flagged as 'changed'\n\n\t\t// Display the new value of the Configuration Form Item\n\t\tconfigFormItem._setWidgetValue(); // Display the new value in the config form item\n\t}\n}\n"],"mappings":";;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACndA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;A","sourceRoot":""}
@@ -0,0 +1,5 @@
1
+ <template-module class="module" id="template-module">
2
+ <div>
3
+ This is a template module
4
+ </div>
5
+ </template-module>
@@ -42,7 +42,17 @@
42
42
  <div id="znc-main" class="px-3 pt-3">
43
43
  <div id="znc-tabNavigator">
44
44
  <ul class="nav nav-tabs" id="tabs" role="tablist"></ul>
45
- <div class="tab-content" id="tabPanels"></div>
45
+ <div class="tab-content" id="tabPanels">
46
+ <div class="tab-pane" id="tabpane-wordsFilter" role="tabpanel" aria-labelledby="tab-wordsFilter">
47
+ <!-- A form is created here automatically, containing the words filter settings -->
48
+ <hr/>
49
+ <div class="custom-tab-content">
50
+ <!-- Custom tab content used in Zone configuration only (when displaying Room configuration this tab pane never shows up) -->
51
+ <label class="config-form-separator-label mb-3 font-weight-bold">Words Files Editor</label>
52
+ <words-files-manager id="znc-wordsFilesManager" class="form-group row"></words-files-manager>
53
+ </div>
54
+ </div>
55
+ </div>
46
56
  </div>
47
57
 
48
58
  <div id="znc-buttons" class="sticky-bottom-controls py-3 bg-color-default">
@@ -238,6 +238,14 @@
238
238
  <div class="text-muted">Service provided by <a href="https://www.geoplugin.com" target="_blank">geoPlugin</a>; includes GeoLite data created by <a href="https://www.maxmind.com" target="_blank">MaxMind</a>.</div>
239
239
  </div>
240
240
  <hr/>
241
+ <div class="custom-tab-content">
242
+ <label class="config-form-separator-label mb-3 font-weight-bold">Disconnect User</label>
243
+ <div>Force immediate user disconnection.</div>
244
+ <div class="row no-gutters my-2">
245
+ <button id="znm-disconnectBt" type="button" class="k-button k-primary my-1"><i class="fas fa-plug mr-1"></i>Disconnect</button>
246
+ </div>
247
+ </div>
248
+ <hr/>
241
249
  <div class="custom-tab-content">
242
250
  <label class="config-form-separator-label mb-3 font-weight-bold">Kick User</label>
243
251
  <div>Force user disconnection by kicking him/her; a message can be sent to the user to describe the reason of the action.</div>
@@ -0,0 +1,147 @@
1
+ <zone>
2
+ <name>BasicExamples</name>
3
+ <isCustomLogin>false</isCustomLogin>
4
+ <isForceLogout>false</isForceLogout>
5
+ <applyWordsFilterToUserName>true</applyWordsFilterToUserName>
6
+ <applyWordsFilterToRoomName>true</applyWordsFilterToRoomName>
7
+ <applyWordsFilterToPrivateMessages>true</applyWordsFilterToPrivateMessages>
8
+ <isFilterBuddyMessages>true</isFilterBuddyMessages>
9
+ <isEncrypted>false</isEncrypted>
10
+ <maxUsers>1000</maxUsers>
11
+ <maxUserVariablesAllowed>10</maxUserVariablesAllowed>
12
+ <maxRoomVariablesAllowed>10</maxRoomVariablesAllowed>
13
+ <minRoomNameChars>1</minRoomNameChars>
14
+ <maxRoomNameChars>20</maxRoomNameChars>
15
+ <maxRooms>500</maxRooms>
16
+ <maxRoomsCreatedPerUser>10</maxRoomsCreatedPerUser>
17
+ <userCountChangeUpdateInterval>1000</userCountChangeUpdateInterval>
18
+ <userReconnectionSeconds>0</userReconnectionSeconds>
19
+ <overrideMaxUserIdleTime>300</overrideMaxUserIdleTime>
20
+ <maxFailedLogins>2</maxFailedLogins>
21
+ <allowGuestUsers>true</allowGuestUsers>
22
+ <guestUserNamePrefix>Guest#</guestUserNamePrefix>
23
+ <publicRoomGroups>default,games,chats</publicRoomGroups>
24
+ <defaultRoomGroups>default,games,chats</defaultRoomGroups>
25
+ <defaultPlayerIdGeneratorClass></defaultPlayerIdGeneratorClass>
26
+ <allowInvitationsOnlyForBuddies>true</allowInvitationsOnlyForBuddies>
27
+ <maxUsersPerJoinInvitationRequest>5</maxUsersPerJoinInvitationRequest>
28
+ <geoLocationEnabled>true</geoLocationEnabled>
29
+ <wordsFilter active="false">
30
+ <useWarnings>false</useWarnings>
31
+ <warningsBeforeKick>3</warningsBeforeKick>
32
+ <kicksBeforeBan>2</kicksBeforeBan>
33
+ <banDuration>1440</banDuration>
34
+ <maxBadWordsPerMessage>0</maxBadWordsPerMessage>
35
+ <kicksBeforeBanMinutes>3</kicksBeforeBanMinutes>
36
+ <secondsBeforeBanOrKick>5</secondsBeforeBanOrKick>
37
+ <warningMessage>Stop swearing or you will be banned</warningMessage>
38
+ <kickMessage>Swearing not allowed: you are being kicked</kickMessage>
39
+ <banMessage>Too much swearing: you are banned</banMessage>
40
+ <wordsFile>config/wordsFile.txt</wordsFile>
41
+ <filterMode>BLACKLIST</filterMode>
42
+ <banMode>NAME</banMode>
43
+ <hideBadWordWithCharacter>*</hideBadWordWithCharacter>
44
+ </wordsFilter>
45
+ <floodFilter active="false">
46
+ <banDurationMinutes>1440</banDurationMinutes>
47
+ <maxFloodingAttempts>5</maxFloodingAttempts>
48
+ <secondsBeforeBan>5</secondsBeforeBan>
49
+ <banMode>NAME</banMode>
50
+ <logFloodingAttempts>true</logFloodingAttempts>
51
+ <banMessage>Too much flooding, you are banned</banMessage>
52
+ </floodFilter>
53
+ <rooms>
54
+ <room>
55
+ <name>The Lobby</name>
56
+ <groupId>default</groupId>
57
+ <password></password>
58
+ <maxUsers>150</maxUsers>
59
+ <maxSpectators>0</maxSpectators>
60
+ <isDynamic>false</isDynamic>
61
+ <isGame>false</isGame>
62
+ <isHidden>false</isHidden>
63
+ <autoRemoveMode>DEFAULT</autoRemoveMode>
64
+ <permissions>
65
+ <flags>PUBLIC_MESSAGES</flags>
66
+ <maxRoomVariablesAllowed>10</maxRoomVariablesAllowed>
67
+ </permissions>
68
+ <events>USER_ENTER_EVENT,USER_EXIT_EVENT,USER_COUNT_CHANGE_EVENT,USER_VARIABLES_UPDATE_EVENT</events>
69
+ <badWordsFilter isActive="false"/>
70
+ <roomVariables/>
71
+ <extension>
72
+ <name></name>
73
+ <type>JAVA</type>
74
+ <file></file>
75
+ <propertiesFile></propertiesFile>
76
+ <reloadMode>AUTO</reloadMode>
77
+ </extension>
78
+ </room>
79
+ </rooms>
80
+ <disabledSystemEvents/>
81
+ <privilegeManager active="false">
82
+ <profiles>
83
+ <profile id="0">
84
+ <name>Guest</name>
85
+ <deniedRequests/>
86
+ <permissionFlags>
87
+ <string>ExtensionCalls</string>
88
+ </permissionFlags>
89
+ </profile>
90
+ <profile id="1">
91
+ <name>Standard</name>
92
+ <deniedRequests/>
93
+ <permissionFlags>
94
+ <string>ExtensionCalls</string>
95
+ </permissionFlags>
96
+ </profile>
97
+ <profile id="2">
98
+ <name>Moderator</name>
99
+ <deniedRequests/>
100
+ <permissionFlags>
101
+ <string>ExtensionCalls</string>
102
+ <string>SuperUser</string>
103
+ </permissionFlags>
104
+ </profile>
105
+ <profile id="3">
106
+ <name>Administrator</name>
107
+ <deniedRequests/>
108
+ <permissionFlags>
109
+ <string>ExtensionCalls</string>
110
+ <string>SuperUser</string>
111
+ </permissionFlags>
112
+ </profile>
113
+ </profiles>
114
+ </privilegeManager>
115
+ <extension>
116
+ <name></name>
117
+ <type>JAVA</type>
118
+ <file></file>
119
+ <propertiesFile></propertiesFile>
120
+ <reloadMode>AUTO</reloadMode>
121
+ </extension>
122
+ <buddyList active="true">
123
+ <allowOfflineBuddyVariables>true</allowOfflineBuddyVariables>
124
+ <maxItemsPerList>100</maxItemsPerList>
125
+ <maxBuddyVariables>15</maxBuddyVariables>
126
+ <offlineBuddyVariablesCacheSize>500</offlineBuddyVariablesCacheSize>
127
+ <customStorageClass></customStorageClass>
128
+ <useTempBuddies>true</useTempBuddies>
129
+ <buddyStates>
130
+ <string>Available</string>
131
+ <string>Away</string>
132
+ <string>Occupied</string>
133
+ </buddyStates>
134
+ <badWordsFilter isActive="true"/>
135
+ </buddyList>
136
+ <databaseManager active="false">
137
+ <driverName></driverName>
138
+ <connectionString></connectionString>
139
+ <userName></userName>
140
+ <password></password>
141
+ <testSql></testSql>
142
+ <maxActiveConnections>10</maxActiveConnections>
143
+ <maxIdleConnections>10</maxIdleConnections>
144
+ <exhaustedPoolAction>FAIL</exhaustedPoolAction>
145
+ <blockTime>3000</blockTime>
146
+ </databaseManager>
147
+ </zone>
@@ -1,5 +1,5 @@
1
1
  module Rubyfox
2
2
  module Server
3
- VERSION = "2.16.0.4"
3
+ VERSION = "2.16.3.1"
4
4
  end
5
5
  end
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: rubyfox-server
3
3
  version: !ruby/object:Gem::Version
4
- version: 2.16.0.4
4
+ version: 2.16.3.1
5
5
  platform: ruby
6
6
  authors:
7
7
  - Peter Leitzen
@@ -9,7 +9,7 @@ authors:
9
9
  autorequire:
10
10
  bindir: bin
11
11
  cert_chain: []
12
- date: 2020-12-07 00:00:00.000000000 Z
12
+ date: 2020-12-09 00:00:00.000000000 Z
13
13
  dependencies:
14
14
  - !ruby/object:Gem::Dependency
15
15
  name: thor
@@ -1121,6 +1121,7 @@ files:
1121
1121
  - lib/rubyfox/server/data/www/ROOT/admin/modules/log-viewer.html
1122
1122
  - lib/rubyfox/server/data/www/ROOT/admin/modules/server-configurator.html
1123
1123
  - lib/rubyfox/server/data/www/ROOT/admin/modules/servlet-manager.html
1124
+ - lib/rubyfox/server/data/www/ROOT/admin/modules/template.html
1124
1125
  - lib/rubyfox/server/data/www/ROOT/admin/modules/zone-configurator.html
1125
1126
  - lib/rubyfox/server/data/www/ROOT/admin/modules/zone-monitor.html
1126
1127
  - lib/rubyfox/server/data/www/ROOT/crossdomain.xml
@@ -1154,6 +1155,7 @@ files:
1154
1155
  - lib/rubyfox/server/data/www/manager/status.xsd
1155
1156
  - lib/rubyfox/server/data/www/manager/xform.xsl
1156
1157
  - lib/rubyfox/server/data/zones/.keep
1158
+ - lib/rubyfox/server/data/zones/BasicExamples.zone.xml
1157
1159
  - lib/rubyfox/server/environment.rb
1158
1160
  - lib/rubyfox/server/third-party-licenses/Antlr-license.txt
1159
1161
  - lib/rubyfox/server/third-party-licenses/StringTemplate-license.txt