rubyfox-server 2.17.3.2 → 2.19.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (160) hide show
  1. checksums.yaml +4 -4
  2. data/lib/rubyfox/server/data/config/admin/descriptors/config_room.txt +10 -1
  3. data/lib/rubyfox/server/data/config/admin/descriptors/config_server.txt +90 -20
  4. data/lib/rubyfox/server/data/config/admin/descriptors/config_zone.txt +9 -0
  5. data/lib/rubyfox/server/data/config/admin/descriptors/runtime_room.txt +11 -0
  6. data/lib/rubyfox/server/data/config/admin/descriptors/runtime_user.txt +3 -3
  7. data/lib/rubyfox/server/data/config/core.xml +4 -4
  8. data/lib/rubyfox/server/data/config/default.words.txt +11 -0
  9. data/lib/rubyfox/server/data/config/log4j.properties +1 -2
  10. data/lib/rubyfox/server/data/config/server.xml +1 -1
  11. data/lib/rubyfox/server/data/data/GeoLite2-Country.mmdb +0 -0
  12. data/lib/rubyfox/server/data/data/bannedusers/users.bin +0 -0
  13. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/bootstrap.jar +0 -0
  14. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/catalina-tasks.xml +39 -39
  15. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/catalina.sh +0 -0
  16. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/ciphers.sh +0 -0
  17. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/commons-daemon-native.tar.gz +0 -0
  18. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/commons-daemon.jar +0 -0
  19. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/configtest.sh +0 -0
  20. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/daemon.sh +0 -0
  21. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/digest.sh +0 -0
  22. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/makebase.sh +0 -0
  23. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/setclasspath.sh +0 -0
  24. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/shutdown.sh +0 -0
  25. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/startup.sh +0 -0
  26. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/tomcat-juli.jar +0 -0
  27. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/tomcat-native.tar.gz +0 -0
  28. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/tool-wrapper.sh +0 -0
  29. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/version.sh +0 -0
  30. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/Catalina/localhost/rewrite.config +1 -1
  31. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/catalina.policy +263 -263
  32. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/catalina.properties +209 -207
  33. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/context.xml +31 -31
  34. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/jaspic-providers.xml +23 -23
  35. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/jaspic-providers.xsd +52 -52
  36. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/keystore.jks +0 -0
  37. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/server.xml +177 -161
  38. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/tomcat-users.xml +18 -7
  39. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/tomcat-users.xsd +59 -59
  40. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/web.xml +4740 -4737
  41. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/annotations-api.jar +0 -0
  42. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ant.jar +0 -0
  43. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ha.jar +0 -0
  44. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ssi.jar +0 -0
  45. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-storeconfig.jar +0 -0
  46. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-tribes.jar +0 -0
  47. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina.jar +0 -0
  48. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/el-api.jar +0 -0
  49. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jasper-el.jar +0 -0
  50. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jasper.jar +0 -0
  51. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jaspic-api.jar +0 -0
  52. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jsp-api.jar +0 -0
  53. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/servlet-api.jar +0 -0
  54. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/sfs2x-ws-helper.jar +0 -0
  55. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-api.jar +0 -0
  56. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-coyote.jar +0 -0
  57. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-dbcp.jar +0 -0
  58. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-cs.jar +0 -0
  59. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-de.jar +0 -0
  60. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-es.jar +0 -0
  61. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-fr.jar +0 -0
  62. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ja.jar +0 -0
  63. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ko.jar +0 -0
  64. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-pt-BR.jar +0 -0
  65. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ru.jar +0 -0
  66. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-zh-CN.jar +0 -0
  67. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-jdbc.jar +0 -0
  68. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-jni.jar +0 -0
  69. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-util-scan.jar +0 -0
  70. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-util.jar +0 -0
  71. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-websocket.jar +0 -0
  72. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/websocket-api.jar +0 -0
  73. data/lib/rubyfox/server/data/lib/javax.activation-1.2.0.jar +0 -0
  74. data/lib/rubyfox/server/data/lib/javax.mail.jar +0 -0
  75. data/lib/rubyfox/server/data/lib/js/JSApi.js +2 -1
  76. data/lib/rubyfox/server/data/lib/js/LibApi.js +181 -48
  77. data/lib/rubyfox/server/data/lib/sfs2x-admin.jar +0 -0
  78. data/lib/rubyfox/server/data/lib/sfs2x-cluster.jar +0 -0
  79. data/lib/rubyfox/server/data/lib/sfs2x-core.jar +0 -0
  80. data/lib/rubyfox/server/data/lib/sfs2x.jar +0 -0
  81. data/lib/rubyfox/server/data/sfs2x-service +26 -30
  82. data/lib/rubyfox/server/data/www/BlueBox.war +0 -0
  83. data/lib/rubyfox/server/data/www/HelloServlet/WEB-INF/web.xml +1 -3
  84. data/lib/rubyfox/server/data/www/ROOT/_css_/default.css +14 -6
  85. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/style.css +44 -2
  86. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/application.bundle.js +98 -61
  87. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/endors~mod-0~mod-1~mod-11~mod-12~mod-17~mod-6~mod-7~mod-8~mod-9.bundle.js +17357 -0
  88. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-0.bundle.js +4 -4
  89. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-1.bundle.js +3 -3
  90. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-10.bundle.js +101 -66
  91. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-11.bundle.js +544 -8
  92. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12.bundle.js +915 -1480
  93. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12~module-15~module-16~module-4.bundle.js +2665 -0
  94. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-13.bundle.js +606 -3093
  95. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-13~module-16~module-17~module-4.bundle.js +2665 -0
  96. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-14.bundle.js +764 -0
  97. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-15.bundle.js +71 -0
  98. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-16.bundle.js +1787 -0
  99. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-17.bundle.js +3383 -0
  100. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-4.bundle.js +121 -1009
  101. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-5.bundle.js +1214 -1744
  102. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-6.bundle.js +398 -666
  103. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-7.bundle.js +717 -192
  104. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-8.bundle.js +2117 -665
  105. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-9.bundle.js +613 -690
  106. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~mod-0~mod-1~mod-10~mod-11~mod-16~mod-5~mod-6~mod-7~mod-8.bundle.js +17357 -0
  107. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~mod-0~mod-1~mod-11~mod-12~mod-17~mod-5~mod-6~mod-7~mod-8~mod-9.bundle.js +17357 -0
  108. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/{vendors~module-0~module-1~module-13~module-4~module-5~module-7~module-8.bundle.js → vendors~mod-0~mod-1~mod-11~mod-12~mod-17~mod-5~mod-7~mod-8~mod-9.bundle.js} +2 -2
  109. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-12.bundle.js +807 -0
  110. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-13.bundle.js +807 -0
  111. data/lib/rubyfox/server/data/www/ROOT/admin/modules/cluster-configurator.html +32 -0
  112. data/lib/rubyfox/server/data/www/ROOT/admin/modules/cluster-monitor.html +185 -0
  113. data/lib/rubyfox/server/data/www/ROOT/admin/modules/cluster-updater.html +47 -0
  114. data/lib/rubyfox/server/data/www/ROOT/admin/modules/extension-deployer.html +84 -0
  115. data/lib/rubyfox/server/data/www/ROOT/admin/modules/zone-monitor.html +15 -8
  116. data/lib/rubyfox/server/data/www/ROOT/index.html +13 -23
  117. data/lib/rubyfox/server/data/www/host-manager/META-INF/context.xml +2 -2
  118. data/lib/rubyfox/server/data/www/host-manager/WEB-INF/jsp/404.jsp +2 -2
  119. data/lib/rubyfox/server/data/www/host-manager/{manager.xml → WEB-INF/manager.xml} +5 -1
  120. data/lib/rubyfox/server/data/www/host-manager/WEB-INF/web.xml +17 -0
  121. data/lib/rubyfox/server/data/www/host-manager/css/manager.css +141 -0
  122. data/lib/rubyfox/server/data/www/host-manager/images/tomcat.svg +967 -0
  123. data/lib/rubyfox/server/data/www/manager/META-INF/context.xml +2 -0
  124. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/connectorCerts.jsp +1 -1
  125. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/connectorCiphers.jsp +1 -1
  126. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/connectorTrustedCerts.jsp +1 -1
  127. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/sessionDetail.jsp +3 -3
  128. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/sessionsList.jsp +1 -1
  129. data/lib/rubyfox/server/data/www/manager/WEB-INF/web.xml +17 -0
  130. data/lib/rubyfox/server/data/www/manager/css/manager.css +141 -0
  131. data/lib/rubyfox/server/data/www/manager/images/tomcat.svg +967 -0
  132. data/lib/rubyfox/server/data/www/manager/xform.xsl +74 -59
  133. data/lib/rubyfox/server/version.rb +1 -1
  134. metadata +30 -31
  135. data/lib/rubyfox/server/data/config/admin/icons/Analytics.png +0 -0
  136. data/lib/rubyfox/server/data/config/admin/icons/BanManager.png +0 -0
  137. data/lib/rubyfox/server/data/config/admin/icons/BlueBoxMonitor.png +0 -0
  138. data/lib/rubyfox/server/data/config/admin/icons/Console.png +0 -0
  139. data/lib/rubyfox/server/data/config/admin/icons/Dashboard.png +0 -0
  140. data/lib/rubyfox/server/data/config/admin/icons/ExtensionManager.png +0 -0
  141. data/lib/rubyfox/server/data/config/admin/icons/LicenseManager.png +0 -0
  142. data/lib/rubyfox/server/data/config/admin/icons/LogViewer.png +0 -0
  143. data/lib/rubyfox/server/data/config/admin/icons/ServerConfigurator.png +0 -0
  144. data/lib/rubyfox/server/data/config/admin/icons/ServletManager.png +0 -0
  145. data/lib/rubyfox/server/data/config/admin/icons/ZoneConfigurator.png +0 -0
  146. data/lib/rubyfox/server/data/config/admin/icons/ZoneMonitor.png +0 -0
  147. data/lib/rubyfox/server/data/lib/BlueBox.war +0 -0
  148. data/lib/rubyfox/server/data/lib/apache-tomcat/LICENSE +0 -1061
  149. data/lib/rubyfox/server/data/lib/apache-tomcat/NOTICE +0 -68
  150. data/lib/rubyfox/server/data/lib/apache-tomcat/README.md +0 -81
  151. data/lib/rubyfox/server/data/lib/apache-tomcat/RELEASE-NOTES +0 -174
  152. data/lib/rubyfox/server/data/lib/imap.jar +0 -0
  153. data/lib/rubyfox/server/data/lib/mailapi.jar +0 -0
  154. data/lib/rubyfox/server/data/lib/pop3.jar +0 -0
  155. data/lib/rubyfox/server/data/lib/smtp.jar +0 -0
  156. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12~module-13~module-9.bundle.js +0 -2634
  157. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-9.bundle.js +0 -807
  158. data/lib/rubyfox/server/data/www/host-manager/images/tomcat.gif +0 -0
  159. data/lib/rubyfox/server/data/www/manager/images/tomcat.gif +0 -0
  160. /data/lib/rubyfox/server/data/data/buddylists/{BasicExamples/.keep → .keep} +0 -0
