rubyfox-server 2.17.3.1 → 2.19.2.0

Sign up to get free protection for your applications and to get access to all the features.
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 -264
  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-12~module-15~module-16~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-12~module-15~module-16~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":""}