@@ -0,0 +1,2665 @@
1
+ /*! (c) gotoAndPlay | All rights reserved */
2
+ (window["webpackJsonpapplication"] = window["webpackJsonpapplication"] || []).push([["module-13~module-16~module-17~module-4"],{
3
+
4
+ /***/ "./src/components/uibuilder/config-check-box.js":
5
+ /*!******************************************************!*\
6
+ !*** ./src/components/uibuilder/config-check-box.js ***!
7
+ \******************************************************/
8
+ /*! exports provided: ConfigCheckBox */
9
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
10
+
11
+ "use strict";
12
+ __webpack_require__.r(__webpack_exports__);
13
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "ConfigCheckBox", function() { return ConfigCheckBox; });
14
+ /* harmony import */ var _config_form_item__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./config-form-item */ "./src/components/uibuilder/config-form-item.js");
15
+ /* harmony import */ var _config_label__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./config-label */ "./src/components/uibuilder/config-label.js");
16
+
17
+
18
+
19
+ class ConfigCheckBox extends _config_form_item__WEBPACK_IMPORTED_MODULE_0__["ConfigFormItem"]
20
+ {
21
+ constructor(configParam, editEnabled, inDialog)
22
+ {
23
+ super(configParam, editEnabled, inDialog);
24
+ }
25
+
26
+ /**
27
+ * Create widget to render the ConfigParameter value.
28
+ * If parameter is not editable, a simple label is used.
29
+ * @override
30
+ */
31
+ _generateInnerWidget()
32
+ {
33
+ if (this._data.editable)
34
+ {
35
+ // Set widget configuration
36
+ let config = {
37
+ type: 'checkbox',
38
+ class: '',
39
+ id: this._data.name,
40
+ name: this._data.name,
41
+ 'data-role': 'switch',
42
+ };
43
+
44
+ // Set widget attributes (see parent class)
45
+ this._setWidgetAttributes(config);
46
+
47
+ // Set additional widget attributes based on validation rules (see parent class)
48
+ this._setWidgetValidationAttributes(config);
49
+
50
+ // Create widget's html
51
+ this._widgetHtml = $('<input>', config);
52
+ }
53
+ else
54
+ this._widgetHtml = new _config_label__WEBPACK_IMPORTED_MODULE_1__["ConfigLabel"]();
55
+
56
+ // Return component
57
+ return this._widgetHtml;
58
+ }
59
+
60
+ /**
61
+ * Initialize widget.
62
+ * @override
63
+ */
64
+ _initialize()
65
+ {
66
+ if (this._data.editable)
67
+ {
68
+ // Initialize kendo widget
69
+ kendo.init(this._widgetHtml);
70
+
71
+ // Save ref. to widget
72
+ this._innerWidget = this._widgetHtml.data('kendoSwitch');
73
+
74
+ // Enable value commit binding
75
+ this._innerWidget.bind('change', $.proxy(this._onValueInput, this));
76
+ }
77
+
78
+ // Proceed with initialization
79
+ super._initialize();
80
+ }
81
+
82
+ /**
83
+ * Set widget's value.
84
+ * If parameter is not editable, the label text is set.
85
+ * @override
86
+ */
87
+ _setWidgetValue()
88
+ {
89
+ if (this._data.editable)
90
+ this._innerWidget.value(this._data.value);
91
+ else
92
+ this._widgetHtml.value = this._data.value;
93
+
94
+ // Trigger event
95
+ this._triggerEvent();
96
+ }
97
+
98
+ /**
99
+ * Set widget's disabled state.
100
+ * @override
101
+ */
102
+ _setWidgetEditEnabled()
103
+ {
104
+ if (this._data.editable)
105
+ this._innerWidget.enable(this._editEnabled);
106
+ }
107
+
108
+ /**
109
+ * Update Configuration Parameter value.
110
+ * @override
111
+ */
112
+ _onValueInput(e)
113
+ {
114
+ // Update Configuration Parameter to new value
115
+ this._data.value = this._innerWidget.value();
116
+
117
+ // Trigger event
118
+ this._triggerEvent();
119
+ }
120
+ }
121
+
122
+ // DEFINE COMPONENT
123
+ if (!window.customElements.get('config-check-box'))
124
+ window.customElements.define('config-check-box', ConfigCheckBox);
125
+
126
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
127
+
128
+ /***/ }),
129
+
130
+ /***/ "./src/components/uibuilder/config-drop-down-list.js":
131
+ /*!***********************************************************!*\
132
+ !*** ./src/components/uibuilder/config-drop-down-list.js ***!
133
+ \***********************************************************/
134
+ /*! exports provided: ConfigDropDownList */
135
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
136
+
137
+ "use strict";
138
+ __webpack_require__.r(__webpack_exports__);
139
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "ConfigDropDownList", function() { return ConfigDropDownList; });
140
+ /* harmony import */ var _config_form_item__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./config-form-item */ "./src/components/uibuilder/config-form-item.js");
141
+ /* harmony import */ var _config_label__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./config-label */ "./src/components/uibuilder/config-label.js");
142
+
143
+
144
+
145
+ class ConfigDropDownList extends _config_form_item__WEBPACK_IMPORTED_MODULE_0__["ConfigFormItem"]
146
+ {
147
+ constructor(configParam, editEnabled, inDialog)
148
+ {
149
+ super(configParam, editEnabled, inDialog);
150
+ }
151
+
152
+ /**
153
+ * Create widget to render the ConfigParameter value.
154
+ * If parameter is not editable, a simple label is used.
155
+ * @override
156
+ */
157
+ _generateInnerWidget()
158
+ {
159
+ if (this._data.editable)
160
+ {
161
+ // Set widget configuration
162
+ let config = {
163
+ class: 'form-control',
164
+ id: this._data.name,
165
+ name: this._data.name,
166
+ 'data-role': 'dropdownlist',
167
+ };
168
+
169
+ // Set widget attributes (see parent class)
170
+ this._setWidgetAttributes(config);
171
+
172
+ // Set additional widget attributes based on validation rules (see parent class)
173
+ this._setWidgetValidationAttributes(config);
174
+
175
+ // Create widget's html
176
+ this._widgetHtml = $('<input>', config);
177
+ }
178
+ else
179
+ this._widgetHtml = new _config_label__WEBPACK_IMPORTED_MODULE_1__["ConfigLabel"]();
180
+
181
+ // Return component
182
+ return this._widgetHtml;
183
+ }
184
+
185
+ /**
186
+ * Initialize widget.
187
+ * @override
188
+ */
189
+ _initialize()
190
+ {
191
+ if (this._data.editable)
192
+ {
193
+ // Initialize kendo widget
194
+ kendo.init(this._widgetHtml);
195
+
196
+ // Save ref. to widget
197
+ this._innerWidget = this._widgetHtml.data('kendoDropDownList');
198
+
199
+ // Set list items
200
+ this._innerWidget.setDataSource(this._getDataSource(this._data.dataProvider))
201
+
202
+ // Enable value commit binding
203
+ this._widgetHtml.bind('change', $.proxy(this._onValueInput, this));
204
+ }
205
+
206
+ // Proceed with initialization
207
+ super._initialize();
208
+ }
209
+
210
+ /**
211
+ * Set widget's value.
212
+ * If parameter is not editable, the label text is set.
213
+ * @override
214
+ */
215
+ _setWidgetValue()
216
+ {
217
+ if (this._data.editable)
218
+ this._innerWidget.value(this._data.value);
219
+ else
220
+ this._widgetHtml.value = this._data.value;
221
+
222
+ // Trigger event
223
+ this._triggerEvent();
224
+ }
225
+
226
+ /**
227
+ * Set widget's disabled state.
228
+ * @override
229
+ */
230
+ _setWidgetEditEnabled()
231
+ {
232
+ if (this._data.editable)
233
+ this._innerWidget.wrapper.attr('disabled', !this._editEnabled);
234
+ }
235
+
236
+ /**
237
+ * Update Configuration Parameter value.
238
+ * @override
239
+ */
240
+ _onValueInput(e)
241
+ {
242
+ // Update Configuration Parameter to new value
243
+ this._data.value = this._innerWidget.value();
244
+
245
+ // Trigger event
246
+ this._triggerEvent();
247
+ }
248
+
249
+ _getDataSource(dpString)
250
+ {
251
+ if (dpString)
252
+ return dpString.split(',');
253
+
254
+ // In case the dataprovider is empty, add at least the current value
255
+ else
256
+ return [this._data.value];
257
+ }
258
+ }
259
+
260
+ // DEFINE COMPONENT
261
+ if (!window.customElements.get('config-drop-down-list'))
262
+ window.customElements.define('config-drop-down-list', ConfigDropDownList);
263
+
264
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
265
+
266
+ /***/ }),
267
+
268
+ /***/ "./src/components/uibuilder/config-dual-list.js":
269
+ /*!******************************************************!*\
270
+ !*** ./src/components/uibuilder/config-dual-list.js ***!
271
+ \******************************************************/
272
+ /*! exports provided: ConfigDualList */
273
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
274
+
275
+ "use strict";
276
+ __webpack_require__.r(__webpack_exports__);
277
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "ConfigDualList", function() { return ConfigDualList; });
278
+ /* harmony import */ var _config_form_item__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./config-form-item */ "./src/components/uibuilder/config-form-item.js");
279
+ /* harmony import */ var _config_label__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./config-label */ "./src/components/uibuilder/config-label.js");
280
+
281
+
282
+
283
+ class ConfigDualList extends _config_form_item__WEBPACK_IMPORTED_MODULE_0__["ConfigFormItem"]
284
+ {
285
+ constructor(configParam, editEnabled, inDialog)
286
+ {
287
+ super(configParam, editEnabled, inDialog);
288
+ }
289
+
290
+ /**
291
+ * Create widget to render the ConfigParameter.
292
+ * @override
293
+ */
294
+ _generateInnerWidget()
295
+ {
296
+ this._widgetHtml = $('<div>');
297
+
298
+ const availableId = this._getId(this._data.name, 'available');
299
+ const selectedId = this._getId(this._data.name, 'selected');
300
+
301
+ // Create header for labels
302
+ let header = $('<div>', {class: 'form-label-container dual-list-labels'});
303
+
304
+ header.append($('<label>', {
305
+ class: 'font-italic form-label dual-list-left-col' + (!this._data.editable ? ' no-interact' : ''),
306
+ for: availableId,
307
+ }).text('Available'));
308
+
309
+ header.append($('<label>', {
310
+ class: 'font-italic font-weight-bold form-label dual-list-right-col' + (!this._data.editable ? ' no-interact' : ''),
311
+ for: selectedId,
312
+ }).text('Selected'));
313
+
314
+ this._widgetHtml.append(header);
315
+
316
+ // Add available items list
317
+ this._availableListHtml = $('<select>', {
318
+ id: availableId,
319
+ class: 'dual-list-left-col' + (!this._data.editable ? ' no-interact' : ''),
320
+ });
321
+ this._widgetHtml.append(this._availableListHtml);
322
+
323
+ // Add selected items list
324
+ this._selectedListHtml = $('<select>', {
325
+ id: selectedId,
326
+ class: 'dual-list-right-col' + (!this._data.editable ? ' no-interact' : ''),
327
+ });
328
+ this._widgetHtml.append(this._selectedListHtml);
329
+
330
+ // Return component
331
+ return this._widgetHtml;
332
+ }
333
+
334
+ // IDs containing a "." cause issues to connected lists
335
+ _getId(name, suffix)
336
+ {
337
+ return name.replace('.', '_') + '-' + suffix;
338
+ }
339
+
340
+ /**
341
+ * Initialize widget.
342
+ * @override
343
+ */
344
+ _initialize()
345
+ {
346
+ // Initialize "avalable" listbox
347
+ this._availableList = this._availableListHtml.kendoListBox({
348
+ connectWith: this._getId(this._data.name, 'selected'),
349
+ toolbar: {
350
+ tools: this._data.editable ? ['transferTo', 'transferFrom', 'transferAllTo', 'transferAllFrom'] : []
351
+ },
352
+ template: "<div>#:value#</div>",
353
+ selectable: 'multiple',
354
+ }).data('kendoListBox');
355
+
356
+ // Initialize "selected" listbox
357
+ this._selectedList = this._selectedListHtml.kendoListBox({
358
+ template: "<div>#:value#</div>",
359
+ selectable: 'multiple',
360
+ // The following listeners can't be used because events are fired before the datasource is actually updated
361
+ // We have to use a change event listener on the datasource (see below), even if not optimal
362
+ //add: $.proxy(this._onValueInput, this),
363
+ //remove: $.proxy(this._onValueInput, this),
364
+ }).data('kendoListBox');
365
+
366
+ // Proceed with initialization
367
+ super._initialize();
368
+ }
369
+
370
+ /**
371
+ * Set widget's datasource.
372
+ * @override
373
+ */
374
+ _setWidgetValue()
375
+ {
376
+ let availableArr = this._data.dataProvider != '' ? this._data.dataProvider.split(',') : [];
377
+ let selectedArr = this._data.value != '' ? this._data.value.split(',') : [];
378
+
379
+ // Remove selected values from available values
380
+ if (selectedArr.length > 0)
381
+ {
382
+ let temp = [];
383
+
384
+ for (let val of availableArr)
385
+ {
386
+ if (selectedArr.indexOf(val) == -1)
387
+ temp.push(val);
388
+ }
389
+
390
+ availableArr = temp;
391
+ }
392
+
393
+ // Convert lists of strings to lists of objects
394
+ let availableValues = [];
395
+ for (let val of availableArr)
396
+ availableValues.push({value: val});
397
+
398
+ let selectedValues = [];
399
+ for (let val of selectedArr)
400
+ selectedValues.push({value: val});
401
+
402
+ // Clear selection
403
+ this._availableList.clearSelection();
404
+ this._selectedList.clearSelection();
405
+
406
+ // Set datasources
407
+ this._availableList.setDataSource(new kendo.data.DataSource({
408
+ data: availableValues
409
+ }));
410
+
411
+ this._selectedList.setDataSource(new kendo.data.DataSource({
412
+ data: selectedValues,
413
+ // We listen to the change event instead of the add/remove events on the listbox, because those are fired before the datasource is updated
414
+ // This is not optimal because the event is fired for each item added to or removed from the datasource
415
+ change: $.proxy(this._onValueInput, this)
416
+ }));
417
+
418
+ // Disable editing
419
+ if (!this._data.editable)
420
+ {
421
+ this._availableList.enable('.k-item', false);
422
+ this._selectedList.enable('.k-item', false);
423
+ }
424
+
425
+ // Trigger event
426
+ this._triggerEvent();
427
+ }
428
+
429
+ /**
430
+ * Set widget's disabled state.
431
+ * @override
432
+ */
433
+ _setWidgetEditEnabled()
434
+ {
435
+ if (this._data.editable)
436
+ {
437
+ // Clear selection
438
+ this._availableList.clearSelection();
439
+ this._selectedList.clearSelection();
440
+
441
+ // Enable/disable lists
442
+ this._availableList.wrapper.attr('disabled', !this._editEnabled);
443
+ this._selectedList.wrapper.attr('disabled', !this._editEnabled);
444
+ }
445
+ }
446
+
447
+ /**
448
+ * Update Configuration Parameter value.
449
+ * @override
450
+ */
451
+ _onValueInput(e)
452
+ {
453
+ let listData = this._selectedList.dataSource.data();
454
+
455
+ // Update Configuration Parameter to new value
456
+ this._data.value = listData.map(e => e.value).join(',');
457
+
458
+ // Trigger event
459
+ this._triggerEvent();
460
+ }
461
+ }
462
+
463
+ // DEFINE COMPONENT
464
+ if (!window.customElements.get('config-dual-list'))
465
+ window.customElements.define('config-dual-list', ConfigDualList);
466
+
467
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
468
+
469
+ /***/ }),
470
+
471
+ /***/ "./src/components/uibuilder/config-form-item.js":
472
+ /*!******************************************************!*\
473
+ !*** ./src/components/uibuilder/config-form-item.js ***!
474
+ \******************************************************/
475
+ /*! exports provided: ConfigFormItem */
476
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
477
+
478
+ "use strict";
479
+ __webpack_require__.r(__webpack_exports__);
480
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "ConfigFormItem", function() { return ConfigFormItem; });
481
+ class ConfigFormItem extends HTMLElement
482
+ {
483
+ constructor(configParam, editEnabled, inDialog)
484
+ {
485
+ super();
486
+
487
+ this.id = 'form-item-' + configParam.name;
488
+ this._editEnabled = editEnabled;
489
+ this._data = configParam;
490
+
491
+ // Create form item view
492
+ this._buildView(inDialog);
493
+
494
+ // Initialize form item
495
+ this._initialize();
496
+ }
497
+
498
+ connectedCallback()
499
+ {
500
+ // Trigger event
501
+ // NOTE: when a ConfigFormItem is instantiated, the _triggerEvent method is called as soon as its value is set.
502
+ // When this happens, due to the fact that the object is not yet in the DOM, the event is not catched by the listener
503
+ // (which is attached to the outer container). So forcing the event to trigger again as soon as the ConfigFormItem
504
+ // is appended to the DOM is needed.
505
+ this._triggerEvent();
506
+ }
507
+
508
+ set data(configParam)
509
+ {
510
+ this._data = configParam;
511
+ this._setWidgetValue();
512
+ }
513
+
514
+ get data()
515
+ {
516
+ return this._data;
517
+ }
518
+
519
+ set editEnabled(enable)
520
+ {
521
+ if (enable != this._editEnabled)
522
+ {
523
+ this._editEnabled = enable;
524
+ this._setWidgetEditEnabled();
525
+ }
526
+ }
527
+
528
+ get editEnabled()
529
+ {
530
+ return this._editEnabled;
531
+ }
532
+
533
+ _buildView(isInsideDialog)
534
+ {
535
+ if (!isInsideDialog)
536
+ {
537
+ // Set additional classes for inner widget
538
+ let classNames = '';
539
+
540
+ switch (this._data.type)
541
+ {
542
+ case 'DualList':
543
+ classNames = 'col-sm-7 col-lg-8';
544
+ break;
545
+ case 'DataGrid':
546
+ classNames = 'col-sm'; // Use 'col-sm-7 col-lg-8' for DataGrid too?
547
+ break;
548
+ case 'TextInput':
549
+ classNames = 'col-sm col-lg-5 col-xl-4';
550
+ break;
551
+ default:
552
+ classNames = 'col-sm-auto';
553
+
554
+ }
555
+
556
+ // Generate boilerplate html, surrounding the actual widget (label, numeric stepper, etc)
557
+ this.innerHTML = `
558
+ <div class="form-group position-relative row">
559
+ <div class="col-sm-5 col-lg-4 col-form-label form-label-container">
560
+ <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>${(this._data.immediate ? '<i class="fas fa-bolt text-muted ml-1 help" title="This setting is applied <b>immediately</b> on submit, without requiring a server restart"></i>' : '')}</label>
561
+ </div>
562
+ <div class="inner-widget align-self-center ${classNames}">
563
+ <span class="k-invalid-msg" data-for="${this._data.name}"></span>
564
+ </div>
565
+ </div>
566
+ `;
567
+ }
568
+ else
569
+ {
570
+ this.innerHTML = `
571
+ <div class="form-group position-relative">
572
+ <div class="col-form-label form-label-container">
573
+ <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>
574
+ </div>
575
+ <div class="inner-widget">
576
+ <span class="k-invalid-msg" data-for="${this._data.name}"></span>
577
+ </div>
578
+ </div>
579
+ `;
580
+ }
581
+
582
+ // Create inner widget (must be overridden)
583
+ let widget = this._generateInnerWidget();
584
+
585
+ // Append inner widget
586
+ $(this).find('.inner-widget').prepend(widget);
587
+ }
588
+
589
+ /**
590
+ * TO BE OVERRIDDEN
591
+ */
592
+ _generateInnerWidget()
593
+ {
594
+ // Show an error, should be overridden
595
+ console.error(`Unable to create ${this._data.type} form item for configuration parameter ${this.id}`);
596
+ }
597
+
598
+ /**
599
+ * Set attributes on the widget configuration object.
600
+ */
601
+ _setWidgetAttributes(config)
602
+ {
603
+ const attribs = this._data.attributes;
604
+
605
+ if (attribs)
606
+ {
607
+ for (let attr in attribs)
608
+ {
609
+ config[attr] = attribs[attr];
610
+
611
+ if (attr == 'pattern')
612
+ config['data-pattern-msg'] = 'Contains invalid characters';
613
+ }
614
+ }
615
+ }
616
+
617
+ /**
618
+ * Set additional attributes on the widget configuration object to properly validate input.
619
+ */
620
+ _setWidgetValidationAttributes(config)
621
+ {
622
+ const val = this._data.validator;
623
+
624
+ if (val != null && val != '')
625
+ {
626
+ if (val == 'ip')
627
+ {
628
+ config['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]?)$';
629
+ config['data-pattern-msg'] = 'Invalid IP address';
630
+ config['required'] = true;
631
+ config['data-required-msg'] = 'Required';
632
+ }
633
+
634
+ else if (val == 'ipRange')
635
+ {
636
+ config['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]?)(\/[3][0-2]|\/[1-2]?[0-9])?$';
637
+ config['data-pattern-msg'] = 'Invalid IP address or range';
638
+ config['required'] = true;
639
+ config['data-required-msg'] = 'Required';
640
+ }
641
+
642
+ else if (val == 'notNull')
643
+ {
644
+ config['required'] = true;
645
+ config['data-required-msg'] = 'Required';
646
+ }
647
+
648
+ else if (val == 'pwd')
649
+ {
650
+ config['pattern'] = '^.{6,}$';
651
+ config['data-pattern-msg'] = 'Minimum length: 6 characters';
652
+ }
653
+
654
+ else if (val == 'posNum')
655
+ {
656
+ config['pattern'] = '^[0-9]\d*$';
657
+ config['data-pattern-msg'] = 'Non-negative number required';
658
+ }
659
+
660
+ else if (val == 'aoi')
661
+ {
662
+ // Nothing to do
663
+ // See Kendo validation initialization in config-interface-builder.js
664
+ }
665
+
666
+ else if (val == 'url')
667
+ {
668
+ config['type'] = 'url';
669
+ config['data-url-msg'] = 'Invalid URL';
670
+ }
671
+ }
672
+ }
673
+
674
+ /**
675
+ * Initialize form item.
676
+ *
677
+ * NOTE: must be overridden if inner widget requires special initialization (for example Kendo widgets)
678
+ */
679
+ _initialize()
680
+ {
681
+ // Set value
682
+ this._setWidgetValue();
683
+
684
+ // Set edit enabled
685
+ this._setWidgetEditEnabled();
686
+ }
687
+
688
+ /**
689
+ * TO BE OVERRIDDEN
690
+ */
691
+ _setWidgetValue()
692
+ {
693
+ // Nothing to do, must be overridden
694
+ }
695
+
696
+ /**
697
+ * TO BE OVERRIDDEN
698
+ */
699
+ _setWidgetEditEnabled()
700
+ {
701
+ // Nothing to do, must be overridden
702
+ }
703
+
704
+ /**
705
+ * TO BE OVERRIDDEN
706
+ */
707
+ _onValueInput(e)
708
+ {
709
+ // Nothing to do, must be overridden
710
+ }
711
+
712
+ _triggerEvent()
713
+ {
714
+ if (this._data.trigger)
715
+ {
716
+ let event = new CustomEvent('value-set', {detail: null, bubbles: true, cancelable: true});
717
+ this.dispatchEvent(event);
718
+ }
719
+ }
720
+ }
721
+
722
+ // DEFINE COMPONENT
723
+ if (!window.customElements.get('config-form-item'))
724
+ window.customElements.define('config-form-item', ConfigFormItem);
725
+
726
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
727
+
728
+ /***/ }),
729
+
730
+ /***/ "./src/components/uibuilder/config-grid.js":
731
+ /*!*************************************************!*\
732
+ !*** ./src/components/uibuilder/config-grid.js ***!
733
+ \*************************************************/
734
+ /*! exports provided: ConfigGrid */
735
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
736
+
737
+ "use strict";
738
+ __webpack_require__.r(__webpack_exports__);
739
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "ConfigGrid", function() { return ConfigGrid; });
740
+ /* harmony import */ var _config_form_item__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./config-form-item */ "./src/components/uibuilder/config-form-item.js");
741
+ /* harmony import */ var _config_label__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./config-label */ "./src/components/uibuilder/config-label.js");
742
+ /* harmony import */ var _widgets_list_item_editor__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./widgets/list-item-editor */ "./src/components/uibuilder/widgets/list-item-editor.js");
743
+
744
+
745
+
746
+
747
+ class ConfigGrid extends _config_form_item__WEBPACK_IMPORTED_MODULE_0__["ConfigFormItem"]
748
+ {
749
+ constructor(configParam, editEnabled, inDialog)
750
+ {
751
+ super(configParam, editEnabled, inDialog);
752
+ }
753
+
754
+ /**
755
+ * Create widget to render the ConfigParameter.
756
+ * @override
757
+ */
758
+ _generateInnerWidget()
759
+ {
760
+ // Create main widget's html
761
+ this._widgetHtml = $('<div>', {class: ''});
762
+
763
+ // Set grid widget configuration
764
+ let gridConfig = {
765
+ id: this._data.name,
766
+ name: this._data.name,
767
+ class: 'limited-height' + (!this._data.editable ? ' no-interact' : '')
768
+ };
769
+
770
+ // Append grid to main html; grid will be converted to Kendo widget during initialization
771
+ this._widgetHtml.append($('<div>', gridConfig));
772
+
773
+ if (this._data.editable)
774
+ {
775
+ // BUTTONS
776
+
777
+ // Create buttons container
778
+ let buttons = $('<div>', {class: 'mt-2 text-right'});
779
+
780
+ // Append buttons to container
781
+ this._addButton = $('<button>', {type: 'button', class: 'k-button k-secondary', title: 'Add'}).append($('<i class="fas fa-plus"></i>'));
782
+ this._editButton = $('<button>', {type: 'button', class: 'k-button k-secondary ml-2', title: 'Edit', disabled: true}).append($('<i class="fas fa-pen"></i>'));
783
+ this._removeButton = $('<button>', {type: 'button', class: 'k-button k-secondary ml-2', title: 'Remove', disabled: true}).append($('<i class="fas fa-times"></i>'));
784
+
785
+ buttons.append(this._addButton);
786
+ buttons.append(this._editButton);
787
+ buttons.append(this._removeButton);
788
+
789
+ // Append buttons container to main html
790
+ this._widgetHtml.append(buttons);
791
+
792
+ // Create edit dialog
793
+ // NOTE: data-dismiss="modal" on the close/cancel buttons was removed to work around an issue with nested modals;
794
+ // the custom "data-cancel" attribute is used to add a custom listener to the buttons
795
+ this._editDialog = $(`
796
+ <div class="modal" tabindex="-1" role="dialog" aria-labelledby="modalTitle" aria-hidden="true" data-keyboard="false" data-backdrop="static">
797
+ <div class="modal-dialog modal-dialog-centered" role="document">
798
+ <div class="modal-content">
799
+ <div class="modal-header">
800
+ <h5 class="modal-title text-primary" id="modalTitle">${this._data.label}</h5>
801
+ <button type="button" class="close" aria-label="Close" data-cancel="modal">
802
+ <span aria-hidden="true">&times;</span>
803
+ </button>
804
+ </div>
805
+ <div class="modal-body in-flow-invalid-msg">
806
+
807
+ </div>
808
+ <div class="modal-footer flex-column">
809
+ <div class="d-flex w-100">
810
+ <div class="flex-grow-1 text-left">
811
+ <button type="button" class="k-button k-primary">...</button>
812
+ </div>
813
+ <div class="flex-grow-1 text-right">
814
+ <button type="button" class="k-button k-secondary" data-cancel="modal">Cancel</button>
815
+ </div>
816
+ </div>
817
+ </div>
818
+ </div>
819
+ </div>
820
+ </div>
821
+ `);
822
+
823
+ // Add listener to dialog hide event
824
+ this._editDialog.on('hidden.bs.modal', $.proxy(this._onEditPanelHidden, this));
825
+
826
+ // Add listener to main button click event
827
+ $('button.k-primary', this._editDialog).on('click', $.proxy(this._onSubmitBtClick, this));
828
+
829
+ // Add listener to close/cancel buttons click event
830
+ $('button[data-cancel="modal"]', this._editDialog).on('click', $.proxy(this._onCancelBtClick, this));
831
+
832
+ // Append edit dialog to main html
833
+ this._widgetHtml.append(this._editDialog);
834
+ }
835
+
836
+ // Return component
837
+ return this._widgetHtml;
838
+ }
839
+
840
+ /**
841
+ * Initialize widget.
842
+ * @override
843
+ */
844
+ _initialize()
845
+ {
846
+ let columns = [];
847
+ for (let subConfigParam of this._data.defaultListItem)
848
+ {
849
+ let col = {
850
+ field: subConfigParam.name,
851
+ title: subConfigParam.label,
852
+ width: 120
853
+ }
854
+
855
+ // Display V or X for booleans
856
+ if (typeof subConfigParam.value === 'boolean')
857
+ col.template = `#= ${subConfigParam.name} ? '<i class="fas fa-check"></i>' : '<i class="fas fa-times"></i>' #`;
858
+
859
+ // Hide passwords
860
+ if (subConfigParam.type == 'TextInput' && subConfigParam.attributes != null && subConfigParam.attributes.type == 'password')
861
+ col.template = `#= '•'.repeat(data.${subConfigParam.name}.length) #`;
862
+
863
+ columns.push(col);
864
+ }
865
+
866
+ // Initialize grid
867
+ let gridHtml = this._widgetHtml.find(`#${$.escapeSelector(this._data.name)}`);
868
+
869
+ gridHtml.kendoGrid({
870
+ resizable: true,
871
+ selectable: this._data.editable ? 'row' : false,
872
+ change: $.proxy(this._onGridSelectionChange, this),
873
+ columns: columns,
874
+ noRecords: {
875
+ template: 'No items.'
876
+ }
877
+ });
878
+
879
+ // Save ref. to widget
880
+ this._gridWidget = gridHtml.data('kendoGrid');
881
+
882
+ // Show tootip if grid's cell content exceeds cell width (ellipsis is displayed by Kendo Grid)
883
+ gridHtml.kendoTooltip({
884
+ filter: 'td',
885
+ show: function(e) {
886
+ // Never show tooltip...
887
+ this.content.parent().css('visibility', 'hidden');
888
+
889
+ // ...unless content is returned (see below) due to cell width being exceeded
890
+ if (this.content.text() != '')
891
+ this.content.parent().css('visibility', 'visible');
892
+ },
893
+ hide: function() {
894
+ this.content.parent().css('visibility', 'hidden');
895
+ },
896
+ content: function(e) {
897
+ let element = e.target[0];
898
+ if (element.offsetWidth < element.scrollWidth)
899
+ return e.target.text();
900
+ else
901
+ return '';
902
+ }
903
+ });
904
+
905
+ /*
906
+ // Initialize button tooltips
907
+ this._widgetHtml.kendoTooltip({
908
+ filter: 'button',
909
+ content: function(e) {
910
+ return `<div class="help-tooltip">${e.target.data('title')}</div>`;
911
+ }
912
+ });
913
+ */
914
+
915
+ // Add button listeners
916
+ if (this._data.editable)
917
+ {
918
+ this._addButton.click($.proxy(this._onAddClick, this));
919
+ this._editButton.click($.proxy(this._onEditClick, this));
920
+ this._removeButton.click($.proxy(this._onRemoveClick, this));
921
+ }
922
+
923
+ // Proceed with initialization
924
+ super._initialize();
925
+ }
926
+
927
+ /**
928
+ * Set widget's datasource.
929
+ * @override
930
+ */
931
+ _setWidgetValue()
932
+ {
933
+ let dataSource = new kendo.data.DataSource({
934
+ data: this._data.listValues
935
+ });
936
+
937
+ // Read current horizontal scroll value
938
+ const scrollLeft = $('.k-grid-content', this._gridWidget.wrapper).scrollLeft();
939
+
940
+ // Clear grid selection if any
941
+ this._gridWidget.clearSelection();
942
+
943
+ // Set updated grid's datasource
944
+ this._gridWidget.setDataSource(dataSource);
945
+
946
+ // Set horizontal scroll
947
+ $('.k-grid-content', this._gridWidget.wrapper).scrollLeft(scrollLeft);
948
+ }
949
+
950
+ /**
951
+ * Set widget's disabled state.
952
+ * @override
953
+ */
954
+ _setWidgetEditEnabled()
955
+ {
956
+ if (this._data.editable)
957
+ {
958
+ // Deselect item
959
+ this._gridWidget.clearSelection();
960
+
961
+ // Enable/disable grid
962
+ this._gridWidget.wrapper.attr('disabled', !this._editEnabled);
963
+
964
+ // Enable "Add" button
965
+ if (this._editEnabled)
966
+ this._addButton.attr('disabled', false);
967
+
968
+ // Disable all buttons
969
+ else
970
+ {
971
+ this._addButton.attr('disabled', true);
972
+ this._editButton.attr('disabled', true);
973
+ this._removeButton.attr('disabled', true);
974
+ }
975
+ }
976
+ }
977
+
978
+ _onGridSelectionChange(e)
979
+ {
980
+ let selectedRows = this._gridWidget.select();
981
+ let selectedDataItems = [];
982
+
983
+ for (let i = 0; i < selectedRows.length; i++)
984
+ {
985
+ let dataItem = this._gridWidget.dataItem(selectedRows[i]);
986
+ selectedDataItems.push(dataItem);
987
+ }
988
+
989
+ // Enable/disable edit button
990
+ if (this._editButton)
991
+ this._editButton.prop('disabled', selectedDataItems.length == 0);
992
+
993
+ // Enable/disable remove button
994
+ if (this._removeButton)
995
+ {
996
+ // Always disable button if no list item is selected
997
+ this._removeButton.prop('disabled', selectedDataItems.length == 0);
998
+
999
+ // Also disable button if denyEmpty == true and just one item remains in the list
1000
+ if (this._data.denyEmpty)
1001
+ this._removeButton.prop('disabled', this._data.listItems.length <= 1);
1002
+ }
1003
+ }
1004
+
1005
+ _onRemoveClick()
1006
+ {
1007
+ let selectedIndex = this._gridWidget.select().index();
1008
+
1009
+ // Remove item from list
1010
+ this._data.removeListItem(selectedIndex);
1011
+
1012
+ // Regenerate datagrid's datasource
1013
+ this._setWidgetValue();
1014
+ }
1015
+
1016
+ _onAddClick()
1017
+ {
1018
+ // Clone default item and add to list
1019
+ let newListItem = [];
1020
+ for (let subCP of this._data.defaultListItem)
1021
+ newListItem.push(subCP.clone(true));
1022
+
1023
+ // Create edit popup
1024
+ this._openEditPanel(newListItem);
1025
+ }
1026
+
1027
+ _onEditClick()
1028
+ {
1029
+ let selectedIndex = this._gridWidget.select().index();
1030
+
1031
+ // Clone selected item and add to list
1032
+ let clonedListItem = [];
1033
+ for (let subCP of this._data.listItems[selectedIndex])
1034
+ clonedListItem.push(subCP.clone(true));
1035
+
1036
+ // Create edit popup
1037
+ this._openEditPanel(clonedListItem, selectedIndex);
1038
+ }
1039
+
1040
+ _openEditPanel(subConfigParamsArray, editIndex = -1)
1041
+ {
1042
+ // Check if this configuration item is inside a modal window;
1043
+ // if yes, the edit panel (which is a modal as well) must be configured to remove the dark background
1044
+ if ($(this).parents('.modal').length > 0)
1045
+ $('.modal', $(this)).attr('data-backdrop', false);
1046
+
1047
+ // Create dialog content
1048
+ this._itemEditor = new _widgets_list_item_editor__WEBPACK_IMPORTED_MODULE_2__["ListItemEditor"]();
1049
+ this._itemEditor.data = subConfigParamsArray;
1050
+ this._itemEditor.index = editIndex;
1051
+
1052
+ let itemEditor = $(this._itemEditor);
1053
+
1054
+ // Append content to dialog
1055
+ $('.modal-body', this._editDialog).append(itemEditor);
1056
+
1057
+ // Set dialog main button text
1058
+ $('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');
1059
+
1060
+ // Display dialog
1061
+ this._editDialog.modal('show');
1062
+ }
1063
+
1064
+ _onSubmitBtClick()
1065
+ {
1066
+ if (this._itemEditor.validate())
1067
+ {
1068
+ let data = this._itemEditor.data;
1069
+ let index = this._itemEditor.index;
1070
+
1071
+ // Hide modal
1072
+ this._editDialog.modal('hide');
1073
+
1074
+ // Complete editing
1075
+ this._onEditComplete(data, index);
1076
+ }
1077
+ }
1078
+
1079
+ _onCancelBtClick()
1080
+ {
1081
+ // Hide modal
1082
+ this._editDialog.modal('hide');
1083
+ }
1084
+
1085
+ _onEditPanelHidden(e)
1086
+ {
1087
+ // Remove content from dialog
1088
+ this._itemEditor.remove();
1089
+
1090
+ // Set dialog main button text
1091
+ $('button.k-primary', this._editDialog).html('...');
1092
+
1093
+ this._itemEditor = null;
1094
+ }
1095
+
1096
+ _onEditComplete(listItem, editIndex)
1097
+ {
1098
+ // An existing list item was updated
1099
+ if (editIndex > -1)
1100
+ this._data.updateListItem(listItem, editIndex);
1101
+
1102
+ // A new list item was added; add it to the configuration parameter
1103
+ else
1104
+ this._data.addListItem(listItem);
1105
+
1106
+ // Regenerate datagrid's datasource
1107
+ this._setWidgetValue();
1108
+ }
1109
+ }
1110
+
1111
+ // DEFINE COMPONENT
1112
+ if (!window.customElements.get('config-grid'))
1113
+ window.customElements.define('config-grid', ConfigGrid);
1114
+
1115
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
1116
+
1117
+ /***/ }),
1118
+
1119
+ /***/ "./src/components/uibuilder/config-label.js":
1120
+ /*!**************************************************!*\
1121
+ !*** ./src/components/uibuilder/config-label.js ***!
1122
+ \**************************************************/
1123
+ /*! exports provided: ConfigLabel */
1124
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
1125
+
1126
+ "use strict";
1127
+ __webpack_require__.r(__webpack_exports__);
1128
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "ConfigLabel", function() { return ConfigLabel; });
1129
+ class ConfigLabel extends HTMLElement
1130
+ {
1131
+ constructor()
1132
+ {
1133
+ super();
1134
+
1135
+ this.setAttribute('class','config-label');
1136
+ }
1137
+
1138
+ set value(val)
1139
+ {
1140
+ if (typeof val === 'boolean')
1141
+ this.innerHTML = (val ? 'true' : 'false');
1142
+ else if (typeof val === 'number')
1143
+ this.innerHTML = (val ? val : 0);
1144
+ else
1145
+ this.innerHTML = (val != '' ? val : '&mdash;');
1146
+ }
1147
+
1148
+ get value()
1149
+ {
1150
+ return this.textContent;
1151
+ }
1152
+ }
1153
+
1154
+ // DEFINE COMPONENT
1155
+ if (!window.customElements.get('config-label'))
1156
+ window.customElements.define('config-label', ConfigLabel);
1157
+
1158
+
1159
+ /***/ }),
1160
+
1161
+ /***/ "./src/components/uibuilder/config-numeric-stepper.js":
1162
+ /*!************************************************************!*\
1163
+ !*** ./src/components/uibuilder/config-numeric-stepper.js ***!
1164
+ \************************************************************/
1165
+ /*! exports provided: ConfigNumericStepper */
1166
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
1167
+
1168
+ "use strict";
1169
+ __webpack_require__.r(__webpack_exports__);
1170
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "ConfigNumericStepper", function() { return ConfigNumericStepper; });
1171
+ /* harmony import */ var _config_form_item__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./config-form-item */ "./src/components/uibuilder/config-form-item.js");
1172
+ /* harmony import */ var _config_label__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./config-label */ "./src/components/uibuilder/config-label.js");
1173
+
1174
+
1175
+
1176
+ class ConfigNumericStepper extends _config_form_item__WEBPACK_IMPORTED_MODULE_0__["ConfigFormItem"]
1177
+ {
1178
+ constructor(configParam, editEnabled, inDialog)
1179
+ {
1180
+ super(configParam, editEnabled, inDialog);
1181
+ }
1182
+
1183
+ /**
1184
+ * Create widget to render the ConfigParameter value.
1185
+ * If parameter is not editable, a simple label is used.
1186
+ * @override
1187
+ */
1188
+ _generateInnerWidget()
1189
+ {
1190
+ if (this._data.editable)
1191
+ {
1192
+ // Set widget configuration
1193
+ let config = {
1194
+ type: 'number',
1195
+ class: 'form-control',
1196
+ id: this._data.name,
1197
+ name: this._data.name,
1198
+ 'data-role': 'numerictextbox',
1199
+ 'data-required-msg': 'Required',
1200
+ 'data-format': '#',
1201
+ required: 'required',
1202
+ };
1203
+
1204
+ // Set widget attributes (see parent class)
1205
+ this._setWidgetAttributes(config);
1206
+
1207
+ // Set additional widget attributes based on validation rules (see parent class)
1208
+ this._setWidgetValidationAttributes(config);
1209
+
1210
+ // Create widget's html
1211
+ this._widgetHtml = $('<input>', config);
1212
+ }
1213
+ else
1214
+ this._widgetHtml = new _config_label__WEBPACK_IMPORTED_MODULE_1__["ConfigLabel"]();
1215
+
1216
+ // Return component
1217
+ return this._widgetHtml;
1218
+ }
1219
+
1220
+ /**
1221
+ * Initialize widget.
1222
+ * @override
1223
+ */
1224
+ _initialize()
1225
+ {
1226
+ if (this._data.editable)
1227
+ {
1228
+ // Initialize kendo widget
1229
+ kendo.init(this._widgetHtml);
1230
+
1231
+ // Save ref. to widget
1232
+ this._innerWidget = this._widgetHtml.data('kendoNumericTextBox');
1233
+
1234
+ // Enable value commit binding
1235
+ this._innerWidget.bind('change', $.proxy(this._onValueInput, this));
1236
+ }
1237
+
1238
+ // Proceed with initialization
1239
+ super._initialize();
1240
+ }
1241
+
1242
+ /**
1243
+ * Set widget's value.
1244
+ * If parameter is not editable, the label text is set.
1245
+ * @override
1246
+ */
1247
+ _setWidgetValue()
1248
+ {
1249
+ if (this._data.editable)
1250
+ this._innerWidget.value(this._data.value);
1251
+ else
1252
+ this._widgetHtml.value = this._data.value;
1253
+
1254
+ // Trigger event
1255
+ this._triggerEvent();
1256
+ }
1257
+
1258
+ /**
1259
+ * Set widget's disabled state.
1260
+ * @override
1261
+ */
1262
+ _setWidgetEditEnabled()
1263
+ {
1264
+ if (this._data.editable)
1265
+ this._innerWidget.enable(this._editEnabled);
1266
+ }
1267
+
1268
+ /**
1269
+ * Update Configuration Parameter value.
1270
+ * @override
1271
+ */
1272
+ _onValueInput(e)
1273
+ {
1274
+ // Update Configuration Parameter to new value
1275
+ this._data.value = Number(this._innerWidget.value());
1276
+
1277
+ // Trigger event
1278
+ this._triggerEvent();
1279
+ }
1280
+ }
1281
+
1282
+ // DEFINE COMPONENT
1283
+ if (!window.customElements.get('config-numeric-stepper'))
1284
+ window.customElements.define('config-numeric-stepper', ConfigNumericStepper);
1285
+
1286
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
1287
+
1288
+ /***/ }),
1289
+
1290
+ /***/ "./src/components/uibuilder/config-text-input.js":
1291
+ /*!*******************************************************!*\
1292
+ !*** ./src/components/uibuilder/config-text-input.js ***!
1293
+ \*******************************************************/
1294
+ /*! exports provided: ConfigTextInput */
1295
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
1296
+
1297
+ "use strict";
1298
+ __webpack_require__.r(__webpack_exports__);
1299
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "ConfigTextInput", function() { return ConfigTextInput; });
1300
+ /* harmony import */ var _config_form_item__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./config-form-item */ "./src/components/uibuilder/config-form-item.js");
1301
+ /* harmony import */ var _config_label__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./config-label */ "./src/components/uibuilder/config-label.js");
1302
+
1303
+
1304
+
1305
+ class ConfigTextInput extends _config_form_item__WEBPACK_IMPORTED_MODULE_0__["ConfigFormItem"]
1306
+ {
1307
+ constructor(configParam, editEnabled, inDialog)
1308
+ {
1309
+ super(configParam, editEnabled, inDialog);
1310
+ }
1311
+
1312
+ /**
1313
+ * Create widget to render the ConfigParameter value.
1314
+ * If parameter is not editable, a simple label is used.
1315
+ * @override
1316
+ */
1317
+ _generateInnerWidget()
1318
+ {
1319
+ if (this._data.editable)
1320
+ {
1321
+ // Set widget configuration
1322
+ let config = {
1323
+ type: 'text',
1324
+ class: 'form-control k-textbox',
1325
+ id: this._data.name,
1326
+ name: this._data.name,
1327
+ autocomplete: 'off',
1328
+ };
1329
+
1330
+ // Set widget attributes
1331
+ this._setWidgetAttributes(config);
1332
+
1333
+ // Set additional widget attributes based on validation rules
1334
+ this._setWidgetValidationAttributes(config);
1335
+
1336
+ // Create widget's html
1337
+ this._widgetHtml = $('<input>', config);
1338
+
1339
+ // Enable value commit binding
1340
+ this._widgetHtml.on('change', $.proxy(this._onValueInput, this));
1341
+ }
1342
+ else
1343
+ this._widgetHtml = new _config_label__WEBPACK_IMPORTED_MODULE_1__["ConfigLabel"]();
1344
+
1345
+ // Return component
1346
+ return this._widgetHtml;
1347
+ }
1348
+
1349
+ /**
1350
+ * Set widget's value.
1351
+ * If parameter is not editable, the label text is set.
1352
+ * @override
1353
+ */
1354
+ _setWidgetValue()
1355
+ {
1356
+ if (this._data.editable)
1357
+ this._widgetHtml.val(this._data.value);
1358
+ else
1359
+ this._widgetHtml.value = this._data.value;
1360
+
1361
+ // Trigger event
1362
+ this._triggerEvent();
1363
+ }
1364
+
1365
+ /**
1366
+ * Set widget's disabled state.
1367
+ * @override
1368
+ */
1369
+ _setWidgetEditEnabled()
1370
+ {
1371
+ if (this._data.editable)
1372
+ this._widgetHtml.attr('disabled', !this._editEnabled);
1373
+ }
1374
+
1375
+ /**
1376
+ * Update Configuration Parameter value.
1377
+ * @override
1378
+ */
1379
+ _onValueInput(e)
1380
+ {
1381
+ // Update Configuration Parameter to new value
1382
+ this._data.value = this._widgetHtml.val();
1383
+
1384
+ // Trigger event
1385
+ this._triggerEvent();
1386
+ }
1387
+ }
1388
+
1389
+ // DEFINE COMPONENT
1390
+ if (!window.customElements.get('config-text-input'))
1391
+ window.customElements.define('config-text-input', ConfigTextInput);
1392
+
1393
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
1394
+
1395
+ /***/ }),
1396
+
1397
+ /***/ "./src/components/uibuilder/config-vector-3d.js":
1398
+ /*!******************************************************!*\
1399
+ !*** ./src/components/uibuilder/config-vector-3d.js ***!
1400
+ \******************************************************/
1401
+ /*! exports provided: ConfigVector3D */
1402
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
1403
+
1404
+ "use strict";
1405
+ __webpack_require__.r(__webpack_exports__);
1406
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "ConfigVector3D", function() { return ConfigVector3D; });
1407
+ /* harmony import */ var _config_form_item__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./config-form-item */ "./src/components/uibuilder/config-form-item.js");
1408
+ /* harmony import */ var _config_label__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./config-label */ "./src/components/uibuilder/config-label.js");
1409
+ /* harmony import */ var _widgets_vector_3d_input__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./widgets/vector-3d-input */ "./src/components/uibuilder/widgets/vector-3d-input.js");
1410
+
1411
+
1412
+
1413
+
1414
+ class ConfigVector3D extends _config_form_item__WEBPACK_IMPORTED_MODULE_0__["ConfigFormItem"]
1415
+ {
1416
+ constructor(configParam, editEnabled, inDialog)
1417
+ {
1418
+ super(configParam, editEnabled, inDialog);
1419
+ }
1420
+
1421
+ /**
1422
+ * Create widget to render the ConfigParameter value.
1423
+ * If parameter is not editable, a simple label is used.
1424
+ * @override
1425
+ */
1426
+ _generateInnerWidget()
1427
+ {
1428
+ if (this._data.editable)
1429
+ {
1430
+ // Create widget's html
1431
+ this._widgetHtml = new _widgets_vector_3d_input__WEBPACK_IMPORTED_MODULE_2__["Vector3DInput"](this._data.name, this._data.validator == 'aoi');
1432
+
1433
+ // Set widget attributes
1434
+ this._setWidgetAttributes(this._widgetHtml);
1435
+
1436
+ // Enable value commit binding
1437
+ $(this._widgetHtml).on('change', $.proxy(this._onValueInput, this));
1438
+ }
1439
+ else
1440
+ this._widgetHtml = new _config_label__WEBPACK_IMPORTED_MODULE_1__["ConfigLabel"]();
1441
+
1442
+ // Return component
1443
+ return this._widgetHtml;
1444
+ }
1445
+
1446
+ /**
1447
+ * Set widget's value.
1448
+ * If parameter is not editable, the label text is set.
1449
+ * @override
1450
+ */
1451
+ _setWidgetValue()
1452
+ {
1453
+ this._widgetHtml.value = this._data.value;
1454
+
1455
+ // Trigger event
1456
+ this._triggerEvent();
1457
+ }
1458
+
1459
+ /**
1460
+ * Set widget's disabled state.
1461
+ * @override
1462
+ */
1463
+ _setWidgetEditEnabled()
1464
+ {
1465
+ if (this._data.editable)
1466
+ {
1467
+ $(this._widgetHtml).attr('disabled', !this._editEnabled);
1468
+ }
1469
+ }
1470
+
1471
+ /**
1472
+ * Update Configuration Parameter value.
1473
+ * @override
1474
+ */
1475
+ _onValueInput(e)
1476
+ {
1477
+ // Update Configuration Parameter to new value
1478
+ this._data.value = this._widgetHtml.value;
1479
+
1480
+ // Trigger event
1481
+ this._triggerEvent();
1482
+ }
1483
+ }
1484
+
1485
+ // DEFINE COMPONENT
1486
+ if (!window.customElements.get('config-vector-3d'))
1487
+ window.customElements.define('config-vector-3d', ConfigVector3D);
1488
+
1489
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
1490
+
1491
+ /***/ }),
1492
+
1493
+ /***/ "./src/components/uibuilder/widgets/list-item-editor.js":
1494
+ /*!**************************************************************!*\
1495
+ !*** ./src/components/uibuilder/widgets/list-item-editor.js ***!
1496
+ \**************************************************************/
1497
+ /*! exports provided: ListItemEditor */
1498
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
1499
+
1500
+ "use strict";
1501
+ __webpack_require__.r(__webpack_exports__);
1502
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "ListItemEditor", function() { return ListItemEditor; });
1503
+ /* harmony import */ var _utils_uibuilder_config_form_item_factory__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../../utils/uibuilder/config-form-item-factory */ "./src/utils/uibuilder/config-form-item-factory.js");
1504
+
1505
+
1506
+ class ListItemEditor extends HTMLElement
1507
+ {
1508
+ constructor()
1509
+ {
1510
+ super();
1511
+ }
1512
+
1513
+ set data(subConfigParamsArray)
1514
+ {
1515
+ this._data = subConfigParamsArray;
1516
+
1517
+ this._buildView();
1518
+ }
1519
+
1520
+ get data()
1521
+ {
1522
+ return this._data;
1523
+ }
1524
+
1525
+ set index(index)
1526
+ {
1527
+ this._index = index;
1528
+ }
1529
+
1530
+ get index()
1531
+ {
1532
+ return this._index;
1533
+ }
1534
+
1535
+ _buildView()
1536
+ {
1537
+ // Generate container form
1538
+ this._form = $('<form>', {
1539
+ onsubmit: 'return false;'
1540
+ });
1541
+
1542
+ // Append form
1543
+ $(this).append(this._form);
1544
+
1545
+ // Generate form fields
1546
+ for (let configParam of this._data)
1547
+ {
1548
+ // Create form item
1549
+ let formItem = _utils_uibuilder_config_form_item_factory__WEBPACK_IMPORTED_MODULE_0__["ConfigFormItemFactory"].create(configParam, true, true);
1550
+ formItem.data = configParam;
1551
+
1552
+ // Add form item to form
1553
+ this._form.append(formItem);
1554
+ }
1555
+
1556
+ // Initialize kendo validation on form
1557
+ this._validator = this._form.kendoValidator({
1558
+ validateOnBlur: true,
1559
+ rules: {
1560
+ // Add rule to validate AOI form items?
1561
+ // (see: https://demos.telerik.com/kendo-ui/validator/custom-validation)
1562
+ aoi: function (input) {
1563
+ if (input.is('[data-aoi-msg]') && input.val() != '')
1564
+ {
1565
+ if (input.val() == '0,0,0')
1566
+ return false;
1567
+ }
1568
+
1569
+ return true;
1570
+ }
1571
+ }
1572
+ }).data('kendoValidator');
1573
+ }
1574
+
1575
+ validate()
1576
+ {
1577
+ return this._validator.validate();
1578
+ }
1579
+ }
1580
+
1581
+ // DEFINE COMPONENT
1582
+ if (!window.customElements.get('list-item-editor'))
1583
+ window.customElements.define('list-item-editor', ListItemEditor);
1584
+
1585
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
1586
+
1587
+ /***/ }),
1588
+
1589
+ /***/ "./src/components/uibuilder/widgets/vector-3d-input.js":
1590
+ /*!*************************************************************!*\
1591
+ !*** ./src/components/uibuilder/widgets/vector-3d-input.js ***!
1592
+ \*************************************************************/
1593
+ /*! exports provided: Vector3DInput */
1594
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
1595
+
1596
+ "use strict";
1597
+ __webpack_require__.r(__webpack_exports__);
1598
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "Vector3DInput", function() { return Vector3DInput; });
1599
+ class Vector3DInput extends HTMLElement
1600
+ {
1601
+ constructor(id, isValidable)
1602
+ {
1603
+ super();
1604
+
1605
+ this.id = id;
1606
+ this.name = id;
1607
+
1608
+ this._isValidable = isValidable;
1609
+
1610
+ this._initialize();
1611
+ }
1612
+
1613
+ set enableClear(value)
1614
+ {
1615
+ if (value)
1616
+ this._clearButton.show();
1617
+ else
1618
+ this._clearButton.hide();
1619
+ }
1620
+
1621
+ set allowNegative(value)
1622
+ {
1623
+ if (value)
1624
+ {
1625
+ this._widgetX.setOptions( {min: null} );
1626
+ this._widgetY.setOptions( {min: null} );
1627
+ this._widgetZ.setOptions( {min: null} );
1628
+ }
1629
+ }
1630
+
1631
+ set value(val)
1632
+ {
1633
+ var coords = val.split(',');
1634
+
1635
+ if (coords.length >= 1)
1636
+ this._widgetX.value(coords[0]);
1637
+
1638
+ if (coords.length >= 2)
1639
+ this._widgetY.value(coords[1]);
1640
+
1641
+ if (coords.length >= 3)
1642
+ this._widgetZ.value(coords[2]);
1643
+
1644
+ if (this._isValidable)
1645
+ this._inputVal.val(this.value);
1646
+ }
1647
+
1648
+ get value()
1649
+ {
1650
+ if (this._widgetX.value() == null && this._widgetY.value() == null && this._widgetZ.value() == null)
1651
+ return '';
1652
+ else
1653
+ return this._widgetX.value() + ',' + this._widgetY.value() + ',' + this._widgetZ.value();
1654
+ }
1655
+
1656
+ _initialize()
1657
+ {
1658
+ // Generate container form
1659
+ this._container = $('<div>', {
1660
+ class: 'form-inline'
1661
+ });
1662
+
1663
+ // Append container
1664
+ $(this).append(this._container);
1665
+
1666
+ // Set inputs configuration
1667
+ let configHtml = {
1668
+ type: 'number',
1669
+ class: 'form-control short-4',
1670
+ };
1671
+
1672
+ // Set widget configuration
1673
+ let configWidget = {
1674
+ min: 0,
1675
+ spinners: false,
1676
+ format: '#.######',
1677
+ decimals: 6,
1678
+ round: false,
1679
+ spinners: false,
1680
+ restrictDecimals: false,
1681
+ change: $.proxy(this._onChange, this)
1682
+ };
1683
+
1684
+ // Create widgets
1685
+ this._inputX = $('<input>', configHtml);
1686
+ this._container.append(this._inputX);
1687
+ this._widgetX = this._inputX.kendoNumericTextBox(configWidget).data('kendoNumericTextBox');
1688
+
1689
+ this._container.append('<span class="px-1">,</span>');
1690
+
1691
+ this._inputY = $('<input>', configHtml);
1692
+ this._container.append(this._inputY);
1693
+ this._widgetY = this._inputY.kendoNumericTextBox(configWidget).data('kendoNumericTextBox');
1694
+
1695
+ this._container.append('<span class="px-1">,</span>');
1696
+
1697
+ this._inputZ = $('<input>', configHtml);
1698
+ this._container.append(this._inputZ);
1699
+ this._widgetZ = this._inputZ.kendoNumericTextBox(configWidget).data('kendoNumericTextBox');
1700
+
1701
+ this._container.append('<span class="px-1"></span>'); // Additional spacer
1702
+
1703
+ // Create invisible field to apply overall validation
1704
+ if (this._isValidable)
1705
+ {
1706
+ this._inputVal = $('<input>', {name: `${this.name}-custom-validate`, 'data-aoi-msg': 'Values can\'t all be 0'});
1707
+ this._container.append(this._inputVal);
1708
+ this._container.append(`<span class="k-invalid-msg" data-for="${this.name}-custom-validate"></span>`)
1709
+ this._inputVal.hide();
1710
+ }
1711
+
1712
+ // Create and append Clear button
1713
+ this._clearButton = $('<button>', {type: 'button', class: 'k-button k-secondary my-1', title: 'Clear'}).append($('<i class="fas fa-times"></i>'));
1714
+ this._clearButton.on('click', $.proxy(this._onClearClick, this));
1715
+ this._container.append(this._clearButton);
1716
+
1717
+ // Hide button by default
1718
+ this._clearButton.hide();
1719
+ }
1720
+
1721
+ _onChange()
1722
+ {
1723
+ // Empty strings are not allowed
1724
+ if (this._widgetX.value() == null)
1725
+ this._widgetX.value(0);
1726
+
1727
+ if (this._widgetY.value() == null)
1728
+ this._widgetY.value(0);
1729
+
1730
+ if (this._widgetZ.value() == null)
1731
+ this._widgetZ.value(0);
1732
+
1733
+ this._dispatchCommit();
1734
+ }
1735
+
1736
+ _onClearClick()
1737
+ {
1738
+ this._widgetX.value('');
1739
+ this._widgetY.value('');
1740
+ this._widgetZ.value('');
1741
+
1742
+ this._dispatchCommit();
1743
+ }
1744
+
1745
+ _dispatchCommit()
1746
+ {
1747
+ if (this._isValidable)
1748
+ this._inputVal.val(this.value);
1749
+
1750
+ this.dispatchEvent(new Event('change'));
1751
+ }
1752
+ }
1753
+
1754
+ // DEFINE COMPONENT
1755
+ if (!window.customElements.get('vector-3d-input'))
1756
+ window.customElements.define('vector-3d-input', Vector3DInput);
1757
+
1758
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
1759
+
1760
+ /***/ }),
1761
+
1762
+ /***/ "./src/utils/uibuilder/config-form-item-factory.js":
1763
+ /*!*********************************************************!*\
1764
+ !*** ./src/utils/uibuilder/config-form-item-factory.js ***!
1765
+ \*********************************************************/
1766
+ /*! exports provided: ConfigFormItemFactory */
1767
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
1768
+
1769
+ "use strict";
1770
+ __webpack_require__.r(__webpack_exports__);
1771
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "ConfigFormItemFactory", function() { return ConfigFormItemFactory; });
1772
+ /* harmony import */ var _components_uibuilder_config_form_item__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../components/uibuilder/config-form-item */ "./src/components/uibuilder/config-form-item.js");
1773
+ /* harmony import */ var _components_uibuilder_config_numeric_stepper__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../components/uibuilder/config-numeric-stepper */ "./src/components/uibuilder/config-numeric-stepper.js");
1774
+ /* harmony import */ var _components_uibuilder_config_text_input__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../../components/uibuilder/config-text-input */ "./src/components/uibuilder/config-text-input.js");
1775
+ /* harmony import */ var _components_uibuilder_config_check_box__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../../components/uibuilder/config-check-box */ "./src/components/uibuilder/config-check-box.js");
1776
+ /* harmony import */ var _components_uibuilder_config_drop_down_list__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../../components/uibuilder/config-drop-down-list */ "./src/components/uibuilder/config-drop-down-list.js");
1777
+ /* harmony import */ var _components_uibuilder_config_grid__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ../../components/uibuilder/config-grid */ "./src/components/uibuilder/config-grid.js");
1778
+ /* harmony import */ var _components_uibuilder_config_dual_list__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ../../components/uibuilder/config-dual-list */ "./src/components/uibuilder/config-dual-list.js");
1779
+ /* harmony import */ var _components_uibuilder_config_vector_3d__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ../../components/uibuilder/config-vector-3d */ "./src/components/uibuilder/config-vector-3d.js");
1780
+
1781
+
1782
+
1783
+
1784
+
1785
+
1786
+
1787
+
1788
+
1789
+
1790
+ class ConfigFormItemFactory
1791
+ {
1792
+ static create(configParam, editEnabled, inDialog = false)
1793
+ {
1794
+ switch (configParam.type)
1795
+ {
1796
+ case 'TextInput':
1797
+ return new _components_uibuilder_config_text_input__WEBPACK_IMPORTED_MODULE_2__["ConfigTextInput"](configParam, editEnabled, inDialog);
1798
+ break;
1799
+
1800
+ case 'CheckBox':
1801
+ return new _components_uibuilder_config_check_box__WEBPACK_IMPORTED_MODULE_3__["ConfigCheckBox"](configParam, editEnabled, inDialog);
1802
+ break;
1803
+
1804
+ case 'NumericStepper':
1805
+ return new _components_uibuilder_config_numeric_stepper__WEBPACK_IMPORTED_MODULE_1__["ConfigNumericStepper"](configParam, editEnabled, inDialog);
1806
+ break;
1807
+
1808
+ case 'ComboBox':
1809
+ return new _components_uibuilder_config_drop_down_list__WEBPACK_IMPORTED_MODULE_4__["ConfigDropDownList"](configParam, editEnabled, inDialog);
1810
+ break;
1811
+
1812
+ case 'DataGrid':
1813
+ return new _components_uibuilder_config_grid__WEBPACK_IMPORTED_MODULE_5__["ConfigGrid"](configParam, editEnabled, inDialog);
1814
+ break;
1815
+
1816
+ case 'DualList':
1817
+ return new _components_uibuilder_config_dual_list__WEBPACK_IMPORTED_MODULE_6__["ConfigDualList"](configParam, editEnabled, inDialog);
1818
+ break;
1819
+
1820
+ case 'Vector3D':
1821
+ return new _components_uibuilder_config_vector_3d__WEBPACK_IMPORTED_MODULE_7__["ConfigVector3D"](configParam, editEnabled, inDialog);
1822
+ break;
1823
+
1824
+ default:
1825
+ return new _components_uibuilder_config_form_item__WEBPACK_IMPORTED_MODULE_0__["ConfigFormItem"](configParam, editEnabled, inDialog); // Will log an error for missing form item type
1826
+ }
1827
+ }
1828
+ }
1829
+
1830
+
1831
+ /***/ }),
1832
+
1833
+ /***/ "./src/utils/uibuilder/config-interface-builder.js":
1834
+ /*!*********************************************************!*\
1835
+ !*** ./src/utils/uibuilder/config-interface-builder.js ***!
1836
+ \*********************************************************/
1837
+ /*! exports provided: ConfigInterfaceBuilder */
1838
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
1839
+
1840
+ "use strict";
1841
+ __webpack_require__.r(__webpack_exports__);
1842
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "ConfigInterfaceBuilder", function() { return ConfigInterfaceBuilder; });
1843
+ /* harmony import */ var _configuration_parameter__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./configuration-parameter */ "./src/utils/uibuilder/configuration-parameter.js");
1844
+ /* harmony import */ var _config_form_item_factory__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./config-form-item-factory */ "./src/utils/uibuilder/config-form-item-factory.js");
1845
+
1846
+
1847
+
1848
+ class ConfigInterfaceBuilder
1849
+ {
1850
+ constructor()
1851
+ {
1852
+ // Set some constants
1853
+ this.TAB_PREFIX = 'tab-'
1854
+ this.TAB_PANE_PREFIX = 'tabpane-';
1855
+ this.SEPARATOR_BEFORE = 'before';
1856
+ this.SEPARATOR_AFTER = 'after';
1857
+ }
1858
+
1859
+ dump(modifiedOnly = false)
1860
+ {
1861
+ let dumpStr = '';
1862
+
1863
+ for (let cp of this._configParams)
1864
+ {
1865
+ if (modifiedOnly)
1866
+ {
1867
+ if (cp.isModified)
1868
+ dumpStr += cp.toString() + '\n';
1869
+ }
1870
+ else
1871
+ dumpStr += cp.toString() + '\n';
1872
+ }
1873
+
1874
+ console.log(dumpStr);
1875
+ }
1876
+
1877
+ buildInterface(data, mainContainerId, disableEdit = false, tabSuffix = '')
1878
+ {
1879
+ this._mainContainerId = mainContainerId;
1880
+ this._configParams = new Array();
1881
+ this._validator = null;
1882
+
1883
+ let hasNewFormItem = false;
1884
+
1885
+ //console.log(data.getDump())
1886
+
1887
+ for (let i = 0; i < data.size(); i++)
1888
+ {
1889
+ // PARSE DATA
1890
+
1891
+ let configParam = _configuration_parameter__WEBPACK_IMPORTED_MODULE_0__["ConfigurationParameter"].fromSfsObject(data.get(i));
1892
+ this._configParams.push(configParam);
1893
+
1894
+ // Get tab and tab pane id from group id
1895
+ const tabId = this.TAB_PREFIX + configParam.categoryId + (tabSuffix ? '_' + tabSuffix : '');
1896
+ const tabPaneId = this.TAB_PANE_PREFIX + configParam.categoryId + (tabSuffix ? '_' + tabSuffix : '');
1897
+
1898
+ // BUILD INTERFACE :: TABS
1899
+
1900
+ // Check if a tab specific for this group already exists inside the mainContainer: if not, create it
1901
+ // (a tab already exists if it was created in a previous loop)
1902
+ let tab = $(`#${mainContainerId} > #tabs #${tabId}`);
1903
+
1904
+ if (tab.length == 0)
1905
+ {
1906
+ // Create tab for tab pane
1907
+ tab = $('<li>', {class: 'nav-item'});
1908
+ tab.append($('<a>', {
1909
+ class: 'nav-link' + (i == 0 ? ' active' : ''),
1910
+ id: tabId,
1911
+ 'data-toggle': 'tab',
1912
+ href: '#' + tabPaneId,
1913
+ role: 'tab',
1914
+ 'aria-controls': tabPaneId,
1915
+ 'aria-selected': (i == 0 ? 'true' : 'false'),
1916
+ html: configParam.category,
1917
+ }));
1918
+
1919
+ // Add tab to container
1920
+ $(`#${mainContainerId} > #tabs`).append(tab);
1921
+ }
1922
+
1923
+ // BUILD INTERFACE :: TAB PANES
1924
+
1925
+ // Check if a tab pane specific for this group already exists inside the mainContainer: if not, create it
1926
+ // (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)
1927
+ let tabPane = $(`#${mainContainerId} > #tabPanels > #${tabPaneId}`);
1928
+
1929
+ if (tabPane.length == 0)
1930
+ {
1931
+ // Create tab pane
1932
+ tabPane = $('<div>', {
1933
+ class: 'tab-pane' + (i == 0 ? ' show active' : ''),
1934
+ id: tabPaneId,
1935
+ role: 'tabpanel',
1936
+ 'aria-labelledby': tabId,
1937
+ 'data-dynamic': 'true',
1938
+ });
1939
+
1940
+ // Add tab pane to container
1941
+ $(`#${mainContainerId} > #tabPanels`).append(tabPane);
1942
+ }
1943
+
1944
+ // BUILD INTERFACE :: TAB PANES' FORM
1945
+
1946
+ // Check if a form already exists inside the tab pane: if not, create it
1947
+ let form = tabPane.find('form');
1948
+
1949
+ if (form.length == 0)
1950
+ {
1951
+ // Create form
1952
+ form = $('<form>', {
1953
+ class: '',
1954
+ autocomplete: 'off'
1955
+ });
1956
+
1957
+ // 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)
1958
+ form.append(
1959
+ $('<fieldset>', {
1960
+ class: ''
1961
+ })
1962
+ );
1963
+
1964
+ // Add form to tab pane
1965
+ tabPane.prepend(form);
1966
+
1967
+ // form.on("keypress", function(event) {
1968
+ // // If the user presses the "Enter" key on the keyboard
1969
+ // if (event.key === "Enter") {
1970
+ // // Cancel the default action, if needed
1971
+ // event.preventDefault();
1972
+ // console.log("ENTER")
1973
+ // }
1974
+ // });
1975
+ form.submit(function(evt) {
1976
+ console.log("SUBMIT")
1977
+ evt.preventDefault();
1978
+ });
1979
+ }
1980
+
1981
+ // Get fieldset, which is the actual form items container
1982
+ let fieldset = form.find('fieldset');
1983
+
1984
+ // BUILD INTERFACE :: TAB PANES' FORM ITEMS
1985
+
1986
+ // Check if form item already exists in fieldset; if yes, just update its data
1987
+ let formItem = fieldset.find(`#form-item-${$.escapeSelector(configParam.name)}`);
1988
+
1989
+ if (formItem.length == 0)
1990
+ {
1991
+ hasNewFormItem = true;
1992
+
1993
+ formItem = _config_form_item_factory__WEBPACK_IMPORTED_MODULE_1__["ConfigFormItemFactory"].create(configParam, !disableEdit);
1994
+
1995
+ // Add separator before
1996
+ if (configParam.separator != null && configParam.separator.pos == 'before')
1997
+ fieldset.append(this._buildSeparator(configParam.separator));
1998
+
1999
+ // Add form item to form
2000
+ fieldset.append(formItem);
2001
+
2002
+ // Add separator after
2003
+ if (configParam.separator != null && configParam.separator.pos == 'after')
2004
+ fieldset.append(this._buildSeparator(configParam.separator));
2005
+ }
2006
+ else
2007
+ formItem[0].data = configParam;
2008
+ }
2009
+
2010
+ // Add listener to show help tooltips
2011
+ let allTabPanes = $(`#${mainContainerId} > #tabPanels > div.tab-pane`);
2012
+ allTabPanes.kendoTooltip({
2013
+ filter: 'i[title].help',
2014
+ position: 'right',
2015
+ width: '250px',
2016
+ content: function(e) {
2017
+ return `<div class="help-tooltip">${e.target.data('title')}</div>`;
2018
+ }
2019
+ });
2020
+
2021
+ // Initialize kendo validation on forms' main container
2022
+ this._validator = $(`#${mainContainerId}`).kendoValidator({
2023
+ validateOnBlur: true,
2024
+ rules: {
2025
+ // Add rule to validate AOI form items
2026
+ // (see: https://demos.telerik.com/kendo-ui/validator/custom-validation)
2027
+ aoi: function (input) {
2028
+ if (input.is('[data-aoi-msg]') && input.val() != '')
2029
+ {
2030
+ if (input.val() == '0,0,0')
2031
+ return false;
2032
+ }
2033
+
2034
+ return true;
2035
+ }
2036
+ }
2037
+ }).data('kendoValidator');
2038
+ }
2039
+
2040
+ destroyInterface()
2041
+ {
2042
+ // Destroy all Kendo widgets in forms
2043
+ kendo.destroy($(`#${this._mainContainerId} > #tabPanels > div.tab-pane > form`));
2044
+
2045
+ // Remove all tabs
2046
+ $(`#${this._mainContainerId} > #tabs`).empty();
2047
+
2048
+ // Remove dynamic tab panes (tab panes created by Interface Builder)
2049
+ $(`#${this._mainContainerId} > #tabPanels > div.tab-pane[data-dynamic="true"]`).remove();
2050
+
2051
+ // Remove form inside static tab panes (predefined tab panes in html)
2052
+ $(`#${this._mainContainerId} > #tabPanels > div.tab-pane > form`).remove();
2053
+
2054
+ // Remove "active" class from static tab panes (otherwise this class messes with the tab navigator functioning)
2055
+ $(`#${this._mainContainerId} > #tabPanels > div.tab-pane`).removeClass('active');
2056
+ }
2057
+
2058
+ disableInterface(disable)
2059
+ {
2060
+ // Enable/disable all config form items
2061
+ $(`#${this._mainContainerId} *[id^='form-item-']`).prop('editEnabled', !disable);
2062
+ }
2063
+
2064
+ _buildSeparator(separator)
2065
+ {
2066
+ if (separator.text == null)
2067
+ return $(`<hr class="config-form-separator">`);
2068
+
2069
+ else
2070
+ return $(`<label class="config-form-separator-label mb-3">${separator.text}</label>`);
2071
+ }
2072
+
2073
+ getChangedData()
2074
+ {
2075
+ let changes = new SFS2X.SFSArray();
2076
+
2077
+ for (var cp of this._configParams)
2078
+ {
2079
+ if (cp.isModified)
2080
+ changes.addSFSObject(cp.toSfsObject());
2081
+ }
2082
+
2083
+ return changes;
2084
+ }
2085
+
2086
+ resetIsModified()
2087
+ {
2088
+ for (let cp of this._configParams)
2089
+ {
2090
+ if (cp.isModified)
2091
+ cp.resetIsModified();
2092
+ }
2093
+ }
2094
+
2095
+ checkIsValid()
2096
+ {
2097
+ return this._validator.validate();
2098
+ }
2099
+
2100
+ resetValidation()
2101
+ {
2102
+ this._validator.hideMessages();
2103
+
2104
+ // The method above doesn't remove the k-invalid classes and aria-invalid="true" attributes from inputs
2105
+ // Let's do it manually
2106
+ $(`#${this._mainContainerId} .k-invalid`).removeClass('k-invalid');
2107
+ $(`#${this._mainContainerId} [aria-invalid="true"]`).removeAttr('aria-invalid');
2108
+ }
2109
+
2110
+ getConfigFormItem(configParamName)
2111
+ {
2112
+ let formItem = $(`#${this._mainContainerId}`).find(`#form-item-${$.escapeSelector(configParamName)}`);
2113
+
2114
+ if (formItem.length > 0)
2115
+ return formItem[0];
2116
+ else
2117
+ return null;
2118
+ }
2119
+
2120
+ activateFirstTabPanel()
2121
+ {
2122
+ let configParam = this._configParams[0];
2123
+ const tabPaneId = this.TAB_PANE_PREFIX + configParam.categoryId;
2124
+ let tabPane = $(`#${this._mainContainerId} > #tabPanels > #${tabPaneId}`);
2125
+ tabPane.addClass('show active');
2126
+ }
2127
+ }
2128
+
2129
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
2130
+
2131
+ /***/ }),
2132
+
2133
+ /***/ "./src/utils/uibuilder/configuration-parameter.js":
2134
+ /*!********************************************************!*\
2135
+ !*** ./src/utils/uibuilder/configuration-parameter.js ***!
2136
+ \********************************************************/
2137
+ /*! exports provided: ConfigurationParameter */
2138
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
2139
+
2140
+ "use strict";
2141
+ __webpack_require__.r(__webpack_exports__);
2142
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "ConfigurationParameter", function() { return ConfigurationParameter; });
2143
+ class ConfigurationParameter
2144
+ {
2145
+ static fromSfsObject(element)
2146
+ {
2147
+ let cp = new ConfigurationParameter();
2148
+
2149
+ // Parse common data
2150
+ cp.name = element.getUtfString('name');
2151
+ cp.label = element.getUtfString('label');
2152
+ cp.category = element.getUtfString('category');
2153
+ cp.immediate = (element.containsKey('immediate') ? element.getBool('immediate') : false);
2154
+ cp.tooltip = element.getUtfString('tooltip');
2155
+ cp.type = element.getUtfString('type');
2156
+ cp.value = element.get('value');
2157
+ cp.validator = element.getUtfString('validator');
2158
+ cp.editable = (element.containsKey('edit') ? element.getBool('edit') : true);
2159
+ cp.trigger = (element.containsKey('trigger') ? element.getBool('trigger') : false);
2160
+ cp.triggerData = element.getSFSArray('triggerData');
2161
+ cp.clientOnly = (element.containsKey('clientOnly') ? element.getBool('clientOnly') : false);
2162
+ cp.dataProvider = element.getUtfString('dataProvider');
2163
+
2164
+ // Parse component specific attributes
2165
+ let tmpAttributes = element.getSFSObject('attributes');
2166
+ if (tmpAttributes != null)
2167
+ {
2168
+ let attributes = {};
2169
+
2170
+ let keys = tmpAttributes.getKeysArray();
2171
+ for (let key of keys)
2172
+ attributes[key] = tmpAttributes.get(key);
2173
+
2174
+ cp.attributes = attributes;
2175
+ }
2176
+
2177
+ // Parse separator settings
2178
+ let tmpSeparator = element.getSFSObject('separator');
2179
+ if (tmpSeparator != null)
2180
+ {
2181
+ let separator = {};
2182
+
2183
+ let keys1 = tmpSeparator.getKeysArray();
2184
+ for (let key1 of keys1)
2185
+ separator[key1] = tmpSeparator.get(key1);
2186
+
2187
+ cp.separator = separator;
2188
+ }
2189
+
2190
+ // Parse default list item
2191
+ let tmpDefaultListItem = element.getSFSArray('defaultListItem');
2192
+ if (tmpDefaultListItem != null && tmpDefaultListItem.size() > 0)
2193
+ {
2194
+ let defaultListItem = [];
2195
+
2196
+ for (let i = 0; i < tmpDefaultListItem.size(); i++)
2197
+ defaultListItem.push(ConfigurationParameter.fromSfsObject(tmpDefaultListItem.getSFSObject(i)));
2198
+
2199
+ cp.defaultListItem = defaultListItem;
2200
+
2201
+ // Parse list values
2202
+ let listValues = [];
2203
+
2204
+ let tmpListValues = element.getSFSArray('listValues');
2205
+ if (tmpListValues != null && tmpListValues.size() > 0)
2206
+ {
2207
+ for (let v = 0; v < tmpListValues.size(); v++)
2208
+ {
2209
+ let listValueObj = tmpListValues.getSFSObject(v);
2210
+ let obj = {};
2211
+
2212
+ let keys2 = listValueObj.getKeysArray();
2213
+ for (let key2 of keys2)
2214
+ obj[key2] = listValueObj.get(key2);
2215
+
2216
+ listValues.push(obj);
2217
+ }
2218
+ }
2219
+
2220
+ cp.listValues = listValues;
2221
+
2222
+ // If we have a list, on the server-side items could be represented by a class
2223
+ cp.clazz = element.getUtfString('clazz');
2224
+
2225
+ // Avoid list to be emptied
2226
+ cp.denyEmpty = (element.containsKey('denyEmpty') ? element.getBool('denyEmpty') : false);
2227
+ }
2228
+
2229
+ return cp;
2230
+ }
2231
+
2232
+ constructor()
2233
+ {
2234
+ /* CONSTANTS */
2235
+ this.DEFAULT_CATEGORY_NAME = 'General';
2236
+ this.DEFAULT_CATEGORY_ID = 'general';
2237
+
2238
+ /* PUBLIC VARS */
2239
+
2240
+ this.name = '';
2241
+ this.label = '';
2242
+ this.tooltip = '';
2243
+ this.type = null;
2244
+ this.trigger = false;
2245
+ this.triggerData = null;
2246
+ this.clientOnly = false;
2247
+ this.editable = true;
2248
+ this.attributes = null;
2249
+ this.dataProvider = null;
2250
+
2251
+ this.separator = null; // Parameter used to create a separator before or after the config parameter
2252
+ this.defaultListItem = null; // List of sub-ConfigurationParameters, each containing the default values
2253
+ this.clazz = null; // Name of the class representing the list item (not used in case of primiteve data types)
2254
+ this.denyEmpty = false; // Disallow to empty a list (DataGrid config parameter type only)
2255
+
2256
+ this.immediate = false; // Shows icon indicating that setting will be applied immediately on submit, without the need for a server restart
2257
+
2258
+ /* PRIVATE VARS */
2259
+
2260
+ this._category = this.DEFAULT_CATEGORY_NAME;
2261
+ this._categoryId = this.DEFAULT_CATEGORY_ID;
2262
+ this._value = null;
2263
+ this._initialValue = null; // Save the initial value of the configuration parameter, to check if the value was modified
2264
+ this._validator = null;
2265
+
2266
+ this._listItems = []; // Array of arrays of ConfigurationParameters
2267
+ this._listItemsChanged = false; // Flag to be set in case a list item is added or removed
2268
+ }
2269
+
2270
+ //---------------------------------------------
2271
+ // GETTERS / SETTERS
2272
+ //---------------------------------------------
2273
+
2274
+ set category(val)
2275
+ {
2276
+ if (val)
2277
+ {
2278
+ this._category = val;
2279
+ this._setIdFromCategoryName(this._category);
2280
+ }
2281
+ }
2282
+
2283
+ get category()
2284
+ {
2285
+ return this._category;
2286
+ }
2287
+
2288
+ set value(val)
2289
+ {
2290
+ if (this._value != val)
2291
+ {
2292
+ // If value is null, then we are setting this for the first time and
2293
+ // we want to save the initial value, to check later if it has been modified
2294
+ if (this._value == null)
2295
+ this._initialValue = val;
2296
+
2297
+ this._value = val;
2298
+ }
2299
+ }
2300
+
2301
+ get value()
2302
+ {
2303
+ return this._value;
2304
+ }
2305
+
2306
+ set validator(val)
2307
+ {
2308
+ if (val)
2309
+ this._validator = val;
2310
+ }
2311
+
2312
+ get validator()
2313
+ {
2314
+ return this._validator;
2315
+ }
2316
+
2317
+ /**
2318
+ * An array of objects; each object contains the name-value pairs used to
2319
+ * populate the list of sub-configuration parameters arrays, based on defaultListItem.
2320
+ */
2321
+ set listValues(arr)
2322
+ {
2323
+ this._setSubConfigurationParams(arr);
2324
+ }
2325
+
2326
+ get listValues()
2327
+ {
2328
+ return this._getSubConfigurationParamsValues();
2329
+ }
2330
+
2331
+ //---------------------------------------------
2332
+ // GETTERS ONLY
2333
+ //---------------------------------------------
2334
+
2335
+ get isModified()
2336
+ {
2337
+ let _isModified = false;
2338
+
2339
+ // If the parameter is used on the client only (for example in a custom trigger)
2340
+ // then we never have to consider it as modified, to prevent it being sent to the server
2341
+ if (!this.clientOnly)
2342
+ {
2343
+ if (this._value != this._initialValue || this._listItemsChanged)
2344
+ _isModified = true;
2345
+ else
2346
+ {
2347
+ // Check sub parameters
2348
+ outerLoop: for (let listItem of this._listItems)
2349
+ {
2350
+ for (let subCP of listItem)
2351
+ {
2352
+ if (subCP.isModified)
2353
+ {
2354
+ _isModified = true;
2355
+ break outerLoop;
2356
+ }
2357
+ }
2358
+ }
2359
+ }
2360
+ }
2361
+
2362
+ return _isModified;
2363
+ }
2364
+
2365
+ get categoryId()
2366
+ {
2367
+ return this._categoryId;
2368
+ }
2369
+
2370
+ get listItems()
2371
+ {
2372
+ return this._listItems;
2373
+ }
2374
+
2375
+ //---------------------------------------------
2376
+ // PUBLIC METHODS
2377
+ //---------------------------------------------
2378
+
2379
+ /**
2380
+ * Return a clone of this ConfigurationParameter.
2381
+ */
2382
+ clone(cloneValue = false)
2383
+ {
2384
+ let cp = new ConfigurationParameter();
2385
+ cp.name = this.name;
2386
+ cp.label = this.label;
2387
+ cp.category = this.category;
2388
+ cp.tooltip = this.tooltip;
2389
+ cp.type = this.type;
2390
+ cp.validator = this.validator;
2391
+ cp.trigger = this.trigger;
2392
+ cp.triggerData = (this.triggerData != null ? SFS2X.SFSArray.newFromBinaryData(this.triggerData.toBinary()) : null);
2393
+ cp.clientOnly = this.clientOnly;
2394
+ cp.dataProvider = this.dataProvider;
2395
+
2396
+ if (cloneValue)
2397
+ cp.value = this.value;
2398
+
2399
+ if (this.attributes != null)
2400
+ {
2401
+ cp.attributes = new Object();
2402
+ for (let s1 in this.attributes)
2403
+ cp.attributes[s1] = this.attributes[s1];
2404
+ }
2405
+
2406
+ if (this.separator != null)
2407
+ {
2408
+ cp.separator = new Object()
2409
+ for (let s2 in this.separator)
2410
+ cp.separator[s2] = this.separator[s2];
2411
+ }
2412
+
2413
+ if (this.defaultListItem != null)
2414
+ {
2415
+ let clonedDefaultListItems = [];
2416
+
2417
+ for (let subCP of this.defaultListItem)
2418
+ clonedDefaultListItems.push(subCP.clone(cloneValue));
2419
+
2420
+ cp.defaultListItem = clonedDefaultListItems;
2421
+ }
2422
+
2423
+ cp.listValues = this.listValues; // No need to clone this, as the listValues setter already does it
2424
+ cp.clazz = this.clazz;
2425
+ cp.denyEmpty = this.denyEmpty;
2426
+ cp.immediate = this.immediate;
2427
+
2428
+ return cp;
2429
+ }
2430
+
2431
+ /**
2432
+ * Reset initial value by copying the current value.
2433
+ */
2434
+ resetIsModified()
2435
+ {
2436
+ this._initialValue = this._value;
2437
+
2438
+ // Reset sub-parameters
2439
+ if (this._listItems != null)
2440
+ {
2441
+ for (let listItem of this._listItems)
2442
+ {
2443
+ for (let subCP of listItem)
2444
+ subCP.resetIsModified();
2445
+ }
2446
+ }
2447
+
2448
+ this._listItemsChanged = false;
2449
+ }
2450
+
2451
+ addListItem(newListItem)
2452
+ {
2453
+ this._listItems.push(newListItem);
2454
+ this._listItemsChanged = true;
2455
+ }
2456
+
2457
+ updateListItem(listItem, itemIndex)
2458
+ {
2459
+ this._listItems[itemIndex] = listItem;
2460
+ this._listItemsChanged = true;
2461
+ }
2462
+
2463
+ removeListItem(itemIndex)
2464
+ {
2465
+ this._listItems.splice(itemIndex, 1);
2466
+ this._listItemsChanged = true;
2467
+ }
2468
+
2469
+ toSfsObject()
2470
+ {
2471
+ let obj = new SFS2X.SFSObject();
2472
+
2473
+ // Set changed setting name
2474
+ obj.putUtfString('name', this.name);
2475
+
2476
+ // Set changed setting class, if any
2477
+ if (this.clazz != null)
2478
+ obj.putUtfString('clazz', this.clazz);
2479
+
2480
+ if (this.value != null)
2481
+ {
2482
+ // Set changed setting value
2483
+ if (typeof this.value === 'boolean')
2484
+ obj.putBool('value', this.value);
2485
+ else if (typeof this.value === 'number')
2486
+ obj.putInt('value', this.value);
2487
+ else
2488
+ obj.putText('value', this.value);
2489
+ }
2490
+ else
2491
+ {
2492
+ // Set changed setting list of values
2493
+
2494
+ let listItems = new SFS2X.SFSArray();
2495
+
2496
+ for (let a of this._listItems)
2497
+ {
2498
+ if (a.length == 1) // We have just one sub config param; no need to parse it complitely
2499
+ {
2500
+ // Simple list
2501
+ let tempObj = a[0].toSfsObject();
2502
+ let wa = tempObj.getWrappedItem('value');
2503
+ listItems.add(wa.value, wa.type);
2504
+ }
2505
+ else
2506
+ {
2507
+ // Complex list
2508
+
2509
+ let values = new SFS2X.SFSArray();
2510
+
2511
+ for (let subCp of a)
2512
+ values.addSFSObject(subCp.toSfsObject());
2513
+
2514
+ listItems.addSFSArray(values);
2515
+ }
2516
+ }
2517
+
2518
+ obj.putSFSArray('value', listItems);
2519
+ }
2520
+
2521
+ return obj;
2522
+ }
2523
+
2524
+ /**
2525
+ * Return a description of the ConfigurationParameter instance.
2526
+ */
2527
+ toString()
2528
+ {
2529
+ let s = ``;
2530
+ s += `Configuration parameter: ${this.name}\n`;
2531
+ s += `\ttype: ${this.type}\n`;
2532
+ s += `\tlabel: ${this.label}\n`;
2533
+ s += `\tcategory name: ${this.category}\n`;
2534
+ s += `\tcategory id: ${this.categoryId}\n`;
2535
+ s += `\timmediate: ${this.immediate}\n`;
2536
+ s += `\ttooltip: ${this.tooltip}\n`;
2537
+ s += `\tvalue: ${this.value}\n`;
2538
+ s += `\ttrigger: ${this.trigger}\n`;
2539
+ s += `\ttrigger data: ${this.triggerData}\n`;
2540
+ s += `\tclient only: ${this.clientOnly}\n`;
2541
+ s += `\tvalidator: ${this.validator}\n`;
2542
+ s += `\tis modified: ${this.isModified}\n`;
2543
+
2544
+ if (this.attributes != null)
2545
+ {
2546
+ s += `\tcomponent attributes:\n`;
2547
+
2548
+ for (let s1 in this.attributes)
2549
+ s += `\t\t${s1} --> ${this.attributes[s1]}\n`;
2550
+ }
2551
+
2552
+ if (this.dataProvider != null)
2553
+ s += `\tdata provider: ${this.dataProvider}\n`;
2554
+
2555
+ if (this.separator != null)
2556
+ {
2557
+ s += `\tseparator:\n`;
2558
+
2559
+ for (let s2 in this.separator)
2560
+ s += `\t\t${s2} --> ${this.separator[s2]}\n`;
2561
+ }
2562
+
2563
+ if (this._listItems != null && this._listItems.length > 0)
2564
+ {
2565
+ s += `\t# list items: ${this._listItems.length}\n`;
2566
+
2567
+ for (let i = 0; i < this._listItems.length; i++)
2568
+ {
2569
+ s += `\tlist item ${i} sub-parameters:\n`;
2570
+ for (let e = 0; e < this._listItems[i].length; e++)
2571
+ s += `\t\t${this._listItems[i][e].toCompactString()}\n`;
2572
+ }
2573
+
2574
+ s += `\tclass name: ${this.clazz}\n`;
2575
+ s += `\tdeny empty list: ${this.denyEmpty}\n`;
2576
+ }
2577
+
2578
+ return s;
2579
+ }
2580
+
2581
+ /**
2582
+ * Return a compact description of the ConfigurationParameter instance.
2583
+ */
2584
+ toCompactString()
2585
+ {
2586
+ return `Configuration parameter '${this.name}': ${this.value} ${this.isModified ? '[X]' : '[ ]'}`;
2587
+ }
2588
+
2589
+ //---------------------------------------------
2590
+ // PRIVATE METHODS
2591
+ //---------------------------------------------
2592
+
2593
+ /**
2594
+ * Retrieve the category id form the category name.
2595
+ * Spaces and invalid characters are removed; words are separated using capitals.
2596
+ */
2597
+ _setIdFromCategoryName(categoryName)
2598
+ {
2599
+ this._categoryId = categoryName;
2600
+
2601
+ // Strip invalid characters
2602
+ var pattern = /[^0-9a-zA-Z]/g;
2603
+ this._categoryId = this._categoryId.replace(pattern, ' ');
2604
+
2605
+ // Capitalize words
2606
+ var words = this._categoryId.split(' ');
2607
+ this._categoryId = '';
2608
+
2609
+ for (let i = 0; i < words.length; i++)
2610
+ {
2611
+ let word = words[i];
2612
+ if (word.length > 0)
2613
+ this._categoryId += (i > 0 ? word.substr(0,1).toUpperCase() : word.substr(0,1).toLowerCase()) + (word.length > 1 ? word.substr(1) : "");
2614
+ }
2615
+
2616
+ if (this._categoryId.length == 0)
2617
+ this._categoryId = this.DEFAULT_CATEGORY_ID;
2618
+ }
2619
+
2620
+ _setSubConfigurationParams(_listValues)
2621
+ {
2622
+ this._listItems = [];
2623
+
2624
+ for (let obj of _listValues)
2625
+ {
2626
+ let listItem = [];
2627
+
2628
+ for (let defaultCP of this.defaultListItem)
2629
+ {
2630
+ let subCP = defaultCP.clone(false);
2631
+ subCP.value = obj[subCP.name];
2632
+
2633
+ listItem.push(subCP);
2634
+ }
2635
+
2636
+ this._listItems.push(listItem);
2637
+ }
2638
+ }
2639
+
2640
+ _getSubConfigurationParamsValues()
2641
+ {
2642
+ let _listValues = [];
2643
+
2644
+ for (let listItem of this._listItems)
2645
+ {
2646
+ let obj = {};
2647
+
2648
+ for (let subCP of listItem)
2649
+ {
2650
+ if (subCP.value != null)
2651
+ obj[subCP.name] = subCP.value;
2652
+ }
2653
+
2654
+ _listValues.push(obj);
2655
+ }
2656
+
2657
+ return _listValues;
2658
+ }
2659
+ }
2660
+
2661
+
2662
+ /***/ })
2663
+
2664
+ }]);
2665
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-13~module-16~module-17~module-4.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\t_initialize()\n\t{\n\t\tif (this._data.editable)\n\t\t{\n\t\t\t// Initialize kendo widget\n\t\t\tkendo.init(this._widgetHtml);\n\n\t\t\t// Save ref. to widget\n\t\t\tthis._innerWidget = this._widgetHtml.data('kendoDropDownList');\n\n\t\t\t// Set list items\n\t\t\tthis._innerWidget.setDataSource(this._getDataSource(this._data.dataProvider))\n\n\t\t\t// Enable value commit binding\n\t\t\tthis._widgetHtml.bind('change', $.proxy(this._onValueInput, 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 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 happens, 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\tcase 'TextInput':\n\t\t\t\t\tclassNames = 'col-sm col-lg-5 col-xl-4';\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>${(this._data.immediate ? '<i class=\"fas fa-bolt text-muted ml-1 help\" title=\"This setting is applied <b>immediately</b> on submit, without requiring a server restart\"></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 == 'ipRange')\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]?)(\\/[3][0-2]|\\/[1-2]?[0-9])?$';\n\t\t\t\tconfig['data-pattern-msg'] = 'Invalid IP address or range';\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{\n\t\t\t// Always disable button if no list item is selected\n\t\t\tthis._removeButton.prop('disabled', selectedDataItems.length == 0);\n\n\t\t\t// Also disable button if denyEmpty == true and just one item remains in the list\n\t\t\tif (this._data.denyEmpty)\n\t\t\t\tthis._removeButton.prop('disabled', this._data.listItems.length <= 1);\n\t\t}\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\t\t\tonsubmit: 'return false;'\n\t\t});\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\n\t\t\t\t// form.on(\"keypress\", function(event) {\n\t\t\t\t// \t// If the user presses the \"Enter\" key on the keyboard\n\t\t\t\t// \tif (event.key === \"Enter\") {\n\t\t\t\t// \t\t// Cancel the default action, if needed\n\t\t\t\t// \t\tevent.preventDefault();\n\t\t\t\t// \t\tconsole.log(\"ENTER\")\n\t\t\t\t// \t}\n\t\t\t\t// });\n\t\t\t\tform.submit(function(evt) {\n\t\t\t\t\tconsole.log(\"SUBMIT\")\n\t\t\t\t\tevt.preventDefault();\n\t\t\t\t});\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.immediate = (element.containsKey('immediate') ? element.getBool('immediate') : false);\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\n\t\t\t// Avoid list to be emptied\n\t\t\tcp.denyEmpty = (element.containsKey('denyEmpty') ? element.getBool('denyEmpty') : false);\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// 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\t\tthis.denyEmpty = false;\t\t\t// Disallow to empty a list (DataGrid config parameter type only)\n\n\t\tthis.immediate = false;\t\t\t// Shows icon indicating that setting will be applied immediately on submit, without the need for a server restart\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\t\tcp.denyEmpty = this.denyEmpty;\n\t\tcp.immediate = this.immediate;\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 += `\\timmediate: ${this.immediate}\\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\ts += `\\tdeny empty list: ${this.denyEmpty}\\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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACnPA;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;;;;;;;;;;;;;;AClXA;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;AACA;AACA;;;;;;;;;;;;;;AC/EA;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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC1RA;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;;;;;A","sourceRoot":""}